Notadd 底層實現了 passport 機制,有統一的受權管理,主要支持兩種方式進行 API 受權,一個是 client,領一個是 passport,這個在其餘文檔中有作詳細的說明。php
這裏主要說的是,如何基於 Notadd 進行 API 接口的開發。web
熟悉 Laravel 的同窗都應該知道,Laravel 遵循這樣的業務邏輯實現:json
路由(route) -> 控制器(controller) -> 業務邏輯(model) -> 數據輸出(view)
而 Notadd 的 API 業務邏輯實現一樣遵循相似的流程:api
路由(route) -> 控制器(controller) -> API 處理器(handler) -> 模型(model) -> 數據輸出(json)
其中,主要的差別在於,API 處理器提供了對數據輸出格式的輸出,返回的數據格式統一爲:跨域
[ 'code' => 200, // API 接口返回的狀態碼,默認爲 200 'data' => [], // API 接口返回的數據,主要爲數組形式 'message' => 'success!', // API 接口返回的提示信息,能夠包含錯誤信息或成功信息 ]
Notadd 在實現 API 受權的時候,使用的是有 路由中間件(middleware) 的方式來實現的。數組
具體實現方式,是在路由的中間配置參數中添加 auth:api 。dom
例如,在實現 api/setting/all 和 api/setting/set 兩個 API 的時候,添加 auth:api 的中間件,代碼參考以下:post
$this->router->group(['middleware' => ['auth:api', 'web'], 'prefix' => 'api/setting'], function () { $this->router->post('all', 'Notadd\Foundation\Setting\Controllers\SettingController@all'); $this->router->post('set', 'Notadd\Foundation\Setting\Controllers\SettingController@set'); });
Notadd 針對須要跨域的 API 還提供了 cross 的路由中間件,以實現 API 跨域的功能。this
例如,爲前兩個 API 提供跨域的功能實現,代碼參考以下:spa
$this->router->group(['middleware' => ['auth:api', 'cross', 'web'], 'prefix' => 'api/setting'], function () { $this->router->post('all', 'Notadd\Foundation\Setting\Controllers\SettingController@all'); $this->router->post('set', 'Notadd\Foundation\Setting\Controllers\SettingController@set'); });
因爲有了獨立的 API處理器 ,控制器層能夠製做簡單處理,僅需向控制器注入 handler,並由 handler 提供的輔助方法返回 API 數據給前臺,便可。
例如,在前面路由調用的 SettingController 中,僅須要注入 AllHandler ,使用方法 toResponse 和 generateHttpResponse 來返回結果給前臺,代碼參考以下:
<?php /** * This file is part of Notadd. * * @author TwilRoad <269044570@qq.com> * @copyright (c) 2016, iBenchu.org * @datetime 2016-11-08 17:01 */ namespace Notadd\Foundation\Setting\Controllers; use Notadd\Foundation\Routing\Abstracts\Controller; use Notadd\Foundation\Setting\Contracts\SettingsRepository; use Notadd\Foundation\Setting\Handlers\AllHandler; use Notadd\Foundation\Setting\Handlers\SetHandler; /** * Class SettingController. */ class SettingController extends Controller { /** * @var \Notadd\Foundation\Setting\Contracts\SettingsRepository */ protected $settings; /** * SettingController constructor. * * @param \Notadd\Foundation\Setting\Contracts\SettingsRepository $settings * * @throws \Illuminate\Contracts\Container\BindingResolutionException */ public function __construct(SettingsRepository $settings) { parent::__construct(); $this->settings = $settings; } /** * All handler. * * @param \Notadd\Foundation\Setting\Handlers\AllHandler $handler * * @return \Notadd\Foundation\Passport\Responses\ApiResponse * @throws \Exception */ public function all(AllHandler $handler) { return $handler->toResponse()->generateHttpResponse(); } /** * Set handler. * * @param \Notadd\Foundation\Setting\Handlers\SetHandler $handler * * @return \Notadd\Foundation\Passport\Responses\ApiResponse * @throws \Exception */ public function set(SetHandler $handler) { return $handler->toResponse()->generateHttpResponse(); } }
在 API Handler中提供了模型的操做接口。
在 Notadd 中,提供了兩類 API Handler,一類是 DataHandler,另外一類是 SetHandler,顧名思義,DataHandler 僅提供數據返回接口,而 SetHandler 不只提供數據返回接口,還提供其餘操做處理的接口。
具體差別體如今,DataHandler 在返回數據接口時僅調用方法 data,而 SetHandler 在調用 data 方法前還有調用 execute 方法。
例如,在前面的 SettingController 中使用的 AllHandler 爲 DataHandler 類 Handler,提供返回全部 配置項 的 API 功能,SetHandler 爲 SetHandler 類 Handler,提供 修改配置項 並返回全部 配置項 的 API 功能。
AllHandler 的代碼以下:
<?php /** * This file is part of Notadd. * * @author TwilRoad <269044570@qq.com> * @copyright (c) 2016, iBenchu.org * @datetime 2016-11-23 14:44 */ namespace Notadd\Foundation\Setting\Handlers; use Illuminate\Container\Container; use Notadd\Foundation\Passport\Abstracts\DataHandler; use Notadd\Foundation\Setting\Contracts\SettingsRepository; /** * Class AllHandler. */ class AllHandler extends DataHandler { /** * @var \Notadd\Foundation\Setting\Contracts\SettingsRepository */ protected $settings; /** * AllHandler constructor. * * @param \Illuminate\Container\Container $container * @param \Notadd\Foundation\Setting\Contracts\SettingsRepository $settings */ public function __construct( Container $container, SettingsRepository $settings ) { parent::__construct($container); $this->settings = $settings; } /** * Http code. * * @return int */ public function code() // 定義 API 操做結果的狀態碼 { return 200; } /** * Data for handler. * * @return array */ public function data() // 定義 API 返回的數據 { return $this->settings->all()->toArray(); } /** * Errors for handler. * * @return array */ public function errors() // 定義 API 操做失敗時返回的信息 { return [ '獲取全局設置失敗!', ]; } /** * Messages for handler. * * @return array */ public function messages() // 定義 API 操做成功時返回的信息 { return [ '獲取全局設置成功!', ]; } }
SetHandler 的代碼以下:
<?php /** * This file is part of Notadd. * * @author TwilRoad <269044570@qq.com> * @copyright (c) 2016, iBenchu.org * @datetime 2016-11-23 15:09 */ namespace Notadd\Foundation\Setting\Handlers; use Illuminate\Container\Container; use Notadd\Foundation\Passport\Abstracts\SetHandler as AbstractSetHandler; use Notadd\Foundation\Setting\Contracts\SettingsRepository; /** * Class SetHandler. */ class SetHandler extends AbstractSetHandler { /** * @var \Notadd\Foundation\Setting\Contracts\SettingsRepository */ protected $settings; /** * SetHandler constructor. * * @param \Illuminate\Container\Container $container * @param \Notadd\Foundation\Setting\Contracts\SettingsRepository $settings */ public function __construct( Container $container, SettingsRepository $settings ) { parent::__construct($container); $this->settings = $settings; } /** * Data for handler. * * @return array */ public function data() // 定義 API 返回的數據 { return $this->settings->all()->toArray(); } /** * Errors for handler. * * @return array */ public function errors() // 定義 API 操做失敗時返回的信息 { return [ '修改設置失敗!', ]; } /** * Execute Handler. * * @return bool */ public function execute() // 定義 API 執行的修改操做 { $this->settings->set('site.enabled', $this->request->input('enabled')); $this->settings->set('site.name', $this->request->input('name')); $this->settings->set('site.domain', $this->request->input('domain')); $this->settings->set('site.beian', $this->request->input('beian')); $this->settings->set('site.company', $this->request->input('company')); $this->settings->set('site.copyright', $this->request->input('copyright')); $this->settings->set('site.statistics', $this->request->input('statistics')); return true; } /** * Messages for handler. * * @return array */ public function messages() // 定義 API 操做成功時返回的信息 { return [ '修改設置成功!', ]; } }
API 結果的數據輸出,已經在 控制器(controller) 中作了處理。
至此,一個完整的 API 開發完成。