Для авторизованных пользователей я установил ограничение на частоту запросов к системе. Если количество запросов за данный интервал времени больше заданного, возвращается ответ с кодом состояния 429
, В моем конкретном случае это не более одного запроса за пять секунд.
В классе идентичности я реализовал yii\filters\RateLimitInterface
:
<?php
namespace common\models;
use yii\db\ActiveRecord;
use yii\web\IdentityInterface;
use yii\filters\RateLimitInterface;
/**
* User model
*
* ...
* @property integer $id
* ...
*
* @property integer $allowance
* @property integer $allowance_updated_at
*/
class User extends ActiveRecord implements IdentityInterface, RateLimitInterface
{
...
/**
* @inheritdoc
*/
public static function tableName()
{
return '{{%user}}';
}
/**
* Returns the maximum number of allowed requests and the window size.
* @param \yii\web\Request $request the current request
* @param \yii\base\Action $action the action to be executed
* @return array an array of two elements. The first element is the maximum number of allowed requests,
* and the second element is the size of the window in seconds.
*/
public function getRateLimit($request, $action)
{
return [1, 5];
}
/**
* Loads the number of allowed requests and the corresponding timestamp from a persistent storage.
* @param \yii\web\Request $request the current request
* @param \yii\base\Action $action the action to be executed
* @return array an array of two elements. The first element is the number of allowed requests,
* and the second element is the corresponding UNIX timestamp.
*/
public function loadAllowance($request, $action)
{
return [$this->allowance, $this->allowance_updated_at];
}
/**
* Saves the number of allowed requests and the corresponding timestamp to a persistent storage.
* @param \yii\web\Request $request the current request
* @param \yii\base\Action $action the action to be executed
* @param integer $allowance the number of allowed requests remaining.
* @param integer $timestamp the current timestamp.
*/
public function saveAllowance($request, $action, $allowance, $timestamp)
{
$this->allowance = $allowance;
$this->allowance_updated_at = $timestamp;
$this->save();
}
...
}
В поведении я добавил следующее:
'rateLimiter' => [
'class' => RateLimiter::className(),
]
К столу tbl_user
Я добавил два поля:
[allowance] [int] NULL
[allowance_updated_at] [int] NULL
В локальной среде все работает нормально, и если пользователь сделал более одного запроса за 5 секунд, он увидит исключение yii\web\TooManyRequestsHttpException
и эта страница:
Хорошо. Затем я создал два файла миграции и применил их на тестовом сервере и на рабочем сервере:
...
yii migrate/create add_allowance_column_to_tbl_user --fields=allowance:integer
yii migrate/create add_allowance_updated_at_column_to_tbl_user --fields=allowance_updated_at:integer
yii migrate
...
Миграционные файлы:
<?php
use yii\db\Migration;
class m160722_073856_add_allowance_column_to_tbl_user extends Migration
{
public function up()
{
$this->addColumn('tbl_user', 'allowance', $this->integer());
}
public function down()
{
$this->dropColumn('tbl_user', 'allowance');
}
}
...
<?php
use yii\db\Migration;
class m160722_073932_add_allowance_updated_at_column_to_tbl_user extends Migration
{
public function up()
{
$this->addColumn('tbl_user', 'allowance_updated_at', $this->integer());
}
public function down()
{
$this->dropColumn('tbl_user', 'allowance_updated_at');
}
}
На тестовом сервере все работает отлично.
Затем я перенес решение на рабочий сервер и у меня возникла проблема:
2016-07-22 12:17:37 [10.1.80.59][25112][snadvdlp3cb15fs9kc0j3dk5c3][error][yii\base\UnknownPropertyException] exception 'yii\base\UnknownPropertyException' with message 'Getting unknown property: common\models\User::allowance' in D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\Component.php:143
Stack trace:
#0 D:\wwwroot\aisnew\vendor\yiisoft\yii2\db\BaseActiveRecord.php(247): yii\base\Component->__get('allowance')
#1 D:\wwwroot\aisnew\common\models\User.php(388): yii\db\BaseActiveRecord->__get('allowance')
#2 D:\wwwroot\aisnew\vendor\yiisoft\yii2\filters\RateLimiter.php(100): common\models\User->loadAllowance(Object(yii\web\Request), Object(yii\base\InlineAction))
#3 D:\wwwroot\aisnew\vendor\yiisoft\yii2\filters\RateLimiter.php(78): yii\filters\RateLimiter->checkRateLimit(Object(common\models\User), Object(yii\web\Request), Object(yii\web\Response), Object(yii\base\InlineAction))
#4 D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\ActionFilter.php(70): yii\filters\RateLimiter->beforeAction(Object(yii\base\InlineAction))
#5 [internal function]: yii\base\ActionFilter->beforeFilter(Object(yii\base\ActionEvent))
#6 D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\Component.php(541): call_user_func(Array, Object(yii\base\ActionEvent))
#7 D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\Controller.php(269): yii\base\Component->trigger('beforeAction', Object(yii\base\ActionEvent))
#8 D:\wwwroot\aisnew\vendor\yiisoft\yii2\web\Controller.php(121): yii\base\Controller->beforeAction(Object(yii\base\InlineAction))
#9 D:\wwwroot\aisnew\common\components\AisUniversityController.php(22): yii\web\Controller->beforeAction(Object(yii\base\InlineAction))
#10 D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\Controller.php(152): common\components\AisUniversityController->beforeAction(Object(yii\base\InlineAction))
#11 D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\Module.php(454): yii\base\Controller->runAction('readyoperator', Array)
#12 D:\wwwroot\aisnew\vendor\yiisoft\yii2\web\Application.php(84): yii\base\Module->runAction('queue/queue/rea...', Array)
#13 D:\wwwroot\aisnew\vendor\yiisoft\yii2\base\Application.php(375): yii\web\Application->handleRequest(Object(yii\web\Request))
#14 D:\wwwroot\aisnew\backend\web\index.php(18): yii\base\Application->run()
Я переключаю локальную / тестовую среду на промышленную базу данных, и все работает ..
В чем может быть причина?
Буду очень признателен за информацию. Спасибо всем.
Задача ещё не решена.
Других решений пока нет …