CommonGateway 介紹
CommonGateway (index.php) 是一個簡化的 MVC 容器。按照 MVC 的設計模式,它將一個 Web 應用服務分成三個部份。 資料模型(Model)、流程控制項(Controller)與視圖(View)。 Model 主要就是透過 schema-database 或其他資料庫函數完成。 因此程序人員需要實作的內容就是控制項與視圖。我實作 CommonGateway 便是要簡化一部份工作。
CommonGateway 主要負責為程序人員完成下列工作:
- CommonGateway 會根據 URL 路徑(PATH_INFO) 選擇 Web 應用服務的控制項。 正是這「依路徑選擇目標」的行為,而它又是 Common Gateway Interface (CGI) 的實作項目,故我將之命名為 CommonGateway 。
- 它會將客戶端送出的文件資料,預先處理成關聯式陣列結構。 除了傳統的 Query string 與 Form data ,它也能處理 HTTP PUT Method 會送出的資料。它也支援 JSON 型態的資料。
- 它會根據 RESTful 的原則,調用對應的控制項方法。
- 它會根據控制項方法的回傳結果與客戶端期望的文件回應型態,處理對應的視圖樣板。
- 它會儘量透過外部注入的方式,將其他資源放入控制項 (即 IoC 模式),減少對原有程式碼的侵入性。
CommonGateway 純粹是一個 MVC 容器,不是框架也不是函數庫。 它的定位更像是一個前台載入器 (front loader),不以整合性為目標,不會整合 Log, Config, Database 等等的功能。 它是面向 PHP 熟練者的工具。 如果你原本就有偏好的工具庫,你可以繼續使用,不必為了使用 CommonGateway 而另外學習一套方法。
- CommonGateway 的源碼託管於 github: common-gateway 。
- CommonGateway 的源碼只有一個 index.php 。 查看 CommonGateway 源碼: 「index.php」。
- 在源碼專案網站上,我也維護了一份 CommonGateway 的 wiki 頁面, 請參考「CommonGateway wiki」。
- CommonGateway的實作概念可以參考「RESTful 介面實作示範」。
程式文件目錄結構
模仿 Ruby on Rails 規劃文件的目錄結構。
- 控制器: controllers. 每一個控制器應設計為一個類別。類別名稱即控制器名稱。
- 視圖: views. 每一個服務分配一個同名的目錄,再依動作名稱規劃視圖樣板的文件名稱。
- 視圖輔助器: helps. 每一個服務對應一個視圖輔助器,其中定義的函數或變數,都將成為視圖內的區域性資源。
樹狀結構圖:
- index.php
- controllers/
- {app_name}.php
- views/
- {app_name}/{action}.p{view_format}
- helpers/
- _global.php
- {app_name}.php
- controllers/
例如:
- index.php - controllers/ - book.php - views/ - book/index.phtml - book/index.pxml - book/get.phtml - book/get.pxml - helpers/ - _global.php - book.php
路徑選擇策略
index.php 就是 CommonGateway 本身。 它的應用服務定址方式採用 RESTful 模式,即以 URL 路徑表示服務內容。 例如有一個叫 book 的書籍服務,它的 URL 應為 http://your.host/index.php/book 。它的路徑規則是:
index.php/{app_name}/{id or action_name}/{option_segments/...}
index.php 之後的路徑內容,會被拆成小節(segments)。 第一小節的 app_name 視為控制項名稱。 第二小節則為資源識別字或是動作識別字。 接下來的其他小節都視為要傳給動作的參數清單。
至於控制項的動作(action),則是依循下列兩個原則選用方法:
原則1: 根據 REQUEST_METHOD
首先根據 RESTful 模式,以 $_SERVER['REQUEST_METHOD'] 欄位為選用方法的目標,找尋控制項內是否有實作同名的方法。 例如 $_SERVER['REQUEST_METHOD'] 之值為 GET 時,CommonGateway 就會嘗試調用控制項的 get() - 方法名稱為小寫。 $_SERVER['REQUEST_METHOD'] 之值為 DELETE 時,CommonGateway 就會嘗試調用控制項的 delete()。
注意,當 $_SERVER['REQUEST_METHOD'] 為 GET ,但路徑沒有第二小節時,它會調用控制項的 index() 。
原則2: 根據 PATH
如果第一個原則找不到相符的控制項方法,而且路徑還有第二段名稱時, CommonGateway 會將第二段名稱視為動作識別字,嘗試調用控制項中同名的方法。 例如 URL 為 index.php/book/info ,則會調用 book 的 info()。
名稱慣例
CommonGateway 首先根據 URL 的服務名稱搜尋控制器的程式碼文件名稱。 再根據找到的程式碼文件名稱決定控制器的類別名稱以及服務的應用名稱(app_name)。 基本上,URL 的服務名稱全為小寫,若有多字組成時,以底線 _ 分隔。控制器的類別名稱,首字母大寫; 程式碼文件名稱應同 URL 的服務名稱。
CommonGateway 結合了一些常見的名稱慣例,它的名稱可能組合與優先順序如下:
第一例: URL 的服務名稱為 book 。控制器類別名稱為 Book,各文件名稱組合的採用順序如下:
controllers/book.php, views/book/, helpers/book.php. controllers/Book.php, views/Book/, helpers/Book.php.
第二例: URL 的服務名稱為 Book 。控制器類別名稱為 Book,各文件名稱組合的採用順序如下:
controllers/Book.php, views/Book/, helpers/Book.php. controllers/book.php, views/book/, helpers/book.php.
第三例: URL 的服務名稱為 book_profile 。控制器類別名稱為 Book_Profile,各文件名稱組合的採用順序如下:
controllers/book_profile.php, views/book_profile/, helpers/book_profile.php. controllers/Book_Profile.php, views/Book_Profile/, helpers/Book_Profile.php. controllers/BookProfile.php, views/BookProfile/, helpers/BookProfile.php.
第四例: URL 的服務名稱為 Book_Profile 。控制器類別名稱為 Book_Profile,各文件名稱組合的採用順序如下:
controllers/Book_Profile.php, views/Book_Profile/, helpers/Book_Profile.php. controllers/BookProfile.php, views/BookProfile/, helpers/BookProfile.php.
第五例: URL 的服務名稱為 bookProfile 。控制器類別名稱為 BookProfile,各文件名稱組合的採用順序如下:
controllers/bookProfile.php, views/bookProfile/, helpers/bookProfile.php. controllers/BookProfile.php, views/BookProfile/, helpers/BookProfile.php.
視圖調用策略
控制項動作的回傳值,決定視圖的處理工作。
CommonGateway 會自動根據服務名稱與 $_SERVER['HTTP_ACCEPT'] 內容,載入對應的視圖。
視圖的副檔名按 Ruby on Rails 型式,開頭為 p ,後接文件型態名稱。
例如 HTML 文件的視圖副檔名為 phtml (這正是 PHP 最早期使用的副檔名); XML 文件的視圖副檔名為 pxml。
比較特別的是 JSON 文件的視圖,其副檔名為 pjs ,不是 pjson 。
或者可以省略視圖,此時將自動調用 json_encode($model)
。
CommonGateway 會根據控制項回傳資料的型態,決定傳給視圖的資料來源(model)內容為何。 回傳值處理規則如下列。
- 介於100 ~ 599間的整數,視為控制項直接回傳 HTTP 狀態碼。 CommonGateway 會將該狀態碼回傳給瀏覽器,而不載入任何視圖。
- null (或無回傳值) : 大部份控制項的動作沒有函數回傳值,故這是預設行為。 此時會將控制項的公開屬性當作資料來源(model),將控制項的公開屬性內容展開成視圖活動範圍內的區域變數。 例如控制項有公開屬性 title ,CommonGateway 會將此屬性指派為視圖的區域變數 $title 。
- true : 同回傳 null 的情形。
- false : 視同控制項自行處理回應工作, CommonGateway 不會繼續載入視圖。
- array : CommonGateway 會將回傳的陣列視為資料來源, 指派為視圖內的區域變數 $model,並將陣列內容展開成為視圖內的區域變數。 注意,若陣列為數字索引陣列,則展開後的區域變數名稱之字首為 data_ 。 例如 $model = array('a', 'b') ,則視圖內展開的區域變數內容將是 $data_0 == 'a', $data_1 == 'b' ,餘類推。
- object : CommonGateway 會將回傳的個體視為資料來源, 指派為視圖內的區域變數 $model。 此時在視圖內將可以調用該個體的方法。這可以取代視圖輔助器(helper)。
- 若 $model 型態為 array 或 object ,則視圖內將同時分配一個和控制項名稱相同的別名。 例如控制項 MyBook 回傳的資料為 object ,則視圖內的 $model 等於 $MyBook 。
其實 CommonGateway 的源碼和 wiki 頁面,我大概一年前就已經放入源碼庫了。 但後來事情一忙,就忘了發佈在 blog 上。我還一直以為我已經發佈到 blog 了。 最近打算在動手修改一些功能,先把這個介紹內容發佈出來,日後再更新修改內容。
CommonGateway 除了引入 RoR 的設計概念外,實作時的 PHP 主要參考對象是 CodeIgniter 。 只是 CommonGateway 只著重在「web service」的設計工作,而不是網站設計。 如果你需要一個簡潔有效網站設計框架,我會建議你使用 CodeIgniter 。
樂多舊回應