1、涉及至第 1 张表,企鹅号的文章:cpa_qq_article,需要验证其字段:title 的唯一性,表结构如图1
2、涉及至第 2 张表,渠道的应用的任务:cpa_channel_app_task,字段:task_id 与 企鹅号的文章的字段:task_id 存在关联关系(多对一),字段:channel_app_source_uuid 为条件(变量、请求参数),字段:status 为条件(常量),表结构如图2
3、filter:用于检查输入值唯一性必然会进行数据库查询,而该属性为用于进一步筛选该查询的过滤条件。样式为 function ($query) 的匿名函数, $query 参数为你希望在该函数内进行修改的 Query 对象。模型中的验证规则声明如下:
/**
* {@inheritdoc}
*/
public function scenarios()
{
$scenarios = parent::scenarios();
$scenarios[self::SCENARIO_CREATE] = ['group_id', 'article_category_id', 'title', 'author', 'source_article_id'];
return $scenarios;
}
/**
* @inheritdoc
*/
public function rules()
{
$rules = [
['title', 'unique', 'filter' => function ($query) {
/* @var $query ChannelAppTaskQuery */
$query->joinWith('channelAppTask', false)->andWhere([ChannelAppTask::tableName() . '.status' => ChannelAppTask::STATUS_PLATFORM_PUBLISHED]);
}, 'on' => self::SCENARIO_CREATE],
];
$parentRules = parent::rules();
return ArrayHelper::merge($rules, $parentRules);
}
/*
* 声明了一个渠道的应用的任务关联
*/
public function getChannelAppTask()
{
return $this->hasOne(ChannelAppTask::className(), ['task_id' => 'task_id']);
}
4、验证请求参数时,生成的 SQL 如下:
SELECT EXISTS(SELECT `cpa_qq_article`.* FROM `cpa_qq_article` LEFT JOIN `cpa_channel_app_task` ON `cpa_qq_article`.`task_id` = `cpa_channel_app_task`.`task_id` WHERE (`cpa_qq_article`.`title`='未来 10 年,“星光中国芯工程”计划投资 100 亿元用于芯片研发及大规模产业化。') AND (`cpa_channel_app_task`.`status`=6))
5、请求参数中,存在 channel_app_source_uuids 字段,其值为数组:[“8d72b7cc2ac911eab85a54ee75d2ebc1”, “18cf06d22ac911eaa31854ee75d2ebc1”],需要将 channel_app_source_uuids 的值做为条件应用于唯一性验证的查询条件中。在执行模型的验证之前,赋值给应用的全局参数变量 Yii::$app->params。如图3
$requestParams = Yii::$app->getRequest()->getBodyParams(); Yii::$app->params['channelAppSourceUuids'] = $requestParams['channel_app_source_uuids'];
6、现在可在验证规则中使用 Yii::$app->params,模型中的验证规则声明调整如下:
/**
* @inheritdoc
*/
public function rules()
{
$rules = [
['title', 'unique', 'filter' => function ($query) {
/* @var $query ChannelAppTaskQuery */
$query->joinWith('channelAppTask', false)->andWhere(['and', ['in', ChannelAppTask::tableName() . '.channel_app_source_uuid', Yii::$app->params['channelAppSourceUuids']], [ChannelAppTask::tableName() . '.status' => ChannelAppTask::STATUS_PLATFORM_PUBLISHED]]);
}, 'on' => self::SCENARIO_CREATE],
];
$parentRules = parent::rules();
return ArrayHelper::merge($rules, $parentRules);
}
7、验证请求参数时,生成的 SQL 如下,请求参数已经应用于查询条件中,符合预期
SELECT EXISTS(SELECT `cpa_qq_article`.* FROM `cpa_qq_article` LEFT JOIN `cpa_channel_app_task` ON `cpa_qq_article`.`task_id` = `cpa_channel_app_task`.`task_id` WHERE (`cpa_qq_article`.`title`='未来 10 年,“星光中国芯工程”计划投资 100 亿元用于芯片研发及大规模产业化。') AND ((`cpa_channel_app_task`.`channel_app_source_uuid` IN ('8d72b7cc2ac911eab85a54ee75d2ebc1', '18cf06d22ac911eaa31854ee75d2ebc1')) AND (`cpa_channel_app_task`.`status`=6)))
8、核心验证器:unique (唯一性) 支持多个模型与变量的实现,验证失败时,响应如下,如图4
{
"code": 226004,
"message": "数据验证失败:标题的值/"未来 10 年,“星光中国芯工程”计划投资 100 亿元用于芯片研发及大规模产业化。/"已经被占用了。"
}
原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/tech/pnotes/250515.html
