reach ($fields as $key) { if (isset($data[$key])) { $map[$key] = $data[$key]; } } } elseif (strpos($key, '=')) { parse_str($key, $map); } elseif (isset($data[$field])) { $map[$key] = $data[$field]; } else { $map = []; } $pk = isset($rule[3]) ? $rule[3] : $db->getPk(); if (is_string($pk)) { if (isset($rule[2])) { $map[$pk] = ['neq', $rule[2]]; } elseif (isset($data[$pk])) { $map[$pk] = ['neq', $data[$pk]]; } } if ($db->where($map)->field($pk)->find()) { return false; } return true; } /** * 使用行为类验证 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return mixed */ protected function behavior($value, $rule, $data) { return Hook::exec($rule, '', $data); } /** * 使用filter_var方式验证 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function filter($value, $rule) { if (is_string($rule) && strpos($rule, ',')) { list($rule, $param) = explode(',', $rule); } elseif (is_array($rule)) { $param = isset($rule[1]) ? $rule[1] : null; $rule = $rule[0]; } else { $param = null; } return false !== filter_var($value, is_int($rule) ? $rule : filter_id($rule), $param); } /** * 验证某个字段等于某个值的时候必须 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function requireIf($value, $rule, $data) { list($field, $val) = explode(',', $rule); if ($this->getDataValue($data, $field) == $val) { return !empty($value) || '0' == $value; } else { return true; } } /** * 通过回调方法验证某个字段是否必须 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function requireCallback($value, $rule, $data) { $result = call_user_func_array($rule, [$value, $data]); if ($result) { return !empty($value) || '0' == $value; } else { return true; } } /** * 验证某个字段有值的情况下必须 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function requireWith($value, $rule, $data) { $val = $this->getDataValue($data, $rule); if (!empty($val)) { return !empty($value) || '0' == $value; } else { return true; } } /** * 验证是否在范围内 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function in($value, $rule) { return in_array($value, is_array($rule) ? $rule : explode(',', $rule)); } /** * 验证是否不在某个范围 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function notIn($value, $rule) { return !in_array($value, is_array($rule) ? $rule : explode(',', $rule)); } /** * between验证数据 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function between($value, $rule) { if (is_string($rule)) { $rule = explode(',', $rule); } list($min, $max) = $rule; return $value >= $min && $value <= $max; } /** * 使用notbetween验证数据 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function notBetween($value, $rule) { if (is_string($rule)) { $rule = explode(',', $rule); } list($min, $max) = $rule; return $value < $min || $value > $max; } /** * 验证数据长度 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function length($value, $rule) { if (is_array($value)) { $length = count($value); } elseif ($value instanceof File) { $length = $value->getSize(); } else { $length = mb_strlen((string) $value); } if (strpos($rule, ',')) { // 长度区间 list($min, $max) = explode(',', $rule); return $length >= $min && $length <= $max; } else { // 指定长度 return $length == $rule; } } /** * 验证数据最大长度 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function max($value, $rule) { if (is_array($value)) { $length = count($value); } elseif ($value instanceof File) { $length = $value->getSize(); } else { $length = mb_strlen((string) $value); } return $length <= $rule; } /** * 验证数据最小长度 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function min($value, $rule) { if (is_array($value)) { $length = count($value); } elseif ($value instanceof File) { $length = $value->getSize(); } else { $length = mb_strlen((string) $value); } return $length >= $rule; } /** * 验证日期 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function after($value, $rule, $data) { return strtotime($value) >= strtotime($rule); } /** * 验证日期 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function before($value, $rule, $data) { return strtotime($value) <= strtotime($rule); } /** * 验证日期字段 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function afterWith($value, $rule, $data) { $rule = $this->getDataValue($data, $rule); return !is_null($rule) && strtotime($value) >= strtotime($rule); } /** * 验证日期字段 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function beforeWith($value, $rule, $data) { $rule = $this->getDataValue($data, $rule); return !is_null($rule) && strtotime($value) <= strtotime($rule); } /** * 验证有效期 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @return bool */ protected function expire($value, $rule) { if (is_string($rule)) { $rule = explode(',', $rule); } list($start, $end) = $rule; if (!is_numeric($start)) { $start = strtotime($start); } if (!is_numeric($end)) { $end = strtotime($end); } return $_SERVER['REQUEST_TIME'] >= $start && $_SERVER['REQUEST_TIME'] <= $end; } /** * 验证IP许可 * @access protected * @param string $value 字段值 * @param mixed $rule 验证规则 * @return mixed */ protected function allowIp($value, $rule) { return in_array($_SERVER['REMOTE_ADDR'], is_array($rule) ? $rule : explode(',', $rule)); } /** * 验证IP禁用 * @access protected * @param string $value 字段值 * @param mixed $rule 验证规则 * @return mixed */ protected function denyIp($value, $rule) { return !in_array($_SERVER['REMOTE_ADDR'], is_array($rule) ? $rule : explode(',', $rule)); } /** * 使用正则验证数据 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 正则规则或者预定义正则名 * @return mixed */ protected function regex($value, $rule) { if (isset($this->regex[$rule])) { $rule = $this->regex[$rule]; } if (0 !== strpos($rule, '/') && !preg_match('/\/[imsU]{0,4}$/', $rule)) { // 不是正则表达式则两端补上/ $rule = '/^' . $rule . '$/'; } return is_scalar($value) && 1 === preg_match($rule, (string) $value); } /** * 验证表单令牌 * @access protected * @param mixed $value 字段值 * @param mixed $rule 验证规则 * @param array $data 数据 * @return bool */ protected function token($value, $rule, $data) { $rule = !empty($rule) ? $rule : '__token__'; if (!isset($data[$rule]) || !Session::has($rule)) { // 令牌数据无效 return false; } // 令牌验证 if (isset($data[$rule]) && Session::get($rule) === $data[$rule]) { // 防止重复提交 Session::delete($rule); // 验证完成销毁session return true; } // 开启TOKEN重置 Session::delete($rule); return false; } // 获取错误信息 public function getError() { return $this->error; } /** * 获取数据值 * @access protected * @param array $data 数据 * @param string $key 数据标识 支持二维 * @return mixed */ protected function getDataValue($data, $key) { if (is_numeric($key)) { $value = $key; } elseif (strpos($key, '.')) { // 支持二维数组验证 list($name1, $name2) = explode('.', $key); $value = isset($data[$name1][$name2]) ? $data[$name1][$name2] : null; } else { $value = isset($data[$key]) ? $data[$key] : null; } return $value; } /** * 获取验证规则的错误提示信息 * @access protected * @param string $attribute 字段英文名 * @param string $title 字段描述名 * @param string $type 验证规则名称 * @param mixed $rule 验证规则数据 * @return string */ protected function getRuleMsg($attribute, $title, $type, $rule) { if (isset($this->message[$attribute . '.' . $type])) { $msg = $this->message[$attribute . '.' . $type]; } elseif (isset($this->message[$attribute][$type])) { $msg = $this->message[$attribute][$type]; } elseif (isset($this->message[$attribute])) { $msg = $this->message[$attribute]; } elseif (isset(self::$typeMsg[$type])) { $msg = self::$typeMsg[$type]; } elseif (0 === strpos($type, 'require')) { $msg = self::$typeMsg['require']; } else { $msg = $title . Lang::get('not conform to the rules'); } if (is_string($msg) && 0 === strpos($msg, '{%')) { $msg = Lang::get(substr($msg, 2, -1)); } elseif (Lang::has($msg)) { $msg = Lang::get($msg); } if (is_string($msg) && is_scalar($rule) && false !== strpos($msg, ':')) { // 变量替换 if (is_string($rule) && strpos($rule, ',')) { $array = array_pad(explode(',', $rule), 3, ''); } else { $array = array_pad([], 3, ''); } $msg = str_replace( [':attribute', ':rule', ':1', ':2', ':3'], [$title, (string) $rule, $array[0], $array[1], $array[2]], $msg); } return $msg; } /** * 获取数据验证的场景 * @access protected * @param string $scene 验证场景 * @return array */ protected function getScene($scene = '') { if (empty($scene)) { // 读取指定场景 $scene = $this->currentScene; } if (!empty($scene) && isset($this->scene[$scene])) { // 如果设置了验证适用场景 $scene = $this->scene[$scene]; if (is_string($scene)) { $scene = explode(',', $scene); } } else { $scene = []; } return $scene; } public static function __callStatic($method, $params) { $class = self::make(); if (method_exists($class, $method)) { return call_user_func_array([$class, $method], $params); } else { throw new \BadMethodCallException('method not exists:' . __CLASS__ . '->' . $method); } } }