Zend_Controller-Router.xml 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 24249 -->
  4. <sect1 id="zend.controller.router" xmlns:xi="http://www.w3.org/2001/XInclude">
  5. <title>標準のルータ</title>
  6. <sect2 id="zend.controller.router.introduction">
  7. <title>導入</title>
  8. <para>
  9. <classname>Zend_Controller_Router_Rewrite</classname> は、標準のルータです。
  10. ルーティングとは、<acronym>URI</acronym> (ベース <acronym>URL</acronym> から取得した <acronym>URI</acronym> の一部)
  11. を展開し、どのコントローラのどのアクションが
  12. リクエストを処理するのかを決める処理のことです。
  13. モジュールやコントローラ、アクション、そしてその他のパラメータが
  14. <classname>Zend_Controller_Request_Http</classname> オブジェクトにまとめられます。
  15. このオブジェクトを処理するのが <classname>Zend_Controller_Dispatcher_Standard</classname> です。
  16. ルーティングが行われるのは一度だけ、すなわちリクエストを最初に受け取ってから
  17. 最初のコントローラに処理が渡される際だけです。
  18. </para>
  19. <para>
  20. <classname>Zend_Controller_Router_Rewrite</classname> は、mod_rewrite 風の機能を
  21. <acronym>PHP</acronym> だけで実現できるように設計されています。
  22. この処理は Ruby on Rails のルーティングを多少参考にしており、
  23. ウェブサーバの <acronym>URL</acronym> 書き換えに関する前提知識を必要としません。
  24. 以下の単純な mod_rewrite ルール (のいずれか) で動作するように設計されています。
  25. </para>
  26. <programlisting language="php"><![CDATA[
  27. RewriteEngine on
  28. RewriteRule !\.(js|ico|gif|jpg|png|css|html)$ index.php
  29. ]]></programlisting>
  30. <para>
  31. あるいは (推奨)
  32. </para>
  33. <programlisting language="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. Rewrite ルータを <acronym>IIS</acronym> ウェブサーバ (バージョン &lt;= 7.0) で使用するには
  43. <ulink url="http://www.isapirewrite.com">Isapi_Rewrite</ulink>
  44. を Isapi 拡張モジュールとしてインストールします。そして次のようなルールを記述します。
  45. </para>
  46. <programlisting language="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. <acronym>IIS</acronym> を使用すると、<varname>$_SERVER['REQUEST_URI']</varname>
  53. が存在しないか空の文字列に設定されます。このような場合、
  54. <classname>Zend_Controller_Request_Http</classname> は
  55. <varname>$_SERVER['HTTP_X_REWRITE_URL']</varname> の値を使用します。これは
  56. <classname>Isapi_Rewrite</classname> 拡張モジュールが設定します。
  57. </para>
  58. </note>
  59. <para>
  60. <acronym>IIS</acronym> 7.0 ではネイティブの <acronym>URL</acronym> リライトモジュールが登場しました。
  61. 次のように設定して使います。
  62. </para>
  63. <programlisting language="xml"><![CDATA[
  64. <?xml version="1.0" encoding="UTF-8"?>
  65. <configuration>
  66. <system.webServer>
  67. <rewrite>
  68. <rules>
  69. <rule name="Imported Rule 1" stopProcessing="true">
  70. <match url="^.*$" />
  71. <conditions logicalGrouping="MatchAny">
  72. <add input="{REQUEST_FILENAME}"
  73. matchType="IsFile" pattern=""
  74. ignoreCase="false" />
  75. <add input="{REQUEST_FILENAME}"
  76. matchType="IsDirectory"
  77. pattern="" ignoreCase="false" />
  78. </conditions>
  79. <action type="None" />
  80. </rule>
  81. <rule name="Imported Rule 2" stopProcessing="true">
  82. <match url="^.*$" />
  83. <action type="Rewrite" url="index.php" />
  84. </rule>
  85. </rules>
  86. </rewrite>
  87. </system.webServer>
  88. </configuration>
  89. ]]></programlisting>
  90. <para>
  91. Lighttpd の場合は、次のようなルールを使用します。
  92. </para>
  93. <programlisting language="lighttpd"><![CDATA[
  94. url.rewrite-once = (
  95. ".*\?(.*)$" => "/index.php?$1",
  96. ".*\.(js|ico|gif|jpg|png|css|html)$" => "$0",
  97. "" => "/index.php"
  98. )
  99. ]]></programlisting>
  100. </sect2>
  101. <sect2 id="zend.controller.router.usage">
  102. <title>ルータの使用法</title>
  103. <para>
  104. Rewrite ルータを適切に使用するには、まずそのインスタンスを作成し、
  105. 次にユーザ定義のルーティングを追加し、それをコントローラに注入しなければなりません。
  106. 以下にコードの例を示します。
  107. </para>
  108. <programlisting language="php"><![CDATA[
  109. // ルータを作成します
  110. $router = $ctrl->getRouter(); // デフォルトで rewrite ルータを返します
  111. $router->addRoute(
  112. 'user',
  113. new Zend_Controller_Router_Route('user/:username',
  114. array('controller' => 'user',
  115. 'action' => 'info'))
  116. );
  117. ]]></programlisting>
  118. </sect2>
  119. <sect2 id="zend.controller.router.basic">
  120. <title>基本的な RewriteRouter の操作法</title>
  121. <para>
  122. RewriteRouter で最も重要なのが、ユーザ定義のルーティングです。
  123. これは、RewriteRouter の addRoute メソッドをコールして追加します。
  124. このメソッドに、<classname>Zend_Controller_Router_Route_Interface</classname>
  125. を実装したクラスの新しいインスタンスを渡します。
  126. </para>
  127. <programlisting language="php"><![CDATA[
  128. $router->addRoute('user',
  129. new Zend_Controller_Router_Route('user/:username'));
  130. ]]></programlisting>
  131. <para>
  132. Rewrite ルータには、6 種類の基本的なルーティング方式があります
  133. (そのうちのひとつは特別なものです)。
  134. </para>
  135. <itemizedlist mark="opencircle">
  136. <listitem>
  137. <para>
  138. <link
  139. linkend="zend.controller.router.routes.standard">Zend_Controller_Router_Route</link>
  140. </para>
  141. </listitem>
  142. <listitem>
  143. <para>
  144. <link
  145. linkend="zend.controller.router.routes.static">Zend_Controller_Router_Route_Static</link>
  146. </para>
  147. </listitem>
  148. <listitem>
  149. <para>
  150. <link
  151. linkend="zend.controller.router.routes.regex">Zend_Controller_Router_Route_Regex</link>
  152. </para>
  153. </listitem>
  154. <listitem>
  155. <para>
  156. <link
  157. linkend="zend.controller.router.routes.hostname">Zend_Controller_Router_Route_Hostname</link>
  158. </para>
  159. </listitem>
  160. <listitem>
  161. <para>
  162. <link
  163. linkend="zend.controller.router.routes.chain">Zend_Controller_Router_Route_Chain</link>
  164. </para>
  165. </listitem>
  166. <listitem>
  167. <para>
  168. <link
  169. linkend="zend.controller.router.default-routes">Zend_Controller_Router_Rewrite</link>
  170. *
  171. </para>
  172. </listitem>
  173. </itemizedlist>
  174. <para>
  175. これらのルーティングは、チェインやユーザ定義のルーティング方式を作成する際に何度も使用します。
  176. 任意の設定でお好みの数のルーティングを使用できますが、
  177. Module ルートだけは例外です。これを使用するのは一度だけで、
  178. もっとも汎用的なルート (デフォルト) として使用します。
  179. 個々のルーティング方式については、後ほど詳細に説明します。
  180. </para>
  181. <para>
  182. addRoute への最初のパラメータはルートの名前です。
  183. これを使用して、ルータがルートを処理します。
  184. たとえば <acronym>URL</acronym> の生成などに使用します。
  185. 二番目のパラメータはルート自身となります。
  186. </para>
  187. <note>
  188. <para>
  189. ルート名のもっとも一般的な使用例は、
  190. <classname>Zend_View</classname> の url ヘルパーです。
  191. </para>
  192. <programlisting language="php"><![CDATA[
  193. <a href=
  194. "<?php echo $this->url(array('username' => 'martel'), 'user') ?>">Martel</a>
  195. ]]></programlisting>
  196. <para>
  197. これは <filename>user/martel</filename> へのリンクとなります。
  198. </para>
  199. </note>
  200. <para>
  201. ルーティング処理は、定義されたすべてのルートから
  202. リクエスト <acronym>URI</acronym> にマッチする定義を探すことによって行います。
  203. マッチするものが見つかれば、ルートのインスタンスから変数の値が返され、
  204. それを Zend_Controller_Request オブジェクトに注入します。
  205. これを、後にディスパッチャやユーザが作成したコントローラで使用します。
  206. マッチするものが見つからない場合は、チェイン内の次のルートを調べます。
  207. </para>
  208. <para>
  209. どのルートがマッチしたかを知りたい場合は
  210. <methodname>getCurrentRouteName()</methodname> メソッドを使用します。
  211. これは、ルートをルータに登録する際に使用した識別子を返します。
  212. ルートオブジェクトそのものを取得したい場合は
  213. <methodname>getCurrentRoute()</methodname> を使用します。
  214. </para>
  215. <note>
  216. <title>定義の順番</title>
  217. <para>
  218. 一番最後にマッチしたルートが適用されるので、
  219. 汎用的なルートは最初に定義するようにしましょう。
  220. </para>
  221. </note>
  222. <note>
  223. <title>返される値</title>
  224. <para>
  225. ルーティングの結果返される値は、<acronym>URL</acronym> パラメータあるいは
  226. ユーザ定義のルータのデフォルト値です。これらの値は、後ほど
  227. <methodname>Zend_Controller_Request::getParam()</methodname> あるいは
  228. <methodname>Zend_Controller_Action::_getParam()</methodname>
  229. メソッドでアクセスできます。
  230. </para>
  231. </note>
  232. <para>
  233. ルートで使用される変数のうち、'module'、'controller' および 'action'
  234. の 3 つは特別な扱いとなります。これらの特殊変数は、<classname>Zend_Controller_Dispatcher</classname>
  235. がディスパッチ先のコントローラとアクションを決定するために使用されます。
  236. </para>
  237. <note>
  238. <title>特殊変数</title>
  239. <para>
  240. これらの特殊変数の名前を変更することもできます。その場合は
  241. <classname>Zend_Controller_Request_Http</classname> の
  242. <methodname>setControllerKey()</methodname> メソッドや
  243. <methodname>setActionKey()</methodname> メソッドを使用します。
  244. </para>
  245. </note>
  246. </sect2>
  247. <sect2 id="zend.controller.router.default-routes">
  248. <title>デフォルトのルート</title>
  249. <para>
  250. <classname>Zend_Controller_Router_Rewrite</classname> がデフォルトのルートとして設定されています。
  251. これは <filename>controller/action</filename> 形式の <acronym>URI</acronym> にマッチします。
  252. さらに、パス要素の最初の部分にモジュール名を指定できます。つまり
  253. <filename>module/controller/action</filename> のような <acronym>URI</acronym> も可能です。
  254. また、<acronym>URI</acronym> にパラメータを追加した形式、つまり
  255. <filename>controller/action/var1/value1/var2/value2</filename>
  256. のような <acronym>URI</acronym> にもデフォルトで対応しています。
  257. </para>
  258. <para>
  259. ルータのマッチ処理についての例を示します。
  260. </para>
  261. <programlisting language="php"><![CDATA[
  262. // 以下の設定を前提とします
  263. $ctrl->setControllerDirectory(
  264. array(
  265. 'default' => '/path/to/default/controllers',
  266. 'news' => '/path/to/news/controllers',
  267. 'blog' => '/path/to/blog/controllers'
  268. )
  269. );
  270. モジュールのみ
  271. http://example/news
  272. module == news
  273. 無効なモジュール名は、コントローラ名として扱われます
  274. http://example/foo
  275. controller == foo
  276. モジュール + コントローラ
  277. http://example/blog/archive
  278. module == blog
  279. controller == archive
  280. モジュール + コントローラ + アクション
  281. http://example/blog/archive/list
  282. module == blog
  283. controller == archive
  284. action == list
  285. モジュール + コントローラ + アクション + パラメータ
  286. http://example/blog/archive/list/sort/alpha/date/desc
  287. module == blog
  288. controller == archive
  289. action == list
  290. sort == alpha
  291. date == desc
  292. ]]></programlisting>
  293. <para>
  294. デフォルトのルートは、<classname>Zend_Controller_Router_Route_Module</classname>
  295. オブジェクトを 'default' という名前 (インデックス) で
  296. RewriteRouter に保存したものです。
  297. これは、以下のようにして作成します。
  298. </para>
  299. <programlisting language="php"><![CDATA[
  300. $compat = new Zend_Controller_Router_Route_Module(array(),
  301. $dispatcher,
  302. $request);
  303. $this->addRoute('default', $compat);
  304. ]]></programlisting>
  305. <para>
  306. このデフォルトルートが不要な場合は、独自の 'デフォルト' ルートで上書きします
  307. (つまり、'default' という名前で保存します)。
  308. あるいは、<methodname>removeDefaultRoutes()</methodname>
  309. で削除することもできます。
  310. </para>
  311. <programlisting language="php"><![CDATA[
  312. // すべてのデフォルトルートを削除します
  313. $router->removeDefaultRoutes();
  314. ]]></programlisting>
  315. </sect2>
  316. <sect2 id="zend.controller.router.rewritebase">
  317. <title>ベース URL およびサブディレクトリ</title>
  318. <para>
  319. Rewrite ルータはサブディレクトリ
  320. (例. <filename>http://domain.com/user/application-root/</filename>)
  321. 内でも使用可能です。この場合、アプリケーションのベース <acronym>URL</acronym>
  322. (<filename>/user/application-root</filename>) の自動検出が
  323. <classname>Zend_Controller_Request_Http</classname> によって行われ、適切に使用されます。
  324. </para>
  325. <para>
  326. ベース <acronym>URL</acronym> の検出に失敗する場合は、
  327. <classname>Zend_Controller_Request_Http</classname> のメソッド <methodname>setBaseUrl()</methodname>
  328. を使用してベースパスを上書き指定できます
  329. (<link linkend="zend.controller.request.http.baseurl">ベース URL およびサブディレクトリ</link>を参照ください)。
  330. </para>
  331. <programlisting language="php"><![CDATA[
  332. $request->setBaseUrl('/~user/application-root/');
  333. ]]></programlisting>
  334. </sect2>
  335. <sect2 id="zend.controller.router.global.parameters">
  336. <title>グローバルパラメータ</title>
  337. <para>
  338. グローバルパラメータをルータ内で設定できます。
  339. これは <methodname>setGlobalParam()</methodname>
  340. によってルートに自動的に適用されます。
  341. グローバルパラメータが設定されているにもかかわらず
  342. 直接メソッドによっても設定された場合は、
  343. ユーザが設定したパラメータのほうがグローバルパラメータより優先されます。
  344. グローバルパラメータは、このように設定します。
  345. </para>
  346. <programlisting language="php"><![CDATA[
  347. $router->setGlobalParam('lang', 'en');
  348. ]]></programlisting>
  349. </sect2>
  350. <sect2 id="zend.controller.router.routes">
  351. <title>ルートの型</title>
  352. <xi:include href="Zend_Controller-Router-Route.xml" />
  353. <xi:include href="Zend_Controller-Router-Route-Static.xml" />
  354. <xi:include href="Zend_Controller-Router-Route-Regex.xml" />
  355. <xi:include href="Zend_Controller-Router-Route-Hostname.xml" />
  356. <xi:include href="Zend_Controller-Router-Route-Chain.xml" />
  357. <xi:include href="Zend_Controller-Router-Route-Rest.xml" />
  358. </sect2>
  359. <sect2 id="zend.controller.router.add-config">
  360. <title>RewriteRouter での Zend_Config の使用法</title>
  361. <para>
  362. 新しいルートを追加する際に、
  363. いちいちコードを書き換えるのではなく設定ファイルの変更で対応できると便利でしょう。
  364. そんなときには <methodname>addConfig()</methodname> メソッドを使用します。基本的な使用法は、
  365. まず <classname>Zend_Config</classname> 互換の設定を作成し、それをコードに読み込み、
  366. そして RewriteRouter に渡すことです。
  367. </para>
  368. <para>
  369. 例として、次のような <acronym>INI</acronym> ファイルを考えてみましょう。
  370. </para>
  371. <programlisting language="php"><![CDATA[
  372. [production]
  373. routes.archive.route = "archive/:year/*"
  374. routes.archive.defaults.controller = archive
  375. routes.archive.defaults.action = show
  376. routes.archive.defaults.year = 2000
  377. routes.archive.reqs.year = "\d+"
  378. routes.news.type = "Zend_Controller_Router_Route_Static"
  379. routes.news.route = "news"
  380. routes.news.defaults.controller = "news"
  381. routes.news.defaults.action = "list"
  382. routes.archive.type = "Zend_Controller_Router_Route_Regex"
  383. routes.archive.route = "archive/(\d+)"
  384. routes.archive.defaults.controller = "archive"
  385. routes.archive.defaults.action = "show"
  386. routes.archive.map.1 = "year"
  387. ; あるいは: routes.archive.map.year = 1
  388. ]]></programlisting>
  389. <para>
  390. 上の <acronym>INI</acronym> ファイルを、次のようにして
  391. <classname>Zend_Config</classname> オブジェクトに読み込みます。
  392. </para>
  393. <programlisting language="php"><![CDATA[
  394. $config = new Zend_Config_Ini('/path/to/config.ini', 'production');
  395. $router = new Zend_Controller_Router_Rewrite();
  396. $router->addConfig($config, 'routes');
  397. ]]></programlisting>
  398. <para>
  399. 上の例では、<acronym>INI</acronym> ファイルの 'routes' セクションを使用してルートを決めるよう、
  400. ルータに指定しています。このセクションの第一レベルのキーがルート名に対応します。
  401. 上の例だと 'archive' と 'news' がこれにあたります。
  402. ルートの各エントリには、最低限 'route' エントリとひとつ以上の 'defaults'
  403. エントリが必要となります。また、オプションでひとつ以上の 'reqs'
  404. ('required' の略) も指定できます。ここで指定したものが、それぞれ
  405. <classname>Zend_Controller_Router_Route_Interface</classname>
  406. オブジェクトに対する引数となります。オプションのキー 'type' を使用すると、
  407. 特定のルートで使用するルートクラスの型を指定できます。デフォルトでは、これは
  408. <classname>Zend_Controller_Router_Route</classname> となります。上の例では、
  409. 'news' ルートで
  410. <classname>Zend_Controller_Router_Route_Static</classname>
  411. を使用するようにしています。
  412. </para>
  413. </sect2>
  414. <sect2 id="zend.controller.router.subclassing">
  415. <title>ルータのサブクラスの作成</title>
  416. <para>
  417. 標準の rewrite ルータには、必要となるであろう機能のほとんどが組み込まれています。
  418. もし新しいルータ型を作成する必要があるとすれば、
  419. それは既存のルートに対して新しい機能を追加したり機能を変更したりしたい場合くらいでしょう。
  420. </para>
  421. <para>
  422. どこかで、既存のものとはまったく異なるルーティング処理が必要となったとしましょう。
  423. そんな場合には <classname>Zend_Controller_Router_Interface</classname>
  424. を使用します。これは、ルータとして最低限必要なひとつのメソッドのみを定義したインターフェイスです。
  425. </para>
  426. <programlisting language="php"><![CDATA[
  427. interface Zend_Controller_Router_Interface
  428. {
  429. /**
  430. * @param Zend_Controller_Request_Abstract $request
  431. * @throws Zend_Controller_Router_Exception
  432. * @return Zend_Controller_Request_Abstract
  433. */
  434. public function route(Zend_Controller_Request_Abstract $request);
  435. }
  436. ]]></programlisting>
  437. <para>
  438. ルーティング処理は、システムが最初にリクエストを受け取った際に一度だけ行われます。
  439. ルータの役割は、リクエストの内容に応じてコントローラやアクションとオプションパラメータを決定し、
  440. それをリクエストに設定することです。
  441. その後、リクエストオブジェクトがディスパッチャに渡されます。
  442. ルートに対応するディスパッチトークンがない場合は、ルータは何も行いません。
  443. </para>
  444. </sect2>
  445. </sect1>
  446. <!--
  447. vim:se ts=4 sw=4 et:
  448. -->