|
resource
リソース
Manual 導入BEARはリソース指向のフレームワークです。 モデルをオブジェクト設計されたものとして扱わず、リソースとして扱います。BEARのリソースもHTMLと同じようにROA(リソース指向設計)の次の4つの特徴を持ちます。
リソースアクセスの結果はリソースに関わらず同じRo型(リソースオブジェクト型)が返ってきます。リソース内部でエラーや例外が発生したりしてもそれをアクセスしたクライアントには同じRo型が返ってきます。 HTMLと同じような仕組みと考えていただけると分かりやすいと思います。HTMLにはURLがあり、操作は制限され(GET/POST)リソースリクエストに状態を持たず他のリソースにリンクがが用意されている場合があります。一方、BEARもリソースにURIがあり、操作は制限され(CRUDやlink)、リソースリクエストに状態を持たず他のリソースにリンクが用意されている場合があります。 詳細リソースが何処にどのようなフォーマットで保存してあるかに関わらず利用する側は共通の方法でアクセスを行います。 リソースアクセス結果もリソースオブジェクト(RO)として共通のフォーマットで受け取ります。 データがファイルにテキスト/CSVファイルでもあってもDBレコードでもあってもリモートサイトのXMLでもあっても、あるいはリソースの状態(中身)がPEARエラーや例外でさえ、受け取るのは同じリソースオブジェクト(Ro)です。 ROA4つの特徴を以下に順番の説明します。 アドレス可能性(URI)リソースはURIを1つだけ持ち、URIを特定するとリソースが特定されます。リソースはアプリケーションリソースとアプリケーション外部の外部リソースがあり、アプリケーションリソースはプロジェクトフォルダの<プロジェクト>/App/Ro/(リソースオブジェクトフォルダ)に設置します。 アプリケーションリソースアプリケーションリソースのうち、拡張子があるファイルをスタティックリソースとして扱います。csvファイルやtxtファイル、ymlファイルなどは内容が読み込まれパースされ配列としてクライアントに返されます。スタティクリソースは原則時間無制限でキャッシュされます。 BEARのURIは基本的にURLと同じフォーマットですが、アプリケーション内部がもつアプリケーションリソースはスキーマやサービス(ホスト名)を省略します。※scpコマンドでローカルファイルを指定するときに<localhost>は省略できるイメージです。 スタティックリソース例)サービス規約リソース
※read操作をすると単一の文字列がbodyに入ったリソースオブジェクトがクライアントに返ります。 例)サービスQ&Aリソース
※read操作をするとQ&Aが配列になったbodyのリソースオブジェクトがクライアントに返ります。※リソースアクセスのオプションでページグネーションが利用できます。 スタティックリソースや、検索性や更新性などを考慮してもスタティックファイルとして管理した方が向いている用途用のリソースです。 RoリソースBEARが標準的にあつかうアプリケーションダイナミックリソースです。引数やDBや他リソースなど外部要因に依存してレスポンスが変わります。内部でロジックを実装できるためです。BEAR_Roを継承した個別のリソースクラスを用意し、onRead()やonLink()などリソース操作に応じたメソッドを用意します。 例)ユーザーのプロフィールリソース
このRoリソースが一般的web dbアプリにおいて標準的なリソースです。サービスレイヤーとして機能します。内部でDBアクセス、他webサービスの呼び出しやドメインロジック実装を行います。 httpリソースhttpリソースはhttp://で始まるwebのURLを指定します。サイトがRSS(xml)のサイトなら配列データとして、プレーンHTMLなら文字列として取得されます。 オリジナルスキーマリソースレガシーなモデル群や、他のフレームワーク、ライブラリを利用したモデルやデータソース、他リモートAPIを利用するときにオリジナルのスキーマリソースを用意することができます。 例) 例えば古い社内システムのデータアクセスにAPIやクラスファイルが用意されているとしましょう。myofficeなどというスキーマを設計してURIによるリソースアクセスできるようにできます。 例)社員番号1600の社員プロフィールリソースのURI例 myoffice://self/staff/profile?id=1 ※selfはローカルホストの自分自身のサービスを表します URIとクエリーを受け取った”myoffice”スキーマ実行クラスが社内システムにアクセスしその値を返します。 ページから直接、外部オブジェクトやAPIを利用することに対して、オリジナルスキーマを用意してURIを持つリソースとして扱う事のメリットはなんでしょうか? - スタティックファイルを直接ページから扱わずスタティックリソースとして利用する場合と同じです。すなわち共通のキャッシュ機構、ページグネーション、ログ管理、レイジーセット、ビューからのPull取得など共通のリソースユーティリティが利用できます。またコンポーネント間の結合が疎になるので、メンテナンス性も向上するでしょう。「実装に依存」してたリソースクライアントが「リソースAPIへの依存」へと変わるからです。ログやデバッグが容易になります。 統一インターフェイス
リソースは単なるデータと違い、操作のインターフェイスが持てます。ですが自由にインターフェイスを持てるわけではなく、限られたインターフェイスしか持てません。この制限されたインターフェイスをRESTではユニファイドインターフェイスといいます。BEARのリソースは CRUD 、create(作成), read(読み込み), update(変更), delete(削除) の操作しかできません。※ もしブログの記事リソースに「検索」の機能を加えたいときでもBlog/Entryエントリーに"検索"という操作はできません(検索インターフェイスはもてません)かわりに(Blog/Entry/Search)というリソースを作成してreadという操作を行います。 インターフェイスは4つ全てを備える必要はありません。必要なインターフェイスだけを実装します。実装されていないインターフェイス(メソッド)でアクセスされると400(Bad Request)が返されます。HTTPでは405(Method not allowed)に相当しますが、BEARでは簡略化した3つのコードしかもたないために400(Bad Request)が結果コードとして返ります。 スキーマによっても利用できるインターフェイスは変わります。例えば現在httpリソースはread操作しか用意されていません。mもしread以外の操作を行うとコード400(Bad Reqeust)のリソースオブジェクトが返されます。 ステートレスオブジェクト設計されたモデルと違いリソースは通常状態を持ちません。リクエストの準備をすべて済ませてからリクエストは1度に行います。HTTPと同様です。単純化した例を下にコードで示します。 例)オブジェクトモデル $user = new User(); $user->id = 3 //id=3という状態 $user->read(); BEARリソースアクセス $params = array('uri' => 'User?id=3');
$resource->read($params)->getBody(); //リソース内部ではid=3の情報は保持していない
または
$params = array('uri' => 'User', 'values' => array('id' => 3);
$resource->read($params)->getBody();;ステートレスのリクエストは単純な分、失敗のリカバリーや記録、キャッシュ化が容易といった特徴があります。 リンクRoリソースはリンクによってリソースと他のリソースを結びつけることができます。例えばユーザーリソースはブログリンクよってブログリソースにリンクされます。ブログリソースは最新記事リンクや人気記事リンクで記事リソースにリンクされます。 リンクはリソース内部でカプセル化され、外部からはリソースからはリンクをたどるだけです。htmlのaタグのイメージです。hrefで指定されたリンク情報はクライアントは管理しません。利用するだけです。もしリンク先リンク方法が変わったとしても利用の仕方は変わりません。 roリソース以外のリソースにもリンクできます。例えばdbのコラムにhatena_idとあれば、それをつかってhatena web apiに直接リンクを張る事ができます。リソース間の結合をROAにしているからリンク記述も容易です。基本的にはリンク先がどんなデータソースでもURIを指定するだけです。この点がDBのリレーションやいわゆるアソシエーション機能とBEARのリンクと大きく違うところです。 リソースクライアントページクラスでは初期化ハンドラでリソースアクセスクラス(BEAR_Resource)をつかってリソース操作をし単数または複数のリソースをアクセスしてそのリソースをread、続けてページにsetし出力ハンドラでHTMLとして可視化します。 リソースはページ以外からも扱えます。CLIやソケット通信もクライアントになります。作成したリソースはページクラス以外からでも動作するので以下の用途にも使えます。
他言語やリモートサイトからはRESTやソケットを介してアクセスします。 リソースアクセスに必要なものBEARでのリソースアクセスは以下を指定して行います。
メソッドは4つしかありません。いわゆるCRUD(create/read/update/delete)インターフェイスでHTTPのPUT/GET/POST/DELETE、DBではCREATE/SELECT/UPDATE/DELETEに相当します。全てのリソースにこの4つのアクセスでしかリソースアクセスできず、RESTではこれをユニファイド(統一された)インターフェイスといいます。 リソースはURIと引数で特定されます。例えばid=5のユーザーはURIは'User'、引数はarray('id'=>5)です。これをクエリー形式でUser?id=1とURIだけで表現することもできます。 リソースをreadして得られる結果をRESTではリソース状態といいます。 オプションは、キャッシュのリソース状態の加工(ページグネーションやコールバック)やキャッシュ使用などの指定に利用します。ユーザー定義したオプションも利用できます。 何を(URI + 引数)どうやって(オプション)どうするのか(メソッド)と考えてみましょう。 リソースオブジェクトリソースアクセスの結果はリソースオブジェクト(Ro)として返ります。Roはミニwebのようなものです。4つのプロパティを持ちます。
RoコードコードはHTTPレスポンスコードに準じた3つのコードがリソースの状態を表しています。
参考: http://ja.wikipedia.org/wiki/HTTPステータスコード Roヘッダーリソース状態(ボディ)のメタ情報が入っています。例えば、ユーザーリソースの結果が配列で返されたとします。それが「何件あるのか」「いつ作成されたのか」といったリソース状態そのものを表しているのではないが、リソース状態に依存した属性情報を格納するのに用います。HTMLでいうと<header></header>の部分です。HTMLの<body>のメタの属性情報(ドキュメントタイトル、作成日、使用文字コード)に<header>が使われています。 Roボディリソース状態です。HTMLでいうと<body></body>の部分です。通常、ビューにアサインされる配列や文字列が入りますが、オブジェクトであってもかまわず型に制限はありません。 RoリンクHTTPと違うのはこの部分です。リンクが独立したプロパティになっています。 リソースは他のリソースにリンクを張れます。HTMLでいうとタグで指定された他ページへのリンクです。リンクがあるおかげで、関連リソースへのアクセス方法を知る事なしに(カプセル化して)リソースからリソースに情報をたどることができます。HTMLでも最も重要な機能です。 サンプルコードリソースの取得とページ化例をあげます。「明日の天気」はリソースです。リソース状態が「晴れ」です。 リソースはアドレス(URI)を持ちます。 リソースにアクセスするためにはリソースアクセスクラスである、BEAR_Resourceクラスを使います。bearコマンドで作成したApp_PageにはページプロパティとしてBEAR_Resourceオブジェクトが注入されプロパティオブジェクトとして利用可能になっています。 public function onInject()
{
$this->_resource = BEAR::dependency('BEAR_Resource');
}リソースを読むのにリソースの場所(URI)と、読み出しに必要な引数「地域は東京」など(values)が必要です。読み込みを実行した後にgetBody()でその結果の内容を取得します。 $params = array(
'uri'=>'Whether/Tommmorow',
'values'=>array('region' => 'tokyo')
);
$wheather = $this->_resource->read($params)->getBody();結果をビューにセットします。 $this->set('wheater', $wheater);以下のように「[http://capsctrl.que.jp/kdmsnr/wiki/bliki/?FluentInterface 流れるようなインターフェイス]」(fluent interface)でsetできます。※この場合のreadはonInit()終了時にlazy実行されます。 $this->_resource->read($params)->set('wheater');リソースの考え方はwebそのものです。例えばhttp://www.example.com/wheather /tommorow?location=tokyoというURIリクエストそのものがリソースで、表示されたページで得られる結果がリソース状態として考えると上記のリソースアクセスが理解しやすいでしょう。 ダイナミックリソースとスタティックリソースwebアプリは一般にデータの格納にDBを使います。DBやリモートサイトのXML/JSON、動的にリソース状態が変化するこれらのリソースをダイナミックリソースと呼びます。 対してスタティックなファイルをリソースファイルとして配置することもできます。これがスタティックリソースです。CSVファイルやYAML、iniファイルなどのスタティックなデータファイルを設置できます。 例えば郵便番号CSVファイルをそのままApp/Ro/以下のフォルダに配置すると、そのcsvファイルはスタティックリソースとしてそのまま使えます。クライアントから読み込むとPHPの連想配列になっています。 BEARではURIに拡張子がないものをダイナミックリソース、あるものをスタティックリソースとして扱います。ダイナミックリソースファイルは拡張子phpのphpファイルです。 スタティックリソースも、キャッシュやページングといった機能はダイナミックリソース同様に扱えます。CSVファイルのデータがDBと同じようにページングできます。 リソースファイルの種類ローカルリソースファイルはCRUD,それぞれのメソッドを記述するApp_Roクラスファイルか、インターフェイスを持たないファイル関数ファイル、あるいはスタティックファイルのいずれかのファイルで用意されます。 リソースオブジェクトクラス(BEAR_Ro, App_Ro)BEAR_RoまたはApp_Roクラスを継承して個別のリソースクラスを実装します。以下の用にインターフェイスを実装します。 //ユーザーリソース
class User extend App_Ro
{
public function onCreate(array $values){}
public function onRead(array $values){}
public function onUpdate(array $values){}
public function onDelete(array $values){}
public function onLink(array $links){}
}readしかできないリソースならonReadだけとリソースに応じて実装します。実装必須のメソッドは特にありません。 リソース関数ファイル新規作成には非推奨ですが、レガシー関数ファイルを使いたい場合に関数もリソースファイルとして使用できます。 関数のみのリソースファイルです。CRUDインターフェイスを持ちません。関数ファイルではリソースそのものに動作を持たすようなネーミングになります。addUserとかeditUser、もしくはuser?mode=add やuser?mode=editという風に引数に動作を表すようにします。 リソースオプションreadオプションリソースをread(取得)する時に、「どのように取得するか」「取得した後の後処理をどうするか」をリソースreadオプションとして指定できます。以下の例では以下のオプションを同時に設定しています。
というオプションを同時に指定しています。キャッシュオプションを指定すると、リソース取得(DBでのセレクト)だけでなく、オプション処理全てがキャッシュされます。 $options = array();
$options['cache']['life'] = 10; // キャッシュ時間
$options['cache']['key'] = $id // キャッシュキー
$options['pager'] = 5; // ページャー
$options['callbackr'] = array('App', 'htmlEscape'); //コールバック
$options['template'] = 'list/user'; // リソーステンプレート指定
$params = ('uri'=>$uri, 'values'=>$values, 'options'=>$options);
$resource->read(params)->set('entry', 'object');なお、readオプションで指定するページングやテンプレートアサインなどのポストプロセスは以下の順番で実行されます。
※callbackは指定したコールバック関数(メソッド)にcall_user_funcを使ってリソース結果配列が渡され、callbackrではarray_walk_recursiveで結果の各要素が渡されます。 POE(Post Once Exactly), CSRF(クロスサイトスクリプトフォージェリ対策)オプションRESTの用語で「サイドエフェクト」「べき等性」という用語があります。 CRUD の4つのインターフェイスのうち、readはどのようなreadであってもリソース状態(内容)は変更されません。対してupdate、deleteはリソースの内容に変更があります(サイドエフェクトがあるといいます) 一方、updateとdeleteは同じ操作を何回してもリソース状態の変更はありません。(「id=1の nameを"kuma"に」(updae) 「id=2のデータを消去」(delete)は同じ事を何回実行しても2回目からリソース状態に変化はありません。) createは違います。「name=kumaでユーザーリソースを作成」を3回実行すればリソースが3つ作成されてしまいます。「べき等性」がないといいます。フォームの多重送信の問題はこれです。また通信不良の多い携帯のwebでも問題になることがあります。 例)通信不良時のcreateの問題
クライアント: ユーザーID=kumaでユーザー登録サブミット→ サーバー:DBに登録し「登録しました」画面を表示。ところが通信不良でこのレスポンスが返らない。→ クライアント: おかしいなとユーザーID=kumaでもう一度ユーザー登録サブミット → サーバー:そのユーザーは登録済みです画面を表示 ※サーバーもクライアントも正しい動作をしても、ユーザー体験として成立しない。 リソースリクエストのpoeオプションでこの問題が解決できます。 $values = array('id'=>'bear');
$options = array('poe'=>true);
$params = ('uri'=>'user', 'values'=>$values, 'options'=>$options);
$this->_resource->create(params)->request();poeオプション指定時には全く同一のリソースcreateリクエストはキャッシュされた結果を返すのみで、onCreateメソッドやリソース関数は1度しか実行されません。これによりDBのinsertアクセスは一度だけ、結果は同じものが何度でも返る、という処理が実現でき上記の通信不良の問題も起りません。DBのinsertアクセスは一度だけですが、「登録できました」画面は何度リクエストしても表示されます。 csrfオプションも同様です。 $values = array('id'=>'bear');
$options = array('csrf'=>true);
$params = ('uri'=>'user', 'values'=>$values, 'options'=>$options);
$this->_resource->create(params)->request();csrfのチェックをパスしない場合は例外が投げられます。poeの場合は同じトークンでのリクエストがあればリソースリクエストを行いません。(エラーメッセージは例外などはありません。) リクエスト単位ではなく全体に適用するにはapp.ymlで指定します。 BEAR_Resource_Request: poe: false csrf: false どちらもトークン用のストレージに標準でセッションを使っています。オプションを利用していてセッションが切れてしまうと、そのサブミットはcsrfチェックに引っかかってしまいます。 トークン用のストレージをセッションを用いずアプリケーションが独自に用意することができます。`BEAR_Form_Token'のインジェクターonInjcet()を切り用意したサービスをインジェクトします。 リソースオブジェクト(BEAR_Ro)リソースファイルでリソース状態を返すには2つ方法があります。1つは得られたリソース結果をそのまま連想配列やスカラー値として返す方法です。DBのクエリー結果などの連想配列変数をそのまま返すことができます。もう1つの方法はリソースオブジェクト(BEAR_Ro)を利用する方法です。 例)サーバー変数をかえすServerリソースにアクセス bodyだけ(配列変数)を返す場合 /**
* @return array
*/
public function onRead($values)
{
$array = $_SERVER;
return $array;
}ヘッダーにcount情報を付加したRO(リソースオブジェクト)を返す場合 /**
* @return BEAR_Ro
*/
public function onRead($values)
{
$result = ...
$ro = $this->setHeader('count', count($result))->setBody($result);
return $ro;
}上記のいずれを返しても受け取る方はROで受け取ります。 $ro = $this->_resource->read($params)->getRo(); $body = $ro->getBody(); // bodyの取得 $headers = $ro->getHeaders(); // headerの取得 リソースオブジェクトはwebページに似ています。最も重要なのは<body></body>タグで囲まれたボディ部分ですが、そのbodyコンテンツのメタ情報が<header></header>でヘッダーとして表されます。ページには他のページへの「リンク」があります。 リソースオブジェクトのもボディ、ヘッダー、リンクのプロパティがあります。それに加えリソース状態を表すコードプロパティがあります。 例をあげます。「検索」リソースではDBページャーを用いて検索結果を返します。データ量は膨大なのでDBページャーを使い1ページ辺り10件の検索結果を「検索」リソース状態として返すことにしましょう。リソースオブジェクトではbodyプロパティに結果が入ります。しかしユーザーは「1ページの検索結果」以外のデータも必要でしょう。「検索結果の全体件数」「現在表示中のページ番号」などといったメタ情報です。これは「header」プロパティに格納されます。そして最後に、次ページや前ページといった現在のリソースから他のリソースへのリンクがあります。「link」プロパティが使用されます。 リソースアクセスサンプルコードこのようにBEAR_Roはリソースの状態(結果)を保持する機能と、リソースアクセスのメソッドを実装(Create, Read, Update, Delete)する機能と2つ大きな機能があります。 Ro(リソースオブジェクト)はHTTP通信の仕組みに似ています。ヘッダーとボディを持ち、POST/GET/PUT/DELETEに対応するインターフェイス(onCreate, onRead, onUpdate, onDelete)を持ちます。リクエストはステートレスで行われエラーが起こっても同じフォーマットでレスポンスが返ります。 クエリー形式の引数指定下記はどちらも同じreadです。 $this->_resource->read(array('uri'=>'wheather', 'values'=>array('area'=>'tokyo', 'time'=>'tomorrow'))->set('weather_tokyo');
$this->_resource->read(array('uri'=>'wheather?area=tokyo&time=tomorrow'))->set('weather_tommorow');CLIでのクエリーCLIでアクセスする場合には&をうまく認識してもらうためにシングルクオートを使います $ bear create 'User?name=kuma&age=10' RESTとBEARREST自身は、多くの異なる技術を利用して実装されうる高水準なスタイルなのです。RESTは、リソースや統一インターフェースの考えを持っています。つまり、全てのリソースが同じメソッドによって応答するという考えなのです。しかし、RESTはどのメソッドでなければならないのかや、メソッドの数がどれくらい必要であるのかといったことには言及していません。REST スタイルを「具現化」したものの1つが、HTTP(と、URIなどの関連する標準セット)です。もしくは、少し抽象的になりますが、Webのアーキテクチャそれ自身であるとも言えます。HTTPは、RESTの統一インターフェースをHTTPの動詞(操作)からなる特別な形で「具体化」したものと言えます。 作者 Stefan Tilkov, 翻訳者 松本 清一 http://www.infoq.com/jp/articles/rest-introduction BEARの最もコアな部分はこのRESTに関連する部分です。HTTPやwebのアーキテクチャのように振る舞うRESTです。 WebブラウザがPOST/GETでWebアプリにアクセスし、webアプリ内部ではページがリソースにCRUD操作します。リソース内部ではDBにINSERT/SELECT/UPDATE/DELETE操作をしています。 ユーザーの入力からDB操作までのデータフローが比較的フラットです。途中のマッピングがなくコンポーネント間でのデータがリニアに受け渡しされやすくなっています。 参考リンク注)このマニュアルではリソース、リソース状態、などのRESTの用語の言葉の厳密性はあえて問わずルーズにしています。ご了承ください。 | ||||||||||||||||||||||||||||||