| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404 |
- <?xml version="1.0" encoding="UTF-8"?>
- <!-- Reviewed: no -->
- <!-- EN-Revision: 24249 -->
- <sect1 id="zend.infocard.basics">
- <title>導入</title>
- <!-- Skip-EN-Revisions: 22747 -->
- <para>注意:このドキュメントでは、英語版のリビジョン 22747 の更新内容をスキップしています。</para>
- <para>
- <classname>Zend_InfoCard</classname> コンポーネントは、
- 情報カード (Information Cards) の relying-party
- サポートを実装したものです。
- 情報カードは、インターネット上でのユーザ識別情報の管理や
- ウェブサイトのユーザ認証に用いるものです。
- 最終的にユーザ認証を行う先のウェブサイトのことを
- <emphasis>relying-party</emphasis> といいます。
- </para>
- <para>
- 情報カードについて、
- あるいはインターネット上の識別メタシステムにおける情報カードの重要性については、
- <ulink url="http://www.identityblog.com/">IdentityBlog</ulink>
- を参照ください。
- </para>
- <sect2 id="zend.infocard.basics.theory">
- <title>基本的な使用法</title>
- <para>
- <classname>Zend_InfoCard</classname> の使用法は、
- <classname>Zend_Auth</classname> コンポーネントの一部として
- <classname>Zend_InfoCard</classname> 認証アダプタを使用するか、
- あるいは単体のコンポーネントとして使用するかのいずれかです。
- どちらの場合についても、ユーザから情報カードを受け取るには
- <acronym>HTML</acronym> のログインフォームの中で次のような <acronym>HTML</acronym> ブロックを使用します。
- </para>
- <programlisting language="html"><![CDATA[
- <form action="http://example.com/server" method="POST">
- <input type='image' src='/images/ic.png' align='center'
- width='120px' style='cursor:pointer' />
- <object type="application/x-informationCard"
- name="xmlToken">
- <param name="tokenType"
- value="urn:oasis:names:tc:SAML:1.0:assertion" />
- <param name="requiredClaims"
- value="http://.../claims/privatepersonalidentifier
- http://.../claims/givenname
- http://.../claims/surname" />
- </object>
- </form>
- ]]></programlisting>
- <para>
- この例において、<code>requiredClaims</code>
- <code><param></code> タグで表しているのが、
- claim (人の姓名など) と呼ばれる識別情報です。
- これは、ウェブサイト ("relying party")
- が情報カードによる認証を行うために必要となります。
- </para>
- <para>
- 上の <acronym>HTML</acronym> をユーザが実行する (クリックする) と、
- ブラウザはカード選択プログラムを実行します。
- これは、そのサイトの要求を満たす情報カードを表示させるだけでなく、
- 条件を満たす情報カードが複数存在する場合に好きなものを選択させることができます。
- この情報カードは、指定した <acronym>URL</acronym> に <acronym>XML</acronym> ドキュメントとして
- <code>POST</code> され、これを
- <classname>Zend_InfoCard</classname> コンポーネントで処理することになります。
- </para>
- <para>
- 情報カードは、<acronym>SSL</acronym> で暗号化した <acronym>URL</acronym> への
- <code>HTTP POST</code> しかできないことに注意しましょう。
- <acronym>SSL</acronym> による暗号化を設定する方法については、
- ウェブサーバのドキュメントを参照ください。
- </para>
- </sect2>
- <sect2 id="zend.infocard.basics.auth">
- <title>Zend_Auth の部品としての使用法</title>
- <para>
- このコンポーネントを <classname>Zend_Auth</classname>
- 認証システムと組み合わせて使用するには、
- <classname>Zend_Auth_Adapter_InfoCard</classname> を使用する必要があります
- (これは、単体で配布されている <classname>Zend_InfoCard</classname>
- には存在しません)。
- この手法での使用例を以下に示します。
- </para>
- <programlisting language="php"><![CDATA[
- <?php
- if (isset($_POST['xmlToken'])) {
- $adapter = new Zend_Auth_Adapter_InfoCard($_POST['xmlToken']);
- $adapter->addCertificatePair('/usr/local/Zend/apache2/conf/server.key',
- '/usr/local/Zend/apache2/conf/server.crt');
- $auth = Zend_Auth::getInstance();
- $result = $auth->authenticate($adapter);
- switch ($result->getCode()) {
- case Zend_Auth_Result::SUCCESS:
- $claims = $result->getIdentity();
- print "Given Name: {$claims->givenname}<br />";
- print "Surname: {$claims->surname}<br />";
- print "Email Address: {$claims->emailaddress}<br />";
- print "PPI: {$claims->getCardID()}<br />";
- break;
- case Zend_Auth_Result::FAILURE_CREDENTIAL_INVALID:
- print "The Credential you provided did not pass validation";
- break;
- default:
- case Zend_Auth_Result::FAILURE:
- print "There was an error processing your credentials.";
- break;
- }
- if (count($result->getMessages()) > 0) {
- print "<pre>";
- var_dump($result->getMessages());
- print "</pre>";
- }
- }
- ?>
- <hr />
- <div id="login" style="font-family: arial; font-size: 2em;">
- <p>Simple Login Demo</p>
- <form method="post">
- <input type="submit" value="Login" />
- <object type="application/x-informationCard" name="xmlToken">
- <param name="tokenType"
- value="urn:oasis:names:tc:SAML:1.0:assertion" />
- <param name="requiredClaims"
- value="http://.../claims/givenname
- http://.../claims/surname
- http://.../claims/emailaddress
- http://.../claims/privatepersonalidentifier" />
- </object>
- </form>
- </div>
- ]]></programlisting>
- <para>
- 上の例では、まず最初に
- <classname>Zend_Auth_Adapter_InfoCard</classname> のインスタンスを作成して、
- カードセレクタから送信された <acronym>XML</acronym> データをそこに渡しています。
- インスタンスを作成したら、次に <acronym>SSL</acronym> 証明書の秘密鍵/公開鍵
- ペアを指定する必要があります。
- このペアは、<code>HTTP POST</code>
- を受け取ったウェブサーバで使用しているものです。
- これらのファイルを使用して、サーバに送信された情報のあて先の検証を行います。
- 情報カードを使用するときにはこれらが必要となります。
- </para>
- <para>
- アダプタの設定がすんだら、あとは
- <classname>Zend_Auth</classname> の標準機能を使って情報カードトークンの検証を行い、
- <methodname>getIdentity()</methodname> で取得した識別情報をもとにユーザの認証を行います。
- </para>
- </sect2>
- <sect2 id="zend.infocard.basics.standalone">
- <title>Zend_InfoCard コンポーネント単体での使用法</title>
- <para>
- <classname>Zend_InfoCard</classname> コンポーネントを、
- それ単体で使用することも可能です。その場合は
- <classname>Zend_InfoCard</classname> クラスを直接操作します。
- <classname>Zend_InfoCard</classname> クラスの使用法は、<classname>Zend_Auth</classname>
- コンポーネントと組み合わせて使用する場合とほぼ同じです。
- 以下に使用例を示します。
- </para>
- <programlisting language="php"><![CDATA[
- <?php
- if (isset($_POST['xmlToken'])) {
- $infocard = new Zend_InfoCard();
- $infocard->addCertificatePair('/usr/local/Zend/apache2/conf/server.key',
- '/usr/local/Zend/apache2/conf/server.crt');
- $claims = $infocard->process($_POST['xmlToken']);
- if($claims->isValid()) {
- print "Given Name: {$claims->givenname}<br />";
- print "Surname: {$claims->surname}<br />";
- print "Email Address: {$claims->emailaddress}<br />";
- print "PPI: {$claims->getCardID()}<br />";
- } else {
- print "Error Validating identity: {$claims->getErrorMsg()}";
- }
- }
- ?>
- <hr />
- <div id="login" style="font-family: arial; font-size: 2em;">
- <p>Simple Login Demo</p>
- <form method="post">
- <input type="submit" value="Login" />
- <object type="application/x-informationCard" name="xmlToken">
- <param name="tokenType"
- value="urn:oasis:names:tc:SAML:1.0:assertion" />
- <param name="requiredClaims"
- value="http://.../claims/givenname
- http://.../claims/surname
- http://.../claims/emailaddress
- http://.../claims/privatepersonalidentifier" />
- </object>
- </form>
- </div>
- ]]></programlisting>
- <para>
- 上の例では、<classname>Zend_InfoCard</classname>
- コンポーネントを単体で使用して、ユーザが送信したトークンを検証しています。
- <classname>Zend_Auth_Adapter_InfoCard</classname> の場合と同様、
- <classname>Zend_InfoCard</classname> のインスタンスを作成してから
- ウェブサーバの <acronym>SSL</acronym> 証明書の公開キー/秘密キーペアを設定します。
- 設定がすんだら、<methodname>process()</methodname>
- メソッドで情報カードの処理を行ってその結果を返します。
- </para>
- </sect2>
- <sect2 id="zend.infocard.basics.claims">
- <title>Claims オブジェクトの使用法</title>
- <para>
- <classname>Zend_InfoCard</classname> の使用方法
- (単体で使用するか、あるいは <classname>Zend_Auth</classname> の一部として
- <classname>Zend_Auth_Adapter_InfoCard</classname> 経由で使用するか)
- にかかわらず、情報カードを処理した結果は
- <classname>Zend_InfoCard_Claims</classname> オブジェクトとして返されます。
- このオブジェクトには assertions (claims) が含まれます。
- これは、ユーザ認証の際にあなたのサイトが出した要求にもとづいて、
- ユーザが送信したデータから作成したものです。
- 上の例で示したように、情報カードの妥当性を確認するには
- <methodname>Zend_InfoCard_Claims::isValid()</methodname>
- メソッドをコールします。claims そのものを取得するには、
- 単純に識別子 (<code>givenname</code> など)
- をオブジェクトのプロパティとして指定してアクセスするか、
- あるいは <methodname>getClaim()</methodname> メソッドを使用します。
- </para>
- <para>
- ほとんどの場合においては <methodname>getClaim()</methodname>
- メソッドを使用する必要はありません。
- しかし、もし <code>requiredClaims</code>
- が複数の異なるソース/名前空間からの情報を要求している場合は、
- それをこのメソッドで明示的に取り出す必要があります
- (claim の完全な <acronym>URI</acronym> を私、情報カードの中からその値を取得します)。
- 一般論として、<classname>Zend_InfoCard</classname>
- コンポーネントがデフォルトで設定する claim 用 <acronym>URI</acronym>
- は情報カードの中で最もよく用いられるものです。
- この場合は単純にプロパティを使用してアクセスできます。
- </para>
- <para>
- 検証処理の中で開発者が行わなければならない部分は、
- 情報カード内の claim の発行元を調べて
- それが信頼できる情報元かどうかを判定するところです。
- これを行うために、<classname>Zend_InfoCard_Claims</classname>
- オブジェクトには <methodname>getIssuer()</methodname> メソッドが用意されています。
- このメソッドは、情報カードの claim の発行元 <acronym>URI</acronym> を返します。
- </para>
- </sect2>
- <sect2 id="zend.infocard.basics.attaching">
- <title>既存のアカウントへの情報カードの添付</title>
- <para>
- 既存の認証システムに情報カードのサポートを追加することもできます。
- そのためには、private personal identifier
- (PPI) を昔ながらの認証アカウントに埋め込み、
- 最低限の claim である
- <code>http://schemas.xmlsoap.org/ws/2005/05/identity/claims/privatepersonalidentifier</code>
- をリクエストの <code>requiredClaims</code>
- に指定します。この claim が要求されると、
- <classname>Zend_InfoCard_Claims</classname>
- オブジェクトはそのカード用の一意な識別子を用意します。
- これは、<methodname>getCardID()</methodname> メソッドによって行います。
- </para>
- <para>
- 情報カードを既存の昔ながらの認証アカウントに添付する例を、
- 以下に示します。
- </para>
- <programlisting language="php"><![CDATA[
- // ...
- public function submitinfocardAction()
- {
- if (!isset($_REQUEST['xmlToken'])) {
- throw new ZBlog_Exception('Expected an encrypted token ' .
- 'but was not provided');
- }
- $infoCard = new Zend_InfoCard();
- $infoCard->addCertificatePair(SSL_CERTIFICATE_PRIVATE,
- SSL_CERTIFICATE_PUB);
- try {
- $claims = $infoCard->process($request['xmlToken']);
- } catch(Zend_InfoCard_Exception $e) {
- // TODO Error processing your request
- throw $e;
- }
- if ($claims->isValid()) {
- $db = ZBlog_Data::getAdapter();
- $ppi = $db->quote($claims->getCardID());
- $fullname = $db->quote("{$claims->givenname} {$claims->surname}");
- $query = "UPDATE blogusers
- SET ppi = $ppi,
- real_name = $fullname
- WHERE username='administrator'";
- try {
- $db->query($query);
- } catch(Exception $e) {
- // TODO Failed to store in DB
- }
- $this->view->render();
- return;
- } else {
- throw new
- ZBlog_Exception("Infomation card failed security checks");
- }
- }
- ]]></programlisting>
- </sect2>
- <sect2 id="zend.infocard.basics.adapters">
- <title>Zend_InfoCard アダプタの作成</title>
- <para>
- <classname>Zend_InfoCard</classname> コンポーネントは、
- 情報カードの標準規格の変化に対応するために
- モジュラー構造を採用しています。
- 現時点では、拡張ポイントの多くは未使用ですので無視できますが、
- 情報カードの実装においてひとつだけ実装すべき点があります。
- それが <classname>Zend_InfoCard_Adapter</classname> です。
- </para>
- <para>
- <classname>Zend_InfoCard</classname> アダプタは、
- コンポーネント内でコールバックを使用してさまざまな処理を行います。
- たとえば、コンポーネントが情報カードを処理する際の
- Assertion ID の保存や取得などを行います。
- 受け取った情報カードの assertion ID の保存は必須ではありませんが、
- もしそれに失敗すると、リプレイ攻撃によって認証が信頼できないものになる可能性が発生します。
- </para>
- <para>
- これを避けるためには、
- <classname>Zend_InfoCard_Adapter_Interface</classname>
- を実装してそのインスタンスを設定してから
- <methodname>process()</methodname> メソッド (単体) あるいは <methodname>authenticate()</methodname>
- メソッド (<classname>Zend_Auth</classname> アダプタ) をコールしなければなりません。
- このインターフェイスを設定するためのメソッドが
- <methodname>setAdapter()</methodname> です。
- 以下の例では、<classname>Zend_InfoCard</classname>
- アダプタを設定してアプリケーション内で使用しています。
- </para>
- <programlisting language="php"><![CDATA[
- class myAdapter implements Zend_InfoCard_Adapter_Interface
- {
- public function storeAssertion($assertionURI,
- $assertionID,
- $conditions)
- {
- /* Store the assertion and its conditions by ID and URI */
- }
- public function retrieveAssertion($assertionURI, $assertionID)
- {
- /* Retrieve the assertion by URI and ID */
- }
- public function removeAssertion($assertionURI, $assertionID)
- {
- /* Delete a given assertion by URI/ID */
- }
- }
- $adapter = new myAdapter();
- $infoCard = new Zend_InfoCard();
- $infoCard->addCertificatePair(SSL_PRIVATE, SSL_PUB);
- $infoCard->setAdapter($adapter);
- $claims = $infoCard->process($_POST['xmlToken']);
- ]]></programlisting>
- </sect2>
- </sect1>
- <!--
- vim:se ts=4 sw=4 et:
- -->
|