Zend_Controller-Plugins-ErrorHandler.xml 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 24249 -->
  4. <sect3 id="zend.controller.plugins.standard.errorhandler">
  5. <title>Zend_Controller_Plugin_ErrorHandler(日本語)</title>
  6. <para>
  7. <classname>Zend_Controller_Plugin_ErrorHandler</classname>
  8. は、アプリケーションからスローされた例外を処理するためのプラグインです。
  9. たとえば、指定したコントローラやアクションが見つからないといったエラーを処理します。
  10. これは、<link linkend="zend.controller.exceptions">MVC の例外についてのセクション</link>
  11. で説明したメソッド群の代わりとして使用できます。
  12. </para>
  13. <para>
  14. このプラグインが主に対象としているのは、次のような例外です。
  15. </para>
  16. <itemizedlist>
  17. <listitem>
  18. <para>一致したルートが見つからない場合に発生する例外</para>
  19. </listitem>
  20. <listitem>
  21. <para>コントローラやアクションメソッドが見つからない場合に発生する例外</para>
  22. </listitem>
  23. <listitem>
  24. <para>アクションコントローラ内で発生する例外</para>
  25. </listitem>
  26. </itemizedlist>
  27. <para>
  28. 言い換えると、<emphasis>ErrorHandler</emphasis> プラグインが想定しているのは、<acronym>HTTP</acronym>
  29. 404 型のエラー (ページが存在しない) と 500 型のエラー (内部エラー)
  30. ということになります。
  31. 他のプラグインで発生した例外の処理は、想定していません。
  32. </para>
  33. <para>
  34. デフォルトでは、<classname>Zend_Controller_Plugin_ErrorHandler</classname>
  35. はデフォルトモジュールの <methodname>ErrorController::errorAction()</methodname>
  36. に処理を転送します。これを変更するには、以下のようなアクセス用メソッドを使用します。
  37. </para>
  38. <itemizedlist>
  39. <listitem>
  40. <para>
  41. <methodname>setErrorHandlerModule()</methodname> は、
  42. 使用するコントローラモジュール名を設定します。
  43. </para>
  44. </listitem>
  45. <listitem>
  46. <para>
  47. <methodname>setErrorHandlerController()</methodname> は、
  48. 使用するコントローラを設定します。
  49. </para>
  50. </listitem>
  51. <listitem>
  52. <para>
  53. <methodname>setErrorHandlerAction()</methodname> は、
  54. 使用するコントローラアクションを設定します。
  55. </para>
  56. </listitem>
  57. <listitem>
  58. <para>
  59. <methodname>setErrorHandler()</methodname> は連想配列を受け取ります。
  60. この連想配列のキーには 'module'、'controller'
  61. あるいは 'action' を指定することができ、
  62. それぞれ対応する値を設定します。
  63. </para>
  64. </listitem>
  65. </itemizedlist>
  66. <para>
  67. さらに、コンストラクタの引数として連想配列を渡すこともできます。
  68. この場合、その配列がそのまま <methodname>setErrorHandler()</methodname>
  69. に渡されます。
  70. </para>
  71. <para>
  72. <classname>Zend_Controller_Plugin_ErrorHandler</classname> は
  73. <methodname>postDispatch()</methodname> フックとして登録され、
  74. <link linkend="zend.controller.response">レスポンスオブジェクト</link>
  75. に格納された例外を確認します。もし何かの例外が見つかったら、
  76. 事前に登録されているエラーハンドラアクションに処理を転送します。
  77. </para>
  78. <para>
  79. エラーハンドラへのディスパッチ中に例外が発生した場合は、
  80. このプラグインはフロントコントローラに例外をスローします。
  81. その際に、レスポンスオブジェクトに格納された直近の例外を再度スローします。
  82. </para>
  83. <sect4 id="zend.controller.plugins.standard.errorhandler.fourohfour">
  84. <title>404 ハンドラとしての ErrorHandler の使用</title>
  85. <para>
  86. <emphasis>ErrorHandler</emphasis> プラグインが捕捉するのは、
  87. アプリケーションのエラーだけではありません。
  88. コントローラチェインが次のコントローラクラスやアクションメソッドを
  89. 見つけられなかった場合に、404 ハンドラとして動作させることもできます。
  90. そのためには、エラーコントローラ内で例外の型を調べる必要があります。
  91. </para>
  92. <para>
  93. 捕捉した例外は、リクエストで登録したオブジェクトの中に記録されます。
  94. これを取得するには、
  95. <methodname>Zend_Controller_Action::_getParam('error_handler')</methodname>
  96. を使用します。
  97. </para>
  98. <programlisting language="php"><![CDATA[
  99. class ErrorController extends Zend_Controller_Action
  100. {
  101. public function errorAction()
  102. {
  103. $errors = $this->_getParam('error_handler');
  104. }
  105. }
  106. ]]></programlisting>
  107. <para>
  108. エラーオブジェクトを取得したら、次に
  109. <command>$errors->type</command> でその型を調べます。
  110. これは、次のいずれかとなります。
  111. </para>
  112. <itemizedlist>
  113. <listitem>
  114. <para>
  115. <constant>Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ROUTE</constant>
  116. は、一致したルートが見つからなかったことを表します。
  117. </para>
  118. </listitem>
  119. <listitem>
  120. <para>
  121. <constant>Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_CONTROLLER</constant>
  122. は、コントローラが見つからなかったことを表します。
  123. </para>
  124. </listitem>
  125. <listitem>
  126. <para>
  127. <constant>Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ACTION</constant>
  128. は、アクションが見つからなかったことを表します。
  129. </para>
  130. </listitem>
  131. <listitem>
  132. <para>
  133. <constant>Zend_Controller_Plugin_ErrorHandler::EXCEPTION_OTHER</constant>
  134. は、その他の例外を表します。
  135. </para>
  136. </listitem>
  137. </itemizedlist>
  138. <para>
  139. 最初の3つの型であった場合に、404 ページを返せばいいわけです。
  140. </para>
  141. <programlisting language="php"><![CDATA[
  142. class ErrorController extends Zend_Controller_Action
  143. {
  144. public function errorAction()
  145. {
  146. $errors = $this->_getParam('error_handler');
  147. switch ($errors->type) {
  148. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ROUTE:
  149. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_CONTROLLER:
  150. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ACTION:
  151. // 404 エラー -- コントローラあるいはアクションが見つかりません
  152. $this->getResponse()
  153. ->setRawHeader('HTTP/1.1 404 Not Found');
  154. // ... 何か、表示する内容を作成します ...
  155. break;
  156. default:
  157. // アプリケーションのエラー。エラーページを表示しますが、
  158. // ステータスコードは変更しません
  159. break;
  160. }
  161. }
  162. }
  163. ]]></programlisting>
  164. <para>
  165. エラーハンドラで発生した礼儀を取得するには、
  166. <emphasis>error_handler</emphasis> オブジェクトのプロパティ
  167. <property>exception</property> を使用します。
  168. </para>
  169. <programlisting language="php"><![CDATA[
  170. public function errorAction()
  171. {
  172. $errors = $this->_getParam('error_handler');
  173. switch ($errors->type) {
  174. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ROUTE:
  175. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_CONTROLLER:
  176. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ACTION:
  177. // 404 エラー -- コントローラあるいはアクションが見つかりません
  178. $this->getResponse()
  179. ->setRawHeader('HTTP/1.1 404 Not Found');
  180. // ... 何か、表示する内容を作成します ...
  181. break;
  182. default:
  183. // アプリケーションのエラー。エラーページを表示しますが、
  184. // ステータスコードは変更しません
  185. // ...
  186. // 例外をログに記録します
  187. $exception = $errors->exception;
  188. $log = new Zend_Log(
  189. new Zend_Log_Writer_Stream(
  190. '/tmp/applicationException.log'
  191. )
  192. );
  193. $log->debug($exception->getMessage() . "\n" .
  194. $exception->getTraceAsString());
  195. break;
  196. }
  197. }
  198. ]]></programlisting>
  199. </sect4>
  200. <sect4 id="zend.controller.plugins.standard.errorhandler.buffer">
  201. <title>前回のレンダリング結果の扱い</title>
  202. <para>
  203. ひとつのリクエストで複数のアクションにディスパッチする場合、
  204. あるいはアクション内で <methodname>render()</methodname> を複数回コールする場合などは、
  205. レスポンスオブジェクト内にすでに内容が格納されていることがあります。
  206. これをうまく処理しないと、本当に必要な内容と
  207. それ以外の内容が混じってしまう恐れがあります。
  208. </para>
  209. <para>
  210. そのような場合にページの中でエラー処理をしたい場合は、
  211. 特に何も手を加える必要はありません。
  212. そのような内容をレンダリングしたくないという場合は、
  213. ビューのレンダリング前に以前の内容を消去しておく必要があります。
  214. </para>
  215. <programlisting language="php"><![CDATA[
  216. $this->getResponse()->clearBody();
  217. ]]></programlisting>
  218. </sect4>
  219. <sect4 id="zend.controller.plugins.standard.errorhandler.examples">
  220. <title>プラグインの使用例</title>
  221. <example id="zend.controller.plugins.standard.errorhandler.examples.example-1">
  222. <title>標準的な使用法</title>
  223. <programlisting language="php"><![CDATA[
  224. $front = Zend_Controller_Front::getInstance();
  225. $front->registerPlugin(new Zend_Controller_Plugin_ErrorHandler());
  226. ]]></programlisting>
  227. </example>
  228. <example id="zend.controller.plugins.standard.errorhandler.examples.example-2">
  229. <title>別のエラーハンドラの設定</title>
  230. <programlisting language="php"><![CDATA[
  231. $front = Zend_Controller_Front::getInstance();
  232. $front->registerPlugin(new Zend_Controller_Plugin_ErrorHandler(array(
  233. 'module' => 'mystuff',
  234. 'controller' => 'static',
  235. 'action' => 'error'
  236. )));
  237. ]]></programlisting>
  238. </example>
  239. <example id="zend.controller.plugins.standard.errorhandler.examples.example-3">
  240. <title>アクセス用メソッドの使用</title>
  241. <programlisting language="php"><![CDATA[
  242. $plugin = new Zend_Controller_Plugin_ErrorHandler();
  243. $plugin->setErrorHandlerModule('mystuff')
  244. ->setErrorHandlerController('static')
  245. ->setErrorHandlerAction('error');
  246. $front = Zend_Controller_Front::getInstance();
  247. $front->registerPlugin($plugin);
  248. ]]></programlisting>
  249. </example>
  250. </sect4>
  251. <sect4 id="zend.controller.plugins.standard.errorhandler.controllerexamples">
  252. <title>エラーコントローラの例</title>
  253. <para>
  254. エラーハンドラプラグインを使用するには、
  255. エラーコントローラが必要です。以下にシンプルな例を示します。
  256. </para>
  257. <programlisting language="php"><![CDATA[
  258. class ErrorController extends Zend_Controller_Action
  259. {
  260. public function errorAction()
  261. {
  262. $errors = $this->_getParam('error_handler');
  263. switch ($errors->type) {
  264. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ROUTE:
  265. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_CONTROLLER:
  266. case Zend_Controller_Plugin_ErrorHandler::EXCEPTION_NO_ACTION:
  267. // 404 エラー -- コントローラあるいはアクションが見つかりません
  268. $this->getResponse()->setRawHeader('HTTP/1.1 404 Not Found');
  269. $content =<<<EOH
  270. <h1>エラー!</h1>
  271. <p>そのページは存在しません。</p>
  272. EOH;
  273. break;
  274. default:
  275. // アプリケーションのエラー
  276. $content =<<<EOH
  277. <h1>エラー!</h1>
  278. <p>予期せぬエラーが発生しました。後でもう一度お試しください。</p>
  279. EOH;
  280. break;
  281. }
  282. // 前回の内容を消去します
  283. $this->getResponse()->clearBody();
  284. $this->view->content = $content;
  285. }
  286. }
  287. ]]></programlisting>
  288. </sect4>
  289. </sect3>
  290. <!--
  291. vim:se ts=4 sw=4 et:
  292. -->