# 会话控制
在现代 Web 程序中,常见的 `HTTP 会话机制` 有多种:
* JWT (JSON Web Token): https://jwt.io/
* Session
* Cookie
* ...
在本章的学习中,我们采用 `session` 来控制会话.
## 会话控制器
在控制器中键入 `php think make:controller user/Session` 创建会话控制器并编辑,
`application\user\controller\Session.php`:
~~~~ php
use app\user\model\User;
...
public function create()
{
$token = $this->request->token('__token__', 'sha1');
$this->assign('token', $token);
return $this->fetch();
}
~~~~
`route\route.php`:
~~~~ php
Route::resource('session', 'user/session');
~~~~
创建模板 `resources\views\user\session\create.blade.php`:
~~~~ html
@extends('_layout.default')
@section('title', '登入')
@section('content')
<div class="col-md-offset-2 col-md-8">
<div class="panel panel-default mt-5">
<div class="panel-heading mb-3">
<h4>登入</h4>
</div>
@if(session('validate'))
<div class="alert alert-warning" role="alert">
{{ session('validate') }}
</div>
@endif
<div class="panel-body">
<form method="POST"
action="{{ url('save') }}">
<input type="hidden" name="__token__" value="{{ $token }}" />
<div class="input-group mb-3">
<div class="input-group-prepend">
<span class="input-group-text">邮箱</span>
</div>
<input type="email"
class="form-control"
name="email">
</div>
<div class="input-group mb-3">
<div class="input-group-prepend">
<span class="input-group-text">密码</span>
</div>
<input type="password"
class="form-control"
name="password">
</div>
<button type="submit"
class="btn btn-primary btn-block">登入</button>
</form>
</div>
</div>
</div>
@stop
~~~~
以上创建的内容与上一张所建内容大多类似,不再详细说明.
现在访问 `http://thinkphp.test/user/session/create` 即可看到登录页面.
同样的,我们创建验证器,在控制台中键入 `php think make:validate user/Session` 并打开创建好的
`application\user\validate\Session.php`:
~~~~ php
protected $rule = [
'__token__' => 'token',
'email|邮件' => 'require|email|max:255',
'password|密码' => 'require|min:6'
];
~~~~
再打开 `application\user\controller\Session.php`:
~~~~ php
public function save(Request $request)
{
$result = $this->validate($request->post(), 'app\user\validate\Session');
if (true !== $result) {
return redirect('user/session/create')->with('validate',$result);
} else {
$user = User::where('email', $request->email)->find();
if ($user !== null && password_verify($request->password, $user->password)) {
return 'Password is valid!';
} else {
return 'Invalid password.';
}
}
}
~~~~
在以上 `save` 方法中, `password_verify($request->password, $user->password)` 对应着上一章所用的 `password_hash` 方法,目的是验证 `hash` 之后的数据, `password_verify(请求数据, 待验证的数据)`,如果验证成功则抛出 `true`: http://php.net/manual/zh/function.password-verify.php
请注意,在 PHP 中判断 `null`:
标识 | empty == null | is_null === null | isset | array_key_exists
---- | ---- | ---- | ---- | ----
ϕ | T | T | F | F
null | T | T | F | T
"" | T | F | T | T
[] | T | F | T | T
0 | T | F | T | T
false | T | F | T | T
true | F | F | T | T
1 | F | F | T | T
\0 | F | F | T | T
* `$user != null` == 在 PHP 运算符中不检查类型
* `$user !== null` === 在 PHP 运算符中检查类型
意味着如果使用 `==`,PHP 将会转换成一致的类型再做判断,这也是动态弱类型语言的一大弱势,如果无法提前知晓接收的值的类型,也没有类型检查,解释器所转换的类型将不可控制.
在要求高精度的程序中(例如财务系统),请使用静态强类型语言,在弱类型语言当中,一些高精度的浮点数将会丢失(int to double),例如 `0.9999999999999^2`,甚至也有 `'0.999999999999' * 0.999999999999`
下面是一些例子:
* 无类型: 汇编
* 弱类型、静态类型 : C/C++
* 弱类型、动态类型检查: Perl/PHP
* 强类型、静态类型检查 :Java/C#
* 强类型、动态类型检查 :Python, Scheme
* 静态显式类型 :Java/C
* 静态隐式类型 :Ocaml, Haskell
* ...
笔者注: 我经常遇到一些朋友想使用 PHP 来做通讯系统,爬虫系统甚至 GUI 程序,做一个一款语言就能集大成的程序,这是非常不可取的,软件行业没有银弹(Silver Bullet),没有任何一款语言能做完所有的事情,也没有任何一款语言能够让你掌握了就吃到老,对阵下药,不要宰牛用杀猪刀.
重新看到 `application\user\controller\Session.php` 并修改:
~~~~ php
if ($user !== null && password_verify($request->password, $user->password)) {
return redirect('user/auth/read')->params(['id' => $user->id]);
} else {
return redirect('user/session/create')->with('validate','邮件地址不存在或密码错误');
}
~~~~
现在如果验证成功,那么就会跳转到上一章所编写的 `user/auth/read` 方法.
虽然现在已经成功验证账户对应的密码,但是还未做到权限的状态管理,用户不管登录是否,都可以 `http://thinkphp.instudy.test/user/auth/read/id/:id` 的地址.
## 中间件拦截
中间件相当于在 `路由至控制器` 之间修建一道门,如果通过中间件的规则则可以进行下一步的操作,我们现在通过中间件来验证 `是否已经登入`: https://www.kancloud.cn/manual/thinkphp5_1/564279
经过笔者两小时的测试,ThinkPHP 的中间件与路由有非常大的逻辑缺陷问题,中间件无法正确挂载至资源路由,资源路由生成规则错误及混乱,URL 绑定在资源路由中不起作用,以下是一些针对这一章节 ThinkPHP 框架的错误:
* 框架版本: V5.1.28 LTS(2018-10-28)
* PHP 版本: PHP 7.2.11-4+ubuntu18.04.1+deb.sury.org+1 (cli) (built: Nov 4 2018 05:11:49) ( NTS )
定义中间件:
~~~~ php
public function handle($request, \Closure $next)
{
$request->mid = 'mid';
return $next($request);
}
~~~~
如果中间件不在路由注册:
~~~~ php
Route::resource('auth', 'user/auth')
~->middleware('Auth')~
~~~~
那么在控制器不管怎么定义了中间件, dump 都是 null
~~~~ php
protected $middleware = [
'Auth',
];
~~~~
访问地址: `http://thinkphp.test/user/auth/create.html` 或者 `http://thinkphp.test/auth/create.html`
都会提示: `/home/vagrant/code/instudy/thinkphp/thinkphp/library/think/Debug.php:226:null`
而如果在路由中进行注册中间件:
~~~~ php
Route::resource('auth', 'user/auth')->middleware('Auth')
~~~~
基于完整的控制器路径 `http://thinkphp.test/user/auth/create.html` 仍然输出 `null`
而进入绑定之后的地址 `http://thinkphp.test/auth/create.html` 却有值 `mid`
更加神奇的是,如果要使用 URL 生成路由 `url('user/auth/create')`,只会生成到 完整的控制器路径(http://thinkphp.test/user/auth/create.html),而不会生成绑定的地址(http://thinkphp.test/auth/create.html)
如果你要强行用绑定的地址,只能写死路径
`return redirect('/auth/read')->params(['id' => $user->id]);`
当写死路径之后,`params()` 方法将不会传值到 `auth/read/:id`,只会跳转到 `/auth/read`
关于路由的错误,在前面的章节已经遇到并且不得已才琢磨出了匹配完整路径的方法,可惜 URL 生成:https://www.kancloud.cn/manual/thinkphp5_1/353977 根本不工作,导致了路由中间件挂载无效.
同时,无论怎么样在资源控制器中挂载中间件,都是无效的,不执行任何东西.
基于以上的问题,我们不再采用中间件的方式进行会话拦截.
## 手动拦截
此方法比较繁琐,代码复用程度极差,是一个临时解决中间件问题的办法.
`application\user\controller\Session.php`:
~~~~ php
use think\facade\Session as SessionFacade;
...
if ($user !== null && password_verify($request->password, $user->password)) {
SessionFacade::set('user', $user);
return redirect('user/auth/read')->params(['id' => $user->id]);
}
~~~~
可以看到,我们在顶部引入了 Session 的库,再引入之后,类就会默认挂载 `class Session`,可是这与我们当前 `Session` 的命名出现了冲突,所以使用 `as SessionFacade` 来为 `Session 外部类` 增加别名.
现在重新进行登录操作,就会重定向至 `'user/auth/read/:id'` 并且附带一个 `user` 的 `session`.
现在我们打开 `application\user\controller\Auth.php`:
~~~~ php
use think\facade\Session;
...
public function read($id)
{
if (Session::has('user')) {
$user = User::find($id);
$this->assign('user', $user);
return $this->fetch();
} else {
return redirect('user/session/create')->with('validate','请先登录');
}
}
~~~~
`Session::has(param)` 是判断 `param` 的 `session` 值是否存在,现在访问 `http://thinkphp.instudy.test/user/auth/read/id/1.html` 则会跳转至登录页面.
- 第一章. 基础信息
- 1.1 序言
- 1.2 关于作者
- 1.3 本书源码
- 1.4 反馈纠错
- 1.5 安全指南
- 1.6 捐助作者
- 第二章. 开发环境布置
- 2.1 编辑器选用
- 2.2 命令行工具
- 2.3 开发环境搭建
- 2.4 浏览器选择
- 2.5 第一个应用
- 2.6 Git 工作流
- 第三章. 构建页面
- 3.1 章节说明
- 3.2 静态页面
- 3.3 Think 命令
- 3.4 小结
- 第四章. 优化页面
- 4.1 章节说明
- 4.2 样式美化
- 4.3 局部视图
- 4.4 路由链接
- 4.5 用户注册页面
- 4.6 集中视图
- 4.7 小结
- 第五章. 用户模型
- 5.1 章节说明
- 5.2 数据库迁移
- 5.3 查看数据表
- 5.4 模型文件
- 5.5 小结
- 第六章. 用户注册
- 6.1 章节说明
- 6.2 注册表单
- 6.3 用户数据验证
- 6.4 注册失败错误信息
- 6.5 注册成功
- 6.6 小结
- 第七章. 会话管理
- 7.1 章节说明
- 7.2 会话
- 7.3 用户登录
- 7.4 退出
- 7.5 小结
- 第八章. 用户 CRUD
- 8.1 章节说明
- 8.2 重构代码
- 8.3 更新用户
- 8.4 权限系统
- 8.5 列出所有用户
- 8.6 删除用户
- 8.7 访客模式
- 8.8 优化前端
- 8.9 小结
- 第九章. 微博 CRUD
- 9.1 章节说明
- 9.2 微博模型
- 9.3 显示微博
- 9.4 发布微博
- 9.5 微博数据流
- 9.6 删除微博
- 9.7 小结