yii2 restful api 風(fēng)格搭建(一)

最近在研究如何利用 yii2 搭建 restful api,將 yii2 restful api / yii2 rest api 搭建心得寫下咽斧,歡迎一起討論
使用yii2.0.13 advanced 版阅束,將 frontend 整個作為 api 接口項目呼胚,除了接口的路由規(guī)則可以認證通過外,其他的路由規(guī)則都返回請求錯誤的格式

1息裸、數(shù)據(jù)庫結(jié)構(gòu)
CREATE TABLE `goods` (
  `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
  `name` varchar(255) NOT NULL,
  `price` int(11) unsigned NOT NULL,
  `status` tinyint(1) unsigned NOT NULL,
  `create_time` int(11) unsigned NOT NULL,
  `modify_time` int(11) unsigned NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
2蝇更、使用 gii 創(chuàng)建 goods model
3、創(chuàng)建 api modules

在 frontend 下新建文件夾 modules
使用 gii 在 modules 文件中創(chuàng)建 v1 module(防止以后接口更新替換時向前兼容,接口增加 v1 v2 等版本控制)

4呼盆、將 v1 moduel 寫到配置文件中
'modules' => [
    'v1' => [
        'class' => 'frontend\modules\v1\Module',
    ],
],
5年扩、修改 frontend 下 main.php 中,user 的配置(以下說的配置文件访圃,都是 frontend 下 main.php)
'user' => [
    'identityClass'     => 'frontend\models\User',
    'enableAutoLogin'   => false,
    'enableSession'     => false,
    'loginUrl'          => null,
],
6厨幻、新建 frontend\models\User,繼承 common\models\User
<?php

namespace frontend\models;

class User extends \common\models\User
{

}
7腿时、啟用并修改配置文件中的 urlManager

調(diào)試 urlManager 的時候要小心况脆,他會將生成好的路由寫入緩存(默認是文件緩存),有些更改可能不會立馬生效

'urlManager' => [
    //用于表明 urlManager 是否啟用 URL 美化功能
    //默認不啟用批糟。但實際使用中格了,特別是產(chǎn)品環(huán)境,一般都會啟用
    'enablePrettyUrl'       => true,
    //是否啟用嚴格解析徽鼎,如啟用嚴格解析盛末,要求當前請求應(yīng)至少匹配1個路由規(guī)則弹惦,否則認為是無效路由。
    //這個選項僅在 enablePrettyUrl 啟用后才有效悄但。
    //如果開啟棠隐,表示只有配置在 rules 里的規(guī)則才有效
    //由于項目會將一些 url 進行優(yōu)化,所以這里需要設(shè)置為 true
    'enableStrictParsing'   => true,
    //指定是否在URL在保留入口腳本 index.php
    'showScriptName'        => false,
    'rules' => [
        //當然算墨,如果自帶的路由無法滿足需求宵荒,可以自己增加規(guī)則
        'GET <module:(v)\d+>/<controller:\w+>/search' => '<module>/<controller>/search',
        [
            'class'         => 'yii\rest\UrlRule',
            'controller'    => ['v1/goods'],
            // 由于 resetful 風(fēng)格規(guī)定 URL 保持格式一致并且始終使用復(fù)數(shù)形式
            // 所以如果你的 controller 是單數(shù)的名稱比如 UserController
            // 設(shè)置 pluralize 為 true (默認為 true)的話,url 地址必須是 users 才可訪問
            // 如果 pluralize 設(shè)置為 false, url 地址必須是 user 也可訪問
            // 如果你的 controller 本身是復(fù)數(shù)名稱 UsersController 净嘀,此參數(shù)沒用报咳,url 地址必須是 users
            'pluralize'     => false,
        ],
    ],
],
8、去掉配置文件中的 errorHandler 配置(整個 frontend 都是接口挖藏,不需要 html 的響應(yīng)格式)
9暑刃、將內(nèi)容協(xié)商配置到引導(dǎo)文件中(因為整個 frontend 都需要)
'bootstrap' => [
    'log',
    //全局內(nèi)容協(xié)商
    [
        //ContentNegotiator 類可以分析request的header然后指派所需的響應(yīng)格式給客戶端,不需要我們?nèi)斯ぶ付?        'class'     => 'yii\filters\ContentNegotiator',
        'formats' => [
            'application/json' => yii\web\Response::FORMAT_JSON,
            'application/xml' => yii\web\Response::FORMAT_XML,
            //api 端目前只需要json 和 xml
            //還可以增加 yii\web\Response 類內(nèi)置的響應(yīng)格式膜眠,或者自己增加響應(yīng)格式
        ],
    ]
],
10岩臣、配置文件中,components 配置 response宵膨,返回格式
'response' => [
    'class'     => 'yii\web\Response',
    //設(shè)置 api 返回格式,錯誤碼不在 header 里實現(xiàn)架谎,而是放到 body里
    'as resBeforeSend' => [
        'class'         => 'frontend\extensions\ResBeforeSendBehavior',
        'defaultCode'   => 500,
        'defaultMsg'    => 'error',
    ],
    //ps:components 中綁定事件,可以用兩種方法
    //'on eventName' => $eventHandler,
    //'as behaviorName' => $behaviorConfig,
    //參考 http://www.yiiframework.com/doc-2.0/guide-concept-configurations.html#configuration-format
],
11辟躏、編寫 frontend\extensions\ResBeforeSendBehavior 代碼
<?php

namespace frontend\extensions;

use Yii;
use yii\web\Response;
use yii\base\Behavior;

class ResBeforeSendBehavior extends Behavior{

    public $defaultCode = 500;

    public $defaultMsg = 'error';

    // 重載events() 使得在事件觸發(fā)時谷扣,調(diào)用行為中的一些方法
    public function events() {
        // 在 EVENT_BEFORE_SEND 事件觸發(fā)時,調(diào)用成員函數(shù) beforeSend
        return [
            Response::EVENT_BEFORE_SEND => 'beforeSend',
        ];
    }

    // 注意 beforeSend 是行為的成員函數(shù)捎琐,而不是綁定的類的成員函數(shù)会涎。
    // 還要注意,這個函數(shù)的簽名瑞凑,要滿足事件 handler 的要求末秃。
    public function beforeSend($event)
    {
        try {
            $response = $event->sender;
            if($response->data === null){
                $response->data = [
                    'code'  => $this->defaultCode,
                    'msg'   => $this->defaultMsg,
                ];
            } elseif(!$response->isSuccessful) {
                $exception = Yii::$app->getErrorHandler()->exception;
                if(is_object($exception) && !$exception instanceof yii\web\HttpException){
                    throw $exception;
                } else {
                    $rData = $response->data;
                    $response->data = [
                        'code'  => empty($rData['status']) ? $this->defaultCode : $rData['status'],
                        'msg'   => empty($rData['message']) ? $this->defaultMsg : $rData['message'],
                    ];
                }
            } else {
                /**
                 * $response->isSuccessful 表示是否會拋出異常
                 * 值為 true, 代表返回數(shù)據(jù)正常,沒有拋出異常
                 */
                $rData = $response->data;
                $response->data = [
                    'code' => isset($rData['error_code']) ? $rData['error_code'] : 0,
                    'msg' => isset($rData['res_msg']) ? $rData['res_msg'] : $rData,
                ];
                $response->statusCode = 200;
            }
        } catch (\Exception $e) {
            $response->data = [
                'code'  => $this->defaultCode,
                'msg'   => $this->defaultMsg,
            ];
        }
        return true;
    }
}
12籽御、創(chuàng)建 GoodsController.php
<?php

namespace frontend\modules\v1\controllers;  
  
use yii\rest\ActiveController;
  
class GoodsController extends ActiveController 
{  
    public $modelClass = 'common\models\Goods';

    public function actionSearch(){
        return [
            'error_code'    => 20,
            'res_msg'       => 'ok',
        ];
    }
}
13练慕、應(yīng)用入口同級增加.htaccess文件,隱藏index.php篱蝇,以apache為例
RewriteEngine On
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule .* index.php
14贺待、測試
命令:
curl -s -H Accept:application/xml http://local.rest.com/v1/goods/1
返回:
<?xml version="1.0" encoding="UTF-8"?>
<response>
  <code>0</code>
  <msg>
    <id>1</id>
    <name>測試商品1</name>
    <price>600</price>
    <status>1</status>
    <create_time>1520490595</create_time>
    <modify_time>1520490595</modify_time>
  </msg>
</response>
命令:
curl -s -H Accept:application/json http://local.rest.com/v1/goods/1
返回:
{
  "code":0,
  "msg":{
    "id":"1",
    "name":"測試商品1",
    "price":"600",
    "status":1,
    "create_time":"1520490595",
    "modify_time":"1520490595"
  }
}
命令:
curl -s -H Accept:application/json http://local.rest.com/v1/goods11
返回:
{"code":404,"msg":"Page not found."}
命令:
curl -s -H Accept:application/json http://local.rest.com/v1/goods/search
返回:
{"code":20,"msg":"ok"}
15、參考

http://www.yiichina.com/doc/guide/2.0/rest-quick-start
http://www.yiichina.com/doc/guide/2.0/runtime-handling-errors
http://www.yiiframework.com/doc-2.0/guide-concept-configurations.html#configuration-format
http://www.yii-china.com/post/detail/536.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末零截,一起剝皮案震驚了整個濱河市麸塞,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌涧衙,老刑警劉巖哪工,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件奥此,死亡現(xiàn)場離奇詭異,居然都是意外死亡雁比,警方通過查閱死者的電腦和手機稚虎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來偎捎,“玉大人蠢终,你說我怎么就攤上這事≤钏” “怎么了寻拂?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長丈牢。 經(jīng)常有香客問我祭钉,道長,這世上最難降的妖魔是什么己沛? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任慌核,我火速辦了婚禮,結(jié)果婚禮上申尼,老公的妹妹穿的比我還像新娘垮卓。我一直安慰自己,他們只是感情好师幕,可當我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布扒接。 她就那樣靜靜地躺著,像睡著了一般们衙。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上碱呼,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天蒙挑,我揣著相機與錄音,去河邊找鬼愚臀。 笑死忆蚀,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的姑裂。 我是一名探鬼主播馋袜,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼舶斧!你這毒婦竟也來了欣鳖?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤茴厉,失蹤者是張志新(化名)和其女友劉穎泽台,沒想到半個月后什荣,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡怀酷,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年稻爬,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蜕依。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡桅锄,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出样眠,到底是詐尸還是另有隱情友瘤,我是刑警寧澤,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布吹缔,位于F島的核電站商佑,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏厢塘。R本人自食惡果不足惜茶没,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望晚碾。 院中可真熱鬧抓半,春花似錦、人聲如沸格嘁。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽糕簿。三九已至探入,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間懂诗,已是汗流浹背蜂嗽。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留殃恒,地道東北人植旧。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓,卻偏偏與公主長得像离唐,于是被迫代替她去往敵國和親病附。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,979評論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理亥鬓,服務(wù)發(fā)現(xiàn)完沪,斷路器,智...
    卡卡羅2017閱讀 134,659評論 18 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,163評論 25 707
  • 標簽(空格分隔): Yii2 1 什么是Restful Restful 是一種軟件架構(gòu)風(fēng)格贮竟,提供了一組設(shè)計原則和約...
    ahcj_11閱讀 3,943評論 0 1
  • 這段時間的娛樂版塊真的是熱鬧非凡丽焊,可卻負能量十足较剃。 剛剛王寶強與馬蓉離婚事件的熱度降下來,隨即張紀中和樊馨蔓兩口子...
    靜默心谷閱讀 428評論 3 2
  • 上次說到談到一個18萬的大客戶啊送,但是技術(shù)不夠、設(shè)備不夠欣孤,拍不過來忍痛讓給了別人馋没。 做無人機這個項目也是很坎坷,先是...
    后宮貓三千閱讀 510評論 1 7