22. Anwendungsübung – Version 11
Es ist nach wie vor üblich, dass Webdienste ihre Antwort als XML-Feed statt als JSON-Feed senden:
- Der JSON-Stream ist schlanker, aber man benötigt eine Anleitung, um ihn zu verstehen;
- der XML-Feed ist ausführlicher, aber selbsterklärend. Er ist sofort verständlich;
Wir passen die Client/Server-Version 11 so an, dass der Server nun einen XML-Feed als Antwort an seine Clients sendet:

22.1. Der Server

Diese Architektur wird durch die folgenden Skripte implementiert:

22.1.1. Die Klasse [Utilities]
Wir verwenden die seit Version 03 verwendete Klasse [Utilities] wieder (siehe verlinkten Absatz):
<?php
// namespace
namespace Application;
// a class of utility functions
abstract class Utilitaires {
public static function cutNewLinechar(string $ligne): string {
…
}
// from https://stackoverflow.com/questions/1397036/how-to-convert-array-to-simplexml
public static function getXmlForArrayOfAttributes(array $arrayOfAttributes,
\SimpleXmlElement &$node): void {
// scan array attributes
foreach ($arrayOfAttributes as $attribute => $value) {
// is the attribute numeric?
if (is_numeric($attribute)) {
// table index case (but also other cases)
$attribute = 'i' . $attribute;
}
// is $value an array?
if (is_array($value)) {
// we'll explore the [$value] array in turn
// we add a node to the XML graph
$subnode = $node->addChild($attribute);
// recursive call to explore the [$value] array
Utilitaires::getXmlForArrayOfAttributes($value, $subnode);
} else {
// we add the node to the XML graph
$node->addChild("$attribute", htmlspecialchars("$value"));
}
}
}
}
Kommentare
- Zeilen 14–36: Wir führen die statische Methode [getXmlForArrayOfAttributes] ein, die den XML-String eines als Parameter übergebenen Arrays [arrayOfAttributes] zurückgibt. Der zweite Parameter ist die Referenz auf einen XML-Graphknoten vom Typ [SimpleXmlElement]. Nach der Ausführung enthält dieser Knoten den XML-Baum des Arrays [arrayOfAttributes];
Wir schreiben den folgenden Test [testXml.php]:

<?php
// dependency
require __DIR__ . "/Utilitaires.php";
// associative table
$array = ["nom" => "amédée", "prénom" => "sylvain", "âge" => 40,
"enfants" => [["nom" => "amédée", "prénom" => "béatrice", "âge" => 6],
["nom" => "amédée", "prénom" => "bertrand", "âge" => 4]]];
// xml
header("Content-Type: application/xml");
$node = new \SimpleXMLElement("<?xml version='1.0' encoding='UTF-8'?><root></root>");
\Application\Utilitaires::getXmlForArrayOfAttributes($array, $node);
print $node->asXML();
Wenn wir dieses Skript [2] ausführen, erhalten wir im Chrome-Browser Folgendes:

