[更新]ComposerUpdate,修正路由规则

This commit is contained in:
Anyon 2018-06-19 16:10:09 +08:00
parent 77ee26ad45
commit 5110f84d1e
31 changed files with 548 additions and 189 deletions

View File

@ -17,7 +17,7 @@ use think\facade\Route;
use think\Request; use think\Request;
/* 注册微信端路由支持 */ /* 注册微信端路由支持 */
Route::rule('wx', function (Request $request) { Route::rule('wx<_?>', function (Request $request) {
$params = explode('-', $request->path()); $params = explode('-', $request->path());
array_shift($params); array_shift($params);
$controller = array_shift($params) ?: config('app.default_controller'); $controller = array_shift($params) ?: config('app.default_controller');

View File

@ -18,7 +18,7 @@
"require": { "require": {
"php": ">=5.6.0", "php": ">=5.6.0",
"endroid/qr-code": "^1.9", "endroid/qr-code": "^1.9",
"topthink/framework": "5.1.16", "topthink/framework": "5.1.*",
"zoujingli/ip2region": "dev-master", "zoujingli/ip2region": "dev-master",
"aliyuncs/oss-sdk-php": "^2.2", "aliyuncs/oss-sdk-php": "^2.2",
"zoujingli/weopen-developer": "dev-master", "zoujingli/weopen-developer": "dev-master",

View File

@ -20,7 +20,7 @@ use think\route\Dispatch;
*/ */
class App extends Container class App extends Container
{ {
const VERSION = '5.1.16'; const VERSION = '5.1.17';
/** /**
* 当前模块路径 * 当前模块路径
@ -333,7 +333,6 @@ class App extends Container
// 对容器中的对象实例进行配置更新 // 对容器中的对象实例进行配置更新
$this->containerConfigUpdate($module); $this->containerConfigUpdate($module);
} }
} }
protected function containerConfigUpdate($module) protected function containerConfigUpdate($module)
@ -427,30 +426,7 @@ class App extends Container
} }
$this->middleware->add(function (Request $request, $next) use ($dispatch, $data) { $this->middleware->add(function (Request $request, $next) use ($dispatch, $data) {
if (is_null($data)) { return is_null($data) ? $dispatch->run() : $data;
try {
// 执行调度
$data = $dispatch->run();
} catch (HttpResponseException $exception) {
$data = $exception->getResponse();
}
}
// 输出数据到客户端
if ($data instanceof Response) {
$response = $data;
} elseif (!is_null($data)) {
// 默认自动识别响应输出类型
$isAjax = $request->isAjax();
$type = $isAjax ? $this->config('app.default_ajax_return') : $this->config('app.default_return_type');
$response = Response::create($data, $type);
} else {
$data = ob_get_clean();
$status = empty($data) ? 204 : 200;
$response = Response::create($data, '', $status);
}
return $response;
}); });
$response = $this->middleware->dispatch($this->request); $response = $this->middleware->dispatch($this->request);

View File

@ -43,11 +43,17 @@ class Controller
protected $batchValidate = false; protected $batchValidate = false;
/** /**
* 前置操作方法列表 * 前置操作方法列表(即将废弃)
* @var array $beforeActionList * @var array $beforeActionList
*/ */
protected $beforeActionList = []; protected $beforeActionList = [];
/**
* 控制器中间件
* @var array
*/
protected $middleware = [];
/** /**
* 构造方法 * 构造方法
* @access public * @access public
@ -61,7 +67,24 @@ class Controller
// 控制器初始化 // 控制器初始化
$this->initialize(); $this->initialize();
// 前置操作方法 // 控制器中间件
if ($this->middleware) {
foreach ($this->middleware as $key => $val) {
if (!is_int($key)) {
if (isset($val['only']) && !in_array($this->request->action(), $val['only'])) {
continue;
} elseif (isset($val['except']) && in_array($this->request->action(), $val['except'])) {
continue;
} else {
$val = $key;
}
}
$this->app['middleware']->controller($val);
}
}
// 前置操作方法 即将废弃
foreach ((array) $this->beforeActionList as $method => $options) { foreach ((array) $this->beforeActionList as $method => $options) {
is_numeric($method) ? is_numeric($method) ?
$this->beforeAction($options) : $this->beforeAction($options) :

View File

@ -117,7 +117,7 @@ class Log implements LoggerInterface
return; return;
} }
if (is_string($msg)) { if (is_string($msg) && !empty($context)) {
$replace = []; $replace = [];
foreach ($context as $key => $val) { foreach ($context as $key => $val) {
$replace['{' . $key . '}'] = $val; $replace['{' . $key . '}'] = $val;

View File

@ -39,62 +39,95 @@ class Middleware
$this->config = array_merge($this->config, $config); $this->config = array_merge($this->config, $config);
} }
public function import(array $middlewares = []) /**
* 导入中间件
* @access public
* @param array $middlewares
* @param string $type 中间件类型
*/
public function import(array $middlewares = [], $type = 'route')
{ {
foreach ($middlewares as $middleware) { foreach ($middlewares as $middleware) {
$this->add($middleware); $this->add($middleware, $type);
} }
} }
/** /**
* {@inheritdoc} * 注册中间件
* @access public
* @param mixed $middleware
* @param string $type 中间件类型
*/ */
public function add($middleware) public function add($middleware, $type = 'route')
{ {
if (is_null($middleware)) { if (is_null($middleware)) {
return; return;
} }
$middleware = $this->buildMiddleware($middleware); $middleware = $this->buildMiddleware($middleware, $type);
if ($middleware) { if ($middleware) {
$this->queue[] = $middleware; $this->queue[$type][] = $middleware;
} }
} }
/** /**
* {@inheritdoc} * 注册控制器中间件
* @access public
* @param mixed $middleware
*/ */
public function unshift($middleware) public function controller($middleware)
{
return $this->add($middleware, 'controller');
}
/**
* 移除中间件
* @access public
* @param mixed $middleware
* @param string $type 中间件类型
*/
public function unshift($middleware, $type = 'route')
{ {
if (is_null($middleware)) { if (is_null($middleware)) {
return; return;
} }
$middleware = $this->buildMiddleware($middleware); $middleware = $this->buildMiddleware($middleware, $type);
if ($middleware) { if ($middleware) {
array_unshift($this->queue, $middleware); array_unshift($this->queue[$type], $middleware);
} }
} }
/** /**
* {@inheritdoc} * 获取注册的中间件
* @access public
* @param string $type 中间件类型
*/ */
public function all() public function all($type = 'route')
{ {
return $this->queue; return $this->queue[$type] ?: [];
} }
/** /**
* {@inheritdoc} * 中间件调度
* @access public
* @param Request $request
* @param string $type 中间件类型
*/ */
public function dispatch(Request $request) public function dispatch(Request $request, $type = 'route')
{ {
return call_user_func($this->resolve(), $request); return call_user_func($this->resolve($type), $request);
} }
protected function buildMiddleware($middleware) /**
* 解析中间件
* @access protected
* @param mixed $middleware
* @param string $type 中间件类型
*/
protected function buildMiddleware($middleware, $type = 'route')
{ {
if (is_array($middleware)) { if (is_array($middleware)) {
list($middleware, $param) = $middleware; list($middleware, $param) = $middleware;
@ -117,7 +150,7 @@ class Middleware
} }
if (is_array($middleware)) { if (is_array($middleware)) {
return $this->import($middleware); return $this->import($middleware, $type);
} }
if (strpos($middleware, ':')) { if (strpos($middleware, ':')) {
@ -127,10 +160,11 @@ class Middleware
return [[$this->app->make($middleware), 'handle'], isset($param) ? $param : null]; return [[$this->app->make($middleware), 'handle'], isset($param) ? $param : null];
} }
protected function resolve() protected function resolve($type = 'route')
{ {
return function (Request $request) { return function (Request $request) use ($type) {
$middleware = array_shift($this->queue);
$middleware = array_shift($this->queue[$type]);
if (null === $middleware) { if (null === $middleware) {
throw new InvalidArgumentException('The queue was exhausted, with no response returned'); throw new InvalidArgumentException('The queue was exhausted, with no response returned');
@ -139,7 +173,7 @@ class Middleware
list($call, $param) = $middleware; list($call, $param) = $middleware;
try { try {
$response = call_user_func_array($call, [$request, $this->resolve(), $param]); $response = call_user_func_array($call, [$request, $this->resolve($type), $param]);
} catch (HttpResponseException $exception) { } catch (HttpResponseException $exception) {
$response = $exception->getResponse(); $response = $exception->getResponse();
} }

View File

@ -236,7 +236,7 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
// 设置当前模型 确保查询返回模型对象 // 设置当前模型 确保查询返回模型对象
$query = Db::connect($this->connection, false, $this->query); $query = Db::connect($this->connection, false, $this->query);
$query->model($this) $query->model($this)
->json($this->json) ->json($this->json, $this->jsonAssoc)
->setJsonFieldType($this->jsonType); ->setJsonFieldType($this->jsonType);
if (isset(static::$readMaster['*']) || isset(static::$readMaster[static::class])) { if (isset(static::$readMaster['*']) || isset(static::$readMaster[static::class])) {
@ -284,7 +284,7 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
$query = $this->buildQuery(); $query = $this->buildQuery();
// 软删除 // 软删除
if (method_exists($this, 'withNoTrashed')) { if (property_exists($this, 'withTrashed') && !$this->withTrashed) {
$this->withNoTrashed($query); $this->withNoTrashed($query);
} }
@ -750,8 +750,6 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
*/ */
public function saveAll($dataSet, $replace = true) public function saveAll($dataSet, $replace = true)
{ {
$result = [];
$db = $this->db(false); $db = $this->db(false);
$db->startTrans(); $db->startTrans();
@ -762,6 +760,8 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
$auto = true; $auto = true;
} }
$result = [];
foreach ($dataSet as $key => $data) { foreach ($dataSet as $key => $data) {
if ($this->exists || (!empty($auto) && isset($data[$pk]))) { if ($this->exists || (!empty($auto) && isset($data[$pk]))) {
$result[$key] = self::update($data, [], $this->field); $result[$key] = self::update($data, [], $this->field);

View File

@ -11,6 +11,9 @@
namespace think; namespace think;
use think\facade\Cookie;
use think\facade\Session;
class Request class Request
{ {
/** /**
@ -310,7 +313,6 @@ class Request
{ {
$request = new static($config->pull('app')); $request = new static($config->pull('app'));
$request->cookie = $app['cookie']->get();
$request->server = $_SERVER; $request->server = $_SERVER;
$request->env = $app['env']->get(); $request->env = $app['env']->get();
@ -761,18 +763,19 @@ class Request
/** /**
* 当前的请求类型 * 当前的请求类型
* @access public * @access public
* @param bool $method true 获取原始请求类型 * @param bool $origin 是否获取原始请求类型
* @return string * @return string
*/ */
public function method($method = false) public function method($origin = false)
{ {
if (true === $method) { if ($origin) {
// 获取原始请求类型 // 获取原始请求类型
return $this->isCli() ? 'GET' : $this->server('REQUEST_METHOD'); return $this->isCli() ? 'GET' : $this->server('REQUEST_METHOD');
} elseif (!$this->method) { } elseif (!$this->method) {
if (isset($_POST[$this->config['var_method']])) { if (isset($_POST[$this->config['var_method']])) {
$this->method = strtoupper($_POST[$this->config['var_method']]); $this->method = strtoupper($_POST[$this->config['var_method']]);
$this->{$this->method}($_POST); $method = strtolower($this->method);
$this->{$method} = $_POST;
} elseif ($this->server('HTTP_X_HTTP_METHOD_OVERRIDE')) { } elseif ($this->server('HTTP_X_HTTP_METHOD_OVERRIDE')) {
$this->method = strtoupper($this->server('HTTP_X_HTTP_METHOD_OVERRIDE')); $this->method = strtoupper($this->server('HTTP_X_HTTP_METHOD_OVERRIDE'));
} else { } else {
@ -910,6 +913,7 @@ class Request
// 获取包含文件上传信息的数组 // 获取包含文件上传信息的数组
$file = $this->file(); $file = $this->file();
$data = is_array($file) ? array_merge($this->param, $file) : $this->param; $data = is_array($file) ? array_merge($this->param, $file) : $this->param;
return $this->input($data, '', $default, $filter); return $this->input($data, '', $default, $filter);
} }
@ -1060,14 +1064,16 @@ class Request
public function session($name = '', $default = null) public function session($name = '', $default = null)
{ {
if (empty($this->session)) { if (empty($this->session)) {
$this->session = facade\Session::get(); $this->session = Session::get();
} }
if ('' === $name) { if ('' === $name) {
return $this->session; return $this->session;
} }
return isset($this->session[$name]) ? $this->session[$name] : $default; $data = $this->getData($this->session, $name);
return is_null($data) ? $default : $data;
} }
/** /**
@ -1080,8 +1086,12 @@ class Request
*/ */
public function cookie($name = '', $default = null, $filter = '') public function cookie($name = '', $default = null, $filter = '')
{ {
if (empty($this->cookie)) {
$this->cookie = Cookie::get();
}
if (!empty($name)) { if (!empty($name)) {
$data = isset($this->cookie[$name]) ? $this->cookie[$name] : $default; $data = Cookie::has($name) ? Cookie::get($name) : $default;
} else { } else {
$data = $this->cookie; $data = $this->cookie;
} }
@ -1272,14 +1282,10 @@ class Request
list($name, $type) = explode('/', $name); list($name, $type) = explode('/', $name);
} }
// 按.拆分成多维数组进行判断 $data = $this->getData($data, $name);
foreach (explode('.', $name) as $val) {
if (isset($data[$val])) { if (is_null($data)) {
$data = $data[$val]; return $default;
} else {
// 无输入数据,返回默认值
return $default;
}
} }
if (is_object($data)) { if (is_object($data)) {
@ -1305,6 +1311,26 @@ class Request
return $data; return $data;
} }
/**
* 获取数据
* @access public
* @param array $data 数据源
* @param string|false $name 字段名
* @return mixed
*/
protected function getData(array $data, $name)
{
foreach (explode('.', $name) as $val) {
if (isset($data[$val])) {
$data = $data[$val];
} else {
return;
}
}
return $data;
}
/** /**
* 设置或获取当前的过滤规则 * 设置或获取当前的过滤规则
* @access public * @access public

View File

@ -390,6 +390,33 @@ class Route
return $this->app['rule_name']->get($name, $domain); return $this->app['rule_name']->get($name, $domain);
} }
/**
* 读取路由
* @access public
* @param string $rule 路由规则
* @param string $domain 域名
* @return array
*/
public function getRule($rule, $domain = null)
{
if (is_null($domain)) {
$domain = $this->domain;
}
return $this->app['rule_name']->getRule($rule, $domain);
}
/**
* 读取路由
* @access public
* @param string $domain 域名
* @return array
*/
public function getRuleList($domain = null)
{
return $this->app['rule_name']->getRuleList($domain);
}
/** /**
* 批量导入路由标识 * 批量导入路由标识
* @access public * @access public

View File

@ -478,16 +478,21 @@ class Session
public function has($name, $prefix = null) public function has($name, $prefix = null)
{ {
empty($this->init) && $this->boot(); empty($this->init) && $this->boot();
$prefix = !is_null($prefix) ? $prefix : $this->prefix; $prefix = !is_null($prefix) ? $prefix : $this->prefix;
$value = $prefix ? (!empty($_SESSION[$prefix]) ? $_SESSION[$prefix] : []) : $_SESSION;
if (strpos($name, '.')) { $name = explode('.', $name);
// 支持数组
list($name1, $name2) = explode('.', $name);
return $prefix ? isset($_SESSION[$prefix][$name1][$name2]) : isset($_SESSION[$name1][$name2]); foreach ($name as $val) {
} else { if (!isset($value[$val])) {
return $prefix ? isset($_SESSION[$prefix][$name]) : isset($_SESSION[$name]); return false;
} else {
$value = $value[$val];
}
} }
return true;
} }
/** /**

View File

@ -112,23 +112,6 @@ class Validate
*/ */
protected $currentScene = null; protected $currentScene = null;
/**
* 内置正则验证规则
* @var array
*/
protected $regex = [
'alpha' => '/^[A-Za-z]+$/',
'alphaNum' => '/^[A-Za-z0-9]+$/',
'alphaDash' => '/^[A-Za-z0-9\-\_]+$/',
'chs' => '/^[\x{4e00}-\x{9fa5}]+$/u',
'chsAlpha' => '/^[\x{4e00}-\x{9fa5}a-zA-Z]+$/u',
'chsAlphaNum' => '/^[\x{4e00}-\x{9fa5}a-zA-Z0-9]+$/u',
'chsDash' => '/^[\x{4e00}-\x{9fa5}a-zA-Z0-9\_\-]+$/u',
'mobile' => '/^1[3-9][0-9]\d{8}$/',
'idCard' => '/(^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$)|(^[1-9]\d{5}\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{2}$)/',
'zip' => '/\d{6}/',
];
/** /**
* Filter_var 规则 * Filter_var 规则
* @var array * @var array
@ -142,6 +125,21 @@ class Validate
'float' => FILTER_VALIDATE_FLOAT, 'float' => FILTER_VALIDATE_FLOAT,
]; ];
/**
* 内置正则验证规则
* @var array
*/
protected $regex = [
'alphaDash' => '/^[A-Za-z0-9\-\_]+$/',
'chs' => '/^[\x{4e00}-\x{9fa5}]+$/u',
'chsAlpha' => '/^[\x{4e00}-\x{9fa5}a-zA-Z]+$/u',
'chsAlphaNum' => '/^[\x{4e00}-\x{9fa5}a-zA-Z0-9]+$/u',
'chsDash' => '/^[\x{4e00}-\x{9fa5}a-zA-Z0-9\_\-]+$/u',
'mobile' => '/^1[3-9][0-9]\d{8}$/',
'idCard' => '/(^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$)|(^[1-9]\d{5}\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{2}$)/',
'zip' => '/\d{6}/',
];
/** /**
* 验证场景定义 * 验证场景定义
* @var array * @var array
@ -751,6 +749,9 @@ class Validate
case 'number': case 'number':
$result = ctype_digit((string) $value); $result = ctype_digit((string) $value);
break; break;
case 'alphaNum':
$result = ctype_alnum($value);
break;
case 'array': case 'array':
// 是否为数组 // 是否为数组
$result = is_array($value); $result = is_array($value);
@ -768,6 +769,10 @@ class Validate
if (isset(self::$type[$rule])) { if (isset(self::$type[$rule])) {
// 注册的验证规则 // 注册的验证规则
$result = call_user_func_array(self::$type[$rule], [$value]); $result = call_user_func_array(self::$type[$rule], [$value]);
} elseif (function_exists('ctype_' . $rule)) {
// ctype验证规则
$ctypeFun = 'ctype_' . $rule;
$result = $ctypeFun($value);
} elseif (isset($this->filter[$rule])) { } elseif (isset($this->filter[$rule])) {
// Filter_var验证规则 // Filter_var验证规则
$result = $this->filter($value, $this->filter[$rule]); $result = $this->filter($value, $this->filter[$rule]);

View File

@ -1340,11 +1340,8 @@ abstract class Connection
if (empty($options['fetch_sql']) && !empty($options['cache'])) { if (empty($options['fetch_sql']) && !empty($options['cache'])) {
// 判断查询缓存 // 判断查询缓存
$cache = $options['cache']; $cache = $options['cache'];
$result = $this->getCacheData($query, $cache, null, $guid);
$guid = is_string($cache['key']) ? $cache['key'] : $this->getCacheKey($query, $field);
$result = Container::get('cache')->get($guid);
if (false !== $result) { if (false !== $result) {
return $result; return $result;

View File

@ -87,7 +87,7 @@ class Query
* 读取主库的表 * 读取主库的表
* @var array * @var array
*/ */
private static $readMaster = []; protected static $readMaster = [];
/** /**
* 日期查询表达式 * 日期查询表达式
@ -635,11 +635,11 @@ class Query
* COUNT查询 * COUNT查询
* @access public * @access public
* @param string $field 字段名 * @param string $field 字段名
* @return integer|string * @return float|string
*/ */
public function count($field = '*') public function count($field = '*')
{ {
if (isset($this->options['group'])) { if (!empty($this->options['group'])) {
// 支持GROUP // 支持GROUP
$options = $this->getOptions(); $options = $this->getOptions();
$subSql = $this->options($options)->field('count(' . $field . ') AS think_count')->bind($this->bind)->buildSql(); $subSql = $this->options($options)->field('count(' . $field . ') AS think_count')->bind($this->bind)->buildSql();
@ -660,7 +660,7 @@ class Query
* SUM查询 * SUM查询
* @access public * @access public
* @param string $field 字段名 * @param string $field 字段名
* @return float|int * @return float
*/ */
public function sum($field) public function sum($field)
{ {
@ -695,7 +695,7 @@ class Query
* AVG查询 * AVG查询
* @access public * @access public
* @param string $field 字段名 * @param string $field 字段名
* @return float|int * @return float
*/ */
public function avg($field) public function avg($field)
{ {
@ -2124,11 +2124,13 @@ class Query
* 设置JSON字段信息 * 设置JSON字段信息
* @access public * @access public
* @param array $json JSON字段 * @param array $json JSON字段
* @param bool $assoc 是否取出数组
* @return $this * @return $this
*/ */
public function json(array $json = []) public function json(array $json = [], $assoc = false)
{ {
$this->options['json'] = $json; $this->options['json'] = $json;
$this->options['json_assoc'] = $assoc;
return $this; return $this;
} }
@ -2247,6 +2249,32 @@ class Query
return $this->parseWhereExp($logic, $field, strtolower($op) . ' time', $range, [], true); return $this->parseWhereExp($logic, $field, strtolower($op) . ' time', $range, [], true);
} }
/**
* 查询当前时间在两个时间字段范围
* @access public
* @param string $startField 开始时间字段
* @param string $endField 结束时间字段
* @return $this
*/
public function whereBetweenTimeField($startField, $endField)
{
return $this->whereTime($startField, '<=', time())
->whereTime($endField, '>=', time());
}
/**
* 查询当前时间不在两个时间字段范围
* @access public
* @param string $startField 开始时间字段
* @param string $endField 结束时间字段
* @return $this
*/
public function whereNotBetweenTimeField($startField, $endField)
{
return $this->whereTime($startField, '>', time())
->whereTime($endField, '<', time(), 'OR');
}
/** /**
* 查询日期或者时间范围 * 查询日期或者时间范围
* @access public * @access public
@ -2897,7 +2925,7 @@ class Query
protected function resultToModel(&$result, $options = [], $resultSet = false) protected function resultToModel(&$result, $options = [], $resultSet = false)
{ {
if (!empty($options['json'])) { if (!empty($options['json'])) {
$this->jsonResult($result, $options['json']); $this->jsonResult($result, $options['json'], $options['json_assoc']);
} }
$condition = (!$resultSet && isset($options['where']['AND'])) ? $options['where']['AND'] : null; $condition = (!$resultSet && isset($options['where']['AND'])) ? $options['where']['AND'] : null;

View File

@ -13,6 +13,7 @@ namespace think\db\connector;
use PDO; use PDO;
use think\db\Connection; use think\db\Connection;
use think\db\Query;
/** /**
* Sqlsrv数据库驱动 * Sqlsrv数据库驱动
@ -125,6 +126,100 @@ class Sqlsrv extends Connection
return $info; return $info;
} }
/**
* 得到某个列的数组
* @access public
* @param Query $query 查询对象
* @param string $field 字段名 多个字段用逗号分隔
* @param string $key 索引
* @return array
*/
public function column(Query $query, $field, $key = '')
{
$options = $query->getOptions();
if (empty($options['fetch_sql']) && !empty($options['cache'])) {
// 判断查询缓存
$cache = $options['cache'];
$guid = is_string($cache['key']) ? $cache['key'] : $this->getCacheKey($query, $field);
$result = Container::get('cache')->get($guid);
if (false !== $result) {
return $result;
}
}
if (isset($options['field'])) {
$query->removeOption('field');
}
if (is_null($field)) {
$field = '*';
} elseif ($key && '*' != $field) {
$field = $key . ',' . $field;
}
if (is_string($field)) {
$field = array_map('trim', explode(',', $field));
}
$query->setOption('field', $field);
// 生成查询SQL
$sql = $this->builder->select($query);
$bind = $query->getBind();
if (!empty($options['fetch_sql'])) {
// 获取实际执行的SQL语句
return $this->getRealSql($sql, $bind);
}
// 执行查询操作
$pdo = $this->query($sql, $bind, $options['master'], true);
if (1 == $pdo->columnCount()) {
$result = $pdo->fetchAll(PDO::FETCH_COLUMN);
} else {
$resultSet = $pdo->fetchAll(PDO::FETCH_ASSOC);
if ('*' == $field && $key) {
$result = array_column($resultSet, null, $key);
} elseif ($resultSet) {
$fields = array_keys($resultSet[0]);
$count = count($fields);
$key1 = array_shift($fields);
$key2 = $fields ? array_shift($fields) : '';
$key = $key ?: $key1;
if (strpos($key, '.')) {
list($alias, $key) = explode('.', $key);
}
if (3 == $count) {
$column = $key2;
} elseif ($count < 3) {
$column = $key1;
} else {
$column = null;
}
$result = array_column($resultSet, $column, $key);
} else {
$result = [];
}
}
if (isset($cache) && isset($guid)) {
// 缓存数据
$this->cacheData($guid, $result, $cache);
}
return $result;
}
/** /**
* SQL性能分析 * SQL性能分析
* @access protected * @access protected

View File

@ -26,7 +26,7 @@ use think\Facade;
* @method void setName(string $name) static 批量导入路由标识 * @method void setName(string $name) static 批量导入路由标识
* @method void import(array $rules, string $type = '*') static 导入配置文件的路由规则 * @method void import(array $rules, string $type = '*') static 导入配置文件的路由规则
* @method \think\route\RuleItem rule(string $rule, mixed $route, string $method = '*', array $option = [], array $pattern = []) static 注册路由规则 * @method \think\route\RuleItem rule(string $rule, mixed $route, string $method = '*', array $option = [], array $pattern = []) static 注册路由规则
* @method void rules(string $rules, string $method = '*', array $option = [], array $pattern = []) static 批量注册路由规则 * @method void rules(array $rules, string $method = '*', array $option = [], array $pattern = []) static 批量注册路由规则
* @method \think\route\RuleGroup group(string|array $name, mixed $route, string $method = '*', array $option = [], array $pattern = []) static 注册路由分组 * @method \think\route\RuleGroup group(string|array $name, mixed $route, string $method = '*', array $option = [], array $pattern = []) static 注册路由分组
* @method \think\route\RuleItem any(string $rule, mixed $route, array $option = [], array $pattern = []) static 注册路由 * @method \think\route\RuleItem any(string $rule, mixed $route, array $option = [], array $pattern = []) static 注册路由
* @method \think\route\RuleItem get(string $rule, mixed $route, array $option = [], array $pattern = []) static 注册路由 * @method \think\route\RuleItem get(string $rule, mixed $route, array $option = [], array $pattern = []) static 注册路由

View File

@ -72,7 +72,7 @@ class File
$info[$type][] = $this->config['json'] ? $msg : '[ ' . $type . ' ] ' . $msg; $info[$type][] = $this->config['json'] ? $msg : '[ ' . $type . ' ] ' . $msg;
} }
if (!$this->config['json'] && in_array($type, $this->config['apart_level'])) { if (!$this->config['json'] && (true === $this->config['apart_level'] || in_array($type, $this->config['apart_level']))) {
// 独立记录的日志级别 // 独立记录的日志级别
$filename = $this->getApartLevelFile($path, $type); $filename = $this->getApartLevelFile($path, $type);

View File

@ -36,6 +36,12 @@ trait Attribute
*/ */
protected $json = []; protected $json = [];
/**
* JSON数据取出是否需要转换为数组
* @var bool
*/
protected $jsonAssoc = false;
/** /**
* JSON数据表字段类型 * JSON数据表字段类型
* @var array * @var array

View File

@ -10,6 +10,12 @@ use think\db\Query;
trait SoftDelete trait SoftDelete
{ {
/**
* 是否包含软删除数据
* @var bool
*/
protected $withTrashed = false;
/** /**
* 判断当前实例是否被软删除 * 判断当前实例是否被软删除
* @access public * @access public
@ -35,7 +41,19 @@ trait SoftDelete
{ {
$model = new static(); $model = new static();
return $model->db(false); return $model->withTrashedData(true)->db(false);
}
/**
* 是否包含软删除数据
* @access protected
* @param bool $withTrashed 是否包含软删除数据
* @return $this
*/
protected function withTrashedData($withTrashed)
{
$this->withTrashed = $withTrashed;
return $this;
} }
/** /**
@ -87,7 +105,7 @@ trait SoftDelete
$result = $this->isUpdate()->withEvent(false)->save(); $result = $this->isUpdate()->withEvent(false)->save();
$this->withEvent = true; $this->withEvent(true);
} else { } else {
// 读取更新条件 // 读取更新条件
$where = $this->getWhere(); $where = $this->getWhere();

View File

@ -563,7 +563,7 @@ class BelongsToMany extends Relation
$pivot[] = [$this->localKey, '=', $this->parent->$pk]; $pivot[] = [$this->localKey, '=', $this->parent->$pk];
if (isset($id)) { if (isset($id)) {
$pivot[] = is_array($id) ? [$this->foreignKey, 'in', $id] : [$this->foreignKey, '=', $id]; $pivot[] = [$this->foreignKey, is_array($id) ? 'in' : '=', $id];
} }
$result = $this->pivot->where($pivot)->delete(); $result = $this->pivot->where($pivot)->delete();

View File

@ -198,14 +198,14 @@ class MorphTo extends Relation
if ($key == $result->$morphType) { if ($key == $result->$morphType) {
// 关联模型 // 关联模型
if (!isset($data[$result->$morphKey])) { if (!isset($data[$result->$morphKey])) {
throw new Exception('relation data not exists :' . $this->model); $relationModel = null;
} else { } else {
$relationModel = $data[$result->$morphKey]; $relationModel = $data[$result->$morphKey];
$relationModel->setParent(clone $result); $relationModel->setParent(clone $result);
$relationModel->isUpdate(true); $relationModel->isUpdate(true);
$result->setRelation($attr, $relationModel);
} }
$result->setRelation($attr, $relationModel);
} }
} }
} }

View File

@ -162,7 +162,29 @@ abstract class Dispatch
$this->autoValidate($option['validate']); $this->autoValidate($option['validate']);
} }
return $this->exec(); $data = $this->exec();
return $this->autoResponse($data);
}
protected function autoResponse($data)
{
if ($data instanceof Response) {
$response = $data;
} elseif (!is_null($data)) {
// 默认自动识别响应输出类型
$isAjax = $this->request->isAjax();
$type = $isAjax ? $this->rule->getConfig('default_ajax_return') : $this->rule->getConfig('default_return_type');
$response = Response::create($data, $type);
} else {
$data = ob_get_clean();
$data = false === $data ? '' : $data;
$status = '' === $data ? 204 : 200;
$response = Response::create($data, '', $status);
}
return $response;
} }
/** /**

View File

@ -171,6 +171,16 @@ abstract class Rule
return $this->parent; return $this->parent;
} }
/**
* 获取路由所在域名
* @access public
* @return string
*/
public function getDomain()
{
return $this->parent->getDomain();
}
/** /**
* 获取变量规则定义 * 获取变量规则定义
* @access public * @access public

View File

@ -312,33 +312,50 @@ class RuleGroup extends Rule
} }
} }
try { if (empty($regex)) {
if (!empty($regex) && preg_match('/^(?:' . implode('|', $regex) . ')/u', $url, $match)) {
$var = [];
foreach ($match as $key => $val) {
if (is_string($key) && '' !== $val) {
list($name, $pos) = explode('_THINK_', $key);
$var[$name] = $val;
}
}
if (!isset($pos)) {
foreach ($regex as $key => $item) {
if (0 === strpos(str_replace(['\/', '\-', '\\' . $depr], ['/', '-', $depr], $item), $match[0])) {
$pos = $key;
break;
}
}
}
return $items[$pos]->checkRule($request, $url, $var);
}
return false; return false;
}
try {
$result = preg_match('/^(?:' . implode('|', $regex) . ')/u', $url, $match);
} catch (\Exception $e) { } catch (\Exception $e) {
throw new Exception('route pattern error'); throw new Exception('route pattern error');
} }
if ($result) {
$var = [];
foreach ($match as $key => $val) {
if (is_string($key) && '' !== $val) {
list($name, $pos) = explode('_THINK_', $key);
$var[$name] = $val;
}
}
if (!isset($pos)) {
foreach ($regex as $key => $item) {
if (0 === strpos(str_replace(['\/', '\-', '\\' . $depr], ['/', '-', $depr], $item), $match[0])) {
$pos = $key;
break;
}
}
}
$rule = $items[$pos]->getRule();
$array = $this->router->getRule($rule);
foreach ($array as $item) {
if (in_array($item->getMethod(), ['*', strtolower($request->method())])) {
$result = $item->checkRule($request, $url, $var);
if (false !== $result) {
return $result;
}
}
}
}
return false;
} }
/** /**
@ -414,6 +431,11 @@ class RuleGroup extends Rule
$method = strtolower($method); $method = strtolower($method);
if ('/' === $rule || '' === $rule) {
// 首页自动完整匹配
$rule .= '$';
}
// 创建路由规则实例 // 创建路由规则实例
$ruleItem = new RuleItem($this->router, $this, $name, $rule, $route, $method, $option, $pattern); $ruleItem = new RuleItem($this->router, $this, $name, $rule, $route, $method, $option, $pattern);

View File

@ -128,6 +128,7 @@ class RuleItem extends Rule
$value = [$this->rule, $vars, $this->parent->getDomain(), $suffix, $this->method]; $value = [$this->rule, $vars, $this->parent->getDomain(), $suffix, $this->method];
Container::get('rule_name')->set($name, $value, $first); Container::get('rule_name')->set($name, $value, $first);
Container::get('rule_name')->setRule($this->rule, $this);
} }
} }
@ -242,7 +243,7 @@ class RuleItem extends Rule
} }
if (false === strpos($rule, '<')) { if (false === strpos($rule, '<')) {
if (0 === strcasecmp($rule, $url) || (!$completeMatch && 0 === strncasecmp($rule, $url, strlen($rule)))) { if (0 === strcasecmp($rule, $url) || (!$completeMatch && 0 === strncasecmp($rule . $depr, $url . $depr, strlen($rule . $depr)))) {
return $var; return $var;
} }
return false; return false;

View File

@ -14,6 +14,7 @@ namespace think\route;
class RuleName class RuleName
{ {
protected $item = []; protected $item = [];
protected $rule = [];
/** /**
* 注册路由标识 * 注册路由标识
@ -32,6 +33,62 @@ class RuleName
} }
} }
/**
* 注册路由规则
* @access public
* @param string $rule 路由规则
* @param RuleItem $route 路由
* @return void
*/
public function setRule($rule, $route)
{
$this->rule[$route->getDomain()][$rule][$route->getRoute()] = $route;
}
/**
* 根据路由规则获取路由对象(列表)
* @access public
* @param string $name 路由标识
* @param string $domain 域名
* @return array
*/
public function getRule($rule, $domain = null)
{
return isset($this->rule[$domain][$rule]) ? $this->rule[$domain][$rule] : [];
}
/**
* 获取全部路由列表
* @access public
* @param string $domain 域名
* @return array
*/
public function getRuleList($domain = null)
{
$list = [];
foreach ($this->rule as $ruleDomain => $rules) {
foreach ($rules as $rule => $items) {
foreach ($items as $item) {
$val = [];
foreach (['method', 'rule', 'name', 'route', 'pattern', 'option'] as $param) {
$call = 'get' . $param;
$val[$param] = $item->$call();
}
$list[$ruleDomain][] = $val;
}
}
}
if ($domain) {
return isset($list[$domain]) ? $list[$domain] : [];
}
return $list;
}
/** /**
* 导入路由标识 * 导入路由标识
* @access public * @access public

View File

@ -15,6 +15,7 @@ use ReflectionMethod;
use think\exception\ClassNotFoundException; use think\exception\ClassNotFoundException;
use think\exception\HttpException; use think\exception\HttpException;
use think\Loader; use think\Loader;
use think\Request;
use think\route\Dispatch; use think\route\Dispatch;
class Module extends Dispatch class Module extends Dispatch
@ -84,8 +85,8 @@ class Module extends Dispatch
// 监听module_init // 监听module_init
$this->app['hook']->listen('module_init'); $this->app['hook']->listen('module_init');
// 实例化控制器
try { try {
// 实例化控制器
$instance = $this->app->controller($this->controller, $instance = $this->app->controller($this->controller,
$this->rule->getConfig('url_controller_layer'), $this->rule->getConfig('url_controller_layer'),
$this->rule->getConfig('controller_suffix'), $this->rule->getConfig('controller_suffix'),
@ -94,36 +95,42 @@ class Module extends Dispatch
throw new HttpException(404, 'controller not exists:' . $e->getClass()); throw new HttpException(404, 'controller not exists:' . $e->getClass());
} }
// 获取当前操作名 $this->app['middleware']->controller(function (Request $request, $next) use ($instance) {
$action = $this->actionName . $this->rule->getConfig('action_suffix'); // 获取当前操作名
$action = $this->actionName . $this->rule->getConfig('action_suffix');
if (is_callable([$instance, $action])) { if (is_callable([$instance, $action])) {
// 执行操作方法 // 执行操作方法
$call = [$instance, $action]; $call = [$instance, $action];
// 严格获取当前操作方法名 // 严格获取当前操作方法名
$reflect = new ReflectionMethod($instance, $action); $reflect = new ReflectionMethod($instance, $action);
$methodName = $reflect->getName(); $methodName = $reflect->getName();
$suffix = $this->rule->getConfig('action_suffix'); $suffix = $this->rule->getConfig('action_suffix');
$actionName = $suffix ? substr($methodName, 0, -strlen($suffix)) : $methodName; $actionName = $suffix ? substr($methodName, 0, -strlen($suffix)) : $methodName;
$this->request->setAction($actionName); $this->request->setAction($actionName);
// 自动获取请求变量 // 自动获取请求变量
$vars = $this->rule->getConfig('url_param_type') $vars = $this->rule->getConfig('url_param_type')
? $this->request->route() ? $this->request->route()
: $this->request->param(); : $this->request->param();
} elseif (is_callable([$instance, '_empty'])) { } elseif (is_callable([$instance, '_empty'])) {
// 空操作 // 空操作
$call = [$instance, '_empty']; $call = [$instance, '_empty'];
$vars = [$this->actionName]; $vars = [$this->actionName];
$reflect = new ReflectionMethod($instance, '_empty'); $reflect = new ReflectionMethod($instance, '_empty');
} else { } else {
// 操作不存在 // 操作不存在
throw new HttpException(404, 'method not exists:' . get_class($instance) . '->' . $action . '()'); throw new HttpException(404, 'method not exists:' . get_class($instance) . '->' . $action . '()');
} }
$this->app['hook']->listen('action_begin', $call); $this->app['hook']->listen('action_begin', $call);
return $this->app->invokeReflectMethod($instance, $reflect, $vars); $data = $this->app->invokeReflectMethod($instance, $reflect, $vars);
return $this->autoResponse($data);
});
return $this->app['middleware']->dispatch($this->request, 'controller');
} }
} }

View File

@ -5,6 +5,6 @@ class Index{$suffix}
{ {
public function index() public function index()
{ {
return '<style type="text/css">*{ padding: 0; margin: 0; } div{ padding: 4px 48px;} a{color:#2E5CD5;cursor: pointer;text-decoration: none} a:hover{text-decoration:underline; } body{ background: #fff; font-family: "Century Gothic","Microsoft yahei"; color: #333;font-size:18px;} h1{ font-size: 100px; font-weight: normal; margin-bottom: 12px; } p{ line-height: 1.6em; font-size: 42px }</style><div style="padding: 24px 48px;"> <h1>:) 2018新年快乐</h1><p> ThinkPHP V5.1<br/><span style="font-size:30px">12载初心不改2006-2018 - 你值得信赖的PHP框架</span></p></div><script type="text/javascript" src="https://tajs.qq.com/stats?sId=64890268" charset="UTF-8"></script><script type="text/javascript" src="https://e.topthink.com/Public/static/client.js"></script><think id="eab4b9f840753f8e7"></think>'; return '<style type="text/css">*{ padding: 0; margin: 0; } div{ padding: 4px 48px;} a{color:#2E5CD5;cursor: pointer;text-decoration: none} a:hover{text-decoration:underline; } body{ background: #fff; font-family: "Century Gothic","Microsoft yahei"; color: #333;font-size:18px;} h1{ font-size: 100px; font-weight: normal; margin-bottom: 12px; } p{ line-height: 1.6em; font-size: 42px }</style><div style="padding: 24px 48px;"> <h1>:) </h1><p> ThinkPHP V5.1<br/><span style="font-size:30px">12载初心不改2006-2018 - 你值得信赖的PHP框架</span></p></div><script type="text/javascript" src="https://tajs.qq.com/stats?sId=64890268" charset="UTF-8"></script><script type="text/javascript" src="https://e.topthink.com/Public/static/client.js"></script><think id="eab4b9f840753f8e7"></think>';
} }
} }

2
vendor/autoload.php vendored
View File

@ -4,4 +4,4 @@
require_once __DIR__ . '/composer/autoload_real.php'; require_once __DIR__ . '/composer/autoload_real.php';
return ComposerAutoloaderInit0402c38ddd8eba428c182633170c04fe::getLoader(); return ComposerAutoloaderInit3917a65c534c33755c27cf34de19b654::getLoader();

View File

@ -2,7 +2,7 @@
// autoload_real.php @generated by Composer // autoload_real.php @generated by Composer
class ComposerAutoloaderInit0402c38ddd8eba428c182633170c04fe class ComposerAutoloaderInit3917a65c534c33755c27cf34de19b654
{ {
private static $loader; private static $loader;
@ -19,15 +19,15 @@ class ComposerAutoloaderInit0402c38ddd8eba428c182633170c04fe
return self::$loader; return self::$loader;
} }
spl_autoload_register(array('ComposerAutoloaderInit0402c38ddd8eba428c182633170c04fe', 'loadClassLoader'), true, true); spl_autoload_register(array('ComposerAutoloaderInit3917a65c534c33755c27cf34de19b654', 'loadClassLoader'), true, true);
self::$loader = $loader = new \Composer\Autoload\ClassLoader(); self::$loader = $loader = new \Composer\Autoload\ClassLoader();
spl_autoload_unregister(array('ComposerAutoloaderInit0402c38ddd8eba428c182633170c04fe', 'loadClassLoader')); spl_autoload_unregister(array('ComposerAutoloaderInit3917a65c534c33755c27cf34de19b654', 'loadClassLoader'));
$useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded()); $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
if ($useStaticLoader) { if ($useStaticLoader) {
require_once __DIR__ . '/autoload_static.php'; require_once __DIR__ . '/autoload_static.php';
call_user_func(\Composer\Autoload\ComposerStaticInit0402c38ddd8eba428c182633170c04fe::getInitializer($loader)); call_user_func(\Composer\Autoload\ComposerStaticInit3917a65c534c33755c27cf34de19b654::getInitializer($loader));
} else { } else {
$map = require __DIR__ . '/autoload_namespaces.php'; $map = require __DIR__ . '/autoload_namespaces.php';
foreach ($map as $namespace => $path) { foreach ($map as $namespace => $path) {
@ -48,19 +48,19 @@ class ComposerAutoloaderInit0402c38ddd8eba428c182633170c04fe
$loader->register(true); $loader->register(true);
if ($useStaticLoader) { if ($useStaticLoader) {
$includeFiles = Composer\Autoload\ComposerStaticInit0402c38ddd8eba428c182633170c04fe::$files; $includeFiles = Composer\Autoload\ComposerStaticInit3917a65c534c33755c27cf34de19b654::$files;
} else { } else {
$includeFiles = require __DIR__ . '/autoload_files.php'; $includeFiles = require __DIR__ . '/autoload_files.php';
} }
foreach ($includeFiles as $fileIdentifier => $file) { foreach ($includeFiles as $fileIdentifier => $file) {
composerRequire0402c38ddd8eba428c182633170c04fe($fileIdentifier, $file); composerRequire3917a65c534c33755c27cf34de19b654($fileIdentifier, $file);
} }
return $loader; return $loader;
} }
} }
function composerRequire0402c38ddd8eba428c182633170c04fe($fileIdentifier, $file) function composerRequire3917a65c534c33755c27cf34de19b654($fileIdentifier, $file)
{ {
if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) { if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) {
require $file; require $file;

View File

@ -4,7 +4,7 @@
namespace Composer\Autoload; namespace Composer\Autoload;
class ComposerStaticInit0402c38ddd8eba428c182633170c04fe class ComposerStaticInit3917a65c534c33755c27cf34de19b654
{ {
public static $files = array ( public static $files = array (
'1cfd2761b63b0a29ed23657ea394cb2d' => __DIR__ . '/..' . '/topthink/think-captcha/src/helper.php', '1cfd2761b63b0a29ed23657ea394cb2d' => __DIR__ . '/..' . '/topthink/think-captcha/src/helper.php',
@ -303,9 +303,9 @@ class ComposerStaticInit0402c38ddd8eba428c182633170c04fe
public static function getInitializer(ClassLoader $loader) public static function getInitializer(ClassLoader $loader)
{ {
return \Closure::bind(function () use ($loader) { return \Closure::bind(function () use ($loader) {
$loader->prefixLengthsPsr4 = ComposerStaticInit0402c38ddd8eba428c182633170c04fe::$prefixLengthsPsr4; $loader->prefixLengthsPsr4 = ComposerStaticInit3917a65c534c33755c27cf34de19b654::$prefixLengthsPsr4;
$loader->prefixDirsPsr4 = ComposerStaticInit0402c38ddd8eba428c182633170c04fe::$prefixDirsPsr4; $loader->prefixDirsPsr4 = ComposerStaticInit3917a65c534c33755c27cf34de19b654::$prefixDirsPsr4;
$loader->classMap = ComposerStaticInit0402c38ddd8eba428c182633170c04fe::$classMap; $loader->classMap = ComposerStaticInit3917a65c534c33755c27cf34de19b654::$classMap;
}, null, ClassLoader::class); }, null, ClassLoader::class);
} }

View File

@ -191,17 +191,17 @@
}, },
{ {
"name": "topthink/framework", "name": "topthink/framework",
"version": "v5.1.16", "version": "v5.1.17",
"version_normalized": "5.1.16.0", "version_normalized": "5.1.17.0",
"source": { "source": {
"type": "git", "type": "git",
"url": "https://github.com/top-think/framework.git", "url": "https://github.com/top-think/framework.git",
"reference": "e0a8bdc52957a8f9774353fbfa0b9fd6ee206188" "reference": "61a66e8eeecf1584e0dc150a530f0d116e6ccd01"
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://files.phpcomposer.com/files/top-think/framework/e0a8bdc52957a8f9774353fbfa0b9fd6ee206188.zip", "url": "https://files.phpcomposer.com/files/top-think/framework/61a66e8eeecf1584e0dc150a530f0d116e6ccd01.zip",
"reference": "e0a8bdc52957a8f9774353fbfa0b9fd6ee206188", "reference": "61a66e8eeecf1584e0dc150a530f0d116e6ccd01",
"shasum": "" "shasum": ""
}, },
"require": { "require": {
@ -217,7 +217,7 @@
"sebastian/phpcpd": "2.*", "sebastian/phpcpd": "2.*",
"squizlabs/php_codesniffer": "2.*" "squizlabs/php_codesniffer": "2.*"
}, },
"time": "2018-06-08T04:10:59+00:00", "time": "2018-06-19T03:11:41+00:00",
"type": "think-framework", "type": "think-framework",
"installation-source": "dist", "installation-source": "dist",
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",