Zend_Session-AdvancedUsage.xml 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 15157 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.session.advanced_usage">
  5. <title>Fortgeschrittene Benutzung</title>
  6. <para>
  7. Wärend die Beispiele für die Basisnutzung völlig akzeptierbar sind, in ihrem Weg Zend Framework Sessions
  8. zu Benutzen, ist auch die beste Praxis zu bedenken. Diese Sektion beschreibt die näheren Details von
  9. Session Handling und illustriert die fortgeschrittene Benutzung der Zend_Session Komponente.
  10. </para>
  11. <sect2 id="zend.session.advanced_usage.starting_a_session">
  12. <title>Starten einer Session</title>
  13. <para>
  14. Wenn man alle Anfragen einer Session durch Zend_Session bearbeitet haben will,
  15. muß die Session in der Bootstrap Datei gestartet werden:
  16. </para>
  17. <example id="zend.session.advanced_usage.starting_a_session.example">
  18. <title>Starten einer globalen Session</title>
  19. <programlisting role="php"><![CDATA[
  20. Zend_Session::start();
  21. ]]></programlisting>
  22. </example>
  23. <para>
  24. Durch das Starten der Session in der Bootstrap Datei verhindert man das die Session gestartet werden
  25. könnte nachdem die Header an den Browser gesendet wurde, was zu einer Ausnahme und möglicherweise
  26. zu einer fehlerhaften Seiten im Browser führen würde. Viele gehobenen Features benötigen zuerst
  27. <classname>Zend_Session::start()</classname>. (Mehr dazu später in den gehobenen Features)
  28. </para>
  29. <para>
  30. Es gibt vier Wege eine Session zustarten wenn Zend_Session verwendet wird. Zwei sind falsch.
  31. </para>
  32. <orderedlist>
  33. <listitem>
  34. <para>
  35. Falsch: PHP's
  36. <ulink url="http://www.php.net/manual/de/ref.session.php#ini.session.auto-start"><code> session.auto_start</code>
  37. Einstellung</ulink> darf nicht eingeschaltet werden. Wenn keine Möglichkeit besteht
  38. diese Einstellung in php.ini zu deaktivieren, und mod_php (oder ähnliches) verwendet wird,
  39. und die Einstellung schon in <code>php.ini</code> aktiviert ist, kann das folgende in der
  40. <code>.htaccess</code> Datei (normalerweise im HTML Dokumenten Basisverzeichnis) hinzugefügt
  41. werden:
  42. <programlisting role="httpd.conf"><![CDATA[
  43. php_value session.auto_start 0
  44. ]]>
  45. </programlisting>
  46. </para>
  47. </listitem>
  48. <listitem>
  49. <para>
  50. Falsch: PHP's <ulink url="http://www.php.net/session_start"><code>session_start()</code></ulink>
  51. Funktion darf nicht direkt verwendet werden. Wenn <code>session_start()</code> direkt, und
  52. anschließend <classname>Zend_Session_Namespace</classname> verwendet wird, wird von
  53. <classname>Zend_Session::start()</classname> eine Ausnahme geworfen ("session has already been started").
  54. Wenn <classname>Zend_Session::start()</classname> ausgerufen wird, nachdem
  55. <classname>Zend_Session_Namespace</classname> verwendet wird oder <classname>Zend_Session::start()</classname>
  56. explizit verwendet wird, wird ein Fehler vom Level <code>E_NOTICE</code> erzeugt und der Aufruf
  57. wird ignoriert.
  58. </para>
  59. </listitem>
  60. <listitem>
  61. <para>
  62. Richtig: Verwenden von <classname>Zend_Session::start()</classname>. Wenn es gewünscht ist, das alle
  63. Anfragen eine Session haben und verwenden, sollte diese Funktion sehr früh, direkt und
  64. entscheidungslos in der Bootstrap Datei aufgerufen werden. Session haben einigen Overhead.
  65. Wenn einige Anfragen Sessions benötigen aber andere Anfragen keine Sessions verwenden, dann:
  66. </para>
  67. <itemizedlist mark="opencircle">
  68. <listitem>
  69. <para>
  70. Entscheidungslos, die <code>strict</code> Option auf <code>true</code> setzen durch
  71. Verwendung von <classname>Zend_Session::setOptions()</classname> in der Bootstrap Datei.
  72. </para>
  73. </listitem>
  74. <listitem>
  75. <para>
  76. Aufruf von <classname>Zend_Session::start()</classname> nur für die Anfragen die eine Session
  77. verwenden müssen und vor jeglichen <classname>Zend_Session_Namespace</classname> initiiert werden.
  78. </para>
  79. </listitem>
  80. <listitem>
  81. <para>
  82. Normales verwenden von "<code>new Zend_Session_Namespace()</code>" wo es benötigt wird,
  83. aber sicherstellen das davor <classname>Zend_Session::start()</classname> ausgerufen wurde.
  84. </para>
  85. </listitem>
  86. </itemizedlist>
  87. <para>
  88. Die Option <code>strict</code> verhindert das <code>new Zend_Session_Namespace()</code>
  89. automatisch eine Session startet und dabei <classname>Zend_Session::start()</classname> verwendet.
  90. Deshalb hilft diese Option Anwendungs Entwicklern, sich für ein Design entscheiden
  91. zu können welches verhindert das für bestimmte Anfragen Sessions verwendet werden, da es eine
  92. Ausnahme verursachen würde wenn <classname>Zend_Session_Namespace</classname> instanziiert wird,
  93. bevor <classname>Zend_Session::start()</classname> aufgerufen wird. Entwickler sollten vorsichtig
  94. entscheiden welchen Einfluß die Verwendung von <classname>Zend_Session::setOptions()</classname> hat,
  95. da diese Optionen globale Seiteneffekte hat, in Folge der Korrespondenz der darunterliegenden
  96. Optionen für ext/session.
  97. </para>
  98. </listitem>
  99. <listitem>
  100. <para>
  101. Richtig: Einfach <code>new Zend_Session_Namespace()</code> instanzieren wo dies auch immer
  102. notwendig ist, und die darunterliegende PHP Session wird automatisch gestartet. Das bietet
  103. eine extrem simple Handhabung die in den meisten Situationen gut funktioniert. Trotzdem ist man
  104. dann dafür verantwortlich darauf zu schauen das das erste <code>new Zend_Session_Namespace()</code>
  105. passiert <emphasis role="strong">bevor</emphasis> irgendeine Ausgabe (z.B.
  106. <ulink url="http://www.php.net/headers_sent">HTTP headers</ulink> ) von PHP an den Client
  107. gesendet wird, wenn standardmäßige, Cookie-basierte Sessions verwendet werden (sehr
  108. empfehlenswert). Siehe <xref linkend="zend.session.global_session_management.headers_sent" />
  109. für weitere Informationen.
  110. </para>
  111. </listitem>
  112. </orderedlist>
  113. </sect2>
  114. <sect2 id="zend.session.advanced_usage.locking">
  115. <title>Gesperrte Session Namensräume</title>
  116. <para>
  117. Session Namensräume können gesperrt werden um weitere Veränderungen der Daten in diesem Namensraum
  118. zu verhindern. Die Verwendung von <code>lock()</code> macht einen speziellen
  119. Namensraum nur-lesbar, <code>unLock()</code> macht einen nur-lesbaren Namensraum les- und schreibbar,
  120. und <code>isLocked()</code> prüft ob ein Namensraum vorher gesperrt wurde. Sperren sind flüchtig und
  121. bestehen nicht von einer Anfrage zur nächsten. Die Sperre des Namensraumes hat keinen Effekt auf
  122. Setz-Methoden von Objekten welche im Namensraum gespeichert sind, aber sie verhindert die Verwendung
  123. der Setz-Methoden des Namensraumes welche das gespeicherte Objekt direkt im Namensraum löschen oder
  124. ersetzen. Gleichwohl verhindert das Sperren von <classname>Zend_Session_Namespace</classname> Instanzen nicht
  125. die Verwendung von symbolischen Tabellen-Aliasen auf die gleichen Daten (siehe
  126. <ulink url="http://www.php.net/references">PHP references</ulink>).
  127. </para>
  128. <example id="zend.session.advanced_usage.locking.example.basic">
  129. <title>Sperren von Session Namensräumen</title>
  130. <programlisting role="php"><![CDATA[
  131. $userProfileNamespace = new Zend_Session_Namespace('userProfileNamespace');
  132. // marking session as read only locked
  133. $userProfileNamespace->lock();
  134. // unlocking read-only lock
  135. if ($userProfileNamespace->isLocked()) {
  136. $userProfileNamespace->unLock();
  137. }
  138. ]]></programlisting>
  139. </example>
  140. </sect2>
  141. <sect2 id="zend.session.advanced_usage.expiration">
  142. <title>Verfall von Namensräumen</title>
  143. <para>
  144. Limits können plaziert werden an der Lebensdauer von beidem, Namensräumen und individuellen Schlüsseln
  145. in Namensräumen. Normale Anwendungsfälle beinhalten das durchlaufen von temporären Informationen zwischen
  146. Anfragen, und das vermindern der Aufdeckung von vielfältigen Sicherheitsrisiken durch das Entfernen des
  147. Zugangs zu potentiell sensitiven Informationen, manchmal nachdem Authentifizierung stettgefunden hat.
  148. Das Ende kann entweder auf abgelaufenen Sekunden oder der Anzahl von "Sprüngen" basieren, wobei ein Sprung
  149. für jede folgende Anfrage stattfindet.
  150. </para>
  151. <example id="zend.session.advanced_usage.expiration.example">
  152. <title>Beispiel für den Verfall</title>
  153. <programlisting role="php"><![CDATA[
  154. $s = new Zend_Session_Namespace('expireAll');
  155. $s->a = 'Apfel';
  156. $s->p = 'Pfirsich';
  157. $s->o = 'Orange';
  158. $s->setExpirationSeconds(5, 'a'); // Der Schlüssel "a" läuft in 5 Sekunden ab
  159. // Der komplette Namensraum läuft in 5 "Sprüngen"
  160. $s->setExpirationHops(5);
  161. $s->setExpirationSeconds(60);
  162. // Der "expireAll" Namensraum wird als "abgelaufen" markiert
  163. // sobald der erste Aufruf empfangen wurde und 60 Sekunden
  164. // vergangen sind, oder in 5 Sprüngen, was auch immer zuerst stattfindet
  165. ]]></programlisting>
  166. </example>
  167. <para>
  168. Wenn mit Daten einer Session gearbeitet wird, die in der aktuellen Anfrage ablaufen, sollte Vorsicht
  169. beim Empfangen dieser Daten gehalten werden. Auch wenn diese Daten durch Referenz zurückgegeben werden,
  170. wird die Änderung derselben, diese Daten nicht über diese Abfrage hinweg gültig machen. Um die
  171. Zeit für das Ablaufen zu "resetieren", müssen die Daten in eine temporäre Variable geholt werden,
  172. diese im Namensraum entfernt und anschliessend der entsprechende Schlüssel wieder gesetzt werden.
  173. </para>
  174. </sect2>
  175. <sect2 id="zend.session.advanced_usage.controllers">
  176. <title>Kapseln von Sessions und Kontroller</title>
  177. <para>
  178. Namensräume können auch verwendet werden um den Zugriff auf Sessions durch Kontroller zu seperieren
  179. um Variablen vor Kontaminierung zu schützen. Zum Beispiel könnte ein Authentifizierungs Kontroller seine
  180. Session Daten von allen anderen Kontrollern separat halten um notwendigen Sicherheiten zu entsprechen.
  181. </para>
  182. <example id="zend.session.advanced_usage.controllers.example">
  183. <title>Session Namensräume für Kontroller mit automatischem Verfall</title>
  184. <para>
  185. Der folgende Code ist Teil eines Kontrollers der die Test Frage anzeigt und eine boolsche Variable
  186. initialisiert die anzeigt ob eine geschickte Antwort zur Test Frage akzeptiert werden sollte oder
  187. nicht. In diesem Fall wird dem Benutzer der Anwendung 300 Sekunden Zeit gegeben die angezeigte
  188. Frage zu beantworten.
  189. </para>
  190. <programlisting role="php"><![CDATA[
  191. // ...
  192. // Im Frage-View Kontroller
  193. $testSpace = new Zend_Session_Namespace('testSpace');
  194. $testSpace->setExpirationSeconds(300, 'accept_answer');
  195. // Nur diese Variable ablaufen lassen
  196. $testSpace->accept_answer = true;
  197. //...
  198. ]]></programlisting>
  199. <para>
  200. Danach bestimmt der Kontroller der die Antworten für die Test Fragen bearbeitet ob eine
  201. Antwort akzeptiert wird oder nach basierend darauf ob der Benutzer die Antwort in der erlaubten
  202. Zeit übermittelt hat:
  203. </para>
  204. <programlisting role="php"><![CDATA[
  205. // ...
  206. // Im Frage-Prozess Kontroller
  207. $testSpace = new Zend_Session_Namespace('testSpace');
  208. if ($testSpace->accept_answer === true) {
  209. // innerhalb der Zeit
  210. }
  211. else {
  212. // nicht innerhalb der Zeit
  213. }
  214. // ...
  215. ]]></programlisting>
  216. </example>
  217. </sect2>
  218. <sect2 id="zend.session.advanced_usage.single_instance">
  219. <title>Mehrfache Instanzen pro Namensraum verhindern</title>
  220. <para>
  221. Obwohl <link linkend="zend.session.advanced_usage.locking">session locking</link> einen guten Grad von
  222. Schutz gegen unerlaubte Verwendung von Session Daten in einem Namensraum bietet, bietet
  223. <classname>Zend_Session_Namespace</classname> auch die Fähigkeit die Erzeugung von mehreren Instanzen zu verhindern
  224. die zu einem einzelnen Namensraum korrespondieren.
  225. </para>
  226. <para>
  227. Um dieses Verhalten einzuschalten, muß <code>true</code> als zweites Argument im Konstruktor angegeben
  228. werden wenn die letzte erlaubt Instanz von <classname>Zend_Session_Namespace</classname> erzeugt wurde. Jeder
  229. weitere Versuch den selben Namensraum zu instanzieren wird in einer geworfenen Ausnahme resultieren.
  230. </para>
  231. <example id="zend.session.advanced_usage.single_instance.example">
  232. <title>Zugriff auf Session Namensräume auf eine einzelne Instanz limitieren</title>
  233. <programlisting role="php"><![CDATA[
  234. // Eine Instanz eines Namensraumes erstellen
  235. $authSpaceAccessor1 = new Zend_Session_Namespace('Zend_Auth');
  236. // Eine weitere Instanz des selben Namensraumes erstellen,
  237. // aber weitere Instanzen verbieten
  238. $authSpaceAccessor2 = new Zend_Session_Namespace('Zend_Auth', true);
  239. // Eine Referenz erstellen ist immer noch möglich
  240. $authSpaceAccessor3 = $authSpaceAccessor2;
  241. $authSpaceAccessor1->foo = 'bar';
  242. assert($authSpaceAccessor2->foo, 'bar');
  243. try {
  244. $aNamespaceObject = new Zend_Session_Namespace('Zend_Auth');
  245. } catch (Zend_Session_Exception $e) {
  246. echo 'Dieser Namensraum kann nicht instanziert werden da ' .
  247. '$authSpaceAccessor2 erstellt wurde\n';
  248. }
  249. ]]></programlisting>
  250. </example>
  251. <para>
  252. Der zweite Parameter oben im Konstruktor sagt <classname>Zend_Session_Namespace</classname> das alle zukünftigen
  253. Instanzen mit dem <classname>Zend_Auth</classname> Namensraum nicht erlaubt sind. Der Versuch solche Instanzen zu erstellen
  254. verursacht eine Ausnahme die vom Konstruktor geworfen wird. Der Entwickler wird darauf aufmerksam gemacht
  255. eine Referenz zu einer Instanz des Objektes irgendwo zu speichern (<code>$authSpaceAccessor1</code>,
  256. <code>$authSpaceAccessor2</code>, oder <code>$authSpaceAccessor3</code> im obigen Beispiel), wenn
  257. der Zugriff auf den Namensraum der Session zu einer späteren Zeit wärend des selben Requests benötigt
  258. wird. Zum Beispiel, könnte ein Entwickler die referenz in einer statischen Variable speichern, die
  259. Referenz zu einer <ulink url="http://www.martinfowler.com/eaaCatalog/registry.html">Registry</ulink>
  260. hinzufügen (siehe <xref linkend="zend.registry" />), oder diese andernfalls für andere Methoden
  261. verfügbar zu machen die Zugriff auf den Namensraum der Session benötigen.
  262. </para>
  263. </sect2>
  264. <sect2 id="zend.session.advanced_usage.arrays">
  265. <title>Arbeiten mit Arrays</title>
  266. <para>
  267. Durch die Vergangenheit der Implmentationen der Magischen Methoden in PHP, wird das Ändern von Arrays
  268. innerhalb eines Namensraumes nicht unter PHP Versionen vor 5.2.1 funktionieren. Wenn nur mit PHP 5.2.1
  269. oder neuer gearbeitet wird, kann <link
  270. linkend="zend.session.advanced_usage.objects">zum nächsten Kapitel gesprungen</link> werden.
  271. </para>
  272. <example id="zend.session.advanced_usage.arrays.example.modifying">
  273. <title>Array Daten innerhalb eines Session Namensraumes verändern</title>
  274. <para>
  275. Das folgende illustriert wie das Problem reproduziert werden kann:
  276. </para>
  277. <programlisting role="php"><![CDATA[
  278. $sessionNamespace = new Zend_Session_Namespace();
  279. $sessionNamespace->array = array();
  280. // wird nicht wie gewünscht funktionieren vor PHP 5.2.1
  281. $sessionNamespace->array['testKey'] = 1;
  282. echo $sessionNamespace->array['testKey'];
  283. ]]></programlisting>
  284. </example>
  285. <example id="zend.session.advanced_usage.arrays.example.building_prior">
  286. <title>Arrays erstellen bevor es Session Speicher gab</title>
  287. <para>
  288. Wenn möglich, sollte das Problem vermieden werden indem Array nur dann im Session Namensraum
  289. gespeichert werden nachdem alle gewünschten Arraywerte gesetzt wurden.
  290. </para>
  291. <programlisting role="php"><![CDATA[
  292. $sessionNamespace = new Zend_Session_Namespace('Foo');
  293. $sessionNamespace->array = array('a', 'b', 'c');
  294. ]]></programlisting>
  295. </example>
  296. <para>
  297. Wenn eine betroffene Version von PHP verwendet wird and ein Array modifiziert werden soll nachdem es mit
  298. einem Schlüssel für den Session Namensraum verbunden wurde, kann einer oder beide der folgenden
  299. Workarounds verwendet werden.
  300. </para>
  301. <example id="zend.session.advanced_usage.arrays.example.workaround.reassign">
  302. <title>Workaround: Ein geändertes Array neu Verbinden</title>
  303. <para>
  304. Im folgenden Code wird eine Kopie des gespeicherten Array erstellt, geändert und wieder dem
  305. Platz von dem die Kopie erstellt wurde zugeordnet wobei das originale Array überschrieben wird.
  306. </para>
  307. <programlisting role="php"><![CDATA[
  308. $sessionNamespace = new Zend_Session_Namespace();
  309. // Das ursprüngliche Array hinzufügen
  310. $sessionNamespace->array = array('tree' => 'apple');
  311. // Eine Kopie des Arrays erstellen
  312. $tmp = $sessionNamespace->array;
  313. // Die Kopie des Arrays ändern
  314. $tmp['fruit'] = 'peach';
  315. // Die Kopie des Arrays wieder mit dem Namensraum der Session verknüpfen
  316. $sessionNamespace->array = $tmp;
  317. echo $sessionNamespace->array['fruit']; // gibt "peach" aus
  318. ]]></programlisting>
  319. </example>
  320. <example id="zend.session.advanced_usage.arrays.example.workaround.reference">
  321. <title>Workaround: Array das Referenz enthält speichern</title>
  322. <para>
  323. Alternativ, kann ein Array gespeichert werden das eine Referenz zum gewünschten Array enthält,
  324. die auf dieses dann indirekt zugegriffen werden.
  325. </para>
  326. <programlisting role="php"><![CDATA[
  327. $myNamespace = new Zend_Session_Namespace('myNamespace');
  328. $a = array(1, 2, 3);
  329. $myNamespace->someArray = array( &$a );
  330. $a['foo'] = 'bar';
  331. echo $myNamespace->someArray['foo']; // gibt "bar" aus
  332. ]]></programlisting>
  333. </example>
  334. </sect2>
  335. <sect2 id="zend.session.advanced_usage.objects">
  336. <title>Sessions mit Objekten verwenden</title>
  337. <para>
  338. Wenn Objekte in einer PHP Session fixiert werden sollen, muß bedacht werden das diese für das Speichern
  339. <ulink url="http://www.php.net/manual/de/language.oop.serialization.php">serialisiert</ulink> werden.
  340. Deshalb muß jedes Objekt das in einer PHP Session verewigt wurde deserialisiert werden nachdem es vom
  341. Speicher empfangen wurde. Das impliziert das der Entwickler sicherstellen muß das die Klassen für das
  342. verewigte Objekt definiert werden müssen befor das Objekt vom Session Speicher deserialisiert wird.
  343. Wenn die Klasse eines unserialisierten Objektes nicht definiert wurde, wird es eine Instand von
  344. <code>stdClass</code>.
  345. </para>
  346. </sect2>
  347. <sect2 id="zend.session.advanced_usage.testing">
  348. <title>Verwenden von Sessions mit Unit Tests</title>
  349. <para>
  350. Zend Framework vertraut auf PHPUnit um das Testen von sich selbst zu ermöglichen. Viele
  351. Entwickler erweitern die existierende Sammlung von Unit Tests um den Code in deren Anwendungen
  352. anzudecken. Die Ausnahme "<emphasis role="strong">Zend_Session ist aktuell als nur-lesbar markiert</emphasis>"
  353. wird geworfen wärend Unit Tests durchgeführt werden, wenn irgendeine schreibende Methode verwendet wird
  354. nachdem Ende der Session. Trotzdem benötigen Unit Tests die Zend_Session verwenden besondere Aufmerksamkeit
  355. weil das Schließen (<classname>Zend_Session::writeClose()</classname>) oder Zerstören einer Session
  356. (<classname>Zend_Session::destroy()</classname>) weitere Änderungen oder Rücknahmen von Schlüsseln in jeder
  357. Instanz von <classname>Zend_Session_Namespace</classname> verhindert. Dieses Verhalten ist ein direktes Resultat
  358. des darunterliegenden ext/session Mechanismus und PHP's <code>session_destroy()</code> und
  359. <code>session_write_close()</code> welche keinen "rückgängig machen" Mechanismus unterstützen um
  360. Setup/Teardown innerhalb der Unit Tests zu unterstützen.
  361. </para>
  362. <para>
  363. Um das Umzuarbeiten, siehe den Unit Test <code>testSetExpirationSeconds()</code> in
  364. <code>SessionTest.php</code> und <code>SessionTestHelper.php</code>, beide im
  365. <code>tests/Zend/Session</code> Verzeichnis, welche PHP's <code>exec()</code> verwenden um einen
  366. eigenständigen Prozess zu starten. Der neue Prozess simuliert eine zweite Anfrage eines Browsers,
  367. viel genauer. Der separate Prozess beginnt mit einer "reinen" Session, genauso wie jede PHP Skript
  368. Ausführung für eine web Anfrage. Auch jede Änderung in <code>$_SESSION[]</code> welche im
  369. aufrufenden Prozess gemacht wurde, ist im Kind-Prozess verfügbar, ermöglicht wenn der Elternprozess
  370. die Session beendet hat, bevor <code>exec()</code> verwendet wird.
  371. </para>
  372. <example id="zend.session.advanced_usage.testing.example">
  373. <title>PHPUnit Test Code der auf Zend_Session beruht</title>
  374. <programlisting role="php"><![CDATA[
  375. // testen von setExpirationSeconds()
  376. $script = 'SessionTestHelper.php';
  377. $s = new Zend_Session_Namespace('space');
  378. $s->a = 'apple';
  379. $s->o = 'orange';
  380. $s->setExpirationSeconds(5);
  381. Zend_Session::regenerateId();
  382. $id = Zend_Session::getId();
  383. // Session freigeben damit der untere Prozess Sie verwenden kann
  384. session_write_close();
  385. sleep(4); // nicht lange genug damit die Dinge ablaufen
  386. exec($script . "expireAll $id expireAll", $result);
  387. $result = $this->sortResult($result);
  388. $expect = ';a === apfel;o === orange;p === pfirsich';
  389. $this->assertTrue($result === $expect,
  390. "Iterierung durch standard Zend_Session Namensraum fehlgeschlagen; " .
  391. "erwartet result === '$expect', aber '$result' bekommen");
  392. sleep(2); // lange genug damit die Dinge ablaufen (insgesamt 6 Sekunden
  393. // warten, aber nach 5 Sekunden abgelaufen)
  394. exec($script . "expireAll $id expireAll", $result);
  395. $result = array_pop($result);
  396. $this->assertTrue($result === '',
  397. "Iterierung durch standard Zend_Session Namensraum fehlgeschlagen; " .
  398. "erwartet result === '', aber '$result' bekommen)");
  399. session_start(); // wiederherstellen der vorher eingefrorenen Session
  400. // Das könnte in einen separaten Test abgeteilt werden, aber aktuell, wenn
  401. // irgendwas vom darüberleigenden Test den darunterliegenden Test
  402. // kontaminiert, ist das auch ein Fehler den wir wissen wollen.
  403. $s = new Zend_Session_Namespace('expireGuava');
  404. $s->setExpirationSeconds(5, 'g'); // Versuch nur einen Schlüssel im
  405. // Namensraum ablaufen zu lassen
  406. $s->g = 'guava';
  407. $s->p = 'peach';
  408. $s->p = 'plum';
  409. // Session auflösen damit der untere Prozess sie verwenden kann
  410. session_write_close();
  411. sleep(6); // Nicht lange genug damit die Dinge ablaufen können
  412. exec($script . "expireAll $id expireGuava", $result);
  413. $result = $this->sortResult($result);
  414. session_start(); // Die bestimmte Session wiederherstellen
  415. $this->assertTrue($result === ';p === plum',
  416. "Iterierung durch benannte Zend_Session Namensräume " .
  417. "fehlgeschlaten (result=$result)");
  418. ]]></programlisting>
  419. </example>
  420. </sect2>
  421. </sect1>