简介
Laravel 从 5.2 开始就有了开箱即用的用户验证,5.3 又在 5.2 的基础上又有了一些改变。为了深入了解具体的用户验证实现,只能深入 Laravel 的源码,探究用户验证是怎么处理的。
开始
安装好 Laravel 5.3 的框架后,执行下面的命令
php artisan make:auth
该命令会在项目里添加以下文件(目录)
app/Http/Controller/HomeController.PHP
resources/views/auth/
resources/views/auth/login.blade.php
resources/views/auth/passwords/
resources/views/auth/passwords/email.blade.php
resources/views/auth/passwords/reset.blade.php
resources/views/auth/register.blade.php
resources/views/home.blade.php
resources/views/layouts/
resources/views/layouts/app.blade.php
除了一个 HomeController
是处理用户登陆之后的逻辑,其他都是一些视图,用于显示相应的页面。
在 routes/web.php
里添加了以下内容
Auth::routes();
Route::get(‘/home’, ‘HomeController@index’);
Auth::routes()
是登陆、注册需要的一些路由;下面是定义一个 /home
路由,交给 HomeController@index
处理。
那么,就从路由开始我们的探究之旅吧。
路由
我们首先看看 Auth::routes()
,定义在 vendor/laravel/framework/src/Illuminate/Support/Facades/Auth.php
:
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
这里由 IoC 容器 解析了一个 Illuminate\Routing\Router 类的实例,再调用里面的 auth()
方法。
我们再来看看 auth() 方法,定义在 vendor/laravel/framework/src/Illuminate/Routing/Router.php
:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
这里定义了 登陆 、 注销 、注册 和 密码重置 的路由。
先看看注册部分。
注册
App\Http\Controllers\Auth\RegisterController
负责注册的逻辑,这里 use 了 Illuminate\Foundation\Auth\RegistersUsers
这个 trait ,包含注册时通用的一些逻辑。
路由 get('/register')
所绑定的方法 Auth\RegisterController@showRegistrationForm
就定义在这个 trait 里:
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
很简单,返回一个 auth.register
视图。
auth.register
视图获取用户的输入: name
,email
,password
,然后 POST 提交到 ‘/register’。
再来看看路由 post('/register')
所绑定的方法 Auth\RegisterController@register
。
同样, register
方法定义在 Illuminate\Foundation\Auth\RegistersUsers
里:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
首先使用请求传入的表单调用 validator()
,返回一个验证对象,再调用 validate()
验证表单内容的合法性。
validator()
定义在 App\Http\Controllers\Auth\RegisterController
里:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
在这里给出了输入表单的验证规则,如果我们的用户注册需要的表单与这几个字段不一致(例如需要添加一个手机号),就在这里修改。
返回的 Validator
对象会在 register()
方法里验证。
再回到 register()
方法, 往下走 $this->guard()->login($this->create($request->all()));
。
$this->guard()
这里会调用 Illuminate\Foundation\Auth\RegistersUsers
里的 guard()
:
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
这里无参数调用 Auth::guard()
返回一个默认的 guard ,看一下 config/auth.php
:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
默认的 guard 是 web ; web 这个 guard 采用 session 驱动, 数据提供者是 users ;users 数据提供者使用 eloquent 驱动, 使用 App\User::class
模型。
具体这个 guard 是怎么生成的,这里暂时先不探究,放到登陆验证里再详细说明。
接下来调用 guard 的 login($this->create($request->all()))
。
首先是 $this->create()
,这个方法定义在 App\Http\Controllers\Auth\RegisterController
里:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
使用 User
模型对输入的内容新增一条记录,并返回这个模型的对象。
同样,如果需要修改注册时使用的字段,也是改写这个方法。
生成的 User
对象交给 guard 的 login()
方法,做一系列登录的操作,具体怎么做的,还是放到登陆验证里再详细说明。
最后, return redirect($this->redirectPath());
完成了注册、登陆的操作,最后跳转到我们在 App\Http\Controllers\Auth\RegisterController
里设置的 protected $redirectTo = '/home';
目标 URI。
可以看一下 $this->redirectPath()
方法怎么写的,在 Illuminate\Foundation\Auth\RedirectsUsers
这个 trait 里:
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
如果定义了 $redirectTo 这个属性,就按照这个属性返回;如果没有,返回 ‘/home’。
这里把这个方法写成 trait 是因为这个方法还会在
App\Http\Controllers\Auth\LoginController
登陆控制器里使用,所以就把redirectPath()
这个方法提出来做成一个 trait ,严格遵守 DRY 原则。
到这里,就完成了注册的所有过程。