[更新]ComposerUpdate

This commit is contained in:
Anyon 2018-10-08 11:38:14 +08:00
parent 8b2e4600b2
commit f5dffb2b79
29 changed files with 336 additions and 297 deletions

View File

@ -1,6 +1,6 @@
ThinkPHP遵循Apache2开源协议发布并提供免费使用。 ThinkPHP遵循Apache2开源协议发布并提供免费使用。
版权所有Copyright © 2006-2016 by ThinkPHP (http://thinkphp.cn) 版权所有Copyright © 2006-2018 by ThinkPHP (http://thinkphp.cn)
All rights reserved。 All rights reserved。
ThinkPHP® 商标和著作权所有者为上海顶想信息科技有限公司。 ThinkPHP® 商标和著作权所有者为上海顶想信息科技有限公司。

View File

@ -20,7 +20,7 @@ use think\route\Dispatch;
*/ */
class App extends Container class App extends Container
{ {
const VERSION = '5.1.24'; const VERSION = '5.1.25';
/** /**
* 当前模块路径 * 当前模块路径
@ -606,14 +606,9 @@ class App extends Container
if (!empty($routeKey)) { if (!empty($routeKey)) {
try { try {
if ($option) { if ($option) {
$this->cache $this->cache->connect($option)->tag('route_cache')->set($routeKey, $dispatch);
->connect($option)
->tag('route_cache')
->set($routeKey, $dispatch);
} else { } else {
$this->cache $this->cache->tag('route_cache')->set($routeKey, $dispatch);
->tag('route_cache')
->set($routeKey, $dispatch);
} }
} catch (\Exception $e) { } catch (\Exception $e) {
// 存在闭包的时候缓存无效 // 存在闭包的时候缓存无效

View File

@ -71,7 +71,9 @@ class Config implements \ArrayAccess
*/ */
public function setYaconf($yaconf) public function setYaconf($yaconf)
{ {
$this->yaconf = $yaconf; if ($this->yaconf) {
$this->yaconf = $yaconf;
}
} }
/** /**

View File

@ -573,16 +573,6 @@ class Console
throw new \InvalidArgumentException($message); throw new \InvalidArgumentException($message);
} }
if (count($commands) > 1) {
$commandList = $this->commands;
$commands = array_filter($commands, function ($nameOrAlias) use ($commandList, $commands) {
$commandName = $commandList[$nameOrAlias]->getName();
return $commandName === $nameOrAlias || !in_array($commandName, $commands);
});
}
$exact = in_array($name, $commands, true); $exact = in_array($name, $commands, true);
if (count($commands) > 1 && !$exact) { if (count($commands) > 1 && !$exact) {
$suggestions = $this->getAbbreviationSuggestions(array_values($commands)); $suggestions = $this->getAbbreviationSuggestions(array_values($commands));

View File

@ -71,9 +71,21 @@ class Controller
if ($this->middleware) { if ($this->middleware) {
foreach ($this->middleware as $key => $val) { foreach ($this->middleware as $key => $val) {
if (!is_int($key)) { if (!is_int($key)) {
if (isset($val['only']) && !in_array($this->request->action(), $val['only'])) { $only = $except = null;
if (isset($val['only'])) {
$only = array_map(function ($item) {
return strtolower($item);
}, $val['only']);
} elseif (isset($val['except'])) {
$except = array_map(function ($item) {
return strtolower($item);
}, $val['except']);
}
if (isset($only) && !in_array($this->request->action(), $only)) {
continue; continue;
} elseif (isset($val['except']) && in_array($this->request->action(), $val['except'])) { } elseif (isset($except) && in_array($this->request->action(), $except)) {
continue; continue;
} else { } else {
$val = $key; $val = $key;
@ -108,14 +120,24 @@ class Controller
if (is_string($options['only'])) { if (is_string($options['only'])) {
$options['only'] = explode(',', $options['only']); $options['only'] = explode(',', $options['only']);
} }
if (!in_array($this->request->action(), $options['only'])) {
$only = array_map(function ($val) {
return strtolower($val);
}, $options['only']);
if (!in_array($this->request->action(), $only)) {
return; return;
} }
} elseif (isset($options['except'])) { } elseif (isset($options['except'])) {
if (is_string($options['except'])) { if (is_string($options['except'])) {
$options['except'] = explode(',', $options['except']); $options['except'] = explode(',', $options['except']);
} }
if (in_array($this->request->action(), $options['except'])) {
$except = array_map(function ($val) {
return strtolower($val);
}, $options['except']);
if (in_array($this->request->action(), $except)) {
return; return;
} }
} }

View File

@ -187,8 +187,6 @@ class Hook
*/ */
protected function execTag($class, $tag = '', $params = null) protected function execTag($class, $tag = '', $params = null)
{ {
$this->app->isDebug() && $this->app['debug']->remark('behavior_start', 'time');
$method = Loader::parseName($tag, 1, false); $method = Loader::parseName($tag, 1, false);
if ($class instanceof \Closure) { if ($class instanceof \Closure) {
@ -209,12 +207,6 @@ class Hook
$result = $this->app->invoke($call, [$params]); $result = $this->app->invoke($call, [$params]);
if ($this->app->isDebug()) {
$debug = $this->app['debug'];
$debug->remark('behavior_end', 'time');
$this->app->log('[ BEHAVIOR ] Run ' . $class . ' @' . $tag . ' [ RunTime:' . $debug->getRangeTime('behavior_start', 'behavior_end') . 's ]');
}
return $result; return $result;
} }

View File

@ -211,12 +211,18 @@ class Log implements LoggerInterface
if (!$this->app->isDebug() && isset($log['debug'])) { if (!$this->app->isDebug() && isset($log['debug'])) {
unset($log['debug']); unset($log['debug']);
} }
foreach ($log as $level => $info) {
$this->app['hook']->listen('log_level', [$level, $info]);
}
} else { } else {
// 记录允许级别 // 记录允许级别
$log = []; $log = [];
foreach ($this->config['level'] as $level) { foreach ($this->config['level'] as $level) {
if (isset($this->log[$level])) { if (isset($this->log[$level])) {
$log[$level] = $this->log[$level]; $log[$level] = $this->log[$level];
// 监听log_level
$this->app['hook']->listen('log_level', [$level, $log[$level]]);
} }
} }
} }

View File

@ -957,6 +957,17 @@ class Route
return $item; return $item;
} }
/**
* 清空路由规则
* @access public
* @return void
*/
public function clear()
{
$this->app['rule_name']->clear();
$this->group->clear();
}
/** /**
* 设置全局的路由分组参数 * 设置全局的路由分组参数
* @access public * @access public

View File

@ -87,10 +87,9 @@ abstract class Builder
* @param array $data 数据 * @param array $data 数据
* @param array $fields 字段信息 * @param array $fields 字段信息
* @param array $bind 参数绑定 * @param array $bind 参数绑定
* @param string $suffix 参数绑定后缀
* @return array * @return array
*/ */
protected function parseData(Query $query, $data = [], $fields = [], $bind = [], $suffix = '') protected function parseData(Query $query, $data = [], $fields = [], $bind = [])
{ {
if (empty($data)) { if (empty($data)) {
return []; return [];
@ -120,7 +119,7 @@ abstract class Builder
$result[$item] = $val->getValue(); $result[$item] = $val->getValue();
continue; continue;
} elseif (!is_scalar($val) && (in_array($key, (array) $query->getOptions('json')) || 'json' == $this->connection->getFieldsType($options['table'], $key))) { } elseif (!is_scalar($val) && (in_array($key, (array) $query->getOptions('json')) || 'json' == $this->connection->getFieldsType($options['table'], $key))) {
$val = json_encode($val); $val = json_encode($val, JSON_UNESCAPED_UNICODE);
} elseif (is_object($val) && method_exists($val, '__toString')) { } elseif (is_object($val) && method_exists($val, '__toString')) {
// 对象数据写入 // 对象数据写入
$val = $val->__toString(); $val = $val->__toString();
@ -129,7 +128,7 @@ abstract class Builder
if (false !== strpos($key, '->')) { if (false !== strpos($key, '->')) {
list($key, $name) = explode('->', $key); list($key, $name) = explode('->', $key);
$item = $this->parseKey($query, $key); $item = $this->parseKey($query, $key);
$result[$item] = 'json_set(' . $item . ', \'$.' . $name . '\', ' . $this->parseDataBind($query, $key, $val, $bind, $suffix) . ')'; $result[$item] = 'json_set(' . $item . ', \'$.' . $name . '\', ' . $this->parseDataBind($query, $key, $val, $bind) . ')';
} elseif (false === strpos($key, '.') && !in_array($key, $fields, true)) { } elseif (false === strpos($key, '.') && !in_array($key, $fields, true)) {
if ($options['strict']) { if ($options['strict']) {
throw new Exception('fields not exists:[' . $key . ']'); throw new Exception('fields not exists:[' . $key . ']');
@ -149,7 +148,7 @@ abstract class Builder
} }
} elseif (is_scalar($val)) { } elseif (is_scalar($val)) {
// 过滤非标量数据 // 过滤非标量数据
$result[$item] = $this->parseDataBind($query, $key, $val, $bind, $suffix); $result[$item] = $this->parseDataBind($query, $key, $val, $bind);
} }
} }
@ -163,22 +162,17 @@ abstract class Builder
* @param string $key 字段名 * @param string $key 字段名
* @param mixed $data 数据 * @param mixed $data 数据
* @param array $bind 绑定数据 * @param array $bind 绑定数据
* @param string $suffix 绑定后缀
* @return string * @return string
*/ */
protected function parseDataBind(Query $query, $key, $data, $bind = [], $suffix = '') protected function parseDataBind(Query $query, $key, $data, $bind = [])
{ {
// 过滤非标量数据 if ($data instanceof Expression) {
if (0 === strpos($data, ':') && $query->isBind(substr($data, 1))) { return $data->getValue();
return $data;
} }
$key = str_replace(['.', '->'], '_', $key); $query->bind($data, isset($bind[$key]) ? $bind[$key] : PDO::PARAM_STR);
$name = 'data__' . $key . $suffix;
$query->bind($name, $data, isset($bind[$key]) ? $bind[$key] : PDO::PARAM_STR); return '?';
return ':' . $name;
} }
/** /**
@ -362,7 +356,7 @@ abstract class Builder
} }
// where子单元分析 // where子单元分析
protected function parseWhereItem(Query $query, $field, $val, $rule = '', $binds = [], $bindName = null) protected function parseWhereItem(Query $query, $field, $val, $rule = '', $binds = [])
{ {
// 字段分析 // 字段分析
$key = $field ? $this->parseKey($query, $field, true) : ''; $key = $field ? $this->parseKey($query, $field, true) : '';
@ -386,8 +380,7 @@ abstract class Builder
} }
foreach ($val as $k => $item) { foreach ($val as $k => $item) {
$bindName = 'where_' . str_replace('.', '_', $field) . '_' . $k; $str[] = $this->parseWhereItem($query, $field, $item, $rule, $binds);
$str[] = $this->parseWhereItem($query, $field, $item, $rule, $binds, $bindName);
} }
return '( ' . implode(' ' . $rule . ' ', $str) . ' )'; return '( ' . implode(' ' . $rule . ' ', $str) . ' )';
@ -399,13 +392,6 @@ abstract class Builder
$exp = $this->exp[$exp]; $exp = $this->exp[$exp];
} }
$bindName = $bindName ?: 'where_' . $rule . '_' . str_replace(['.', '-'], '_', $field);
if (preg_match('/\W/', $bindName)) {
// 处理带非单词字符的字段名
$bindName = md5($bindName);
}
if ($value instanceof Expression) { if ($value instanceof Expression) {
} elseif (is_object($value) && method_exists($value, '__toString')) { } elseif (is_object($value) && method_exists($value, '__toString')) {
@ -421,20 +407,14 @@ abstract class Builder
} }
if (is_scalar($value) && !in_array($exp, ['EXP', 'NOT NULL', 'NULL', 'IN', 'NOT IN', 'BETWEEN', 'NOT BETWEEN']) && strpos($exp, 'TIME') === false) { if (is_scalar($value) && !in_array($exp, ['EXP', 'NOT NULL', 'NULL', 'IN', 'NOT IN', 'BETWEEN', 'NOT BETWEEN']) && strpos($exp, 'TIME') === false) {
if (strpos($value, ':') !== 0 || !$query->isBind(substr($value, 1))) { $query->bind($value, $bindType);
if ($query->isBind($bindName)) { $value = '?';
$bindName .= '_' . str_replace('.', '_', uniqid('', true));
}
$query->bind($bindName, $value, $bindType);
$value = ':' . $bindName;
}
} }
// 解析查询表达式 // 解析查询表达式
foreach ($this->parser as $fun => $parse) { foreach ($this->parser as $fun => $parse) {
if (in_array($exp, $parse)) { if (in_array($exp, $parse)) {
$whereStr = $this->$fun($query, $key, $exp, $value, $field, $bindName, $bindType, isset($val[2]) ? $val[2] : 'AND'); $whereStr = $this->$fun($query, $key, $exp, $value, $field, $bindType, isset($val[2]) ? $val[2] : 'AND');
break; break;
} }
} }
@ -454,19 +434,17 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @param string $logic * @param string $logic
* @return string * @return string
*/ */
protected function parseLike(Query $query, $key, $exp, $value, $field, $bindName, $bindType, $logic) protected function parseLike(Query $query, $key, $exp, $value, $field, $bindType, $logic)
{ {
// 模糊匹配 // 模糊匹配
if (is_array($value)) { if (is_array($value)) {
foreach ($value as $k => $item) { foreach ($value as $item) {
$bindKey = $bindName . '_' . intval($k); $bind[] = [$item, $bindType];
$bind[$bindKey] = [$item, $bindType]; $array[] = $key . ' ' . $exp . ' ?';
$array[] = $key . ' ' . $exp . ' :' . $bindKey;
} }
$query->bind($bind); $query->bind($bind);
@ -487,11 +465,10 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param array $value * @param array $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseColumn(Query $query, $key, $exp, array $value, $field, $bindName, $bindType) protected function parseColumn(Query $query, $key, $exp, array $value, $field, $bindType)
{ {
// 字段比较查询 // 字段比较查询
list($op, $field2) = $value; list($op, $field2) = $value;
@ -511,11 +488,10 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param Expression $value * @param Expression $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseExp(Query $query, $key, $exp, Expression $value, $field, $bindName, $bindType) protected function parseExp(Query $query, $key, $exp, Expression $value, $field, $bindType)
{ {
// 表达式查询 // 表达式查询
return '( ' . $key . ' ' . $value->getValue() . ' )'; return '( ' . $key . ' ' . $value->getValue() . ' )';
@ -529,11 +505,10 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseNull(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseNull(Query $query, $key, $exp, $value, $field, $bindType)
{ {
// NULL 查询 // NULL 查询
return $key . ' IS ' . $exp; return $key . ' IS ' . $exp;
@ -547,33 +522,20 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseBetween(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseBetween(Query $query, $key, $exp, $value, $field, $bindType)
{ {
// BETWEEN 查询 // BETWEEN 查询
$data = is_array($value) ? $value : explode(',', $value); $data = is_array($value) ? $value : explode(',', $value);
if ($query->isBind($bindName . '_between_1')) { $bind[] = [$data[0], $bindType];
$bindKey1 = $bindName . '_between_1' . uniqid(); $bind[] = [$data[1], $bindType];
$bindKey2 = $bindName . '_between_2' . uniqid();
} else {
$bindKey1 = $bindName . '_between_1';
$bindKey2 = $bindName . '_between_2';
}
$bind = [
$bindKey1 => [$data[0], $bindType],
$bindKey2 => [$data[1], $bindType],
];
$query->bind($bind); $query->bind($bind);
$between = ':' . $bindKey1 . ' AND :' . $bindKey2; return $key . ' ' . $exp . ' ? AND ? ';
return $key . ' ' . $exp . ' ' . $between;
} }
/** /**
@ -584,11 +546,10 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseExists(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseExists(Query $query, $key, $exp, $value, $field, $bindType)
{ {
// EXISTS 查询 // EXISTS 查询
if ($value instanceof \Closure) { if ($value instanceof \Closure) {
@ -610,13 +571,12 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseTime(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseTime(Query $query, $key, $exp, $value, $field, $bindType)
{ {
return $key . ' ' . substr($exp, 0, 2) . ' ' . $this->parseDateTime($query, $value, $field, $bindName, $bindType); return $key . ' ' . substr($exp, 0, 2) . ' ' . $this->parseDateTime($query, $value, $field, $bindType);
} }
/** /**
@ -627,11 +587,10 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseCompare(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseCompare(Query $query, $key, $exp, $value, $field, $bindType)
{ {
if (is_array($value)) { if (is_array($value)) {
throw new Exception('where express error:' . $exp . var_export($value, true)); throw new Exception('where express error:' . $exp . var_export($value, true));
@ -653,20 +612,19 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseBetweenTime(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseBetweenTime(Query $query, $key, $exp, $value, $field, $bindType)
{ {
if (is_string($value)) { if (is_string($value)) {
$value = explode(',', $value); $value = explode(',', $value);
} }
return $key . ' ' . substr($exp, 0, -4) return $key . ' ' . substr($exp, 0, -4)
. $this->parseDateTime($query, $value[0], $field, $bindName . '_between_1', $bindType) . $this->parseDateTime($query, $value[0], $field, $bindType)
. ' AND ' . ' AND '
. $this->parseDateTime($query, $value[1], $field, $bindName . '_between_2', $bindType); . $this->parseDateTime($query, $value[1], $field, $bindType);
} }
@ -678,11 +636,10 @@ abstract class Builder
* @param string $exp * @param string $exp
* @param mixed $value * @param mixed $value
* @param string $field * @param string $field
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseIn(Query $query, $key, $exp, $value, $field, $bindName, $bindType) protected function parseIn(Query $query, $key, $exp, $value, $field, $bindType)
{ {
// IN 查询 // IN 查询
if ($value instanceof \Closure) { if ($value instanceof \Closure) {
@ -692,17 +649,10 @@ abstract class Builder
$bind = []; $bind = [];
$array = []; $array = [];
$i = 0;
foreach ($value as $k => $v) { foreach ($value as $k => $v) {
$i++; $bind[] = [$v, $bindType];
if ($query->isBind($bindName . '_in_' . $i)) { $array[] = '?';
$bindKey = $bindName . '_in_' . uniqid() . '_' . $i;
} else {
$bindKey = $bindName . '_in_' . $i;
}
$bind[$bindKey] = [$v, $bindType];
$array[] = ':' . $bindKey;
} }
$zone = implode(',', $array); $zone = implode(',', $array);
@ -736,12 +686,10 @@ abstract class Builder
* @param Query $query 查询对象 * @param Query $query 查询对象
* @param string $value * @param string $value
* @param string $key * @param string $key
* @param array $options
* @param string $bindName
* @param integer $bindType * @param integer $bindType
* @return string * @return string
*/ */
protected function parseDateTime(Query $query, $value, $key, $bindName = null, $bindType = null) protected function parseDateTime(Query $query, $value, $key, $bindType = null)
{ {
$options = $query->getOptions(); $options = $query->getOptions();
@ -776,15 +724,9 @@ abstract class Builder
} }
} }
$bindName = $bindName ?: $key; $query->bind($value, $bindType);
if ($query->isBind($bindName)) { return '?';
$bindName .= '_' . str_replace('.', '_', uniqid('', true));
}
$query->bind($bindName, $value, $bindType);
return ':' . $bindName;
} }
/** /**
@ -855,11 +797,11 @@ abstract class Builder
foreach ($order as $key => $val) { foreach ($order as $key => $val) {
if ($val instanceof Expression) { if ($val instanceof Expression) {
$array[] = $val->getValue(); $array[] = $val->getValue();
} elseif (is_array($val)) { } elseif (is_array($val) && !preg_match('/\W/', $key)) {
$array[] = $this->parseOrderField($query, $key, $val); $array[] = $this->parseOrderField($query, $key, $val);
} elseif ('[rand]' == $val) { } elseif ('[rand]' == $val) {
$array[] = $this->parseRand($query); $array[] = $this->parseRand($query);
} else { } elseif (is_string($val)) {
if (is_numeric($key)) { if (is_numeric($key)) {
list($key, $sort) = explode(' ', strpos($val, ' ') ? $val : $val . ' '); list($key, $sort) = explode(' ', strpos($val, ' ') ? $val : $val . ' ');
} else { } else {
@ -901,7 +843,7 @@ abstract class Builder
$bind = $this->connection->getFieldsBind($options['table']); $bind = $this->connection->getFieldsBind($options['table']);
foreach ($val as $k => $item) { foreach ($val as $k => $item) {
$val[$k] = $this->parseDataBind($query, $key, $item, $bind, $k); $val[$k] = $this->parseDataBind($query, $key, $item, $bind);
} }
return 'field(' . $this->parseKey($query, $key, true) . ',' . implode(',', $val) . ')' . $sort; return 'field(' . $this->parseKey($query, $key, true) . ',' . implode(',', $val) . ')' . $sort;
@ -1114,8 +1056,8 @@ abstract class Builder
// 获取绑定信息 // 获取绑定信息
$bind = $this->connection->getFieldsBind($options['table']); $bind = $this->connection->getFieldsBind($options['table']);
foreach ($dataSet as $k => $data) { foreach ($dataSet as $data) {
$data = $this->parseData($query, $data, $allowFields, $bind, '_' . $k); $data = $this->parseData($query, $data, $allowFields, $bind);
$values[] = 'SELECT ' . implode(',', array_values($data)); $values[] = 'SELECT ' . implode(',', array_values($data));

View File

@ -24,6 +24,7 @@ use think\Loader;
abstract class Connection abstract class Connection
{ {
const PARAM_FLOAT = 21;
protected static $instance = []; protected static $instance = [];
/** @var PDOStatement PDO操作实例 */ /** @var PDOStatement PDO操作实例 */
protected $PDOStatement; protected $PDOStatement;
@ -303,7 +304,9 @@ abstract class Connection
{ {
if (0 === strpos($type, 'set') || 0 === strpos($type, 'enum')) { if (0 === strpos($type, 'set') || 0 === strpos($type, 'enum')) {
$bind = PDO::PARAM_STR; $bind = PDO::PARAM_STR;
} elseif (preg_match('/(int|double|float|decimal|real|numeric|serial|bit)/is', $type)) { } elseif (preg_match('/(double|float|decimal|real|numeric)/is', $type)) {
$bind = self::PARAM_FLOAT;
} elseif (preg_match('/(int|serial|bit)/is', $type)) {
$bind = PDO::PARAM_INT; $bind = PDO::PARAM_INT;
} elseif (preg_match('/bool/is', $type)) { } elseif (preg_match('/bool/is', $type)) {
$bind = PDO::PARAM_BOOL; $bind = PDO::PARAM_BOOL;
@ -1343,13 +1346,14 @@ abstract class Connection
} }
if (is_null($field)) { if (is_null($field)) {
$field = '*'; $field = ['*'];
} elseif ($key && '*' != $field) { } elseif (is_string($field)) {
$field = $key . ',' . $field; $field = array_map('trim', explode(',', $field));
} }
if (is_string($field)) { if ($key && ['*'] != $field) {
$field = array_map('trim', explode(',', $field)); array_unshift($field, $key);
$field = array_unique($field);
} }
$query->setOption('field', $field); $query->setOption('field', $field);
@ -1357,6 +1361,7 @@ abstract class Connection
// 生成查询SQL // 生成查询SQL
$sql = $this->builder->select($query); $sql = $this->builder->select($query);
// 还原field参数
if (isset($options['field'])) { if (isset($options['field'])) {
$query->setOption('field', $options['field']); $query->setOption('field', $options['field']);
} else { } else {
@ -1378,7 +1383,7 @@ abstract class Connection
} else { } else {
$resultSet = $pdo->fetchAll(PDO::FETCH_ASSOC); $resultSet = $pdo->fetchAll(PDO::FETCH_ASSOC);
if ('*' == $field && $key) { if (['*'] == $field && $key) {
$result = array_column($resultSet, null, $key); $result = array_column($resultSet, null, $key);
} elseif ($resultSet) { } elseif ($resultSet) {
$fields = array_keys($resultSet[0]); $fields = array_keys($resultSet[0]);
@ -1454,10 +1459,10 @@ abstract class Connection
$value = is_array($val) ? $val[0] : $val; $value = is_array($val) ? $val[0] : $val;
$type = is_array($val) ? $val[1] : PDO::PARAM_STR; $type = is_array($val) ? $val[1] : PDO::PARAM_STR;
if (PDO::PARAM_STR == $type) { if (PDO::PARAM_INT == $type || self::PARAM_FLOAT == $type) {
$value = '\'' . addslashes($value) . '\'';
} elseif (PDO::PARAM_INT == $type) {
$value = (float) $value; $value = (float) $value;
} elseif (PDO::PARAM_STR == $type) {
$value = '\'' . addslashes($value) . '\'';
} }
// 判断占位符 // 判断占位符
@ -1490,7 +1495,11 @@ abstract class Connection
if (is_array($val)) { if (is_array($val)) {
if (PDO::PARAM_INT == $val[1] && '' === $val[0]) { if (PDO::PARAM_INT == $val[1] && '' === $val[0]) {
$val[0] = 0; $val[0] = 0;
} elseif (self::PARAM_FLOAT == $val[1]) {
$val[0] = (float) $val[0];
$val[1] = PDO::PARAM_STR;
} }
$result = $this->PDOStatement->bindValue($param, $val[0], $val[1]); $result = $this->PDOStatement->bindValue($param, $val[0], $val[1]);
} else { } else {
$result = $this->PDOStatement->bindValue($param, $val); $result = $this->PDOStatement->bindValue($param, $val);
@ -2023,7 +2032,7 @@ abstract class Connection
// 分布式数据库配置解析 // 分布式数据库配置解析
foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) { foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) {
$_config[$name] = explode(',', $this->config[$name]); $_config[$name] = is_string($this->config[$name]) ? explode(',', $this->config[$name]) : $this->config[$name];
} }
// 主服务器序号 // 主服务器序号

View File

@ -177,7 +177,7 @@ class Query
call_user_func_array([$this->model, $method], $args); call_user_func_array([$this->model, $method], $args);
return $this; return $this;
} else { } else {
throw new Exception('method not exist:' . static::class . '->' . $method); throw new Exception('method not exist:' . ($this->model ? get_class($this->model) : static::class) . '->' . $method);
} }
} }
@ -628,6 +628,9 @@ class Query
$result = (float) $result; $result = (float) $result;
} }
// 查询完成后清空聚合字段信息
$this->removeOption('field');
return $result; return $result;
} }
@ -653,10 +656,12 @@ class Query
$query->fetchSql(true); $query->fetchSql(true);
} }
return $query->aggregate('COUNT', '*', true); $count = $query->aggregate('COUNT', '*');
} else {
$count = $this->aggregate('COUNT', $field);
} }
return $this->aggregate('COUNT', $field, true); return is_string($count) ? $count : (int) $count;
} }
/** /**
@ -1024,17 +1029,12 @@ class Query
* 表达式方式指定查询字段 * 表达式方式指定查询字段
* @access public * @access public
* @param string $field 字段名 * @param string $field 字段名
* @param array $bind 参数绑定
* @return $this * @return $this
*/ */
public function fieldRaw($field, array $bind = []) public function fieldRaw($field)
{ {
$this->options['field'][] = $this->raw($field); $this->options['field'][] = $this->raw($field);
if ($bind) {
$this->bind($bind);
}
return $this; return $this;
} }
@ -1424,11 +1424,17 @@ class Query
*/ */
public function whereExp($field, $condition, $bind = [], $logic = 'AND') public function whereExp($field, $condition, $bind = [], $logic = 'AND')
{ {
if ($bind) {
foreach ($bind as $key => $value) {
if (!is_numeric($key)) {
$where = str_replace(':' . $key, '?', $where);
}
}
$this->bind(array_values($bind));
}
$this->options['where'][$logic][] = [$field, 'EXP', $this->raw($condition)]; $this->options['where'][$logic][] = [$field, 'EXP', $this->raw($condition)];
if ($bind) {
$this->bind($bind);
}
return $this; return $this;
} }
@ -1442,12 +1448,18 @@ class Query
*/ */
public function whereRaw($where, $bind = [], $logic = 'AND') public function whereRaw($where, $bind = [], $logic = 'AND')
{ {
$this->options['where'][$logic][] = $this->raw($where);
if ($bind) { if ($bind) {
$this->bind($bind); foreach ($bind as $key => $value) {
if (!is_numeric($key)) {
$where = str_replace(':' . $key, '?', $where);
}
}
$this->bind(array_values($bind));
} }
$this->options['where'][$logic][] = $this->raw($where);
return $this; return $this;
} }
@ -1875,17 +1887,12 @@ class Query
* 表达式方式指定Field排序 * 表达式方式指定Field排序
* @access public * @access public
* @param string $field 排序字段 * @param string $field 排序字段
* @param array $bind 参数绑定
* @return $this * @return $this
*/ */
public function orderRaw($field, array $bind = []) public function orderRaw($field)
{ {
$this->options['order'][] = $this->raw($field); $this->options['order'][] = $this->raw($field);
if ($bind) {
$this->bind($bind);
}
return $this; return $this;
} }
@ -2424,17 +2431,16 @@ class Query
/** /**
* 参数绑定 * 参数绑定
* @access public * @access public
* @param mixed $key 参数名
* @param mixed $value 绑定变量值 * @param mixed $value 绑定变量值
* @param integer $type 绑定类型 * @param integer $type 绑定类型
* @return $this * @return $this
*/ */
public function bind($key, $value = false, $type = PDO::PARAM_STR) public function bind($value = false, $type = PDO::PARAM_STR)
{ {
if (is_array($key)) { if (is_array($value)) {
$this->bind = array_merge($this->bind, $key); $this->bind = array_merge($this->bind, $value);
} else { } else {
$this->bind[$key] = [$value, $type]; $this->bind[] = [$value, $type];
} }
return $this; return $this;
@ -3197,7 +3203,7 @@ class Query
{ {
$result = $this->with($with)->cache($cache); $result = $this->with($with)->cache($cache);
if (is_array($data) && key($data) !== 0) { if ((is_array($data) && key($data) !== 0) || $data instanceof Where) {
$result = $result->where($data); $result = $result->where($data);
$data = null; $data = null;
} elseif ($data instanceof \Closure) { } elseif ($data instanceof \Closure) {

View File

@ -79,8 +79,7 @@ class Where implements ArrayAccess
* 分析查询表达式 * 分析查询表达式
* @access protected * @access protected
* @param string $field 查询字段 * @param string $field 查询字段
* @param string $op 查询表达式 * @param array $where 查询条件
* @param mixed $condition 查询条件
* @return array * @return array
*/ */
protected function parseItem($field, $where = []) protected function parseItem($field, $where = [])

View File

@ -129,6 +129,17 @@ class File
*/ */
protected function getMasterLogFile() protected function getMasterLogFile()
{ {
if ($this->config['max_files']) {
$files = glob($this->config['path'] . '*.log');
try {
if (count($files) > $this->config['max_files']) {
unlink($files[0]);
}
} catch (\Exception $e) {
}
}
if ($this->config['single']) { if ($this->config['single']) {
$name = is_string($this->config['single']) ? $this->config['single'] : 'single'; $name = is_string($this->config['single']) ? $this->config['single'] : 'single';
@ -138,14 +149,6 @@ class File
if ($this->config['max_files']) { if ($this->config['max_files']) {
$filename = date('Ymd') . $cli . '.log'; $filename = date('Ymd') . $cli . '.log';
$files = glob($this->config['path'] . '*.log');
try {
if (count($files) > $this->config['max_files']) {
unlink($files[0]);
}
} catch (\Exception $e) {
}
} else { } else {
$filename = date('Ym') . DIRECTORY_SEPARATOR . date('d') . $cli . '.log'; $filename = date('Ym') . DIRECTORY_SEPARATOR . date('d') . $cli . '.log';
} }

View File

@ -30,6 +30,8 @@ class Socket
'force_client_ids' => [], 'force_client_ids' => [],
// 限制允许读取日志的client_id // 限制允许读取日志的client_id
'allow_client_ids' => [], 'allow_client_ids' => [],
//输出到浏览器默认展开的日志级别
'expand_level' => ['debug'],
]; ];
protected $css = [ protected $css = [
@ -95,7 +97,7 @@ class Socket
foreach ($log as $type => $val) { foreach ($log as $type => $val) {
$trace[] = [ $trace[] = [
'type' => 'groupCollapsed', 'type' => in_array($type, $this->config['expand_level']) ? 'group' : 'groupCollapsed',
'msg' => '[ ' . $type . ' ]', 'msg' => '[ ' . $type . ' ]',
'css' => isset($this->css[$type]) ? $this->css[$type] : '', 'css' => isset($this->css[$type]) ? $this->css[$type] : '',
]; ];

View File

@ -68,6 +68,26 @@ class BelongsTo extends OneToOne
return $relationModel; return $relationModel;
} }
/**
* 创建关联统计子查询
* @access public
* @param \Closure $closure 闭包
* @param string $aggregate 聚合查询方法
* @param string $field 字段
* @return string
*/
public function getRelationCountQuery($closure, $aggregate = 'count', $field = '*')
{
if ($closure) {
$closure($this->query);
}
return $this->query
->whereExp($this->localKey, '=' . $this->parent->getTable() . '.' . $this->foreignKey)
->fetchSql()
->$aggregate($field);
}
/** /**
* 根据关联条件查询当前模型 * 根据关联条件查询当前模型
* @access public * @access public

View File

@ -68,6 +68,26 @@ class HasOne extends OneToOne
return $relationModel; return $relationModel;
} }
/**
* 创建关联统计子查询
* @access public
* @param \Closure $closure 闭包
* @param string $aggregate 聚合查询方法
* @param string $field 字段
* @return string
*/
public function getRelationCountQuery($closure, $aggregate = 'count', $field = '*')
{
if ($closure) {
$closure($this->query);
}
return $this->query
->whereExp($this->foreignKey, '=' . $this->parent->getTable() . '.' . $this->parent->getPk())
->fetchSql()
->$aggregate($field);
}
/** /**
* 根据关联条件查询当前模型 * 根据关联条件查询当前模型
* @access public * @access public

View File

@ -195,7 +195,7 @@ class MorphMany extends Relation
if (isset($result->$pk)) { if (isset($result->$pk)) {
if ($closure) { if ($closure) {
$closur($this->query); $closure($this->query);
} }
$count = $this->query $count = $this->query

View File

@ -99,15 +99,18 @@ class Redirect extends Response
/** /**
* 跳转到上次记住的url * 跳转到上次记住的url
* @access public * @access public
* @param string $url 闪存数据不存在时的跳转地址
* @return $this * @return $this
*/ */
public function restore() public function restore($url = null)
{ {
$session = $this->app['session']; $session = $this->app['session'];
if ($session->has('redirect_url')) { if ($session->has('redirect_url')) {
$this->data = $session->get('redirect_url'); $this->data = $session->get('redirect_url');
$session->delete('redirect_url'); $session->delete('redirect_url');
} elseif ($url) {
$this->data = $url;
} }
return $this; return $this;

View File

@ -75,6 +75,11 @@ class AliasRule extends Domain
$this->mergeGroupOptions(); $this->mergeGroupOptions();
} }
if (isset($this->option['ext'])) {
// 路由ext参数 优先于系统配置的URL伪静态后缀参数
$bind = preg_replace('/\.(' . $request->ext() . ')$/i', '', $bind);
}
$this->parseBindAppendParam($this->route); $this->parseBindAppendParam($this->route);
if (0 === strpos($this->route, '\\')) { if (0 === strpos($this->route, '\\')) {

View File

@ -584,4 +584,22 @@ class RuleGroup extends Rule
return isset($this->rules[strtolower($method)]) ? $this->rules[strtolower($method)] : []; return isset($this->rules[strtolower($method)]) ? $this->rules[strtolower($method)] : [];
} }
/**
* 清空分组下的路由规则
* @access public
* @return void
*/
public function clear()
{
$this->rules = [
'*' => [],
'get' => [],
'post' => [],
'put' => [],
'patch' => [],
'delete' => [],
'head' => [],
'options' => [],
];
}
} }

View File

@ -133,4 +133,14 @@ class RuleName
return $result; return $result;
} }
/**
* 清空路由规则
* @access public
* @return void
*/
public function clear()
{
$this->item = [];
$this->rule = [];
}
} }

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 ComposerAutoloaderInit1d3094fd985c01a522304485af2b5f26::getLoader(); return ComposerAutoloaderInit7ac0a16c6bb261e6fc9b66fbcc229d16::getLoader();

View File

@ -2,7 +2,7 @@
// autoload_real.php @generated by Composer // autoload_real.php @generated by Composer
class ComposerAutoloaderInit1d3094fd985c01a522304485af2b5f26 class ComposerAutoloaderInit7ac0a16c6bb261e6fc9b66fbcc229d16
{ {
private static $loader; private static $loader;
@ -19,15 +19,15 @@ class ComposerAutoloaderInit1d3094fd985c01a522304485af2b5f26
return self::$loader; return self::$loader;
} }
spl_autoload_register(array('ComposerAutoloaderInit1d3094fd985c01a522304485af2b5f26', 'loadClassLoader'), true, true); spl_autoload_register(array('ComposerAutoloaderInit7ac0a16c6bb261e6fc9b66fbcc229d16', 'loadClassLoader'), true, true);
self::$loader = $loader = new \Composer\Autoload\ClassLoader(); self::$loader = $loader = new \Composer\Autoload\ClassLoader();
spl_autoload_unregister(array('ComposerAutoloaderInit1d3094fd985c01a522304485af2b5f26', 'loadClassLoader')); spl_autoload_unregister(array('ComposerAutoloaderInit7ac0a16c6bb261e6fc9b66fbcc229d16', '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\ComposerStaticInit1d3094fd985c01a522304485af2b5f26::getInitializer($loader)); call_user_func(\Composer\Autoload\ComposerStaticInit7ac0a16c6bb261e6fc9b66fbcc229d16::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 ComposerAutoloaderInit1d3094fd985c01a522304485af2b5f26
$loader->register(true); $loader->register(true);
if ($useStaticLoader) { if ($useStaticLoader) {
$includeFiles = Composer\Autoload\ComposerStaticInit1d3094fd985c01a522304485af2b5f26::$files; $includeFiles = Composer\Autoload\ComposerStaticInit7ac0a16c6bb261e6fc9b66fbcc229d16::$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) {
composerRequire1d3094fd985c01a522304485af2b5f26($fileIdentifier, $file); composerRequire7ac0a16c6bb261e6fc9b66fbcc229d16($fileIdentifier, $file);
} }
return $loader; return $loader;
} }
} }
function composerRequire1d3094fd985c01a522304485af2b5f26($fileIdentifier, $file) function composerRequire7ac0a16c6bb261e6fc9b66fbcc229d16($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 ComposerStaticInit1d3094fd985c01a522304485af2b5f26 class ComposerStaticInit7ac0a16c6bb261e6fc9b66fbcc229d16
{ {
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 ComposerStaticInit1d3094fd985c01a522304485af2b5f26
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 = ComposerStaticInit1d3094fd985c01a522304485af2b5f26::$prefixLengthsPsr4; $loader->prefixLengthsPsr4 = ComposerStaticInit7ac0a16c6bb261e6fc9b66fbcc229d16::$prefixLengthsPsr4;
$loader->prefixDirsPsr4 = ComposerStaticInit1d3094fd985c01a522304485af2b5f26::$prefixDirsPsr4; $loader->prefixDirsPsr4 = ComposerStaticInit7ac0a16c6bb261e6fc9b66fbcc229d16::$prefixDirsPsr4;
$loader->classMap = ComposerStaticInit1d3094fd985c01a522304485af2b5f26::$classMap; $loader->classMap = ComposerStaticInit7ac0a16c6bb261e6fc9b66fbcc229d16::$classMap;
}, null, ClassLoader::class); }, null, ClassLoader::class);
} }

View File

@ -50,17 +50,17 @@
}, },
{ {
"name": "topthink/framework", "name": "topthink/framework",
"version": "v5.1.24", "version": "v5.1.25",
"version_normalized": "5.1.24.0", "version_normalized": "5.1.25.0",
"source": { "source": {
"type": "git", "type": "git",
"url": "https://github.com/top-think/framework.git", "url": "https://github.com/top-think/framework.git",
"reference": "03ee17d1ea9c432a698fa1d2411df53974420187" "reference": "2e684d5fa38bd2d8ee864982f3114a28992da895"
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://api.github.com/repos/top-think/framework/zipball/03ee17d1ea9c432a698fa1d2411df53974420187", "url": "https://api.github.com/repos/top-think/framework/zipball/2e684d5fa38bd2d8ee864982f3114a28992da895",
"reference": "03ee17d1ea9c432a698fa1d2411df53974420187", "reference": "2e684d5fa38bd2d8ee864982f3114a28992da895",
"shasum": "", "shasum": "",
"mirrors": [ "mirrors": [
{ {
@ -82,7 +82,7 @@
"sebastian/phpcpd": "2.*", "sebastian/phpcpd": "2.*",
"squizlabs/php_codesniffer": "2.*" "squizlabs/php_codesniffer": "2.*"
}, },
"time": "2018-09-05T03:37:46+00:00", "time": "2018-10-04T05:04:50+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/",
@ -215,17 +215,17 @@
}, },
{ {
"name": "symfony/options-resolver", "name": "symfony/options-resolver",
"version": "v3.4.15", "version": "v3.4.17",
"version_normalized": "3.4.15.0", "version_normalized": "3.4.17.0",
"source": { "source": {
"type": "git", "type": "git",
"url": "https://github.com/symfony/options-resolver.git", "url": "https://github.com/symfony/options-resolver.git",
"reference": "6debc476953a45969ab39afe8dee0b825f356dc7" "reference": "1cf7d8e704a9cc4164c92e430f2dfa3e6983661d"
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://api.github.com/repos/symfony/options-resolver/zipball/6debc476953a45969ab39afe8dee0b825f356dc7", "url": "https://api.github.com/repos/symfony/options-resolver/zipball/1cf7d8e704a9cc4164c92e430f2dfa3e6983661d",
"reference": "6debc476953a45969ab39afe8dee0b825f356dc7", "reference": "1cf7d8e704a9cc4164c92e430f2dfa3e6983661d",
"shasum": "", "shasum": "",
"mirrors": [ "mirrors": [
{ {
@ -237,7 +237,7 @@
"require": { "require": {
"php": "^5.5.9|>=7.0.8" "php": "^5.5.9|>=7.0.8"
}, },
"time": "2018-07-26T08:45:46+00:00", "time": "2018-09-17T17:29:18+00:00",
"type": "library", "type": "library",
"extra": { "extra": {
"branch-alias": { "branch-alias": {
@ -443,17 +443,17 @@
}, },
{ {
"name": "zoujingli/wechat-developer", "name": "zoujingli/wechat-developer",
"version": "v1.1.13", "version": "v1.1.14",
"version_normalized": "1.1.13.0", "version_normalized": "1.1.14.0",
"source": { "source": {
"type": "git", "type": "git",
"url": "https://github.com/zoujingli/WeChatDeveloper.git", "url": "https://github.com/zoujingli/WeChatDeveloper.git",
"reference": "39682a047e5c8f9c66107924dfed4e2080e97eb4" "reference": "328d21d32c3d06f5f050185a740f00fe475b6a03"
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://api.github.com/repos/zoujingli/WeChatDeveloper/zipball/39682a047e5c8f9c66107924dfed4e2080e97eb4", "url": "https://api.github.com/repos/zoujingli/WeChatDeveloper/zipball/328d21d32c3d06f5f050185a740f00fe475b6a03",
"reference": "39682a047e5c8f9c66107924dfed4e2080e97eb4", "reference": "328d21d32c3d06f5f050185a740f00fe475b6a03",
"shasum": "", "shasum": "",
"mirrors": [ "mirrors": [
{ {
@ -467,7 +467,7 @@
"ext-openssl": "*", "ext-openssl": "*",
"php": ">=5.4" "php": ">=5.4"
}, },
"time": "2018-07-11T02:02:41+00:00", "time": "2018-09-12T06:12:49+00:00",
"type": "library", "type": "library",
"installation-source": "dist", "installation-source": "dist",
"autoload": { "autoload": {

View File

@ -353,11 +353,9 @@ class OptionsResolver implements Options
* *
* The normalizer should be a closure with the following signature: * The normalizer should be a closure with the following signature:
* *
* ```php * function (Options $options, $value) {
* function (Options $options, $value) { * // ...
* // ... * }
* }
* ```
* *
* The closure is invoked when {@link resolve()} is called. The closure * The closure is invoked when {@link resolve()} is called. The closure
* has access to the resolved values of other options through the passed * has access to the resolved values of other options through the passed
@ -383,11 +381,7 @@ class OptionsResolver implements Options
} }
if (!isset($this->defined[$option])) { if (!isset($this->defined[$option])) {
throw new UndefinedOptionsException(sprintf( throw new UndefinedOptionsException(sprintf('The option "%s" does not exist. Defined options are: "%s".', $option, implode('", "', array_keys($this->defined))));
'The option "%s" does not exist. Defined options are: "%s".',
$option,
implode('", "', array_keys($this->defined))
));
} }
$this->normalizers[$option] = $normalizer; $this->normalizers[$option] = $normalizer;
@ -426,11 +420,7 @@ class OptionsResolver implements Options
} }
if (!isset($this->defined[$option])) { if (!isset($this->defined[$option])) {
throw new UndefinedOptionsException(sprintf( throw new UndefinedOptionsException(sprintf('The option "%s" does not exist. Defined options are: "%s".', $option, implode('", "', array_keys($this->defined))));
'The option "%s" does not exist. Defined options are: "%s".',
$option,
implode('", "', array_keys($this->defined))
));
} }
$this->allowedValues[$option] = \is_array($allowedValues) ? $allowedValues : array($allowedValues); $this->allowedValues[$option] = \is_array($allowedValues) ? $allowedValues : array($allowedValues);
@ -471,11 +461,7 @@ class OptionsResolver implements Options
} }
if (!isset($this->defined[$option])) { if (!isset($this->defined[$option])) {
throw new UndefinedOptionsException(sprintf( throw new UndefinedOptionsException(sprintf('The option "%s" does not exist. Defined options are: "%s".', $option, implode('", "', array_keys($this->defined))));
'The option "%s" does not exist. Defined options are: "%s".',
$option,
implode('", "', array_keys($this->defined))
));
} }
if (!\is_array($allowedValues)) { if (!\is_array($allowedValues)) {
@ -516,11 +502,7 @@ class OptionsResolver implements Options
} }
if (!isset($this->defined[$option])) { if (!isset($this->defined[$option])) {
throw new UndefinedOptionsException(sprintf( throw new UndefinedOptionsException(sprintf('The option "%s" does not exist. Defined options are: "%s".', $option, implode('", "', array_keys($this->defined))));
'The option "%s" does not exist. Defined options are: "%s".',
$option,
implode('", "', array_keys($this->defined))
));
} }
$this->allowedTypes[$option] = (array) $allowedTypes; $this->allowedTypes[$option] = (array) $allowedTypes;
@ -555,11 +537,7 @@ class OptionsResolver implements Options
} }
if (!isset($this->defined[$option])) { if (!isset($this->defined[$option])) {
throw new UndefinedOptionsException(sprintf( throw new UndefinedOptionsException(sprintf('The option "%s" does not exist. Defined options are: "%s".', $option, implode('", "', array_keys($this->defined))));
'The option "%s" does not exist. Defined options are: "%s".',
$option,
implode('", "', array_keys($this->defined))
));
} }
if (!isset($this->allowedTypes[$option])) { if (!isset($this->allowedTypes[$option])) {
@ -664,11 +642,7 @@ class OptionsResolver implements Options
ksort($clone->defined); ksort($clone->defined);
ksort($diff); ksort($diff);
throw new UndefinedOptionsException(sprintf( throw new UndefinedOptionsException(sprintf((\count($diff) > 1 ? 'The options "%s" do not exist.' : 'The option "%s" does not exist.').' Defined options are: "%s".', implode('", "', array_keys($diff)), implode('", "', array_keys($clone->defined))));
(\count($diff) > 1 ? 'The options "%s" do not exist.' : 'The option "%s" does not exist.').' Defined options are: "%s".',
implode('", "', array_keys($diff)),
implode('", "', array_keys($clone->defined))
));
} }
// Override options set by the user // Override options set by the user
@ -683,10 +657,7 @@ class OptionsResolver implements Options
if (\count($diff) > 0) { if (\count($diff) > 0) {
ksort($diff); ksort($diff);
throw new MissingOptionsException(sprintf( throw new MissingOptionsException(sprintf(\count($diff) > 1 ? 'The required options "%s" are missing.' : 'The required option "%s" is missing.', implode('", "', array_keys($diff))));
\count($diff) > 1 ? 'The required options "%s" are missing.' : 'The required option "%s" is missing.',
implode('", "', array_keys($diff))
));
} }
// Lock the container // Lock the container
@ -730,17 +701,10 @@ class OptionsResolver implements Options
// Check whether the option is set at all // Check whether the option is set at all
if (!array_key_exists($option, $this->defaults)) { if (!array_key_exists($option, $this->defaults)) {
if (!isset($this->defined[$option])) { if (!isset($this->defined[$option])) {
throw new NoSuchOptionException(sprintf( throw new NoSuchOptionException(sprintf('The option "%s" does not exist. Defined options are: "%s".', $option, implode('", "', array_keys($this->defined))));
'The option "%s" does not exist. Defined options are: "%s".',
$option,
implode('", "', array_keys($this->defined))
));
} }
throw new NoSuchOptionException(sprintf( throw new NoSuchOptionException(sprintf('The optional option "%s" has no value set. You should make sure it is set with "isset" before reading it.', $option));
'The optional option "%s" has no value set. You should make sure it is set with "isset" before reading it.',
$option
));
} }
$value = $this->defaults[$option]; $value = $this->defaults[$option];
@ -750,10 +714,7 @@ class OptionsResolver implements Options
// If the closure is already being called, we have a cyclic // If the closure is already being called, we have a cyclic
// dependency // dependency
if (isset($this->calling[$option])) { if (isset($this->calling[$option])) {
throw new OptionDefinitionException(sprintf( throw new OptionDefinitionException(sprintf('The options "%s" have a cyclic dependency.', implode('", "', array_keys($this->calling))));
'The options "%s" have a cyclic dependency.',
implode('", "', array_keys($this->calling))
));
} }
// The following section must be protected from cyclic // The following section must be protected from cyclic
@ -809,7 +770,9 @@ class OptionsResolver implements Options
// Don't include closures in the exception message // Don't include closures in the exception message
continue; continue;
} elseif ($value === $allowedValue) { }
if ($value === $allowedValue) {
$success = true; $success = true;
break; break;
} }
@ -840,10 +803,7 @@ class OptionsResolver implements Options
// If the closure is already being called, we have a cyclic // If the closure is already being called, we have a cyclic
// dependency // dependency
if (isset($this->calling[$option])) { if (isset($this->calling[$option])) {
throw new OptionDefinitionException(sprintf( throw new OptionDefinitionException(sprintf('The options "%s" have a cyclic dependency.', implode('", "', array_keys($this->calling))));
'The options "%s" have a cyclic dependency.',
implode('", "', array_keys($this->calling))
));
} }
$normalizer = $this->normalizers[$option]; $normalizer = $this->normalizers[$option];
@ -1102,20 +1062,4 @@ class OptionsResolver implements Options
{ {
return (\function_exists($isFunction = 'is_'.$type) && $isFunction($value)) || $value instanceof $type; return (\function_exists($isFunction = 'is_'.$type) && $isFunction($value)) || $value instanceof $type;
} }
/**
* @return array
*/
private function getInvalidValues(array $arrayValues, $type)
{
$invalidValues = array();
foreach ($arrayValues as $key => $value) {
if (!self::isValueValidType($type, $value)) {
$invalidValues[$key] = $value;
}
}
return $invalidValues;
}
} }

View File

@ -108,6 +108,25 @@ class BasicWeChat
return $this->access_token = $result['access_token']; return $this->access_token = $result['access_token'];
} }
/**
* 设置外部接口 AccessToken
* @param string $access_token
* @throws \WeChat\Exceptions\LocalCacheException
* @author 高一平 <iam@gaoyiping.com>
*
* 当用户使用自己的缓存驱动时,直接实例化对象后可直接设置 AccessToekn
* - 多用于分布式项目时保持 AccessToken 统一
* - 使用此方法后就由用户来保证传入的 AccessToekn 为有效 AccessToekn
*/
public function setAccessToken($access_token)
{
if (!is_string($access_token)) {
throw new InvalidArgumentException("Invalid AccessToken type, need string.");
}
$cache = $this->config->get('appid') . '_access_token';
Tools::setCache($cache, $this->access_token = $access_token);
}
/** /**
* 清理删除accessToken * 清理删除accessToken
* @return bool * @return bool

View File

@ -28,14 +28,13 @@ class Custom extends BasicWeChat
* 添加客服帐号 * 添加客服帐号
* @param string $kf_account 客服账号 * @param string $kf_account 客服账号
* @param string $nickname 客服昵称 * @param string $nickname 客服昵称
* @param string $password 账号密码
* @return array * @return array
* @throws Exceptions\InvalidResponseException * @throws Exceptions\InvalidResponseException
* @throws Exceptions\LocalCacheException * @throws Exceptions\LocalCacheException
*/ */
public function addAccount($kf_account, $nickname, $password) public function addAccount($kf_account, $nickname)
{ {
$data = ['kf_account' => $kf_account, 'nickname' => $nickname, 'password' => $password]; $data = ['kf_account' => $kf_account, 'nickname' => $nickname];
$url = "https://api.weixin.qq.com/customservice/kfaccount/add?access_token=ACCESS_TOKEN"; $url = "https://api.weixin.qq.com/customservice/kfaccount/add?access_token=ACCESS_TOKEN";
$this->registerApi($url, __FUNCTION__, func_get_args()); $this->registerApi($url, __FUNCTION__, func_get_args());
return $this->httpPostForJson($url, $data); return $this->httpPostForJson($url, $data);
@ -45,14 +44,13 @@ class Custom extends BasicWeChat
* 修改客服帐号 * 修改客服帐号
* @param string $kf_account 客服账号 * @param string $kf_account 客服账号
* @param string $nickname 客服昵称 * @param string $nickname 客服昵称
* @param string $password 账号密码
* @return array * @return array
* @throws Exceptions\InvalidResponseException * @throws Exceptions\InvalidResponseException
* @throws Exceptions\LocalCacheException * @throws Exceptions\LocalCacheException
*/ */
public function updateAccount($kf_account, $nickname, $password) public function updateAccount($kf_account, $nickname)
{ {
$data = ['kf_account' => $kf_account, 'nickname' => $nickname, 'password' => $password]; $data = ['kf_account' => $kf_account, 'nickname' => $nickname];
$url = "https://api.weixin.qq.com/customservice/kfaccount/update?access_token=ACCESS_TOKEN"; $url = "https://api.weixin.qq.com/customservice/kfaccount/update?access_token=ACCESS_TOKEN";
$this->registerApi($url, __FUNCTION__, func_get_args()); $this->registerApi($url, __FUNCTION__, func_get_args());
return $this->httpPostForJson($url, $data); return $this->httpPostForJson($url, $data);
@ -61,15 +59,13 @@ class Custom extends BasicWeChat
/** /**
* 删除客服帐号 * 删除客服帐号
* @param string $kf_account 客服账号 * @param string $kf_account 客服账号
* @param string $nickname 客服昵称
* @param string $password 账号密码
* @return array * @return array
* @throws Exceptions\InvalidResponseException * @throws Exceptions\InvalidResponseException
* @throws Exceptions\LocalCacheException * @throws Exceptions\LocalCacheException
*/ */
public function deleteAccount($kf_account, $nickname, $password) public function deleteAccount($kf_account)
{ {
$data = ['kf_account' => $kf_account, 'nickname' => $nickname, 'password' => $password]; $data = ['kf_account' => $kf_account];
$url = "https://api.weixin.qq.com/customservice/kfaccount/del?access_token=ACCESS_TOKEN"; $url = "https://api.weixin.qq.com/customservice/kfaccount/del?access_token=ACCESS_TOKEN";
$this->registerApi($url, __FUNCTION__, func_get_args()); $this->registerApi($url, __FUNCTION__, func_get_args());
return $this->httpPostForJson($url, $data); return $this->httpPostForJson($url, $data);

View File

@ -15,6 +15,8 @@
namespace WePay; namespace WePay;
use WeChat\Contracts\BasicPay; use WeChat\Contracts\BasicPay;
use WeChat\Contracts\Tools;
use WeChat\Exceptions\InvalidResponseException;
/** /**
* 微信商户退款 * 微信商户退款
@ -48,4 +50,27 @@ class Refund extends BasicPay
return $this->callPostApi($url, $options); return $this->callPostApi($url, $options);
} }
/**
* 获取退款通知
* @return array
* @throws InvalidResponseException
*/
public function getNotify()
{
$data = Tools::xml2arr(file_get_contents("php://input"));
if (!isset($data['return_code']) || $data['return_code'] !== 'SUCCESS') {
throw new InvalidResponseException('获取退款通知XML失败');
}
if (!class_exists('Prpcrypt', false)) {
include dirname(__DIR__) . '/WeChat/Contracts/Prpcrypt.php';
}
$pc = new \Prpcrypt(md5($this->config->get('mch_key')));
$array = $pc->decrypt(base64_decode($data['req_info']));
if (intval($array[0]) > 0) {
throw new InvalidResponseException($array[1], $array[0]);
}
$data['decode'] = $array[1];
return $data;
}
} }