数据校验

最后更新于:2022-04-01 04:34:33

# 数据校验 [TOC=2,3] 当在 Action 里处理用户的请求时,经常要先获取用户提交过来的数据,然后对其校验,如果校验没问题后才能进行后续的操作。当参数校验完成后,有时候还要进行权限判断,等这些都判断无误后才能进行真正的逻辑处理。如果将这些代码都放在一个 Action 里,势必让 Action 的代码非常复杂且冗长。 为了解决这个问题, ThinkJS 在控制器前面增加了一层 `Logic`,Logic 里的 Action 和控制器里的 Action 一一对应,系统在调用控制器里的 Action 之前会自动调用 Logic 里的 Action。 ## Logic 层 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` 来完成的。 ## 支持的校验类型 ### required 必填项。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "required" //name 的值必填 } } } ~~~ ### requiredIf 当另一个项的值为某些值其中一项时,该项必填。如: ~~~ 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` 的值必填。 ### requiredNotIf 当另一个项的值不在某些值中时,该项必填。如: ~~~ 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` 的值必填。 ### requiredWith 当其他几项有一项值存在时,该项必填。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "requiredWith:email,title" } } } ~~~ 当 `email`, `title` 等项有一项值存在时,`name` 的值必填。 ### requiredWithAll 当其他几项值都存在时,该项必填。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "requiredWithAll:email,title" } } } ~~~ 当 `email`, `title` 等项值都存在时,`name` 的值必填。 ### requiredWithout 当其他几项有一项值不存在时,该项必填。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "requiredWithout:email,title" } } } ~~~ 当 `email`, `title` 等项其中有一项值不存在时,`name` 的值必填。 ### requiredWithoutAll 当其他几项值都存在时,该项必填。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "requiredWithoutAll:email,title" } } } ~~~ 当 `email`, `title` 等项值都不存在时,`name` 的值必填。 ### contains 值需要包含某个特定的值。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "contains:thinkjs" //需要包含字符串 thinkjs。 } } } ~~~ ### equals 和另一项的值相等。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "equals:firstname" } } } ~~~ `name` 的值需要和 `firstname` 的值相等。 ### different 和另一项的值不等。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "different:firstname" } } } ~~~ `name` 的值不能和 `firstname` 的值相等。 ### before 值需要在一个日期之后,默认为需要在当前日期之前。 ~~~ 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 之前。 } } } ~~~ ### after 值需要在一个日期之后,默认为需要在当前日期之后。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { end_time: "after", //需要在当前日期之后。 end_time1: "after:2015/10/10" //需要在 2015/10/10 之后。 } } } ~~~ ### alpha 值只能是 [a-zA-Z] 组成。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { en_name: "alpha" } } } ~~~ `en_name` 的值只能是 [a-zA-Z] 组成。 ### alphaDash 值只能是 [a-zA-Z_] 组成。 ### alphaNumeric 值只能是 [a-zA-Z0-9] 组成。 ### alphaNumericDash 值只能是 [a-zA-Z0-9_] 组成。 ### ascii 值只能是 ascii 字符组成。 ### base64 值必须是 base64 编码。 ### byteLength 字节长度需要在一个区间内。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "byteLength:10" //字节长度不能小于 10 name1: "byteLength:10,100" //字节长度需要在 10 - 100 之间 } } } ~~~ ### creditcard 需要是信用卡数字。 ### currency 需要是货币。 ### date 需要是个日期。 ### decimal 需要是个小数。 ### divisibleBy 需要被一个数整除。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { count: "divisibleBy:3" //可以被 3 整除 } } } ~~~ ### email 需要是个 email 格式。 ### fqdn 需要是个合格的域名。 ### float 需要是个浮点数。 ~~~ 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 } } } ~~~ ### fullWidth 包含宽字节字符。 ### halfWidth 包含半字节字符。 ### hexColor 需要是个十六进制颜色值。 ### hex 需要是十六进制。 ### ip 需要是 ip 格式。 ### ip4 需要是 ip4 格式。 ### ip6 需要是 ip6 格式。 ### isbn 需要是图书编码。 #### isin 需要是证券识别编码。 ### iso8601 需要是 iso8601 日期格式。 ### in 在某些值中。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { version: "in:1.2,2.0" //需要是 1.2,2.0 其中一个 } } } ~~~ ### noin 不能在某些值中。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { version: "noin:1.2,2.0" //不能是 1.2,2.0 其中一个 } } } ~~~ ### int 需要是 int 型。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { value: "int" //需要是 int 型 value1: "int:1" //不能小于1 value2: "int:10,100" //需要在 10 - 100 之间 } } } ~~~ ### min 不能小于某个值。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { value: "min:10" //不能小于10 } } } ~~~ ### max 不能大于某个值。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { value: "max:10" //不能大于10 } } } ~~~ ### length 长度需要在某个范围。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "length:10" //长度不能小于10 name1: "length:10,100" //长度需要在 10 - 100 之间 } } } ~~~ ### minLength 长度不能小于最小长度。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "minLength:10" //长度不能小于10 } } } ~~~ ### maxLength 长度不能大于最大长度。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { name: "maxLength:10" //长度不能大于10 } } } ~~~ ### lowercase 需要都是小写字母。 ### uppercase 需要都是大小字母。 ### mobile 需要手机号。 ~~~ export default class extends think.logic.base { indexAction(){ let rules = { mobile: "mobile:zh-CN" //必须为中国的手机号 } } } ~~~ ### mongoId 是 MongoDB 的 ObjectID。 ### multibyte 包含多字节字符。 ### url 是个 url。 ### order 数据库查询 order,如:name DESC。 ### field 数据库查询的字段,如:name,title。 ### image 上传的文件是否是个图片。 ### startWith 以某些字符打头。 ### endWith 以某些字符结束。 ### string 值为字符串。 ### array 值为数组。 ### boolean 值为布尔类型。 ### object 值为对象。 ## 扩展校验类型 如果默认支持的校验类型不能满足需求,可以通过 `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; }) ~~~
';