当在 Action 里处理用户的请求时,经常要先获取用户提交过来的数据,然后对其校验,如果校验没问题后才能进行后续的操作。当参数校验完成后,有时候还要进行权限判断,等这些都判断无误后才能进行真正的逻辑处理。如果将这些代码都放在一个 Action 里,势必让 Action 的代码非常复杂且冗长。
为了解决这个问题, ThinkJS 在控制器前面增加了一层 Logic
,Logic 里的 Action 和控制器里的 Action 一一对应,系统在调用控制器里的 Action 之前会自动调用 Logic 里的 Action。
Logic 目录在 src/[module]/logic
,在通过命令 thinkjs controller [name]
创建 Controller 时会自动创建对应的 Logic。Logic 代码类似如下:
'use strict';
/**
* logic
* @param {} []
* @return {} []
*/
export default class extends think.logic.base {
/**
* index action logic
* @return {} []
*/
indexAction(){
}
}
其中,Logic 里的 Action 和 Controller 里的 Action 一一对应。Logic 里也支持 __before
和 __after
等魔术方法。
数据校验的配置如下:
export default class extends think.logic.base {
indexAction(){
let rules = {
doc: "string|default:index",
version: "string|in:1.2,2.0|default:2.0"
}
}
}
配置格式为 字段名
-> 配置
,每个字段的配置支持多个校验类型,校验类型之间用 |
隔开,校验类型和参数之间用 :
隔开,参数之间用 ,
隔开来支持多个参数。
校验类型后面可以接参数,除了支持用逗号隔开的简单参数外,还可以支持 JSON 格式的复杂参数。如:
export default class extends think.logic.base {
indexAction(){
let rules = {
field1: "array|default:[1,2]", //参数为数组
field2: 'object|default:{\"name\":\"thinkjs\"}' //参数为对象
}
}
}
支持的数据类型有:boolean
、string
、int
、float
、array
、object
,默认为 string
。
使用 default:value
来定义字段的默认值,如果当前字段值为空,会将默认值覆盖过去,后续获取到的值为该默认值。
默认根据当前请求的类型来获取字段对应的值,如果当前请求类型是 GET,那么会通过 this.get('version')
来获取 version
字段的值。如果请求类型是 POST,那么会通过 this.post
来获取字段的值。
但有时候在 POST 类型下,可能会获取上传的文件或者获取 URL 上的参数,这时候就需要指定获取数据的方式了。支持的获取数据方式为 get
,post
和 file
。
export default class extends think.logic.base {
/**
* 保存数据,POST 请求
* @return {} []
*/
saveAction(){
let rules = {
name: "required",
image: "object|file|required",
version: "string|get|in:1.2,2.0|default:2.0"
}
}
}
上面示例指定了字段 name
通过 post
方法来获取值,字段 image
通过 file
方式来获取值,字段 version
通过 get
方式来获取值。
上面的配置只是指定了具体的校验规则,并没有指定校验出错后给出的错误信息。错误信息支持国际化,需要在配置文件 src/common/config/locale/[lang].js
中定义。如:
// src/common/config/locale/en.js
export default {
validate_required: '{name} can not be blank',
validate_contains: '{name} need contains {args}',
}
其中 key 为 validate_
+ 校验类型名称
,值里面支持 {name}
和 {args}
2个参数,分别代表字段名称和传递的参数。
如果想定义个特定字段某个错误类型的具体信息,可以通过在后面加上字段名。如:
// src/common/config/locale/en.js
export default {
validate_required: '{name} can not be blank',
validate_required_email: 'email can not be blank', //指定字段 email 的 required 错误信息
}
配置好校验规则后,可以通过 this.validate
方法进行校验。如:
export default class extends think.logic.base {
indexAction(){
let rules = {
doc: "string|default:index",
version: "string|in:1.2,2.0|default:2.0"
}
let flag = this.validate(rules);
if(!flag){
return this.fail('validate error', this.errors());
}
}
}
如果返回值为 false
,那么可以通过 this.errors
方法获取详细的错误信息。拿到错误信息后,可以通过 this.fail
方法把错误信息以 JSON 格式输出,也可以通过 this.display
方法输出一个页面。
错误信息通过 errors
字段赋值到模版里,模版里通过下面的方式显示错误信息(以 ejs 模版为例):
<%for(var field in errors){%>
<%-field%>:<%errors[field]%>
<%}%>
一般情况下,都是校验有问题后,输出一个 JSON 信息。如果每次都要在 Logic 的 Action 手动调用 this.validate
进行校验,势必比较麻烦。可以通过将校验规则赋值给 this.rules
属性进行自动校验。如:
export default class extends think.logic.base {
indexAction(){
this.rules = {
doc: "string|default:index",
version: "string|in:1.2,2.0|default:2.0"
}
}
}
将校验规则赋值给 this.rules
属性后,会在这个 Action 执行完成后自动校验,如果有错误则直接输出 JSON 格式的错误信息。自动校验是通过魔术方法 __after
来完成的。
必填项。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'required' //name 的值必填
}
}
}
当另一个项的值为某些值其中一项时,该项必填。如:
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'requiredIf:email,admin@example.com,admin1@example.com'
}
}
}
当 email
的值为 admin@example.com
,admin1@example.com
等其中一项时, name
的值必填。
当另一个项的值不在某些值中时,该项必填。如:
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'requiredNotIf:email,admin@example.com,admin1@example.com'
}
}
}
当 email
的值不为 admin@example.com
,admin1@example.com
等其中一项时, name
的值必填。
当其他几项有一项值存在时,该项必填。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'requiredWith:email,title'
}
}
}
当 email
, title
等项有一项值存在时,name
的值必填。
当其他几项值都存在时,该项必填。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'requiredWithAll:email,title'
}
}
}
当 email
, title
等项值都存在时,name
的值必填。
当其他几项有一项值不存在时,该项必填。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'requiredWithout:email,title'
}
}
}
当 email
, title
等项其中有一项值不存在时,name
的值必填。
当其他几项值都不存在时,该项必填。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'requiredWithoutAll:email,title'
}
}
}
当 email
, title
等项值都不存在时,name
的值必填。
值需要包含某个特定的值。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'contains:thinkjs' //需要包含字符串 thinkjs。
}
}
}
和另一项的值相等。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'equals:firstname'
}
}
}
name
的值需要和 firstname
的值相等。
和另一项的值不等。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'different:firstname'
}
}
}
name
的值不能和 firstname
的值相等。
值需要在一个日期之后,默认为需要在当前日期之前。
export default class extends think.logic.base {
indexAction(){
let rules = {
start_time: 'before', //需要在当前日期之前。
start_time1: 'before:2015/10/12 10:10:10' //需要在 2015/10/12 10:10:10 之前。
}
}
}
值需要在一个日期之后,默认为需要在当前日期之后。
export default class extends think.logic.base {
indexAction(){
let rules = {
end_time: 'after', //需要在当前日期之后。
end_time1: 'after:2015/10/10' //需要在 2015/10/10 之后。
}
}
}
值只能是 [a-zA-Z] 组成。
export default class extends think.logic.base {
indexAction(){
let rules = {
en_name: 'alpha'
}
}
}
en_name
的值只能是 [a-zA-Z] 组成。
值只能是 [a-zA-Z_] 组成。
值只能是 [a-zA-Z0-9] 组成。
值只能是 [a-zA-Z0-9_] 组成。
值只能是 ascii 字符组成。
值必须是 base64 编码。
字节长度需要在一个区间内。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'byteLength:10' //字节长度不能小于 10
name1: 'byteLength:10,100' //字节长度需要在 10 - 100 之间
}
}
}
需要是信用卡数字。
需要是货币。
需要是个日期。
需要是个小数。
需要被一个数整除。
export default class extends think.logic.base {
indexAction(){
let rules = {
count: 'divisibleBy:3' //可以被 3 整除
}
}
}
需要是个 email 格式。
需要是个合格的域名。
需要是个浮点数。
export default class extends think.logic.base {
indexAction(){
let rules = {
money: 'float' //需要是个浮点数
money1: 'float:3.2' //需要是个浮点数,且最小值为 3.2
money2: 'float:3.2,10.5' //需要是个浮点数,且最小值为 3.2,最大值为 10.5
}
}
}
包含宽字节字符。
包含半字节字符。
需要是个十六进制颜色值。
需要是十六进制。
需要是 ip 格式。
需要是 ip4 格式。
需要是 ip6 格式。
需要是图书编码。
需要是证券识别编码。
需要是 iso8601 日期格式。
在某些值中。
export default class extends think.logic.base {
indexAction(){
let rules = {
version: 'in:1.2,2.0' //需要是 1.2,2.0 其中一个
}
}
}
不能在某些值中。
export default class extends think.logic.base {
indexAction(){
let rules = {
version: 'noin:1.2,2.0' //不能是 1.2,2.0 其中一个
}
}
}
需要是 int 型。
export default class extends think.logic.base {
indexAction(){
let rules = {
value: 'int' //需要是 int 型
value1: 'int:1' //不能小于1
value2: 'int:10,100' //需要在 10 - 100 之间
}
}
}
不能小于某个值。
export default class extends think.logic.base {
indexAction(){
let rules = {
value: 'min:10' //不能小于10
}
}
}
不能大于某个值。
export default class extends think.logic.base {
indexAction(){
let rules = {
value: 'max:10' //不能大于10
}
}
}
长度需要在某个范围。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'length:10' //长度不能小于10
name1: 'length:10,100' //长度需要在 10 - 100 之间
}
}
}
长度不能小于最小长度。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'minLength:10' //长度不能小于10
}
}
}
长度不能大于最大长度。
export default class extends think.logic.base {
indexAction(){
let rules = {
name: 'maxLength:10' //长度不能大于10
}
}
}
需要都是小写字母。
需要都是大小字母。
需要手机号。
export default class extends think.logic.base {
indexAction(){
let rules = {
mobile: 'mobile:zh-cn' //必须为中国的手机号
}
}
}
是 MongoDB 的 ObjectID。
包含多字节字符。
是个 url。
数据库查询 order,如:name DESC。
数据库查询的字段,如:name,title。
上传的文件是否是个图片。
以某些字符打头。
以某些字符结束。
值为字符串。
值为数组。
值为布尔类型。
值为对象。
如果默认支持的校验类型不能满足需求,可以通过 think.validate
方法对校验类型进行扩展。如:
// src/common/bootstrap/validate.js
think.validate('validate_name', (value, ...args) => {
//需要返回 true 或者 false
//true 表示校验成功,false 表示校验失败
})
上面注册了一个名为 validate_name
的校验类型,这样在 Logic 里就可以直接使用该校验类型了。
如果要解析后面的 args
,如:该字段值跟其他字段值进行比较,这时拿到的参数是其他字段名称,但比较的时候肯定需要拿到这个字段值,所以需要将字段名称解析为对应的字段值。
可以通过注册一个解析参数函数来完成。如:上面的校验类型名称为 validate_name
,那么对应的解析参数的名称必须为 _validate_name
,即:_
+ 校验类型
。
think.validate('_validate_name', (args, data) => {
let arg0 = args[0];
args[0] = data[arg0].value; //将第一个参数字段名称解析为对应的参数值
return args;
})