論PHP介面版本控制(如何相容多端介面)

lijianlin0204發表於2020-09-01

在對接第三方介面的時候,總是會看到介面字尾會帶著v1,v2這樣的標識,我們知道這些都是介面版本的概念,那麼如果我方需要提供對外的介面,或者對接web端和APP端的時候,希望公用同一個介面,但是介面所渲染的資料表現形式不太一致,以及介面授權也不太一致的情況下,如何做到使用不同版本,且不同版本直接互不影響且同時共存呢?

首先筆者在考慮到介面設計時,有幾大模組:

  1. 控制器層(controller):筆者將其定義為入口層(相當於java的dao層)
  2. 服務層(services):邏輯服務層,控制器入口層通過版本號標識轉接到不同的服務層,具體的程式碼邏輯實現都在此處編寫
  3. 行為層(behavior):也可理解為事件層,服務中介軟體,行為鉤子都將在此處控制,入口許可權過濾校驗,對接第三方服務擴充套件通過行為鉤子抽出,不讓其加大服務層的程式碼臃腫
  4. 模型層(model):該層根據實際業務和開發習慣而定,可要可不要
  5. 校驗層(validate):筆者認為很有必要,所有獨立拉出一個目錄來做相關校驗,不管是獨立校驗,校驗引擎,還是框架自帶校驗都在該目錄定義,方便維護和擴充套件
  6. 公共層(common):系統公共程式碼,比如附件上傳,下載等
  7. 配置層(config):內部配置,根據需求自定義是否需要
  8. 語言包(lang):根據需求而定
  9. 複用層(tarits):根據實際需求而定
  10. 任務層(job):根據實際需求而定

目錄層級如圖所示:

那麼在入口層如何轉接到服務層呢?因為在這過程我們會將介面中的版本號轉接到不同的版本服務層。
首先在控制器入口層寫一個基類控制器,後續所有的控制器都將會繼承該類,在建構函式中調取行為類中的解析服務層程式碼,將服務層類初始化給基類變數!

    public $service = null;

    /**
     * 建構函式處理頭部請求
     *
     * @return void
     */
    public function __construct($type = 0, Request $request)
    {
        // 登入跳過
        if (!$type) {
            // 註冊行為監聽
            Hook::add('app_init', [
                // 校驗請求介面的身份(身份驗證)
                'app\\saas\\behavior\\AuthToken'
            ]);
            Hook::listen('app_init', []);
        }
        // 立即執行初始化控制器服務應用
        $this->service = Hook::exec('app\\saas\\behavior\\InitializtionService', ['tag' => $type, 'request' => $request]);
    }

服務InitializtionService解析路由,判斷,將服務層例項化

    public function run($params)
    {
        // 相容控制器分層,優化控制器目錄結構
        $controller = request()->controller();
        $controllerArray = explode('.', $controller);
        $controllerLength = count($controllerArray);
        $appendControllerName = '';
        if ($controllerLength == 1) {
            $appendControllerName = $controllerArray[0];
        } else {
            for ($i = 0; $i < $controllerLength - 1; $i++) {
                $appendControllerName .= strtolower($controllerArray[$i]) . '\\';
            }
            $appendControllerName .= ucfirst($controllerArray[($controllerLength - 1)]);
        }

        // $controller =  '\\app\\saas\\controller\\' . request()->controller();
        $controller =  '\\app\\saas\\controller\\' . $appendControllerName;
        $verion = request()->param('version');
        $init_service = function () use ($controller, $verion, $params) {
            // dump($controller);
            // $controller = '\app\saas\controller\test\Test';
            $reflection = new \ReflectionClass($controller);

            if (property_exists($controller, 'versions')
                && isset($reflection->getStaticProperties()['versions'][$verion])
            ) {
                // 預設規則返回,在前在後不允許返回其他資訊
                $service = $reflection->getStaticProperties()['versions'][$verion];
                // 判斷控制器服務檔案是否存在
                return class_exists($service) ? new $service($params['tag'], $params['request']) : Merror::getInstance()->jsonApi(40006);
            } else {
                Merror::getInstance()->jsonApi(40001);
            }
        };

        return is_null($verion) ? Merror::getInstance()->jsonApi(40002) : $init_service();
    }

這樣在控制器中檔案定義如下呼叫服務層邏輯程式碼,而不用關心是屬於哪個服務層類,服務層程式碼只和版本有關

 class Sysorder extends Saas
 {
    /**
     * 版本服務排程屬性--必須預設一個且是v1
     *
     * @var     array
     */
    protected static $versions = [
        'v1' => \app\saas\services\syscenter\Sysorder::class,
    ];

    /**
     * 獲取資訊集許可權目錄
     *
     * @method  POST|GET
     * @name    getSubMenuListCate
     */
    public function getSubMenuListCate()
    {
        return json($this->service->getSubMenuListCate());
    }

    /**
     * 獲取列表
     *
     * @method  POST|GET
     * @name    getSysOrderList
     */
    public function getSysOrderList()
    {
        return json($this->service->getSysOrderList());
    }

結語:此設計拋磚引玉,具體實現看各位phper大顯神通了!
api多版本介面設計模式,可以參考(基於ThinkPhp5.1實現,框架不同,設計理念一致):www.kancloud.cn/lijianlin/ethantp5...
最後推廣一下筆者自研的一套基於laravel設計的工作流流程引擎(https://learnku.com/laravel/t/48967),歡迎研究自研!

本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章