通过 Passport 实现 API 请求认证

准备工作

# 1.
$ composer require laravel/passport # 安装
$ php artisan migrate # 运行数据库迁移命令创建 OAuth 相关数据表
$ php artisan passport:install 

# 2.在对应模型类中使用 Laravel\Passport\HasApiTokens Trait
class User extends Authenticatable
{
    use HasApiTokens, Notifiable;
    ...
}

# 3.在 AuthServiceProvider 的 boot 方法中注册 API 认证相关路由
public function boot()
{
    ...
    Passport::routes();
}

# 4.修改配置文件config/auth.php
'guards' => [
    ...
    'api' => [
        'driver' => 'passport',
        'provider' => 'users',
    ],
],

OAuth 2.0 的四种方式

这里用 blog (后端)项目和新建的 testapp (前端)项目进行演示操作。

密码式 password

如果你高度信任某个应用,RFC 6749 也允许用户把用户名和密码,直接告诉该应用。该应用就使用你的密码,申请令牌,这种方式称为"密码式"(password)。

# 1. 在后端应用中注册,获取Client id 和 secret
# 2.配置 testapp 的 .env 和 config/services.php
# 3.在 testapp 中写控制器方法
# 4.测试后端应用授权移动端认证

# 1.在 blog 中运行命令注册 testapp ,获取到Client id 和 secret
$ php artisan passport:client --password

# 2. 
# 配置 testapp 的 .env 文件
CLIENT_ID=x
CLIENT_SECRET=xxx...
# 配置 config/services.php
'blog' => [
    'appid' => env('CLIENT_ID'),
    'secret' => env('CLIENT_SECRET'),
    'callback' => 'http://app.test/auth/callback'
]

# 3.
// 重写 AuthenticatesUsers 中的 login 方法
public function login(Request $request)
{
    $request->validate([
        'email' => 'required|string',
        'password' => 'required|string',
    ]);

    $http = new Client();
    // 发送相关字段到后端应用获取授权令牌
    $response = $http->post('http://blog.test/oauth/token', [
        'form_params' => [
            'grant_type' => 'password',
            'client_id' => config('services.blog.appid'),
            'client_secret' => config('services.blog.secret'),
            'username' => $request->input('email'),  // 这里传递的是邮箱
            'password' => $request->input('password'), // 传递密码信息
            'scope' => '*'
        ],
    ]);

    return response($response->getBody());
}

# 4.在浏览器访问http://app.test/login, 校验成功后就会返回令牌信息

授权码式 authorization code

授权码(authorization code)方式,指的是第三方应用先申请一个授权码,然后再用该码获取令牌。

# 1.在后端应用中注册,获取Client id 和 secret
# 2.配置 testapp 的 .env
# 3.编写 testapp 应用路由和控制器
# 4.测试通过授权码获取令牌

# 1.
$ php artisan passport:client
# What should we name the client?
$ testapp
# ...redirect...?
$ http://app.test/auth/callback

# 2. .env:
CLIENT_ID=x
CLIENT_SECRET=xxx...

# 3.
# web.php:
Route::get('/auth', 'Auth\LoginController@oauth');
Route::get('/auth/callback', 'Auth\LoginController@callback');
# LoginController.php:
public function oauth()
{

    $query = http_build_query([
        'client_id' => config('services.blog.appid'),
        'redirect_uri' => config('services.blog.callback'),
        'response_type' => 'code',
        'scope' => '',
    ]);

    return redirect('http://blog.test/oauth/authorize?'.$query);
}

public function callback(Request $request)
{
    $code = $request->get('code');
    if (!$code) {
        dd('授权失败');
    }
    $http = new Client();
    $response = $http->post('http://blog.test/oauth/token', [
        'form_params' => [
            'grant_type' => 'authorization_code',
            'client_id' => config('services.blog.appid'),  // your client id
            'client_secret' => config('services.blog.secret'),   // your client secret
            'redirect_uri' => config('services.blog.callback'),
            'code' => $code,
        ],
    ]);

    return response($response->getBody());
}

# 4.访问 http://app.test/auth,跳转授权,获取令牌

凭证式 client credentials

最后一种方式是凭证式(client credentials),适用于没有前端的命令行应用,即在命令行下请求令牌。

