Zend_Acl.xml 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 24249 -->
  4. <sect1 id="zend.acl.introduction">
  5. <title>導入</title>
  6. <para>
  7. <classname>Zend_Acl</classname> は、軽量で柔軟なアクセス制御リスト (<acronym>ACL</acronym>)
  8. 機能と権限管理機能を提供します。アプリケーションでは一般に、
  9. 保護されたオブジェクトへのアクセスを制御するためにこれらの機能を使用します。
  10. </para>
  11. <para>
  12. このドキュメントにおいて、
  13. </para>
  14. <itemizedlist>
  15. <listitem>
  16. <para>
  17. <emphasis>リソース (Resource)</emphasis>
  18. とは、アクセス制御の対象となるオブジェクトのことです。
  19. </para>
  20. </listitem>
  21. <listitem>
  22. <para>
  23. <emphasis>ロール (Role)</emphasis>
  24. とは、リソースに対してのアクセスを要求するオブジェクトのことです。
  25. </para>
  26. </listitem>
  27. </itemizedlist>
  28. <para>
  29. 簡単に言うと、<emphasis>ロールがリソースに対してのアクセスを要求する</emphasis>
  30. ということです。たとえば、ある人が自動車を利用したいと考えているとしましょう。
  31. この場合、自動車を誰が利用できるのかが管理されているとすると
  32. 「ある人」がロールで「自動車」がリソースとなります。
  33. </para>
  34. <para>
  35. アクセス制御リスト (<acronym>ACL</acronym>) での設定によって、アプリケーションは
  36. 要求してきたオブジェクト (ロール) が制限されたオブジェクト (リソース)
  37. へのアクセスを認められているかどうかを制御します。
  38. </para>
  39. <sect2 id="zend.acl.introduction.resources">
  40. <title>リソース</title>
  41. <para>
  42. <classname>Zend_Acl</classname> では、リソースを作成するのは簡単です。<classname>Zend_Acl</classname> の
  43. <classname>Zend_Acl_Resource_Interface</classname> に、
  44. 開発者がリソースを作成するの手助けする機能が含まれています。
  45. リソースクラスは、単にこのインターフェイスを実装するだけで作成できます。
  46. このインターフェイスに含まれるメソッドはひとつだけで、それは
  47. <methodname>getResourceId()</methodname> です。このメソッドにより、
  48. <classname>Zend_Acl</classname> ではそのオブジェクトがリソースであると判断します。さらに、
  49. 基本的なリソースの実装を含む <classname>Zend_Acl_Resource</classname>
  50. が <classname>Zend_Acl</classname> にインクルードされています。
  51. 開発者は、必要な部分だけを拡張することでリソースを作成できます。
  52. </para>
  53. <para>
  54. <classname>Zend_Acl</classname> は、ツリー構造を提供しています。これを用いて複数のリソース
  55. ("アクセス制御されているエリア") を追加できます。
  56. リソースはこのようにツリー構造で管理されるので、全般的なもの
  57. (ルートに近いほう) から特殊なもの (末端に近いほう) までを扱うことができます。
  58. 特定のリソースに対して問い合わせを行うと、
  59. リソースの階層をたどって自動的に上位階層への問い合わせも行われます。
  60. これにより、規則を階層化して管理すること実現できます。
  61. たとえば、ある都市のすべての建物に適用されるデフォルトの規則がある場合に、
  62. それを各建物に適用する代わりに都市に対して適用することができるようになります。
  63. 中にはこの規則の例外となる建物もあるかもしれません。
  64. <classname>Zend_Acl</classname> では、このような状況も簡単に処理できます。
  65. 例外的な規則を適用する建物については、建物に対して直接規則を定義すればいいのです。
  66. リソースは、単一のリソースしか継承することができません。
  67. そしてその継承元のリソースがまた別の親リソースを継承し……
  68. といった具合になります。
  69. </para>
  70. <para>
  71. <classname>Zend_Acl</classname> は、リソースに対する権限 ("create"、"read"、"update"、"delete" など)
  72. もサポートしており、すべての権限あるいは一部の権限に影響を及ぼす規則を、
  73. リソースに対して割り当てることができます。
  74. </para>
  75. </sect2>
  76. <sect2 id="zend.acl.introduction.roles">
  77. <title>ロール</title>
  78. <para>
  79. リソースと同様、ロールを作成するのも簡単です。Zend_Acl の
  80. <classname>Zend_Acl_Role_Interface</classname> に、
  81. 開発者がロールを作成するの手助けする機能が含まれています。
  82. ロールクラスは、単にこのインターフェイスを実装するだけで作成できます。
  83. このインターフェイスに含まれるメソッドはひとつだけで、それは
  84. <methodname>getRoleId()</methodname> です。このメソッドにより、
  85. Zend_Acl はそのオブジェクトがロールであると判断します。さらに、
  86. 基本的なロールの実装を含む <classname>Zend_Acl_Role</classname>
  87. が <classname>Zend_Acl</classname> にインクルードされています。
  88. 開発者は、必要な部分だけを拡張することでリソースを作成できます。
  89. </para>
  90. <para>
  91. <classname>Zend_Acl</classname> では、あるロールは他の複数のロールを継承できます。
  92. これは、それぞれのロールの規則を継承することをサポートするためのものです。
  93. たとえば、"sally" のようなユーザロールは、"editor" かつ "administrator"
  94. のように複数の親ロールに属することもありえるでしょう。
  95. この場合、開発者は "editor" および "administrator" にそれぞれ別に規則を定義します。
  96. そして "sally" がその両方を継承することにします。
  97. "sally" に規則を直接定義する必要はありません。
  98. </para>
  99. <para>
  100. 複数のロールからの継承は非常に便利ですが、
  101. 多重継承は複雑な問題を引き起こすこともあります。
  102. 次の例は、あいまいな条件になったときに <classname>Zend_Acl</classname>
  103. がそれをどう解決するかを示すものです。
  104. </para>
  105. <example id="zend.acl.introduction.roles.example.multiple_inheritance">
  106. <title>ロールの多重継承</title>
  107. <para>
  108. 以下のコードでは、基本となる三つのロール "guest"、"member"
  109. および "admin" を定義しています。他のロールはこれらを継承することになります。
  110. 次に、"someUser" というロールを作成してこれらの三つのロールを継承します。
  111. これらのロールが配列 <code>$parents</code> にあらわれる順序が重要となります。
  112. 問い合わせ先のロール (ここでは "someUser" にアクセス規則が定義されていないが
  113. その継承元 (ここでは "guest"、"member" および "admin")
  114. には定義されているという場合、<classname>Zend_Acl</classname> はそちらを検索します。
  115. </para>
  116. <programlisting language="php"><![CDATA[
  117. $acl = new Zend_Acl();
  118. $acl->addRole(new Zend_Acl_Role('guest'))
  119. ->addRole(new Zend_Acl_Role('member'))
  120. ->addRole(new Zend_Acl_Role('admin'));
  121. $parents = array('guest', 'member', 'admin');
  122. $acl->addRole(new Zend_Acl_Role('someUser'), $parents);
  123. $acl->add(new Zend_Acl_Resource('someResource'));
  124. $acl->deny('guest', 'someResource');
  125. $acl->allow('member', 'someResource');
  126. echo $acl->isAllowed('someUser', 'someResource') ? 'allowed' : 'denied';
  127. ]]></programlisting>
  128. <para>
  129. "someUser" ロールおよび
  130. "someResource" に対する規則は定義されていないので、
  131. <classname>Zend_Acl</classname> は、その規則が "someUser"
  132. の継承元ロールで定義されているものとして検索します。
  133. まず "admin" ロールを探しますが、ここではアクセス規則が定義されていません。
  134. 次に "member" ロールを探し、ここで <classname>Zend_Acl</classname> が規則を発見します。つまり
  135. "member" は "someResource" へのアクセスを許可されているということです。
  136. </para>
  137. <para>
  138. しかし、仮に <classname>Zend_Acl</classname> がさらに別の親に対しても規則の検索を続けたとすると、
  139. "guest" は "someResource" へのアクセスが拒否されていることに気づくことでしょう。
  140. これは問題となります。というのも、"someUser" は
  141. "someResource" へのアクセスが許可されていると同時に拒否されているわけで、
  142. それぞれ別の親ロールから取得した規則が衝突することになるからです。
  143. </para>
  144. <para>
  145. <classname>Zend_Acl</classname> では、このような衝突の可能性を解決するために、
  146. 直近に継承されたロールの優先度が高くなるようにしています。
  147. 今回の場合は、"member" のほうが
  148. "guest" ロールより先に調べられ、例のコードの出力は
  149. "allowed" となります。
  150. </para>
  151. </example>
  152. <note>
  153. <para>
  154. 複数の親をロールに指定する場合は、承認クエリでの規則の検索順を覚えておきましょう。
  155. 最後に指定した親が最初に対象となります。
  156. </para>
  157. </note>
  158. </sect2>
  159. <sect2 id="zend.acl.introduction.creating">
  160. <title>アクセス制御リストの作成</title>
  161. <para>
  162. アクセス制御リスト (<acronym>ACL</acronym>) を使用して、物理的あるいは仮想的なオブジェクトの組み合わせを
  163. お望みどおりに表現できます。しかしここでは、説明用として、
  164. 基本的なコンテンツ管理システム (<acronym>CMS</acronym>) の <acronym>ACL</acronym> を考えます。
  165. これは、さまざまな領域で複数階層のグループを管理するものです。
  166. 新しい <acronym>ACL</acronym> オブジェクトを作成するには、何もパラメータを指定せずに
  167. <acronym>ACL</acronym> のインスタンスを作成します。
  168. </para>
  169. <programlisting language="php"><![CDATA[
  170. $acl = new Zend_Acl();
  171. ]]></programlisting>
  172. <note>
  173. <para>
  174. 開発者が "allow" 規則を指定しない限り、<classname>Zend_Acl</classname>
  175. はあらゆるロールのすべてのリソース上の権限からのアクセスも拒否します。
  176. </para>
  177. </note>
  178. </sect2>
  179. <sect2 id="zend.acl.introduction.role_registry">
  180. <title>ロールの登録</title>
  181. <para>
  182. コンテンツ管理システムでは、ほとんどすべての場面で権限階層の管理が必要となります。
  183. これにより、ユーザの編集権限を決定します。たとえば 'Guest'
  184. グループに対してはデモ用に限定したアクセス権限のみを許可し、
  185. 'Staff' グループは通常の操作をする大半の <acronym>CMS</acronym> ユーザ用に作成し、
  186. 'Editor' グループにはコンテンツの公開やレビュー、保存や削除の権限を与え、
  187. 最後に 'Administrator' は、その他のすべてのグループの権限に加えて
  188. 機密情報の管理やユーザ管理、バックエンドの設定データ、
  189. バックアップ、そしてエクスポート機能を与えるといったようになります。
  190. これらの権限を、ロールレジストリで表すことができます。
  191. 各グループの権限を '親の' グループから継承させ、
  192. そのグループに固有の権限を追加で定義します。
  193. これらの権限を整理すると、次のようになります。
  194. </para>
  195. <table id="zend.acl.introduction.role_registry.table.example_cms_access_controls">
  196. <title>サンプル CMS 用のアクセス制御</title>
  197. <tgroup cols="3">
  198. <thead>
  199. <row>
  200. <entry>名前</entry>
  201. <entry>権限</entry>
  202. <entry>継承する権限の継承元</entry>
  203. </row>
  204. </thead>
  205. <tbody>
  206. <row>
  207. <entry>Guest</entry>
  208. <entry>View</entry>
  209. <entry>なし</entry>
  210. </row>
  211. <row>
  212. <entry>Staff</entry>
  213. <entry>Edit, Submit, Revise</entry>
  214. <entry>Guest</entry>
  215. </row>
  216. <row>
  217. <entry>Editor</entry>
  218. <entry>Publish, Archive, Delete</entry>
  219. <entry>Staff</entry>
  220. </row>
  221. <row>
  222. <entry>Administrator</entry>
  223. <entry>(すべてのアクセスを許可)</entry>
  224. <entry>なし</entry>
  225. </row>
  226. </tbody>
  227. </tgroup>
  228. </table>
  229. <para>
  230. この例では <classname>Zend_Acl_Role</classname> を用いていますが、
  231. <classname>Zend_Acl_Role_Interface</classname> を実装しているオブジェクトなら何でも使用可能です。
  232. これらのグループを、次のようにしてロールレジストリに追加します。
  233. </para>
  234. <programlisting language="php"><![CDATA[
  235. $acl = new Zend_Acl();
  236. // Zend_Acl_Role を使用して、グループをロールレジストリに追加します
  237. // Guest はアクセス制御を受け継ぎません
  238. $roleGuest = new Zend_Acl_Role('guest');
  239. $acl->addRole($roleGuest);
  240. // Staff は guest の権限を継承します
  241. $acl->addRole(new Zend_Acl_Role('staff'), $roleGuest);
  242. /*
  243. あるいは、上の内容は次のように書くこともできます
  244. $acl->addRole(new Zend_Acl_Role('staff'), 'guest');
  245. */
  246. // Editor は staff の権限を継承します
  247. $acl->addRole(new Zend_Acl_Role('editor'), 'staff');
  248. // Administrator はアクセス制御を受け継ぎません
  249. $acl->addRole(new Zend_Acl_Role('administrator'));
  250. ]]></programlisting>
  251. </sect2>
  252. <sect2 id="zend.acl.introduction.defining">
  253. <title>アクセス制御の定義</title>
  254. <para>
  255. <acronym>ACL</acronym> に適切なロールが含まれた状態になりました。これで、リソースに対して
  256. ロールがどのようにアクセスするのかという規則を定義できる状態になりました。
  257. この例では特定のリソースを定義していないことにお気づきかもしれません。
  258. この場合、規則はすべてのリソースに対して適用されます。
  259. <classname>Zend_Acl</classname> を使用すると、全般的なものであろうが特殊なものであろうが
  260. 規則を適用するだけで定義できるようになります。
  261. リソースやロールは、その継承元で定義されている規則を引き継ぐからです。
  262. </para>
  263. <note>
  264. <para>
  265. 一般に、<classname>Zend_Acl</classname> は指定された規則に従います。
  266. ただし、より詳細な規則が別途適用されている場合は例外です。
  267. </para>
  268. </note>
  269. <para>
  270. そのため、複雑な規則の組み合わせを最小限のコードで定義できるようになります。
  271. 上で定義した基本的な権限を適用するには、次のようにします。
  272. </para>
  273. <programlisting language="php"><![CDATA[
  274. $acl = new Zend_Acl();
  275. $roleGuest = new Zend_Acl_Role('guest');
  276. $acl->addRole($roleGuest);
  277. $acl->addRole(new Zend_Acl_Role('staff'), $roleGuest);
  278. $acl->addRole(new Zend_Acl_Role('editor'), 'staff');
  279. $acl->addRole(new Zend_Acl_Role('administrator'));
  280. // Guest は、コンテンツを閲覧することのみが可能です
  281. $acl->allow($roleGuest, null, 'view');
  282. /*
  283. 上と同じ内容を、このように書くこともできます
  284. $acl->allow('guest', null, 'view');
  285. //*/
  286. // Staff は guest の権限をすべて引き継いだうえで、さらに追加の権限を必要とします
  287. $acl->allow('staff', null, array('edit', 'submit', 'revise'));
  288. // Editor は、staff の権限 (view、edit、submit および revise)
  289. // を引き継いだうえで、さらに追加の権限を必要とします
  290. $acl->allow('editor', null, array('publish', 'archive', 'delete'));
  291. // Administrator は何も引き継ぎませんが、すべての権限が認められています
  292. $acl->allow('administrator');
  293. ]]></programlisting>
  294. <para>
  295. 上の <methodname>allow()</methodname> のコールにおける <constant>NULL</constant> は、
  296. 規則をすべてのリソースに対して適用することを意味します。
  297. </para>
  298. </sect2>
  299. <sect2 id="zend.acl.introduction.querying">
  300. <title>ACL への問い合わせ</title>
  301. <para>
  302. これで、柔軟な <acronym>ACL</acronym> が作成できました。これにより、
  303. ウェブアプリケーションの使用者が、
  304. ある機能を使用するために必要な権限を持っているかを調べられるようになりました。
  305. 問い合わせを行うのは簡単で、単に <methodname>isAllowed()</methodname>
  306. メソッドを使用するだけです。
  307. </para>
  308. <programlisting language="php"><![CDATA[
  309. echo $acl->isAllowed('guest', null, 'view') ?
  310. "allowed" : "denied";
  311. // allowed となります
  312. echo $acl->isAllowed('staff', null, 'publish') ?
  313. "allowed" : "denied";
  314. // denied となります
  315. echo $acl->isAllowed('staff', null, 'revise') ?
  316. "allowed" : "denied";
  317. // allowed となります
  318. echo $acl->isAllowed('editor', null, 'view') ?
  319. "allowed" : "denied";
  320. // guest から引き継いでいるので allowed となります
  321. echo $acl->isAllowed('editor', null, 'update') ?
  322. "allowed" : "denied";
  323. // 'update' 用の規則がないので denied となります
  324. echo $acl->isAllowed('administrator', null, 'view') ?
  325. "allowed" : "denied";
  326. // administrator はすべての権限が許可されているので allowed となります
  327. echo $acl->isAllowed('administrator') ?
  328. "allowed" : "denied";
  329. // administrator はすべての権限が許可されているので allowed となります
  330. echo $acl->isAllowed('administrator', null, 'update') ?
  331. "allowed" : "denied";
  332. // administrator はすべての権限が許可されているので allowed となります
  333. ]]></programlisting>
  334. </sect2>
  335. </sect1>
  336. <!--
  337. vim:se ts=4 sw=4 et:
  338. -->