代码改变世界

l5如何通过路由走api版本回退查找设置

2015-04-10 09:59  轩脉刃  阅读(1476)  评论(0编辑  收藏  举报

l5如何通过路由走api版本回退查找设置

具体需求

当前遇到的问题是使用laravel写接口,但是接口是有版本号的,我们把版本号放在url中,比如:

http://yejianfeng.com/api/user/info/?uid=1
http://yejianfeng.com/api1.1/user/info/?uid=1
http://yejianfeng.com/api1.2/user/info/?uid=1

但是实际上api1.1的user/info和api的user/info的action是一样的,但是api1.2的user/info是不一样的

本来路由应该这么写:

<?php
Route::group(array('prefix' => 'api'), function() {
    Route::get('/user/info', ['uses' => 'UserController@userinfo']);
});


Route::group(array('prefix' => 'api1.1'), function() {
    Route::get('/user/info', ['uses' => 'UserController@userinfo']);
});


Route::group(array('prefix' => 'api1.2'), function() {
    Route::get('/user/info', ['uses' => 'UserController@userinfo1_2']);
});

这个感觉还是丑了点,我不希望路由会这么复杂,我希望的是进行版本衰退寻找,api1.1中的user/info那个不需要写,它能自动去寻找api1.1中有没有这个路由,没有的话,去寻找比它版本低的路由。

解决方法

这里当然要使用到middleware,希望路由是:

<?php
Route::group(array('prefix' => 'api'), function() {
    Route::get('/user/info', ['uses' => 'UserController@userinfo']);
});


Route::group(array('prefix' => 'api1.1', 'middleware' => 'downgrade'), function() {
});


Route::group(array('prefix' => 'api1.2', 'middleware' => 'downgrade'), function() {
    Route::get('/user/info', ['uses' => 'UserController@userinfo1_2']);
});

但是非常可惜,这样写的话

http://yejianfeng.com/api1.1/user/info/?uid=1

是进不了middleware的。

我们需要的是有个“匹配所有”的路由能将路由定位定到prefix 1.1的这个里面

所以改成这样:

<?php
Route::group(array('prefix' => 'api'), function() {
    Route::get('/user/info', ['uses' => 'UserController@userinfo']);
});


Route::group(array('prefix' => 'api1.1', 'middleware' => 'downgrade'), function() {
    Route::any('/{c}/{a}', function(){});
});


Route::group(array('prefix' => 'api1.2', 'middleware' => 'downgrade'), function() {

    Route::get('/user/info', ['uses' => 'UserController@userinfo1_2']);

    Route::any('/{c}/{a}', function(){});
});

这里就能将所有的/{version}/{controller}/{action}这样的请求经过downgrade中间件了。

但是中间件怎么写呢?

downgrade中间件的编写

<?php namespace App\Http\Middleware;

use Closure;
use Illuminate\Contracts\Routing\Middleware;


class DownGradeMiddleware implements Middleware {

    public function handle($request, Closure $next)
    {
        $routeAction = $request->route()->getAction();
        $routes = \Route::getRoutes()->getRoutes();
        
        $requestUri = $_SERVER['REQUEST_URI'];
        $querys = explode('?', $requestUri);
        $queryPath = trim($querys[0], '/');
        $querySecs = explode('/', $queryPath);

        // 没有对应的,进行api版本回找
        $versions = ['api', 'api1.1', 'api1.2'];

        $apiversion = $querySecs[0];
        $key = array_search($apiversion, $versions);
        while (1) {
            if ($key < 0) {
                break;
            }
            $querySecs[0] = $versions[$key];
            $queryPath = trim(implode('/', $querySecs), '/');

            foreach ($routes as $route) {
                if ($route->getUri() == $queryPath) {
                    $action = $route->getAction();
                    $routeAction['uses'] = $action['uses'];
                    $request->route()->setAction($routeAction);
                    return $next($request);
                }
            }

            $key--;
        }

        $response = $next($request);

        return $response;
    }
}

这里最重要的点就是将$routeAction的uses字段修改之后,调用

$request->route()->setAction($routeAction);

就可以修改路由对应的action了

其他的就是业务逻辑的问题了。

至于如何挂载middleware,可以参考laravel文档:路由进行挂载

总结

laravel4把匹配全路由的函数去掉了,但是其实使用中间件+any("{a}/{b}/{c}") 的方法也可以近似实现一个这样的功能的。

so,总是有路通向罗马的。