22.1.2. Das Server-Skript
Das Serverskript [impots-server.php] muss ebenso wie seine Konfigurationsdatei [config-server.json] geändert werden:
{
"rootDirectory": "C:/myprograms/laragon-lite/www/php7/scripts-web/impots/version-11",
"databaseFilename": "Data/database.json",
"relativeDependencies": [
"/../version-08/Entities/BaseEntity.php",
"/../version-08/Entities/ExceptionImpots.php",
"/../version-08/Entities/TaxAdminData.php",
"/../version-08/Entities/Database.php",
"/../version-08/Dao/InterfaceServerDao.php",
"/../version-08/Dao/ServerDao.php",
"/../version-09/Dao/ServerDaoWithSession.php",
"/../version-08/Métier/InterfaceServerMetier.php",
"/../version-08/Métier/ServerMetier.php",
"/../version-09/Utilities/Logger.php",
"/../version-09/Utilities/SendAdminMail.php",
"/Utilities/Utilitaires.php"
],
"absoluteDependencies": [
"C:/myprograms/laragon-lite/www/vendor/autoload.php",
"C:/myprograms/laragon-lite/www/vendor/predis/predis/autoload.php"
],
"users": [
{
"login": "admin",
"passwd": "admin"
}
],
"adminMail": {
"smtp-server": "localhost",
"smtp-port": "25",
"from": "guest@localhost",
"to": "guest@localhost",
"subject": "plantage du serveur de calcul d'impôts",
"tls": "FALSE",
"attachments": []
},
"logsFilename": "Data/logs.txt"
}
Kommentare
- Das Projektverzeichnis ist nun der Ordner „Version 11“;
- Zeile 16: Die neue Klasse [Utilities] wurde hinzugefügt;
Die Änderungen am Server-Skript lauten wie folgt:
<?php
// strict adherence to declared types of function parameters
declare (strict_types=1);
// namespace
namespace Application;
…
// prepare JSON server response
$response = new Response();
$response->headers->set("content-type", "application/xml");
$response->setCharset("utf-8");
…
// creation of the [business] layer
$métier = new ServerMetier($dao);
// tAX CALCULATION
$result = $métier->calculerImpot($marié, (int) $enfants, (int) $salaire);
// we return the answer
sendResponse($response, $result, Response::HTTP_OK, [], $logger, $redis);
// end
exit;
function doInternalServerError(string $message, Response $response, array $infos,
…
}
// function to send HTTP response to client
function sendResponse(Response $response, array $result, int $statusCode,
array $headers, Logger $logger = NULL, \Predis\Client $predisClient = NULL) {
// $response : answer HTTP
// $result: results table
// $statusCode: HTTP response status
// $headers: HTTP headers to be included in the response
// $logger: application logger
// $predisClient: a customer [predis]
//
// status HTTTP
$response->setStatusCode($statusCode);
// body XML
$node = new \SimpleXMLElement("<?xml version='1.0' encoding='UTF-8'?><réponse></réponse>");
Utilitaires::getXmlForArrayOfAttributes($result, $node);
$response->setContent($node->asXML());
// headers
$response->headers->add($headers);
// shipping
$response->send();
// log
if ($logger != NULL) {
// log in jSON
$log = \json_encode(["réponse" => $result], JSON_UNESCAPED_UNICODE);
$logger->write("$log\n");
$logger->close();
}
// close connection [redis]
if ($predisClient != NULL) {
$predisClient->disconnect();
}
}
Kommentare
- Zeile 12: gibt an, dass die Antwort vom Typ [application/xml] ist;
- Zeilen 29–59: Die Serverantwort ist nun XML;
- Zeile 41: Erstellung des Stammknotens [<response></response>] des XML-Baums;
- Zeile 42: Dieser Baum wird mit dem XML-Baum aus dem Array [$result] der an den Client zu sendenden Ergebnisse gefüllt;
- Zeile 43: Der XML-Baum wird in eine XML-Zeichenkette konvertiert, um an den Client gesendet zu werden;
Test
Geben Sie direkt in einem Chrome-Browser die URL [http://localhost/php7/scripts-web/impots/version-11/impots-server.php?mari%C3%A9=oui&enfants=2&salaire=60000] ein. Das folgende Ergebnis [1] wird in einem Chrome-Browser angezeigt:

22.2. Der Client
Wir konzentrieren uns nun auf die clientseitige Seite der Anwendung.

Diese Architektur wird durch die folgenden Skripte implementiert:

In der neuen Version gibt es lediglich folgende Änderungen:
- die Konfigurationsdatei [config-client.json];
- die [dao]-Ebene des Clients;
Die Konfigurationsdatei [config-client.json] sieht nun wie folgt aus:
{
"rootDirectory": "C:/Data/st-2019/dev/php7/poly/scripts-console/impots/version-11",
"taxPayersDataFileName": "Data/taxpayersdata.json",
"resultsFileName": "Data/results.json",
"errorsFileName": "Data/errors.json",
"dependencies": [
"/../version-08/Entities/BaseEntity.php",
"/../version-08/Entities/TaxPayerData.php",
"/../version-08/Entities/ExceptionImpots.php",
"/../version-08/Utilities/Utilitaires.php",
"/../version-08/Dao/InterfaceClientDao.php",
"/../version-08/Dao/TraitDao.php",
"/Dao/ClientDao.php",
"/../version-08/Métier/InterfaceClientMetier.php",
"/../version-08/Métier/ClientMetier.php"
],
"absoluteDependencies": [
"C:/myprograms/laragon-lite/www/vendor/autoload.php"
],
"user": {
"login": "admin",
"passwd": "admin"
},
"urlServer": "https://localhost:443/php7/scripts-web/impots/version-11/impots-server.php"
}
22.2.1. Die [dao]-Schicht
Der Client [ClientDao.php] (Zeile 13 oben) wird angepasst, um das neue Antwortformat zu berücksichtigen. Wir verwenden [simpleXML], um es zu verarbeiten:
<?php
namespace Application;
// dependencies
use \Symfony\Component\HttpClient\HttpClient;
class ClientDao implements InterfaceClientDao {
// using a Trait
use TraitDao;
// attributes
private $urlServer;
private $user;
private $sessionCookie;
// manufacturer
public function __construct(string $urlServer, array $user) {
$this->urlServer = $urlServer;
$this->user = $user;
}
// tAX CALCULATION
public function calculerImpot(string $marié, int $enfants, int $salaire): array {
…
// we retrieve the XML response
$réponse = $response->getContent(false);
$xml = new \SimpleXMLElement($réponse);
// logs
// print "$réponse";
// retrieve response status
$statusCode = $response->getStatusCode();
// mistake?
if ($statusCode !== 200) {
// we have an error - we throw an exception
$message = \json_encode(["statut HTTP" => $statusCode, "réponse" => $xml], JSON_UNESCAPED_UNICODE);
throw new ExceptionImpots($message);
}
if (!$this->sessionCookie) {
// retrieve the session cookie
$headers = $response->getHeaders();
if (isset($headers["set-cookie"])) {
// session cookie ?
foreach ($headers["set-cookie"] as $cookie) {
$match = [];
$match = preg_match("/^PHPSESSID=(.+?);/", $cookie, $champs);
if ($match) {
$this->sessionCookie = "PHPSESSID=" . $champs[1];
}
}
}
}
// the answer is given in the form of a table
return \json_decode(\json_encode($xml, JSON_UNESCAPED_UNICODE), true);
}
}
Kommentare
- Zeilen 26–27: Die Antwort des Servers wird gelesen. Es handelt sich um ein XML-Dokument [<response>…</response>]. Aus dem empfangenen XML-Dokument wird ein [SimpleXMLElement]-Objekt erstellt;
- Zeilen 33–37: Im Falle eines Fehlers ist die Ausnahmemeldung der JSON-String der Serverantwort und nicht der XML-String. Der Grund dafür ist, dass der JSON-String prägnanter ist;
- Zeile 53: Das Ergebnis-Array wird in zwei Schritten zurückgegeben:
- Das [$xml]-Objekt vom Typ [\SimpleXMLElement] wird in JSON konvertiert;
- Wir wandeln die resultierende JSON-Zeichenkette in ein assoziatives Array um. Dies ist das zurückzugebende Ergebnis;
Test
Wenn wir den Client in einer geeigneten Umgebung ausführen (Datenbank, Authentifizierung, Protokolle), erhalten wir die üblichen Ergebnisse (siehe die Dateien [taxpayersdata.json, results.txt, errors.json]). Auf der Serverseite sehen die Protokolle wie folgt aus:
06/07/19 07:41:32:877 :
---nouvelle requête
06/07/19 07:41:32:882 : Autentification en cours…
06/07/19 07:41:32:883 : Authentification réussie [admin, admin]
06/07/19 07:41:32:883 : paramètres ['marié'=>oui, 'enfants'=>2, 'salaire'=>55555] valides
06/07/19 07:41:32:908 : données fiscales prises en base de données
06/07/19 07:41:32:959 : {"réponse":{"impôt":2814,"surcôte":0,"décôte":0,"réduction":0,"taux":0.14}}
06/07/19 07:41:33:070 :
---nouvelle requête
06/07/19 07:41:33:077 : Authentification prise en session…
06/07/19 07:41:33:077 : paramètres ['marié'=>oui, 'enfants'=>2, 'salaire'=>50000] valides
06/07/19 07:41:33:099 : données fiscales prises dans redis
06/07/19 07:41:33:100 : {"réponse":{"impôt":1384,"surcôte":0,"décôte":384,"réduction":347,"taux":0.14}}
06/07/19 07:41:33:189 :
---nouvelle requête
06/07/19 07:41:33:202 : Authentification prise en session…
06/07/19 07:41:33:202 : paramètres ['marié'=>oui, 'enfants'=>3, 'salaire'=>50000] valides
06/07/19 07:41:33:233 : données fiscales prises dans redis
06/07/19 07:41:33:233 : {"réponse":{"impôt":0,"surcôte":0,"décôte":720,"réduction":0,"taux":0.14}}
06/07/19 07:41:33:318 :
…
22.2.2. Tests [Codeception]

Der [ClientMetierTest]-Test sieht wie folgt aus:
<?php
// strict adherence to declared types of function parameters
declare (strict_types=1);
// namespace
namespace Application;
// definition of constants
define("ROOT", "C:/Data/st-2019/dev/php7/poly/scripts-console/impots/version-11");
// configuration file path
define("CONFIG_FILENAME", ROOT . "/Data/config-client.json");
// we retrieve the configuration
$config = \json_decode(file_get_contents(CONFIG_FILENAME), true);
…
// test class
class ClientMetierTest extends Unit {
// business layer
private $métier;
public function __construct() {
parent::__construct();
// we retrieve the configuration
$config = \json_decode(\file_get_contents(CONFIG_FILENAME), true);
// creation of the [dao] layer
$clientDao = new ClientDao($config["urlServer"], $config["user"]);
// creation of the [business] layer
$this->métier = new ClientMetier($clientDao);
}
// tests
…
}
Die Testergebnisse lauten wie folgt:
