Dingo Api 1.0在laravel5.2中的简单应用

Dingo Api是为基于laravel的开发提供了一系列工具集,这些工具集可以帮助开发者快速构建APIDingo Api最新的版本是2.0.0-alpha1,这个版本需要php7.0以上的php版本的支撑,笔者这里以Dingo Api 1.0为例简单介绍其在laravel5.2中的应用。

安装Dingo Api

  • composer.json中的require中添加"dingo/api": "1.0.*@dev"

    composer.json

  • 执行composer update

    composer update

配置Dingo Api

  • config/app.phpproviders中增加Dingo\Api\Provider\LaravelServiceProvider::class,
  • 执行php artisan vendor:publish --provider="Dingo\Api\Provider\LaravelServiceProvider"以生成config下配置文件
    vendor:publish
  • config/app.phpaliases中添加'Api' => Dingo\Api\Facade\API::class,
  • 配置config/api.php
    需要配置的内容如下:
'standardsTree' => env('API_STANDARDS_TREE', 'vnd'), // 项目需要能被公开访问

'subtype' => env('API_SUBTYPE', 'blog'), // 根据项目名称配置

'prefix' => env('API_PREFIX', 'api'), // 可以根据需求去设置 这里设置成api

'name' => env('API_NAME', 'Blog Api'), // 根据项目设置

'debug' => env('API_DEBUG', true), // 开启调试便于开发 上线后关闭之

创建API路由

  • 创建App\Http\Controllers\Api\V1\UserController.php

    创建控制器

  • 创建API路由器

// 在app/Http/routes.php中增加
$api = app('Dingo\Api\Routing\Router');
  • 创建以版本为单位的路由组
// 在app/Http/routes.php中增加
$api->version('v1', function ($api) {
    $api->get('user/{id}', ['as' => 'user.show', 'uses' => 'App\Http\Controllers\Api\V1\UserController@show']); // 注意此处controller要写完整命名空间
});
  • 生成URL
// 在app/Http/routes.php中增加
app('Dingo\Api\Routing\UrlGenerator')->version('v1')->route('user.show', 'id');
  • 最终代码概览
// 在app/Http/routes.php中增加
$api = app('Dingo\Api\Routing\Router');
$api->version('v1', function ($api) {
    $api->get('user/{id}', ['as' => 'user.show', 'uses' => 'App\Http\Controllers\Api\V1\UserController@show']); // 注意此处controller要写完整命名空间
});
app('Dingo\Api\Routing\UrlGenerator')->version('v1')->route('user.show', 'id');
  • 在控制台查看路由


    查看路由

创建API并访问

  • 完善App\Http\Controllers\Api\V1\UserController
    添加代码如下:
<?php

namespace App\Http\Controllers\Api\V1;

use App\Http\Controllers\Controller;
// 这里可以去掉没用的内容
use App\User; // 引用模型

class UserController extends Controller
{
    public function show($id) // 接口方法
    {
        return User::findOrFail($id);
    }
}
  • 访问接口


    访问接口

响应构建器

通过响应构建器可以构建更多自定义响应。

  • 创建基类


    创建基类
  • 填充基类内容
    具体代码如下:

<?php

namespace App\Http\Controllers\Api\V1;

// 自动生成的内容可以去掉
use Dingo\Api\Routing\Helpers; // trait
use Illuminate\Routing\Controller; // 增加

class BaseController extends Controller
{
    use Helpers;
}
  • 继承基类
    修改App\Http\Controllers\Api\V1\UserController为:
<?php

namespace App\Http\Controllers\Api\V1;

use App\Http\Controllers\Api\V1\BaseController; // 此处有调整

use App\User; // 引用模型

class UserController extends BaseController // 此处有调整
{
    public function show($id) // 接口方法
    {
        return User::findOrFail($id);
    }
}
  • 数组响应
    修改接口代码为:
public function show($id) // 接口方法
{
    $user = User::findOrFail($id);
    return $this->response->array($user->toArray());
}

增加一个获取全部用户数据的接口

  • 修改路由
$api = app('Dingo\Api\Routing\Router');
$api->version('v1', function ($api) {
    $api->get('user/{id}', ['as' => 'user.show', 'uses' => 'App\Http\Controllers\Api\V1\UserController@show']); // 注意此处controller要写完整命名空间
    $api->get('user', ['as' => 'user.index', 'uses' => 'App\Http\Controllers\Api\V1\UserController@index']); // 注意此处controller要写完整命名空间 增加行
});
app('Dingo\Api\Routing\UrlGenerator')->version('v1')->route('user.show', 'id');
app('Dingo\Api\Routing\UrlGenerator')->version('v1')->route('user.index'); // 增加行
  • 增加接口
    App\Http\Controllers\Api\V1\UserController增加方法:
public function index()
{
    return User::all();
}
user.index
  • 数组响应
    修改App\Http\Controllers\Api\V1\UserControllerindex方法:
public function index()
{
    $users = User::all();
    return $this->response->array($users->toArray());
}
数组响应

构建转换器

转换器用于对数据进行过滤和转化用以提供更加人性化的数据。

  • 添加App\Transformers\Api\V1\UserTransformer.php
    具体代码如下:
<?php

namespace App\Transformers\Api\V1;

use League\Fractal\TransformerAbstract;
use App\User;

class UserTransformer extends TransformerAbstract
{
    public function transform(User $user)
    {
        return [
            'name' => $user->name,
            'email' => $user->email,
            'created_at' => date('Y-m-d', strtotime($user->created_at))
        ];
    }
}
  • 使用转换器
    修改App\Http\Controllers\Api\V1\UserController
<?php

namespace App\Http\Controllers\Api\V1;

use App\Http\Controllers\Api\V1\BaseController;
use App\Transformers\Api\V1\UserTransformer; // 增加行

use App\User; // 引用模型

class UserController extends BaseController
{
    
    public function index()
    {
        $users = User::all();
        return $this->response->array($users->toArray());
    }

    public function show($id) // 接口方法
    {
        $user = User::findOrFail($id);
        return $this->response->item($user, new UserTransformer); //修改
    }
}
transformer user.show

实现分页

  • 执行php artisan make:seeder UsersTableSeeder

    make:seeder

  • 修改database\seeds\UsersTableSeeder.php

<?php

use Illuminate\Database\Seeder;

class UsersTableSeeder extends Seeder
{
    /**
     * Run the database seeds.
     *
     * @return void
     */
    public function run()
    {
        factory(App\User::class, 50)->create(); // 增加一行
    }
}
  • 执行php artisan db:seed --class=UsersTableSeeder

    db:seed

    users

  • 修改App\Http\Controllers\Api\V1\UserControllerindex方法:

public function index()
{
    $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer);
}
paginate user.index

更多响应

  • 指定信息和状态码的错误响应
public function index()
{
    /* $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer); */
    return $this->response->error('This is an error.', 404);
}
指定信息和状态码的错误响应
  • 未找到错误响应
public function index()
{
    /* $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer); */
    return $this->response->errorNotFound();
}
未找到错误响应
  • 坏请求错误响应
public function index()
{
    /* $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer); */
    return $this->response->errorBadRequest();
}
坏请求错误响应
  • 禁止访问错误响应
public function index()
{
    /* $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer); */
    return $this->response->errorForbidden();
}
禁止访问错误响应
  • 内部错误错误响应
public function index()
{
    /* $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer); */
    return $this->response->errorInternal();
}
内部错误错误响应
  • 未认证错误响应
public function index()
{
    /* $users = User::paginate(8);
    return $this->response->paginator($users, new UserTransformer); */
    return $this->response->errorUnauthorized();
}
未认证错误响应
  • 添加响应头
public function show($id) // 接口方法
{
    $user = User::findOrFail($id);
    return $this->response->item($user, new UserTransformer)->withHeader('author', 'stone');
}
添加响应头
  • 添加元数据
public function show($id) // 接口方法
{
    $user = User::findOrFail($id);
    return $this->response->item($user, new UserTransformer)->addMeta('author', 'stone');
}
添加元数据
public function show($id) // 接口方法
{
    $user = User::findOrFail($id);
    return $this->response->item($user, new UserTransformer)->setMeta(['author' => 'stone', 'date' => date('Y-m-d')]);
}
添加多个元数据
  • 设置响应状态码
public function show($id) // 接口方法
{
    $user = User::findOrFail($id);
    return $this->response->item($user, new UserTransformer)->setStatusCode(202);
}
设置响应状态码

为接口增加权限控制

这一步需要提供oAuth的支持,可参考oAuth2.0在laravel5.2中的简单应用

  • 在路由定义上加入oAuth中间件
$api->version('v1', ['middleware' => 'oauth'], function ($api) { // 加上中间件
    $api->get('user/{id}', ['as' => 'user.show', 'uses' => 'App\Http\Controllers\Api\V1\UserController@show']); // 注意此处controller要写完整命名空间
    $api->get('user', ['as' => 'user.index', 'uses' => 'App\Http\Controllers\Api\V1\UserController@index']); // 注意此处controller要写完整命名空间
});
  • 获取access_token
    创建可以通过http://www.test.com/test.php访问的脚本,内容如下:
function post($url, $param){
    $oCurl = curl_init();
    $aPOST = [];
    foreach($param as $key=>$val){
        $aPOST[] = $key.'='.urlencode($val);
    }
    $strPOST =  join('&', $aPOST);
    curl_setopt($oCurl, CURLOPT_URL, $url);
    curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1 );
    curl_setopt($oCurl, CURLOPT_POST,true);
    curl_setopt($oCurl, CURLOPT_POSTFIELDS,$strPOST);
    $sContent = curl_exec($oCurl);
    $aStatus = curl_getinfo($oCurl);
    curl_close($oCurl);
    if(200 == intval($aStatus['http_code'])){
        return $sContent;
    }else{
        return false;
    }
}

$server = 'http://localhost:8000/oauth/access_token';
$params = [
    'grant_type' => 'password',
    'username' => 'admin@admin.com',
    'password' => '123456',
    'client_id' => 'shy7jf8fa93d59c45502c0ae8chj76s',
    'client_secret' => 'bc7f6f8fa93d59c45502c0ae8c4a95d',
];
echo post($server, $params);
获取access_token
  • 访问需要授权的接口


    不带access_token访问
带错误的access_token访问
带正确的access_token访问

本文首发于公众号:programmer_cc,转载请注明出处。


微信公众号.jpg
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,646评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,973评论 25 707
  • 0.1配置1.模板继承2.控制器3.git4.支付宝支付的流程5.路由6.中间件7.请求8.laravel 学习笔...
    云龙789阅读 806评论 0 5
  • 机缘巧合之下报读小v老师的思维导图课程,在21天的训练营里,逻辑思维水平提高不少,但我的阅读速度并没有跟上。 紧接...
    三鱼甘阅读 345评论 0 0
  • css盒模型是理解css布局的基础。 一个页面是由无数个“盒子”组成的,如果大家用Firebug或是其它调试程序...
    有梦不怕路远_阅读 217评论 0 0