Zend_Controller-Router.xml 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 15399 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.controller.router" xmlns:xi="http://www.w3.org/2001/XInclude">
  5. <title>Der Standard Router</title>
  6. <sect2 id="zend.controller.router.introduction">
  7. <title>Einführung</title>
  8. <para>
  9. <classname>Zend_Controller_Router_Rewrite</classname> ist der Standard Router des Frameworks.
  10. Routing ist der Prozess der Übernahme und Zerteilung einer URI, um zu
  11. ermitteln, welches Modul, welcher Controller und welche Aktion des Controllers die
  12. Anfrage erhalten soll. Die Definition des Moduls, des Controllers, der Aktion sowie
  13. weiterer Parameter wird in einem Objekt mit Namen
  14. <classname>Zend_Controller_Dispatcher_Token</classname> gekapselt, das dann vom
  15. <classname>Zend_Controller_Dispatcher_Standard</classname> verarbeitet wird. Das Routing
  16. geschieht nur einmal: wenn zu Beginn die Anfrage erhalten wird und bevor der erste
  17. Controller aufgerufen wird.
  18. </para>
  19. <para>
  20. <classname>Zend_Controller_Router_Rewrite</classname> wurde entwickelt, um mit reinen PHP
  21. Strukturen eine mod_rewrite ähnliche Funktionalität zu erlauben. Es richtet sich
  22. sehr frei nach dem Ruby on Rails Routing und benötigt kein tieferes Wissen über
  23. URL Weiterleitung des Webservers. Es wurde entwickelt, um mit einer einzigen
  24. mod_rewrite Regel zu arbeiten.
  25. </para>
  26. <programlisting role="php"><![CDATA[
  27. RewriteEngine on
  28. RewriteRule !\.(js|ico|gif|jpg|png|css|html)$ index.php
  29. ]]></programlisting>
  30. <para>
  31. oder (bevorzugt):
  32. </para>
  33. <programlisting role="php"><![CDATA[
  34. RewriteEngine On
  35. RewriteCond %{REQUEST_FILENAME} -s [OR]
  36. RewriteCond %{REQUEST_FILENAME} -l [OR]
  37. RewriteCond %{REQUEST_FILENAME} -d
  38. RewriteRule ^.*$ - [NC,L]
  39. RewriteRule ^.*$ index.php [NC,L]
  40. ]]></programlisting>
  41. <para>
  42. Der Rewrite Router kann auch mit dem IIS Webserver verwendet werden (Versionen &lt;= 7.0), wenn
  43. <ulink url="http://www.isapirewrite.com">Isapi_Rewrite</ulink> als Isapi
  44. Erweiterung installiert wurde und folgende Umschreibungsregel verwendet wird:
  45. </para>
  46. <programlisting role="php"><![CDATA[
  47. RewriteRule ^[\w/\%]*(?:\.(?!(?:js|ico|gif|jpg|png|css|html)$)[\w\%]*$)? /index.php [I]
  48. ]]></programlisting>
  49. <note>
  50. <title>IIS Isapi_Rewrite</title>
  51. <para>
  52. Bei Verwendung von IIS, wird <code>$_SERVER['REQUEST_URI']</code> entweder nicht
  53. vorhanden oder auf einen leeren String gesetzt sein. In diesem Fall wird
  54. <classname>Zend_Controller_Request_Http</classname> versuchen, den durch die Isapi_Rewrite
  55. Erweiterung gesetzten Wert <code>$_SERVER['HTTP_X_REWRITE_URL']</code> zu
  56. verwenden.
  57. </para>
  58. </note>
  59. <para>
  60. IIS 7.0 führt ein natives URL Rewriting Modul ein, und kann wie folgt konfiguriert werden:
  61. </para>
  62. <programlisting role="xml"><![CDATA[
  63. <?xml version="1.0" encoding="UTF-8"?>
  64. <configuration>
  65. <system.webServer>
  66. <rewrite>
  67. <rules>
  68. <rule name="Imported Rule 1" stopProcessing="true">
  69. <match url="^.*$" />
  70. <conditions logicalGrouping="MatchAny">
  71. <add input="{REQUEST_FILENAME}"
  72. matchType="IsFile" pattern=""
  73. ignoreCase="false" />
  74. <add input="{REQUEST_FILENAME}"
  75. matchType="IsDirectory"
  76. pattern="" ignoreCase="false" />
  77. </conditions>
  78. <action type="None" />
  79. </rule>
  80. <rule name="Imported Rule 2" stopProcessing="true">
  81. <match url="^.*$" />
  82. <action type="Rewrite" url="index.php" />
  83. </rule>
  84. </rules>
  85. </rewrite>
  86. </system.webServer>
  87. </configuration>
  88. ]]></programlisting>
  89. <para>
  90. Bei der Verwendung von Lighttpd, ist folgende Umschreibungsregel gültig:
  91. </para>
  92. <programlisting role="lighttpd"><![CDATA[
  93. url.rewrite-once = (
  94. ".*\?(.*)$" => "/index.php?$1",
  95. ".*\.(js|ico|gif|jpg|png|css|html)$" => "$0",
  96. "" => "/index.php"
  97. )
  98. ]]></programlisting>
  99. </sect2>
  100. <sect2 id="zend.controller.router.usage">
  101. <title>Einen Router verwenden</title>
  102. <para>
  103. Um den Rewrite Router richtig zu verwenden, muß er instanziiert, einige
  104. benutzerdefinierte Routen hinzufügt und in den Controller einbunden werden. Der folgende
  105. Code veranschaulicht die Vorgehensweise:
  106. </para>
  107. <programlisting role="php"><![CDATA[
  108. // Einen Router erstellen
  109. $router = $ctrl->getRouter(); // gibt standardmäßig einen Rewrite Router zurück
  110. $router->addRoute(
  111. 'user',
  112. new Zend_Controller_Router_Route('user/:username',
  113. array('controller' => 'user',
  114. 'action' => 'info'))
  115. );
  116. ]]></programlisting>
  117. </sect2>
  118. <sect2 id="zend.controller.router.basic">
  119. <title>Grundsätzliche Rewrite Router Operationen</title>
  120. <para>
  121. Das Herz des RewriteRouters ist die Definition von Benutzerdefinierten Routen. Routen werden durch
  122. aufruf der addRoute Methode des RewriteRouters hinzugefügt und übergeben eine neue Instanz einer
  123. Klasse die <classname>Zend_Controller_Router_Route_Interface</classname> implementiert. Z.B.:
  124. </para>
  125. <programlisting role="php"><![CDATA[
  126. $router->addRoute('user',
  127. new Zend_Controller_Router_Route('user/:username'));
  128. ]]></programlisting>
  129. <para>
  130. Der Rewrite Router kommt mit sechs Basistypen von Routen (eine von denen ist speziell):
  131. is special):
  132. </para>
  133. <itemizedlist mark="opencircle">
  134. <listitem><para><xref linkend="zend.controller.router.routes.standard" /></para></listitem>
  135. <listitem><para><xref linkend="zend.controller.router.routes.static" /></para></listitem>
  136. <listitem><para><xref linkend="zend.controller.router.routes.regex" /></para></listitem>
  137. <listitem><para><xref linkend="zend.controller.router.routes.hostname" /></para></listitem>
  138. <listitem><para><xref linkend="zend.controller.router.routes.chain" /></para></listitem>
  139. <listitem><para><xref linkend="zend.controller.router.default-routes" /> *</para></listitem>
  140. </itemizedlist>
  141. <para>
  142. Routen können unzählige Male verwendet werden um eine Kette oder benutzerdefinierte Routing Schemas
  143. von Anwendungen zu erstellen. Es kann jede beliebige Anzahl von Routen in jeder beliebigen Konfiguration
  144. verwendet werden, mit Ausnahme der Modul Route, welche nur einmal verwendet werden sollte, und
  145. möglicherweise die am meisten standardmäßige Route ist (z.B., als ein Standard). Jede Route wird
  146. später detailiert beschrieben.
  147. </para>
  148. <para>
  149. Der erste Parameter für addRoute ist der Name der Route. Er wird als Handle verwendet um die Route
  150. außerhalb des Routers zu erhalten (z.B. für den Zweck der URL Erzeugung). Der zweite Parameter ist
  151. die Route selbst.
  152. </para>
  153. <note>
  154. <para>
  155. Die gewöhnlichste Verwendung des Namens der Route ist gegeben durch die Zwecke des <classname>Zend_View</classname>
  156. Url Helfers:
  157. </para>
  158. <programlisting role="php"><![CDATA[
  159. <a href=
  160. "<?php echo $this->url(array('username' => 'martel'), 'user') ?>">Martel</a>
  161. ]]></programlisting>
  162. <para>
  163. Was zu folgender href führt: <code>user/martel</code>.
  164. </para>
  165. </note>
  166. <para>
  167. Routen ist ein einfacher Prozess des Durchlaufens aller vorhandenen Routen und Vergleichens deren
  168. Definitionen mit der aktuellen Anfrage URI. Wenn ein positiver Vergleich gefunden wird, werden
  169. variable Werte von der Instanz des Routers zurückgegeben, und werden für die spätere Verwendung im
  170. Dispatcher in das <classname>Zend_Controller_Request</classname> Objekt iniziiert, sowie in von Benutzern
  171. erstellten Controllern. Bei einem negativen Ergebnis des Vergleiches, wird die nächste Route in der
  172. Kette geprüft.
  173. </para>
  174. <para>
  175. Wenn man herausfinden will welche Route gepasst hat, kann man die <code>getCurrentRouteName()</code>
  176. Methode verwenden, die den Identifikator zurückgibt der verwendet wurde als die Route im
  177. Router registriert wurde. Wenn man das aktuelle Route Objekt benötigt, kann
  178. <code>getCurrentRoute()</code> verwendet werden.
  179. </para>
  180. <note>
  181. <title>Umgekehrter Vergleich</title>
  182. <para>
  183. Routen werden in umgekehrter Reihenfolge verglichen. Deswegen muß sichergestellt werden das die
  184. generellen Routen zuerst definiert werden.
  185. </para>
  186. </note>
  187. <note>
  188. <title>Zurückgegebene Werte</title>
  189. <para>
  190. Werte die vom Routen zurückgegeben werden kommen von URL Parametern oder Benutzerdefinierten
  191. Router Standards. Diese Variablen sind später durch die
  192. <classname>Zend_Controller_Request::getParam()</classname> oder
  193. <classname>Zend_Controller_Action::_getParam()</classname>Methoden verwendbar.
  194. </para>
  195. </note>
  196. <para>
  197. Es gibt drei spezielle Variablen welche in den Routen verwendet werden können - 'module', 'controller'
  198. und 'action'. Diese speziellen Variablen werden durch <classname>Zend_Controller_Dispatcher</classname> verwendet um
  199. einen Kontroller und die Aktion zu funden zu der verwiesen wird.
  200. </para>
  201. <note>
  202. <title>Spezielle Variablen</title>
  203. <para>
  204. Die Namen dieser speziellen Variablen kann unterschiedlich sein wenn entschieden wird die
  205. Standards in <classname>Zend_Controller_Request_Http</classname> mithilfe der <code>setControllerKey</code>
  206. und <code>setActionKey</code> Methode zu Ändern.
  207. </para>
  208. </note>
  209. </sect2>
  210. <sect2 id="zend.controller.router.default-routes">
  211. <title>Standard Routen</title>
  212. <para>
  213. <classname>Zend_Controller_Router_Rewrite</classname> kommt mit einer Standard Route vorkonfiguriert, welche URIs im Sinn
  214. von <code>controller/action</code> entspricht. Zusätzlich kann ein Modul Name als erstes Pfad Element
  215. definiert werden, welches URIs in der Form von <code>module/controller/action</code> erlaubt.
  216. Letztendlich wird es auch allen zusätzlichen Parametern entsprechen die der URI standardmäßig
  217. hinzugefügt wurden - <code>controller/action/var1/value1/var2/value2</code>.
  218. </para>
  219. <para>
  220. Einige Beispiele wie solche Routen verglichen werden:
  221. </para>
  222. <programlisting role="php"><![CDATA[
  223. // Folgende Annahme:
  224. $ctrl->setControllerDirectory(
  225. array(
  226. 'default' => '/path/to/default/controllers',
  227. 'news' => '/path/to/news/controllers',
  228. 'blog' => '/path/to/blog/controllers'
  229. )
  230. );
  231. Nur Modul:
  232. http://example/news
  233. module == news
  234. Ungültiges Modul, geht an den Controller Namen:
  235. http://example/foo
  236. controller == foo
  237. Modul + Controller:
  238. http://example/blog/archive
  239. module == blog
  240. controller == archive
  241. Modul + Controller + Aktion:
  242. http://example/blog/archive/list
  243. module == blog
  244. controller == archive
  245. action == list
  246. Modul + Controller + Aktion + Parameter:
  247. http://example/blog/archive/list/sort/alpha/date/desc
  248. module == blog
  249. controller == archive
  250. action == list
  251. sort == alpha
  252. date == desc
  253. ]]></programlisting>
  254. <para>
  255. Die Standardroute ist einfach ein <classname>Zend_Controller_Router_Route_Module</classname> Objekt welches unter
  256. dem Namen (Index) 'default' im RewriteRouter gespeichert ist. Es wird mehr oder weniger wie folgt
  257. erstellt:
  258. </para>
  259. <programlisting role="php"><![CDATA[
  260. $compat = new Zend_Controller_Router_Route_Module(array(),
  261. $dispatcher,
  262. $request);
  263. $this->addRoute('default', $compat);
  264. ]]></programlisting>
  265. <para>
  266. Wenn diese spezielle Standard Route im eigenen Routing Schema nicht gewünscht ist, kann Sie durch
  267. Erstellung einer eigenen 'default' Route überschrieben werden (z.B. durch Speichern unter dem Namen
  268. 'default') oder dem kompletten Entfernen durch verwenden von <code>removeDefaultRoutes()</code>:
  269. </para>
  270. <programlisting role="php"><![CDATA[
  271. // Löschen aller Standard Routen
  272. $router->removeDefaultRoutes();
  273. ]]></programlisting>
  274. </sect2>
  275. <sect2 id="zend.controller.router.rewritebase">
  276. <title>Basis URL und Unterverzeichnisse</title>
  277. <para>
  278. Der Rewrite Router kann in Unterverzeichnissen verwendet werden (z.B.
  279. <code>http://domain.com/~user/application-root/</code>) und in diesem Fall sollte die Basis URL der
  280. Anwendung (<code>/~user/application-root</code>) automatisch durch
  281. <classname>Zend_Controller_Request_Http</classname> erkannt und auch verwendet werden.
  282. </para>
  283. <para>
  284. Sollte die Basis URL nicht richtig erkannt werden kann diese mit eigenen Basispfad überschrieben werden
  285. durch Verwendung von <classname>Zend_Controller_Request_Http</classname> und Auruf der <code>setBaseUrl()</code>
  286. Methode (siehe <xref linkend="zend.controller.request.http.baseurl" />):
  287. </para>
  288. <programlisting role="php"><![CDATA[
  289. $request->setBaseUrl('/~user/application-root/');
  290. ]]></programlisting>
  291. </sect2>
  292. <sect2 id="zend.controller.router.global.parameters">
  293. <title>Globale Parameter</title>
  294. <para>
  295. Man kann in einem Router globale Parameter setzen die der Route automatisch zur Verfügung stehen
  296. wenn Sie durch <code>setGlobalParam</code> eingefügt werden. Wenn ein globaler Parameter
  297. gesetzt ist, aber auch direkt an die Assemble Methode gegeben wird, überschreibt der
  298. Benutzer-Parameter den Globalen-Parameter. Globale Parameter können auf folgendem Weg gesetzt werden:
  299. </para>
  300. <programlisting role="php"><![CDATA[
  301. $router->setGlobalParam('lang', 'en');
  302. ]]></programlisting>
  303. </sect2>
  304. <sect2 id="zend.controller.router.routes">
  305. <title>Router Typen</title>
  306. <xi:include href="Zend_Controller-Router-Route.xml" />
  307. <xi:include href="Zend_Controller-Router-Route-Static.xml" />
  308. <xi:include href="Zend_Controller-Router-Route-Regex.xml" />
  309. <xi:include href="Zend_Controller-Router-Route-Hostname.xml" />
  310. <xi:include href="Zend_Controller-Router-Route-Chain.xml" />
  311. </sect2>
  312. <sect2 id="zend.controller.router.add-config">
  313. <title>Zend_Config mit dem RewriteRouter verwenden</title>
  314. <para>
  315. Manchmal ist es praktischer, eine Konfigurationsdatei mit neuen Routen zu
  316. aktualisieren, als den Code zu ändern. Dies ist mit Hilfe der
  317. <code>addConfig()</code> Methode möglich. Im Wesentlichen kann man eine
  318. <classname>Zend_Config</classname> kompatible Konfiguration erstellen, in seinem Code einlesen und an den
  319. RewriteRouter übergeben:
  320. </para>
  321. <para>
  322. Als Beispiel wird die folgende INI Datei angenommen:
  323. </para>
  324. <programlisting role="php"><![CDATA[
  325. [production]
  326. routes.archive.route = "archive/:year/*"
  327. routes.archive.defaults.controller = archive
  328. routes.archive.defaults.action = show
  329. routes.archive.defaults.year = 2000
  330. routes.archive.reqs.year = "\d+"
  331. routes.news.type = "Zend_Controller_Router_Route_Static"
  332. routes.news.route = "news"
  333. routes.news.defaults.controller = "news"
  334. routes.news.defaults.action = "list"
  335. routes.archive.type = "Zend_Controller_Router_Route_Regex"
  336. routes.archive.route = "archive/(\d+)"
  337. routes.archive.defaults.controller = "archive"
  338. routes.archive.defaults.action = "show"
  339. routes.archive.map.1 = "year"
  340. ; OR: routes.archive.map.year = 1
  341. ]]></programlisting>
  342. <para>
  343. Die oben angeführte INI Datei kann dann wie folgt in ein
  344. <classname>Zend_Config</classname> Objekt eingelesen werden:
  345. </para>
  346. <programlisting role="php"><![CDATA[
  347. $config = new Zend_Config_Ini('/path/to/config.ini', 'production');
  348. $router = new Zend_Controller_Router_Rewrite();
  349. $router->addConfig($config, 'routes');
  350. ]]></programlisting>
  351. <para>
  352. Im oberen Beispiel teilen wir dem Router mit, den 'routes' Bereich der INI Datei
  353. für seine Routen zu verwenden. Jeder Schlüssel auf erster Ebene in diesem Bereich
  354. wird verwendet, um den Namen der Routen zu definieren; das obige Beispiel definiert
  355. die Routen 'archive' und 'news'. Jede Route erfordert dann mindestens einen 'route'
  356. Eintrag und einen oder mehrere 'defaults' Einträge; optional können eine oder
  357. mehrere 'reqs' (kurz für 'required', d.h. erforderlich) Einträge angegeben werden.
  358. Alles in allem entspricht dies den drei Argumenten, die an ein
  359. <classname>Zend_Controller_Router_Route_Interface</classname> Objekt übergeben werden. Ein
  360. Optionsschlüssel 'type' kann verwendet werden, um den Typ der Routenklasse für
  361. diese Route anzugeben; standardmäßig wird <classname>Zend_Controller_Router_Route</classname>
  362. verwendet. Im obigen Beispiel wird die 'news' Route definiert, um
  363. <classname>Zend_Controller_Router_Route_Static</classname> zu verwenden.
  364. </para>
  365. </sect2>
  366. <sect2 id="zend.controller.router.subclassing">
  367. <title>Erben vom Router</title>
  368. <para>
  369. Der Standard Rewrite Router sollte die meisten Funktionalitäten die benötigt werden zur Verfügung
  370. stellen; meistens wird es nur notwendig sein einen neuen Router Typen zu erstellen um neue oder
  371. modifizierte Funktionalitäten für die verfügbaren Routen zu bieten.
  372. </para>
  373. <para>
  374. So gesehen, wird man in einigen Fällen ein anderes Routing Paradigma verwenden wollen. Das Interface
  375. <classname>Zend_Controller_Router_Interface</classname> bietet die minimalen Information die benötigt werden um
  376. einen Router er erstellen und besteht aus einer einzigen Methode.
  377. </para>
  378. <programlisting role="php"><![CDATA[
  379. interface Zend_Controller_Router_Interface
  380. {
  381. /**
  382. * @param Zend_Controller_Request_Abstract $request
  383. * @throws Zend_Controller_Router_Exception
  384. * @return Zend_Controller_Request_Abstract
  385. */
  386. public function route(Zend_Controller_Request_Abstract $request);
  387. }
  388. ]]></programlisting>
  389. <para>
  390. Das Routing findet nur einmal statt, wenn die Anfrage das erste Mal vom System erhalten
  391. wird. Der Zweck des Routers ist es, Controller, Aktion und optionale Parameter auf Basis
  392. der Anfrageumgebung zu ermitteln und im Request zu setzen. Das Request Objekt wird dann
  393. an den Dispatcher übergeben. Wenn es nicht möglich ist, eine Route auf einen Dispatch
  394. Token abzubilden, soll der Router nichts mit dem Request Objekt machen.
  395. </para>
  396. </sect2>
  397. </sect1>
  398. <!--
  399. vim:se ts=4 sw=4 et:
  400. -->