Zend_Test_PHPUnit
bietet einen Testfall für MVC-Anwendungen,
der Zusicherungen für Tests auf eine Vielzahl von Verantwortlichkeiten enthält. Um
zu verstehen, was man damit machen kann, ist es wahrscheinlich am einfachsten, sich das folgende
Beispiel anzusehen.
Beispiel 896. Beispiel eines Testfalls für ein Anwendungs-Login
Das folgende ist ein einfacher Testfall für einen UserController
,
um verschiedene Dinge zu prüfen:
-
Das Login-Formular soll nicht-authentifizierten Benutzern angezeigt werden.
-
Wenn sich ein Benutzer einloggt, soll er zu seiner Profilseite umgeleitet werden und diese Profilseite soll relevante Informationen enthalten.
Dieses spezielle Beispiel setzt ein paar Dinge voraus. Zunächst verschieben wir das meiste unseres Bootstrappings in ein Plugin. Das vereinfacht das Setup des Testfalls, da es uns erlaubt, unsere Umgebung gezielt zu definieren und die Anwendung mit einer einzigen Zeile zu starten. Außerdem setzt unser spezielles Beispiel auch voraus, dass das automatische Laden von Klassen aktiviert ist, so dass wir uns nicht um das Laden der benötigten Klassen kümmern müssen (wie die richtigen Controller, Plugins, usw).
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { public function setUp() { $this->bootstrap = array($this, 'appBootstrap'); parent::setUp(); } public function appBootstrap() { $this->frontController ->registerPlugin(new Bugapp_Plugin_Initialize('development')); } public function testCallWithoutActionShouldPullFromIndexAction() { $this->dispatch('/user'); $this->assertController('user'); $this->assertAction('index'); } public function testIndexActionShouldContainLoginForm() { $this->dispatch('/user'); $this->assertAction('index'); $this->assertQueryCount('form#loginForm', 1); } public function testValidLoginShouldGoToProfilePage() { $this->request->setMethod('POST') ->setPost(array( 'username' => 'foobar', 'password' => 'foobar' )); $this->dispatch('/user/login'); $this->assertRedirectTo('/user/view'); $this->resetRequest() ->resetResponse(); $this->request->setMethod('GET') ->setPost(array()); $this->dispatch('/user/view'); $this->assertRoute('default'); $this->assertModule('default'); $this->assertController('user'); $this->assertAction('view'); $this->assertNotRedirect(); $this->assertQuery('dl'); $this->assertQueryContentContains('h2', 'User: foobar'); } }
Dieses Beispiel könnte auch einfacher geschrieben werden -- nicht alle der gezeigten Zusicherungen sind notwendig. Hoffentlich zeigt es, wie einfach es sein kann, die eigene Anwendung zu testen.
Wie im Login-Beispiel gezeigt, sollten
alle MVC-Testfälle
Zend_Test_PHPUnit_ControllerTestCase
erweitern. Diese Klasse
ihrerseits erweitert PHPUnit_Framework_TestCase
und gibt einem alle
Strukturen und Zusicherungen, die man von PHPUnit erwartet -- sowie einiges an Scaffolding und
Zusicherungen, die genau auf die Zend Framework MVC-Implementation zugeschnitten sind.
Um die eigene MVC-Anwendung zu testen, muß diese ein Bootstrap ausführen.
Es gibt verschiedene Wege, dies zu tun, wobei sich alle der öffentlichen
$bootstrap
-Eigenschaft bedienen.
Erstens und möglicherweise am zielgerichtetsten kann man einfach eine Instanz von
Zend_Application
erstellen, wie man es in der
index.php
machen würde und diese der $bootstrap
-Eigenschaft
zuweisen. Normalerweise macht man das in der setUp()
-Methode;
anschließend muss man parent::setUp()
aufrufen:
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { public function setUp() { // Zuordnen und Initiieren in einem Schritt: $this->bootstrap = new Zend_Application( 'testing', APPLICATION_PATH . '/configs/application.ini' ); parent::setUp(); } }
Zweitens kann diese Eigenschaft so gesetzt werden, dass sie auf eine Datei zeigt. Wenn dieser Weg gewählt wird, sollte diese Datei nicht den Front-Controller ausführen, sondern stattdessen den Front-Controller konfigurieren und alles, was die Anwendung an speziellen Anforderungen benötigt.
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { public $bootstrap = '/path/to/bootstrap/file.php' // ... }
Drittens kann ein PHP-Callback angegeben werden, der nach dem Bootstrap der Anwendung ausgeführt wird. Diese Methode kann im Login-Beispiel gesehen werden. Wenn das Callback eine Funktion oder statische Methode ist, könnte sie auch in der Klasse gesetzt werden:
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { public $bootstrap = array('App', 'bootstrap'); // ... }
In Fällen, in denen eine Objektinstanz notwendig ist, empfehlen wir die Durchführung in der
eigenen setUp()
-Methode:
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { public function setUp() { // Verwende die 'start' Methode einer Bootstrap-Objektinstanz: $bootstrap = new Bootstrap('test'); $this->bootstrap = array($bootstrap, 'start'); parent::setUp(); } }
Man beachte, dass parent::setUp()
aufgerufen wird; das ist notwendig, da
die setUp()
-Methode von
Zend_Test_PHPUnit_ControllerTestCase
den Rest des Bootstrap-Prozesses
durchführen wird (was den Aufruf des Callbacks einschließt).
Während der normalen Anwendung wird die setUp()
-Methode das
Bootstrap der Anwendung ausführen. Dieser Prozess wird zunächst das Löschen der Umgebung
enthalten, um einen sauberen Anfragestatus zu erhalten, das Zurücksetzen aller Plugins, Helfer
und Antwortobjekte. Sobald das getan wurde, wird sie anschließend die Datei mit
include()
laden, die in $bootstrap
angegeben
ist oder den spezifizierten Callback aufrufen.
Das Bootstrappen sollte so nahe wie möglich daran sein, wie die Anwendung das Bootstrap durchführt. Trotzdem gibt es einige Fallstricke:
-
Wir bieten keine alternative Implementierung der Anfrage- und Antwortobjekte; diese werden nicht verwendet.
Zend_Test_PHPUnit_ControllerTestCase
verwendet eigene Anfrage- und Antwortobjekte,Zend_Controller_Request_HttpTestCase
undZend_Controller_Response_HttpTestCase
. Diese Objekte stellen Methoden zur Verfügung, um die Anfrageumgebung gezielt aufzusetzen und um auf speziellem Weg die Antwort als Prüfgegenstand abzuholen. -
Man sollte nicht erwarten Server-spezifisches zu testen. Mit anderen Worten, die Tests garantieren nicht, dass der Code in einer speziellen Serverkonfiguration läuft, aber dass die Anwendung wie erwartet funktionieren sollte und der Router eine gegebene Anfrage routen kann. Aus diesem Grund sollten keine Server-spezifischen Header im Anfrageobjekt gesetzt werden.
Sobald die Anwendung das Bootstrapping ausgeführt hat, kann damit begonnen werden, eigene Tests zu erstellen.
Sobald man sein Bootstrap hat, kann man mit dem Testen beginnen. Testen funktioniert grundsätzlich so, wie man es in einer PHPUnit-TestSuite erwarten würde, mit ein paar kleinen Unterschieden.
Zuerst muss man eine URL ausführen, die getestet werden soll, indem die
dispatch()
-Methode des Testfalls ausgeführt wird:
class IndexControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { // ... public function testHomePage() { $this->dispatch('/'); // ... } }
Manchmal ist es trotzdem nötig, zusätzliche Informationen anzugeben --
GET
und POST Variablen, COOKIE Informationen, usw.
Man kann die Anfrage mit folgenden Informationen ausstatten:
class FooControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { // ... public function testBarActionShouldReceiveAllParameters() { // Setzt GET Variablen: $this->request->setQuery(array( 'foo' => 'bar', 'bar' => 'baz', )); // Setzt POST Variablen: $this->request->setPost(array( 'baz' => 'bat', 'lame' => 'bogus', )); // Setzt einen Cookie Wert: $this->request->setCookie('user', 'matthew'); // or many: $this->request->setCookies(array( 'timestamp' => time(), 'host' => 'foobar', )); // Setzt sogar Header: $this->request->setHeader('X-Requested-With', 'XmlHttpRequest'); // Setzt die Anfrage Methode: $this->request->setMethod('POST'); // Ausführung: $this->dispatch('/foo/bar'); // ... } }
Jetzt wurde die Anfrage durchgeführt, es ist also Zeit Zusicherungen zu prüfen.
Wichtig
Der Redirect Action Helper hat Probleme mit der Anweisung exit()
,
wenn die Methode gotoAndExit()
verwendet wird
und wird dann natürlich auch einen Test beenden, der für diese Methode läuft. Um die
eigene Anwendung testbar zu machen, sollte diese Methode nicht am Redirector
verwendet werden.
Durch seine Natur führt das Redirector Action Helper Plugin ein Redirect aus und steigt
nach diesem aus. Weil man Teile einer Anwendung nicht testen kann, wenn diese
Exit-Aufrufe durchführen, deaktiviert
Zend_Test_PHPUnit_ControllerTestCase
den Exit-Teil des
Redirectors automatisch, was zu unterschiedlichen Verhaltensweisen in Tests und echter Anwendung
führen kann. Um sicherzustellen, dass der Redirect richtig arbeitet, sollte man diesen
auf folgendem Weg durchführen:
class MyController extends Zend_Controller_Action { public function indexAction() { if ($someCondition == true) { return $this->_redirect(...); } else if ($anotherCondition == true) { $this->_redirector->gotoSimple("foo"); return; } // Mach was } }
Wichtig
Abhängig von der eigenen Anwendung kann es sein, dass das nicht genug ist, da eine
zusätzliche preDispatch()
- oder
postDispatch()
-Logik ausgeführt werden könnte. Das kann
aktuell mit Zend_Test auf keine vernünftige Weise behandelt werden.
Zusicherungen sind das Herz der UnitTests; sie können verwendet werden um zu prüfen, ob
die Ergebnisse das sind was man erwartet. Zu diesem Zweck bietet
Zend_Test_PHPUnit_ControllerTestCase
eine Anzahl an Zusicherungen, um
das Testen eigener MVC-Anwendungen und Controller einfacher zu machen.
CSS-Selektoren sind ein einfacher Weg um zu prüfen, dass bestimmte Teile im Inhalt der Antwort enthalten sind. Mit ihnen ist es auch trivial sicherzustellen, dass Elemente vorhanden sind, die für Javascript-UIs und/oder AJAX-Integrationen notwendig sind; die meisten JS-Toolkits bieten einige Mechanismen für das Abholen von DOM-Elementen an, die auf CSS-Selektoren basieren, so dass die Syntax die gleiche wäre.
Diese Funktionalität wird über Zend_Dom_Query angeboten und in ein Set von 'Query'-Zusicherungen integriert. Jede dieser Zusicherungen nimmt als erstes Argument einen CSS-Selektor mit optional hinzugefügten Argumenten und/oder einer Fehlermeldung, basierend auf dem Typ der Zusicherung. Die Regeln für das Schreiben der CSS-Selektoren kann im Kapitel Theorie der Anwendung von Zend_Dom_Query gefunden werden. Abfragezusicherungen enthalten:
-
assertQuery($path, $message)
: Nimmt an, dass ein oder mehrere DOM Elemente, die dem gegebenen CSS-Selektor entsprechen, vorhanden sind. Wenn eine$message
vorhanden ist, wird diese jeder fehlgeschlagenen Meldung einer Zusicherung vorangestellt. -
assertQueryContentContains($path, $match, $message)
: Nimmt an, dass ein oder mehrere DOM Elemente, die dem angegebenen CSS-Selektor entsprechen, vorhanden sind, und dass zumindest einer dem Inhalt entspricht, der in$match
angegeben wurde. Wenn eine$message
vorhanden ist, wird diese jeder fehlgeschlagenen Meldung einer Zusicherung vorangestellt. -
assertQueryContentRegex($path, $pattern, $message)
: Nimmt an, dass ein oder mehrere DOM-Elemente vorhanden sind, die dem angegebenen CSS-Selektor entsprechen und dass zumindest einer dem Regulären Ausdruck entspricht, der in$pattern
angegeben wurde, Wenn eine$message
vorhanden ist, wird diese jeder fehlgeschlagenen Meldung einer Zusicherung vorangestellt. -
assertQueryCount($path, $count, $message)
: Nimmt an, dass exakt$count
DOM-Elemente dem angegebenen CSS Selektor entsprechen. Wenn eine$message
vorhanden ist, wird diese jeder fehlgeschlagenen Meldung einer Zusicherung vorangestellt. -
assertQueryCountMin($path, $count, $message)
: Nimmt an, dass zumindest$count
DOM-Element dem angegebenen CSS Selektor entsprechen. Wenn eine$message
vorhanden ist, wird diese jeder fehlgeschlagenen Meldung einer Zusicherung vorangestellt. Achtung: Die Spezifizierung eines Wertes von 1 für$count
ist das Gleiche wie die einfache Verwendung vonassertQuery()
. -
assertQueryCountMax($path, $count, $message)
: Nimmt an, dass es nicht mehr als$count
DOM-Elemente gibt, die dem angegebenen CSS-Selektor entsprechen. Wenn eine$message
vorhanden ist, wird diese jeder fehlgeschlagenen Meldung einer Zusicherung vorangestellt. Achtung: Die Spezifizierung eines Wertes von 1 für$count
ist das Gleiche wie die einfache Verwendung vonassertQuery()
.
Zusätzlich hat jede der obigen Methoden eine 'Not'-Variante, die eine negative Zusicherung
anbietet: assertNotQuery()
,
assertNotQueryContentContains()
,
assertNotQueryContentRegex()
und
assertNotQueryCount()
. (Es ist zu beachten, dass die min und max
Zählen keine dieser Varianten haben, was aus logischen Gründen so ist.)
Einige Entwickler sind mit XPath vertrauter als mit CSS-Selektoren, und deshalb werden für alle Abfrage Zusicherungen auch XPath-Varianten engeboten. Diese sind:
-
assertXpath($path, $message = '')
-
assertNotXpath($path, $message = '')
-
assertXpathContentContains($path, $match, $message = '')
-
assertNotXpathContentContains($path, $match, $message = '')
-
assertXpathContentRegex($path, $pattern, $message = '')
-
assertNotXpathContentRegex($path, $pattern, $message = '')
-
assertXpathCount($path, $count, $message = '')
-
assertNotXpathCount($path, $count, $message = '')
-
assertXpathCountMin($path, $count, $message = '')
-
assertNotXpathCountMax($path, $count, $message = '')
Oft wird eine Aktion umgeleitet. Statt der Umleitung zu folgen, erlaubt es
Zend_Test_PHPUnit_ControllerTestCase
, diese Umleitungen mit einer
handvoll von Zusicherungen zu Testen.
-
assertRedirect($message = '')
: Nimmt einfach an, dass eine Umleitung stattgefunden hat. -
assertNotRedirect($message = '')
: Nimmt einfach an, dass keine Umleitung stattgefunden hat. -
assertRedirectTo($url, $message = '')
: Nimmt an, dass eine Umleitung stattgefunden hat und dass der Wert des Ziel-Headers die angegebene$url
ist. -
assertNotRedirectTo($url, $message = '')
: Nimmt an, dass eine Umleitung entweder NICHT stattgefunden hat oder dass der Wert des Ziel-Headers NICHT die angegebene$url
ist. -
assertRedirectRegex($pattern, $message = '')
: Nimmt an, dass eine Umleitung stattgefunden hat und dass der Wert des Ziel-Headers dem durch$pattern
angegebenen regulären Ausdruck entspricht. -
assertNotRedirectRegex($pattern, $message = '')
: Nimmt an, dass eine Umleitung entweder NICHT stattgefunden hat oder dass der Wert des Ziel-Headers NICHT dem durch$pattern
angegebenen regulären Ausdruck entspricht.
Zusätzlich zur Prüfung auf Umleitungs-Header, ist es oft notwendig auf spezielle HTTP-Antwort-Codes und -Header zu prüfen -- zum Beispiel, um zu erkennen, ob eine Aktion eine 404 oder 500 Antwort hervorruft oder um sicherzustellen, dass JSON-Antworten die entsprechenden Content-Type-Header enthält. Die folgenden Zusicherungen sind vorhanden.
-
assertResponseCode($code, $message = '')
: Nimmt an, dass die Antwort zum gegebenen HTTP-Antwort-Code geführt hat. -
assertHeader($header, $message = '')
: Nimmt an, dass die Antwort den gegebenen Header enthält. -
assertHeaderContains($header, $match, $message)
: Nimmt an, dass die Antwort den gegebenen Header enthält und dass sein Inhalt den gegebenen String enthält. -
assertHeaderRegex($header, $pattern, $message)
: Nimmt an, dass die Antwort den gegebenen Header enthält und dass sein Inhalt der gegebenen Regex entspricht.
Zusätzlich hat jede der obigen Zusicherungen eine 'Not'-Variante für negative Zusicherungen.
Es ist oft sinnvoll gegen die letzte Aktion, den Controller und das Modul zu prüfen; zusätzlich ist es möglich die genommene Route die prüfen. Die folgenden Zusicherungen können in diesen Fällen helfen:
-
assertModule($module, $message = '')
: Nimmt an, dass das angegebene Modul in der letzten Dispatch-Aktion verwendet wurde. -
assertController($controller, $message = '')
: Nimmt an, dass der angegebene Controller in der letzten ausgeführten Aktion ausgewählt wurde. -
assertAction($action, $message = '')
: Nimmt an, dass die angegebene Aktion zuletzt ausgeführt wurde. -
assertRoute($route, $message = '')
: Nimmt an, dass die angegebene benannte Route dem Router entsprochen hat.
Jede hat auch eine 'Not'-Variante für negative Zusicherungen.
Zu wissen, wie man die eigene Infrastruktur für Tests einstellt und wie Zusicherungen zu erstellen sind, ist nur die halbe Miete; jetzt ist es Zeit, sich einige Testszenarien anzuschauen und herauszufinden, wie diese wirksam eingesetzt werden können.
Beispiel 897. Den UserController testen
Betrachten wir eine Standardaufgabe für eine Webseite: Authentifizierung und Registrierung von Benutzern. In unserem Beispiel definieren wir einen UserController, um das zu behandeln und haben die folgenden Anforderungen:
-
Wenn ein Benutzer nicht authentifiziert ist, wird er immer zur Login-Seite des Controllers umgeleitet, unabhängig von der angeforderten Aktion.
-
Die Login-Formularseite wird sowohl das Login-Formular als auch das Registrationsformular anzeigen.
-
Die Angabe von ungültigen Anmeldedaten soll zur Anzeige des Login-Formulars führen.
-
Das Ansehen der Anmeldedaten soll zu einer Umleitung zur Profilseite des Benutzers führen.
-
Die Profilseite soll angepasst werden, um den Benutzernamen des Benutzers anzuzeigen.
-
Authentifizierte Benutzer, welche die Loginseite besuchen, sollen zu ihrer Profilseite umgeleitet werden.
-
Bei der Abmeldung soll ein Benutzer zur Loginseite umgeleitet werden.
-
Mit ungültigen Daten soll die Registrierung fehlschlagen.
Wir können und sollten zusätzliche Tests definieren, aber diese reichen vorerst aus.
Für unsere Anwendung definieren wir ein Plugin, 'Initialisieren' es, damit es bei
routeStartup()
läuft. Das erlaubt es uns, das Bootstrapping in
einem OOP-Interface zu kapseln, was auch einen einfachen Weg bietet, um ein Callback zu
ermöglichen. Schauen wir uns erstmals die Grundlagen dieser Klasse an:
class Bugapp_Plugin_Initialize extends Zend_Controller_Plugin_Abstract { /** * @var Zend_Config */ protected static $_config; /** * @var string Aktuelle Umgebung */ protected $_env; /** * @var Zend_Controller_Front */ protected $_front; /** * @var string Pfad zum Root der Anwendung */ protected $_root; /** * Constructor * * Umgebung, Root Pfad und Konfiguration initialisieren * * @param string $env * @param string|null $root * @return void */ public function __construct($env, $root = null) { $this->_setEnv($env); if (null === $root) { $root = realpath(dirname(__FILE__) . '/../../../'); } $this->_root = $root; $this->initPhpConfig(); $this->_front = Zend_Controller_Front::getInstance(); } /** * Route beginnen * * @return void */ public function routeStartup(Zend_Controller_Request_Abstract $request) { $this->initDb(); $this->initHelpers(); $this->initView(); $this->initPlugins(); $this->initRoutes(); $this->initControllers(); } // Die Definition von Methoden würde hier folgen... }
Das erlaubt es uns einen Bootstrap-Callback wie folgt zu erstellen:
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { public function appBootstrap() { $controller = $this->getFrontController(); $controller->registerPlugin( new Bugapp_Plugin_Initialize('development') ); } public function setUp() { $this->bootstrap = array($this, 'appBootstrap'); parent::setUp(); } // ... }
Sobald das fertig ist, können wir unsere Tests schreiben. Was ist jedoch mit den
Tests, die erfordern, dass der Benutzer angemeldet ist? Die einfache Lösung besteht darin,
dass unsere Anwendungslogik das macht... und ein bisschen trickst, indem die Methoden
resetRequest()
und resetResponse()
verwendet werden, die es uns erlauben eine andere Anfrage abzusetzen.
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { // ... public function loginUser($user, $password) { $this->request->setMethod('POST') ->setPost(array( 'username' => $user, 'password' => $password, )); $this->dispatch('/user/login'); $this->assertRedirectTo('/user/view'); $this->resetRequest() ->resetResponse(); $this->request->setPost(array()); // ... } // ... }
Jetzt schreiben wir Tests:
class UserControllerTest extends Zend_Test_PHPUnit_ControllerTestCase { // ... public function testCallWithoutActionShouldPullFromIndexAction() { $this->dispatch('/user'); $this->assertController('user'); $this->assertAction('index'); } public function testLoginFormShouldContainLoginAndRegistrationForms() { $this->dispatch('/user'); $this->assertQueryCount('form', 2); } public function testInvalidCredentialsShouldResultInRedisplayOfLoginForm() { $request = $this->getRequest(); $request->setMethod('POST') ->setPost(array( 'username' => 'bogus', 'password' => 'reallyReallyBogus', )); $this->dispatch('/user/login'); $this->assertNotRedirect(); $this->assertQuery('form'); } public function testValidLoginShouldRedirectToProfilePage() { $this->loginUser('foobar', 'foobar'); } public function testAuthenticatedUserShouldHaveCustomizedProfilePage() { $this->loginUser('foobar', 'foobar'); $this->request->setMethod('GET'); $this->dispatch('/user/view'); $this->assertNotRedirect(); $this->assertQueryContentContains('h2', 'foobar'); } public function testAuthenticatedUsersShouldBeRedirectedToProfileWhenVisitingLogin() { $this->loginUser('foobar', 'foobar'); $this->request->setMethod('GET'); $this->dispatch('/user'); $this->assertRedirectTo('/user/view'); } public function testUserShouldRedirectToLoginPageOnLogout() { $this->loginUser('foobar', 'foobar'); $this->request->setMethod('GET'); $this->dispatch('/user/logout'); $this->assertRedirectTo('/user'); } public function testRegistrationShouldFailWithInvalidData() { $data = array( 'username' => 'This will not work', 'email' => 'this is an invalid email', 'password' => 'Th1s!s!nv@l1d', 'passwordVerification' => 'wrong!', ); $request = $this->getRequest(); $request->setMethod('POST') ->setPost($data); $this->dispatch('/user/register'); $this->assertNotRedirect(); $this->assertQuery('form .errors'); } }
Es ist zu beachten, dass die Tests knapp sind und größtenteils nicht den aktuellen Inhalt suchen. Stattdessen suchen sie nach Teilen in der Anfrage -- Anfrage Codes und Header sowie DOM-Knoten. Das erlaubt es schnell zu prüfen, dass die Strukturen wie erwartet sind -- und verhindern, dass die Tests jedesmal scheitern, wenn der Site neue Inhalte hinzugefügt werden.
Es ist auch zu beachten, dass wir die Struktur des Dokuments in unseren Tests verwenden. Zum Beispiel suchen wir im letzten Test nach einer Form, die einen Knoten der Klasse "errors" hat; das erlaubt es uns lediglich auf das Vorhandensein von Form-Prüfungsfehlern zu testen und uns keine Sorgen darüber zu machen, warum spezielle Fehler überhaupt geworfen werden.
Diese Anwendung könnte eine Datenbank verwenden. Wenn dem so ist, muss man wahrscheinlich einige Grundlagen ändern um sicherzustellen, dass die Datenbank am Anfang jedes Tests in einer unverfälschten, testbaren Konfiguration ist. PHPUnit bietet bereits Funktionalität um das sicherzustellen; Lesen Sie darüber in der PHPUnit-Dokumentation nach. Wir empfehlen eine separate Datenbank für das Testen zu verwenden statt der Produktionsdatenbank und entweder eine SQLite-Datei oder eine Datenbank im Speicher zu verwenden, da beide Optionen sehr performant sind, keinen separaten Server benötigen und die meisten SQL-Syntax verwenden können.