场景
- laravel5.5中没有使用middleware throttle 但是却可以实现限制访问次数, 那么他是怎么实现的呢?

源码分析
$this->hasTooManyLoginAttempts($request)
- 判断是否已经超过了限制
$this->limiter()->tooManyAttempts
方法和laavel5.5 throttle middleware源码解析 博文中的tooManyAttempts的实现方式是一样的,
$this->throttleKey($request)
唯一键是email && ip的组合体
Str::lower( r e q u e s t − > i n p u t ( request->input( request−>input(this->username())).’|’.$request->ip();`
$this->incrementLoginAttempts($request);
- 内部调用hit方法, 实现create计时器 计数器 or increment 计数器的功能
安全改造(签名IP+Emmail => IP)
laravel5.5对login生成的唯一键是 email . '|' . ip ; 如果某人有一大批用户名 && 密码; 则只要保证一种分钟内尝试登陆的用户名不同,就可以暴力破解密码了;
- 引入自定义配置文件 config/custom.php; 方便随时切换签名
- return [‘throttle_key’ => ‘ip’,]
- App\Http\Controllers\Auth\LoginController 重写throttleKey
改造过的throttleKey
protected function throttleKey(Request $request)
{
if (config('custom.throttle_key') === 'ip') {
return $request->ip();
}
return Str::lower($request->input($this->username())).'|'.$request->ip();
}
源码
public function login(Request $request)
{
$this->validateLogin($request);
if ($this->hasTooManyLoginAttempts($request)) {
$this->fireLockoutEvent($request);
return $this->sendLockoutResponse($request);
}
if ($this->attemptLogin($request)) {
flash('欢迎回来!')->success();
return $this->sendLoginResponse($request);
}
$this->incrementLoginAttempts($request);
return $this->sendFailedLoginResponse($request);
}
protected function hasTooManyLoginAttempts(Request $request)
{
return $this->limiter()->tooManyAttempts(
$this->throttleKey($request), $this->maxAttempts(), $this->decayMinutes()
);
}
protected function throttleKey(Request $request)
{
return Str::lower($request->input($this->username())).'|'.$request->ip();
}
public function maxAttempts()
{
return property_exists($this, 'maxAttempts') ? $this->maxAttempts : 5;
}
public function decayMinutes()
{
return property_exists($this, 'decayMinutes') ? $this->decayMinutes : 1;
}
protected function incrementLoginAttempts(Request $request)
{
$this->limiter()->hit(
$this->throttleKey($request), $this->decayMinutes()
);
}