# 1 2 4 同上
# 3. web.php:
Route::get('/auth/client', 'Auth\LoginController@client');
LoginController.php:
public function client()
{
    $http = new Client();
    $response = $http->post('http://blog.test/oauth/token', [
        'form_params' => [
            'grant_type' => 'client_credentials',
            'client_id' => config('services.blog.appid'),  // your client id
            'client_secret' => config('services.blog.secret'),   // your client secret
            'scope' => '*'
        ],
    ]);

    return response($response->getBody());
}

沙箱测试篇(私人访问令牌)

这种授权方式比较特殊,不需要授权码,也不需要用户输入登录凭证,而是用户给自己颁发访问令牌。将后端应用 blog 类比做开放平台,在这个开发平台上通过的测试应用体验系统提供的认证 API。

# 1. 注册,获取 client id 和secret
$ php artisan passport:client --personal

# 2.在 User 模型类中使用HasApiTokens
class User extends Authenticatable
{
    use HasApiTokens, Notifiable;
    ...
}

# 3.
#在 routes/web.php 中定义一个路由用于测试获取访问令牌
Route::get('auth/personal', 'Auth\LoginController@personal');
# 控制器 LoginController 编写对应的方法 personal
public function personal()
{
    $user = User::where('name', 'Kong')->first();
    $token = $user->createToken('Users')->accessToken;
    dd($token);
}
# 4.访问

隐藏式 implicit

有些 Web 应用是纯前端应用,没有后端。这时就不能用上面的方式了,必须将令牌储存在前端。RFC 6749 就规定了第二种方式,允许直接向前端颁发令牌。这种方式没有授权码这个中间步骤,所以称为(授权码)"隐藏式"(implicit)。

# 在后端系统 AuthServiceProvider 的 boot 方法中调用 enableImplicitGrant 方法
public function boot()
{
    ...
Passport::enableImplicitGrant();
}

# 1 2 3 4 步骤同上

# 1.
$ php artisan passport:client
# ..name..?
$ testapp
# ..rediredt..?
$ http://app.test/auth/implicit/callback

# 2. .env不用我多说了吧
# 在 config/services.php 的 blog 配置项中修改 callback 配置值:
'callback' => 'http://app.test/auth/implicit/callback'

# 3.
# web.php
Route::get('/auth/implicit', 'Auth\LoginController@implicit');
Route::get('/auth/implicit/callback', 'Auth\LoginController@implicitCallback');
# LoginController.php
public function implicit()
{
    $query = http_build_query([
        'client_id' => config('services.blog.appid'),
        'redirect_uri' => config('services.blog.callback'),
        'response_type' => 'token',
        'scope' => '',
    ]);

    return redirect('http://blog.test/oauth/authorize?'.$query);
}

public function implicitCallback(Request $request)
{
    dd($request->get('access_token'));
}


令牌作用域

以授权码获取令牌为例。

# 后端
# 1.在 AuthServiceProvider.php 的 boot 方法中设置令牌作用域
# 2.在 app/Http/Kernel.php 的 $routeMiddleware 中引入两个中间件
# 3.在 routes/api.php 中新增路由
# 前端
# 4.修改授权码令牌对应配置的 scope

# 1. AuthServiceProvider.php:
Passport::tokensCan([
    'basic-user-info' => '获取用户名、邮箱信息',
    'all-user-info' => '获取用户所有信息',
    'get-post-info' => '获取文章详细信息',
]);

# 2.
'scopes' => \Laravel\Passport\Http\Middleware\CheckScopes::class,
'scope' => \Laravel\Passport\Http\Middleware\CheckForAnyScope::class,

# 3.
Route::middleware('auth:api')->group(function () {
    Route::get('/user', function (Request $request) {
        $user = $request->user();
        if ($user->tokenCan('all-user-info')) {
            // 如果用户令牌有获取所有信息权限,返回所有用户字段
            return $user;
        }
        // 否则返回用户名和邮箱等基本信息
        return ['name' => $user->name, 'email' => $user->email];
    })->middleware('scope:basic-user-info,all-user-info');
    Route::get('/post/{id}', function (Request $request, $id) {
        return \App\Post::find($id);
    })->middleware('scopes:get-post-info');
});

# 4.
public function oauth()
{

    $query = http_build_query([
        'client_id' => config('services.blog.appid'),
        'redirect_uri' => config('services.blog.callback'),
        'response_type' => 'code',
        'scope' => 'all-user-info get-post-info',
    ]);

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