重新数据库驱动、增加更多模型高级操作
This commit is contained in:
parent
10ec74c2f7
commit
10a993a1d6
|
@ -21,9 +21,9 @@ class Framework
|
|||
/**
|
||||
* 框架入口
|
||||
* @param string $callable
|
||||
* @param array $namespaceMap
|
||||
* @param array $autoLoadMap
|
||||
*/
|
||||
public static function startApp($callable = '', $namespaceMap = [])
|
||||
public static function startApp($callable = '', $autoLoadMap = [])
|
||||
{
|
||||
|
||||
(is_callable($callable)) && $callable(self::class);
|
||||
|
@ -45,7 +45,7 @@ class Framework
|
|||
!defined('CONFIG_DIR') && define('CONFIG_DIR', APP_PATH . BIND_MODULE . DS . 'config' . DS);
|
||||
|
||||
require 'library/Application.php';
|
||||
Application::run($namespaceMap);
|
||||
Application::run($autoLoadMap);
|
||||
} else echo '请使用Framework::appPath()指定应用目录';
|
||||
}
|
||||
|
||||
|
|
|
@ -18,7 +18,7 @@ return [
|
|||
'prefix' => '',
|
||||
],
|
||||
'db' => [
|
||||
'driver' => 'MySQLi',
|
||||
'driver' => 'Mysql',
|
||||
'host' => '127.0.0.1',
|
||||
'user' => '',
|
||||
'passwd' => '',
|
||||
|
|
|
@ -7,24 +7,37 @@ use top\library\exception\BaseException;
|
|||
use top\library\http\Request;
|
||||
use top\library\http\Response;
|
||||
|
||||
/**
|
||||
* Class Application
|
||||
* @package top\library
|
||||
*/
|
||||
class Application
|
||||
{
|
||||
|
||||
/**
|
||||
* 已获取到的反射实例
|
||||
* @var array
|
||||
*/
|
||||
private static $reflectionClass = [];
|
||||
|
||||
/**
|
||||
* 已获取到的反射方法实例
|
||||
* @var array
|
||||
*/
|
||||
private static $reflectionMethod = [];
|
||||
|
||||
/**
|
||||
* 开始执行程序
|
||||
* @param array $namespaceMap
|
||||
* @param array $autoLoadMap
|
||||
*/
|
||||
public static function run($namespaceMap = [])
|
||||
public static function run($autoLoadMap = [])
|
||||
{
|
||||
// 注册框架自动加载
|
||||
require 'Loader.php';
|
||||
$loader = new Loader();
|
||||
$loader->set('top', FRAMEWORK_PATH);
|
||||
$loader->set(APP_NS, APP_PATH);
|
||||
foreach ($namespaceMap as $prefix => $path) {
|
||||
foreach ($autoLoadMap as $prefix => $path) {
|
||||
$loader->set($prefix, $path);
|
||||
}
|
||||
$loader->register();
|
||||
|
@ -41,51 +54,22 @@ class Application
|
|||
(PHP_VERSION > 5.6) && set_error_handler([new BaseError(), 'handler']);
|
||||
set_exception_handler([new BaseException(), 'handler']);
|
||||
|
||||
// 加载必要文件
|
||||
self::loadFiles();
|
||||
|
||||
// 初始化路由实例
|
||||
$router = Router::instance(Request::instance());
|
||||
|
||||
// 处理请求并得到数据
|
||||
$responseData = Response::instance()->header([
|
||||
'X-Powered-By: TOP-Framework'
|
||||
])->send($router->execute());
|
||||
|
||||
// 输出响应内容
|
||||
echo $responseData->content;
|
||||
}
|
||||
|
||||
/**
|
||||
* 加载必要文件
|
||||
*/
|
||||
private static function loadFiles()
|
||||
{
|
||||
// 加载系统函数库
|
||||
// 系统函数库
|
||||
require FRAMEWORK_PATH . 'library' . DS . 'functions' . DS . 'functions.php';
|
||||
|
||||
// 加载用户函数库
|
||||
// 用户函数库
|
||||
$funcFile = APP_PATH . BIND_MODULE . DS . 'functions.php';
|
||||
(is_file($funcFile)) && require $funcFile;
|
||||
|
||||
$configInstance = Config::instance();
|
||||
|
||||
$sessionConfig = $configInstance->get('session');
|
||||
// session目录
|
||||
$sessionConfig = Config::instance()->get('session');
|
||||
if (!empty($sessionConfig) && $sessionConfig['open'] === true) {
|
||||
session_save_path(SESSION_PATH);
|
||||
session_start();
|
||||
}
|
||||
|
||||
// 数据库驱动
|
||||
$config = $configInstance->get('db');
|
||||
$driver = $config['driver'] ? $config['driver'] : 'MySQLi';
|
||||
Register::set('DBDriver', function () use ($driver) {
|
||||
$class = '\\top\\library\\database\\driver\\' . $driver;
|
||||
return $class::instance();
|
||||
});
|
||||
|
||||
// 配置文件中配置的注册
|
||||
$initRegister = $configInstance->get('register');
|
||||
// 配置文件中注册的类
|
||||
$initRegister = Config::instance()->get('register');
|
||||
if (!empty($initRegister)) {
|
||||
foreach ($initRegister as $key => $value) {
|
||||
Register::set($key, function () use ($value) {
|
||||
|
@ -93,6 +77,18 @@ class Application
|
|||
});
|
||||
}
|
||||
}
|
||||
|
||||
// 初始化路由实例
|
||||
$router = Router::instance(Request::instance());
|
||||
|
||||
// 处理请求并得到数据
|
||||
$response = Response::instance()->header([
|
||||
'X-Powered-By: TOP-Framework'
|
||||
])->send($router->execute());
|
||||
|
||||
// 响应内容
|
||||
echo $response->content;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -49,7 +49,6 @@ class Config
|
|||
* 获取配置
|
||||
* @param string $name
|
||||
* @return array|mixed
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function get($name = '')
|
||||
{
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
namespace top\library;
|
||||
|
||||
use top\library\http\Request;
|
||||
use top\traits\Magic;
|
||||
use top\traits\Json;
|
||||
|
||||
/**
|
||||
|
@ -12,6 +13,8 @@ use top\traits\Json;
|
|||
abstract class Controller
|
||||
{
|
||||
|
||||
use Magic;
|
||||
|
||||
use Json;
|
||||
|
||||
/**
|
||||
|
@ -133,4 +136,5 @@ abstract class Controller
|
|||
]);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -2,7 +2,8 @@
|
|||
|
||||
namespace top\library;
|
||||
|
||||
use top\library\database\ifs\DatabaseIfs;
|
||||
use top\library\database\Base;
|
||||
use top\library\exception\DatabaseException;
|
||||
|
||||
/**
|
||||
* 数据库操作类
|
||||
|
@ -12,10 +13,10 @@ class Database
|
|||
{
|
||||
|
||||
/**
|
||||
* 数据库驱动
|
||||
* @var null
|
||||
* 数据库连接
|
||||
* @var Base
|
||||
*/
|
||||
private static $driver = null;
|
||||
private static $connection = null;
|
||||
|
||||
/**
|
||||
* 当前类实例
|
||||
|
@ -23,12 +24,6 @@ class Database
|
|||
*/
|
||||
private static $instance = [];
|
||||
|
||||
/**
|
||||
* 当前表结构
|
||||
* @var array
|
||||
*/
|
||||
private static $tableDesc = [];
|
||||
|
||||
/**
|
||||
* 数据库配置
|
||||
* @var array
|
||||
|
@ -48,16 +43,16 @@ class Database
|
|||
private $pk = '';
|
||||
|
||||
/**
|
||||
* 多个表(仅delete操作)
|
||||
* 别名
|
||||
* @var null
|
||||
*/
|
||||
private $effect = null;
|
||||
private $alias = null;
|
||||
|
||||
/**
|
||||
* 数据去重
|
||||
* @var null
|
||||
*/
|
||||
private $distinct = null;
|
||||
private $distinct = false;
|
||||
|
||||
/**
|
||||
* 操作的字段
|
||||
|
@ -89,57 +84,30 @@ class Database
|
|||
*/
|
||||
private $join = [];
|
||||
|
||||
/**
|
||||
* 关联
|
||||
* @var array
|
||||
*/
|
||||
private $on = [];
|
||||
|
||||
/**
|
||||
* Database constructor.
|
||||
* @param $table
|
||||
* @param $pk
|
||||
* @param $prefix
|
||||
* @throws \Exception
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
private function __construct($table, $pk, $prefix)
|
||||
{
|
||||
$driver = Register::get('DBDriver');
|
||||
// 获取配置
|
||||
$this->config = Config::instance()->get('db');
|
||||
// 当前操作表名
|
||||
$this->table = $this->getTableName($prefix, $table);
|
||||
// 当前操作表主键
|
||||
$this->pk = $pk;
|
||||
$this->setDriver($driver, $this->config);
|
||||
}
|
||||
|
||||
/**
|
||||
* 指定表
|
||||
* @param $table
|
||||
* @param string $pk
|
||||
* @param string $prefix
|
||||
* @return mixed
|
||||
* @throws \Exception
|
||||
*/
|
||||
public static function table($table, $pk = '', $prefix = '')
|
||||
{
|
||||
$ident = $prefix . $table;
|
||||
if (!isset(self::$instance[$ident])) {
|
||||
self::$instance[$ident] = new self($table, $pk, $prefix);
|
||||
if (!self::$connection) { // 保证只有一个数据库连接
|
||||
// 设置数据库驱动
|
||||
$driver = $this->config['driver'] ? $this->config['driver'] : 'MySQLi';
|
||||
$class = '\\top\\library\\database\\driver\\' . $driver;
|
||||
if (class_exists($class)) {
|
||||
// 获取数据库驱动实例
|
||||
self::$connection = $class::instance()->connect($this->config);
|
||||
} else throw new DatabaseException('不存在的数据库驱动:' . $driver);
|
||||
}
|
||||
return self::$instance[$ident];
|
||||
}
|
||||
|
||||
/**
|
||||
* 指定数据库驱动
|
||||
*
|
||||
* @param DatabaseIfs $driver
|
||||
* @param array $config
|
||||
*/
|
||||
private function setDriver(DatabaseIfs $driver, $config)
|
||||
{
|
||||
if (!self::$driver) {
|
||||
self::$driver = $driver->connect($config);
|
||||
}
|
||||
return self::$driver;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -162,23 +130,39 @@ class Database
|
|||
}
|
||||
|
||||
/**
|
||||
* 指定多张表
|
||||
* @param $effect
|
||||
* 指定表
|
||||
* @param $table
|
||||
* @param string $pk
|
||||
* @param string $prefix
|
||||
* @return $this
|
||||
*/
|
||||
public static function table($table = '', $pk = '', $prefix = '')
|
||||
{
|
||||
$ident = $prefix . $table;
|
||||
if (!isset(self::$instance[$ident])) {
|
||||
self::$instance[$ident] = new self($table, $pk, $prefix);
|
||||
}
|
||||
return self::$instance[$ident];
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置表别名
|
||||
* @param $name
|
||||
* @return \top\library\Database
|
||||
*/
|
||||
public function effect($effect)
|
||||
public function alias($name)
|
||||
{
|
||||
$this->effect = $effect;
|
||||
$this->alias = $name;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $field
|
||||
* @param $flag
|
||||
* @return \top\library\Database
|
||||
*/
|
||||
public function distinct($field)
|
||||
public function distinct($flag = true)
|
||||
{
|
||||
$this->distinct = $field;
|
||||
$this->distinct = $flag ? true : false;
|
||||
return $this;
|
||||
}
|
||||
|
||||
|
@ -260,12 +244,12 @@ class Database
|
|||
/**
|
||||
* 多表
|
||||
*
|
||||
* @param $table
|
||||
* @param $on
|
||||
* @param string $type
|
||||
* @param string $table
|
||||
* @param string $name
|
||||
* @return \top\library\Database
|
||||
*/
|
||||
public function join($type, $table, $name)
|
||||
public function join($table, $on, $type = 'INNER')
|
||||
{
|
||||
$tableName = null;
|
||||
if (is_array($table) && isset($table[0]) && isset($table[1])) {
|
||||
|
@ -274,24 +258,13 @@ class Database
|
|||
$tableName = $this->config['prefix'] . $table;
|
||||
}
|
||||
$this->join[] = [
|
||||
$type,
|
||||
$tableName,
|
||||
$name
|
||||
$on,
|
||||
$type
|
||||
];
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 多表关联
|
||||
* @param string $on
|
||||
* @return \top\library\Database
|
||||
*/
|
||||
public function on($on)
|
||||
{
|
||||
$this->on[] = $on;
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 插入记录
|
||||
*
|
||||
|
@ -300,7 +273,7 @@ class Database
|
|||
*/
|
||||
public function insert($data)
|
||||
{
|
||||
$result = self::$driver->insert($this->table, $data);
|
||||
$result = self::$connection->insert($this->table, $data);
|
||||
return $result;
|
||||
}
|
||||
|
||||
|
@ -311,17 +284,21 @@ class Database
|
|||
*/
|
||||
public function find($param = false)
|
||||
{
|
||||
if (is_callable($param))
|
||||
$param($this);
|
||||
$field = $this->getPk();
|
||||
$pkWhere = [];
|
||||
if (!is_bool($param) && !is_callable($param))
|
||||
$pkWhere = [$field => $param];
|
||||
$result = self::$driver->find([
|
||||
(is_callable($param)) && $param($this);
|
||||
if (!is_bool($param) && !is_callable($param)) {
|
||||
$this->where = array_merge($this->where, [
|
||||
[($this->alias ? $this->alias . '.' : '') . $this->getPk() => $param],
|
||||
]);
|
||||
}
|
||||
$result = self::$connection->find(
|
||||
$this->table,
|
||||
!empty($this->join),
|
||||
$pkWhere
|
||||
], $this->distinct, $this->field, $this->join, $this->on, $this->where, $this->order);
|
||||
$this->alias,
|
||||
$this->distinct,
|
||||
$this->field,
|
||||
$this->join,
|
||||
$this->where,
|
||||
$this->order
|
||||
);
|
||||
$this->_reset();
|
||||
return $result;
|
||||
}
|
||||
|
@ -334,17 +311,22 @@ class Database
|
|||
*/
|
||||
public function select($param = false)
|
||||
{
|
||||
if (is_callable($param))
|
||||
$param($this);
|
||||
$field = $this->getPk();
|
||||
$pkWhere = [];
|
||||
if (!is_bool($param) && !is_callable($param))
|
||||
$pkWhere = [$field => $param];
|
||||
$result = self::$driver->select([
|
||||
(is_callable($param)) && $param($this);
|
||||
if (!is_bool($param) && !is_callable($param)) {
|
||||
$this->where = array_merge($this->where, [
|
||||
[($this->alias ? $this->alias . '.' : '') . $this->getPk() => $param],
|
||||
]);
|
||||
}
|
||||
$result = self::$connection->select(
|
||||
$this->table,
|
||||
!empty($this->join),
|
||||
$pkWhere
|
||||
], $this->distinct, $this->field, $this->join, $this->on, $this->where, $this->order, $this->limit);
|
||||
$this->alias,
|
||||
$this->distinct,
|
||||
$this->field,
|
||||
$this->join,
|
||||
$this->where,
|
||||
$this->order,
|
||||
$this->limit
|
||||
);
|
||||
$this->_reset();
|
||||
foreach ($result as $k => $v)
|
||||
$result[$k] = $v;
|
||||
|
@ -360,17 +342,21 @@ class Database
|
|||
*/
|
||||
public function update($data, $param = false)
|
||||
{
|
||||
if (is_callable($param))
|
||||
$param($this);
|
||||
$field = $this->getPk();
|
||||
$pkWhere = [];
|
||||
if (!is_bool($param) && !is_callable($param))
|
||||
$pkWhere = [$field => $param];
|
||||
$result = self::$driver->update([
|
||||
(is_callable($param)) && $param($this);
|
||||
if (!is_bool($param) && !is_callable($param)) {
|
||||
$this->where = array_merge($this->where, [
|
||||
[($this->alias ? $this->alias . '.' : '') . $this->getPk() => $param],
|
||||
]);
|
||||
}
|
||||
$result = self::$connection->update(
|
||||
$this->table,
|
||||
!empty($this->join),
|
||||
$pkWhere
|
||||
], $this->join, $this->on, $this->where, $this->order, $this->limit, $data);
|
||||
$this->alias,
|
||||
$this->join,
|
||||
$this->where,
|
||||
$this->order,
|
||||
$this->limit,
|
||||
$data
|
||||
);
|
||||
$this->_reset();
|
||||
return $result;
|
||||
}
|
||||
|
@ -383,18 +369,20 @@ class Database
|
|||
*/
|
||||
public function delete($param = false)
|
||||
{
|
||||
if (is_callable($param)) {
|
||||
$param($this);
|
||||
(is_callable($param)) && $param($this);
|
||||
if (!is_bool($param) && !is_callable($param)) {
|
||||
$this->where = array_merge($this->where, [
|
||||
[($this->alias ? $this->alias . '.' : '') . $this->getPk() => $param],
|
||||
]);
|
||||
}
|
||||
$field = $this->getPk();
|
||||
$pkWhere = [];
|
||||
if (!is_bool($param) && !is_callable($param))
|
||||
$pkWhere = [$field => $param];
|
||||
$result = self::$driver->delete($this->effect, [
|
||||
$result = self::$connection->delete(
|
||||
$this->table,
|
||||
!empty($this->join),
|
||||
$pkWhere
|
||||
], $this->join, $this->on, $this->where, $this->order, $this->limit);
|
||||
$this->alias,
|
||||
$this->join,
|
||||
$this->where,
|
||||
$this->order,
|
||||
$this->limit
|
||||
);
|
||||
$this->_reset();
|
||||
|
||||
return $result;
|
||||
|
@ -409,48 +397,32 @@ class Database
|
|||
*/
|
||||
public function common($param, $type)
|
||||
{
|
||||
if (is_callable($param)) {
|
||||
$param($this);
|
||||
}
|
||||
(is_callable($param)) && $param($this);
|
||||
if (empty($this->field) && $param && !is_callable($param)) {
|
||||
$this->field = $param;
|
||||
}
|
||||
$result = self::$driver->common([
|
||||
$result = self::$connection->common(
|
||||
$this->table,
|
||||
!empty($this->join),
|
||||
[]
|
||||
], $this->distinct, $this->field, $this->join, $this->on, $this->where, $type);
|
||||
$this->alias,
|
||||
$this->field,
|
||||
$this->join,
|
||||
$this->where,
|
||||
$type
|
||||
);
|
||||
$this->_reset();
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取表结构
|
||||
*
|
||||
* @param string $table
|
||||
* @return mixed
|
||||
*/
|
||||
public function tableDesc($table = '')
|
||||
{
|
||||
$table = ($table) ? $table : $this->table;
|
||||
if (!isset(self::$tableDesc[$table])) {
|
||||
self::$tableDesc[$table] = self::$driver->tableDesc($table);
|
||||
}
|
||||
|
||||
return self::$tableDesc[$table];
|
||||
}
|
||||
|
||||
/**
|
||||
* 执行一条SQL
|
||||
*
|
||||
* @param string $query
|
||||
* @return resource|bool
|
||||
* @param $query
|
||||
* @param array $params
|
||||
* @return bool|\PDOStatement
|
||||
*/
|
||||
public function query($query)
|
||||
public function query($query, $params = [])
|
||||
{
|
||||
$result = self::$driver->query($query);
|
||||
return $result;
|
||||
return self::$connection->query($query, $params);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -458,7 +430,7 @@ class Database
|
|||
*/
|
||||
public function begin()
|
||||
{
|
||||
self::$driver->begin();
|
||||
self::$connection->begin();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -466,7 +438,7 @@ class Database
|
|||
*/
|
||||
public function commit()
|
||||
{
|
||||
self::$driver->commit();
|
||||
self::$connection->commit();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -474,7 +446,16 @@ class Database
|
|||
*/
|
||||
public function rollback()
|
||||
{
|
||||
self::$driver->rollback();
|
||||
self::$connection->rollback();
|
||||
}
|
||||
|
||||
/**
|
||||
* 返回PDO
|
||||
* @return \PDO
|
||||
*/
|
||||
public function getPDO()
|
||||
{
|
||||
return self::$connection->getPDO();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -482,9 +463,9 @@ class Database
|
|||
*
|
||||
* @return string
|
||||
*/
|
||||
public function _sql()
|
||||
public function sql()
|
||||
{
|
||||
return self::$driver->sql();
|
||||
return self::$connection->sql();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -492,11 +473,9 @@ class Database
|
|||
*/
|
||||
private function _reset()
|
||||
{
|
||||
$this->effect = null;
|
||||
$this->distinct = null;
|
||||
$this->distinct = false;
|
||||
$this->field = null;
|
||||
$this->join = [];
|
||||
$this->on = [];
|
||||
$this->where = [];
|
||||
$this->order = null;
|
||||
$this->limit = null;
|
||||
|
@ -510,15 +489,8 @@ class Database
|
|||
private function getPk()
|
||||
{
|
||||
if (!$this->pk) {
|
||||
$tableInfo = $this->tableDesc();
|
||||
$pk = '';
|
||||
foreach ($tableInfo as $value) {
|
||||
if ($value['Key'] == 'PRI') {
|
||||
$pk = $value['Field'];
|
||||
break;
|
||||
}
|
||||
}
|
||||
return $pk;
|
||||
$pk = self::$connection->getPk($this->table, $this->config['dbname']);
|
||||
return ($pk) ? $pk : 'id';
|
||||
}
|
||||
return $this->pk;
|
||||
}
|
||||
|
|
|
@ -8,6 +8,15 @@ use top\library\exception\DatabaseException;
|
|||
/**
|
||||
* 基础模型
|
||||
* @author topnuomi 2018年11月23日
|
||||
*
|
||||
* @method $this alias($name)
|
||||
* @method $this distinct(bool $distinct)
|
||||
* @method $this field(string|array $field)
|
||||
* @method $this where($field, $condition = null, $value = null)
|
||||
* @method $this order(string $order)
|
||||
* @method $this limit(string|array $limit)
|
||||
* @method $this join(string $table, string $on, string $type = null)
|
||||
* @method $this sql()
|
||||
*/
|
||||
class Model
|
||||
{
|
||||
|
@ -40,7 +49,7 @@ class Model
|
|||
* insert值映射
|
||||
* @var array
|
||||
*/
|
||||
protected $inReplace = [];
|
||||
protected $insertReplace = [];
|
||||
|
||||
/**
|
||||
* update值映射
|
||||
|
@ -84,42 +93,36 @@ class Model
|
|||
if ($table) {
|
||||
$this->table = $table;
|
||||
} else if (!$this->table) {
|
||||
$table = get_table_name(get_called_class());
|
||||
$table = get_table_name(static::class);
|
||||
$this->table = $table;
|
||||
}
|
||||
// $this->getDb() = Database::table($this->table, $this->pk);
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取Database实例
|
||||
* @return mixed
|
||||
*/
|
||||
private function getDb()
|
||||
private function database()
|
||||
{
|
||||
return Database::table($this->table, $this->pk, $this->prefix);
|
||||
}
|
||||
|
||||
// 可以静态调用的方法--开始
|
||||
|
||||
/**
|
||||
* 影响的表(仅多表delete)
|
||||
* @param $effect
|
||||
* @return $this
|
||||
*/
|
||||
private function effect($effect)
|
||||
private function _alias($name)
|
||||
{
|
||||
$this->getDb()->effect($effect);
|
||||
$this->database()->alias($name);
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 过滤重复值的字段
|
||||
* @param $field
|
||||
* 过滤重复值
|
||||
* @param $flag
|
||||
* @return $this
|
||||
*/
|
||||
private function distinct($field)
|
||||
private function _distinct($flag = true)
|
||||
{
|
||||
$this->getDb()->distinct($field);
|
||||
$this->database()->distinct($flag);
|
||||
return $this;
|
||||
}
|
||||
|
||||
|
@ -128,9 +131,9 @@ class Model
|
|||
* @param $field
|
||||
* @return $this
|
||||
*/
|
||||
private function field($field)
|
||||
private function _field($field)
|
||||
{
|
||||
$this->getDb()->field($field);
|
||||
$this->database()->field($field);
|
||||
return $this;
|
||||
}
|
||||
|
||||
|
@ -138,10 +141,10 @@ class Model
|
|||
* 查询条件
|
||||
* @return $this
|
||||
*/
|
||||
private function where()
|
||||
private function _where()
|
||||
{
|
||||
call_user_func_array([
|
||||
$this->getDb(),
|
||||
$this->database(),
|
||||
'where'
|
||||
], func_get_args());
|
||||
return $this;
|
||||
|
@ -151,10 +154,10 @@ class Model
|
|||
* 排序
|
||||
* @return $this
|
||||
*/
|
||||
private function order()
|
||||
private function _order()
|
||||
{
|
||||
call_user_func_array([
|
||||
$this->getDb(),
|
||||
$this->database(),
|
||||
'order'
|
||||
], func_get_args());
|
||||
return $this;
|
||||
|
@ -164,10 +167,10 @@ class Model
|
|||
* 限制
|
||||
* @return $this
|
||||
*/
|
||||
private function limit()
|
||||
private function _limit()
|
||||
{
|
||||
call_user_func_array([
|
||||
$this->getDb(),
|
||||
$this->database(),
|
||||
'limit'
|
||||
], func_get_args());
|
||||
return $this;
|
||||
|
@ -175,26 +178,25 @@ class Model
|
|||
|
||||
/**
|
||||
* 多表
|
||||
* @param $type
|
||||
* @param $table
|
||||
* @param $name
|
||||
* @param $on
|
||||
* @param string $type
|
||||
* @return $this
|
||||
*/
|
||||
private function join($type, $table, $name)
|
||||
private function _join($table, $on, $type = 'INNER')
|
||||
{
|
||||
$this->getDb()->join($type, $table, $name);
|
||||
$this->database()->join($table, $on, $type);
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 多表
|
||||
* @param $on
|
||||
* @return $this
|
||||
* 获取最后一次执行的SQL
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
private function on($on)
|
||||
private function _sql()
|
||||
{
|
||||
$this->getDb()->on($on);
|
||||
return $this;
|
||||
return $this->database()->sql();
|
||||
}
|
||||
|
||||
// 可静态调用的方法--结束
|
||||
|
@ -212,7 +214,7 @@ class Model
|
|||
// 此处取消了数据验证,在$this->>data()方法中验证,减少一次数据库查询
|
||||
// 入库时最后的数据处理
|
||||
$data = $this->inHandle($data);
|
||||
return $this->getDb()->insert($data);
|
||||
return $this->database()->insert($data);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
@ -224,7 +226,7 @@ class Model
|
|||
*/
|
||||
public function delete($param = false)
|
||||
{
|
||||
return $this->getDb()->delete($param);
|
||||
return $this->database()->delete($param);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -241,7 +243,7 @@ class Model
|
|||
// 此处取消了数据验证,在$this->data()方法中验证,减少一次数据库查询
|
||||
// 入库时最后的数据处理
|
||||
$data = $this->inHandle($data);
|
||||
return $this->getDb()->update($data, $param);
|
||||
return $this->database()->update($data, $param);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
@ -254,7 +256,7 @@ class Model
|
|||
*/
|
||||
public function find($param = false, $notRaw = true)
|
||||
{
|
||||
$result = $this->getDb()->find($param);
|
||||
$result = $this->database()->find($param);
|
||||
if ($notRaw) {
|
||||
if (is_array($result)) {
|
||||
$result = $this->outHandle($result);
|
||||
|
@ -271,7 +273,7 @@ class Model
|
|||
*/
|
||||
public function select($param = false, $notRaw = true)
|
||||
{
|
||||
$result = $this->getDb()->select($param);
|
||||
$result = $this->database()->select($param);
|
||||
if ($notRaw) {
|
||||
if (is_array($result)) {
|
||||
$result = $this->outHandle($result);
|
||||
|
@ -287,7 +289,7 @@ class Model
|
|||
*/
|
||||
public function count($param = null)
|
||||
{
|
||||
return $this->getDb()->common($param, 'count');
|
||||
return $this->database()->common($param, 'count');
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -297,7 +299,7 @@ class Model
|
|||
*/
|
||||
public function avg($param = null)
|
||||
{
|
||||
return $this->getDb()->common($param, 'avg');
|
||||
return $this->database()->common($param, 'avg');
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -307,7 +309,7 @@ class Model
|
|||
*/
|
||||
public function max($param = null)
|
||||
{
|
||||
return $this->getDb()->common($param, 'max');
|
||||
return $this->database()->common($param, 'max');
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -317,7 +319,7 @@ class Model
|
|||
*/
|
||||
public function min($param = null)
|
||||
{
|
||||
return $this->getDb()->common($param, 'min');
|
||||
return $this->database()->common($param, 'min');
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -327,27 +329,28 @@ class Model
|
|||
*/
|
||||
public function sum($param = null)
|
||||
{
|
||||
return $this->getDb()->common($param, 'sum');
|
||||
return $this->database()->common($param, 'sum');
|
||||
}
|
||||
|
||||
/**
|
||||
* 执行一条SQL
|
||||
* @param $query
|
||||
* @param array $params
|
||||
* @return mixed
|
||||
*/
|
||||
public function query($query)
|
||||
public function query($query, $params = [])
|
||||
{
|
||||
return $this->getDb()->query($query);
|
||||
return $this->database()->query($query, $params);
|
||||
}
|
||||
|
||||
/**
|
||||
* MySQL事务
|
||||
* @param $action
|
||||
* @return mixed
|
||||
* @return bool
|
||||
*/
|
||||
public function transaction($action)
|
||||
{
|
||||
$db = $this->getDb();
|
||||
$db = $this->database();
|
||||
// 开启事务
|
||||
$db->begin();
|
||||
try {
|
||||
|
@ -363,44 +366,22 @@ class Model
|
|||
}
|
||||
|
||||
/**
|
||||
* 获取最后一次执行的SQL
|
||||
*
|
||||
* @return string
|
||||
* 返回PDO
|
||||
* @return \PDO
|
||||
*/
|
||||
public function _sql()
|
||||
public function getPDO()
|
||||
{
|
||||
return $this->getDb()->_sql();
|
||||
return $this->database()->getPDO();
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取表单数据
|
||||
* @param array $data
|
||||
* @param bool $notRaw
|
||||
* @return array|bool
|
||||
*/
|
||||
public function data($data = [], $notRaw = true)
|
||||
public function data($data = [])
|
||||
{
|
||||
$mapData = $this->processMapped($data);
|
||||
if ($mapData) { // 如果正确处理字段映射并且数据验证通过
|
||||
if (!$notRaw) {
|
||||
return $mapData;
|
||||
} else {
|
||||
$data = [];
|
||||
$prefix = $this->prefix ? $this->prefix : Config::instance()->get('db')['prefix'];
|
||||
$tableDesc = $this->tableDesc($prefix . $this->table);
|
||||
foreach ($tableDesc as $value) {
|
||||
if (array_key_exists($value['Field'], $mapData)) {
|
||||
// 如果表单值为空则赋值为数据库字段默认值
|
||||
if (!$mapData[$value['Field']] && !is_numeric($mapData[$value['Field']])) {
|
||||
$mapData[$value['Field']] = $value['Default'];
|
||||
}
|
||||
$data[$value['Field']] = $mapData[$value['Field']];
|
||||
}
|
||||
}
|
||||
return $data;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
return $this->processMapped($data);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -459,15 +440,14 @@ class Model
|
|||
*/
|
||||
private function inHandle($data)
|
||||
{
|
||||
$replace = ($this->isInsert) ? $this->inReplace : $this->updateReplace;
|
||||
$replace = ($this->isInsert) ? $this->insertReplace : $this->updateReplace;
|
||||
foreach ($replace as $key => $value) {
|
||||
$fieldValue = '';
|
||||
if (!array_key_exists($key, $data)) {
|
||||
$data[$key] = '';
|
||||
}
|
||||
if (is_array($value)) {
|
||||
if (isset($value[1]) && $value[1] === true) {
|
||||
$object = get_called_class();
|
||||
$object = static::class;
|
||||
if (method_exists($object, $value[0])) {
|
||||
$methodName = $value[0];
|
||||
$fieldValue = call_user_func_array([
|
||||
|
@ -476,6 +456,8 @@ class Model
|
|||
], [
|
||||
$data[$key]
|
||||
]);
|
||||
} else {
|
||||
throw new Exception('方法' . $object . '->' . $value[0] . '不存在');
|
||||
}
|
||||
} else if (isset($value[0]) && function_exists($value[0])) {
|
||||
$fieldValue = $value[0]($data[$key]);
|
||||
|
@ -592,16 +574,6 @@ class Model
|
|||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取表结构
|
||||
* @param $table
|
||||
* @return mixed
|
||||
*/
|
||||
public function tableDesc($table)
|
||||
{
|
||||
return $this->getDb()->tableDesc($table);
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取信息
|
||||
* @return string
|
||||
|
@ -611,30 +583,6 @@ class Model
|
|||
return $this->message;
|
||||
}
|
||||
|
||||
/**
|
||||
* 某些方法提供以下替代方式
|
||||
* @param $name
|
||||
* @return array|mixed|null
|
||||
*/
|
||||
public function __get($name)
|
||||
{
|
||||
$data = null;
|
||||
switch ($name) {
|
||||
case 'one':
|
||||
$data = $this->find();
|
||||
break;
|
||||
case 'all':
|
||||
$data = $this->select();
|
||||
break;
|
||||
case 'sql':
|
||||
$data = $this->_sql();
|
||||
break;
|
||||
default:
|
||||
$data = $this->$name();
|
||||
}
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* 非静态调用连贯操作
|
||||
* @param string $name
|
||||
|
@ -643,8 +591,9 @@ class Model
|
|||
*/
|
||||
public function __call($name, $arguments)
|
||||
{
|
||||
if (method_exists($this, $name)) {
|
||||
return $this->{$name}($arguments);
|
||||
$methodName = '_' . $name;
|
||||
if (method_exists($this, $methodName)) {
|
||||
return call_user_func_array([$this, $methodName], $arguments);
|
||||
} else throw new Exception('不存在的方法:' . $name);
|
||||
}
|
||||
|
||||
|
@ -656,10 +605,8 @@ class Model
|
|||
*/
|
||||
public static function __callStatic($name, $arguments)
|
||||
{
|
||||
$methodName = null;
|
||||
if (method_exists(static::class, $name)) {
|
||||
$methodName = $name;
|
||||
} else {
|
||||
$methodName = '_' . $name;
|
||||
if (!method_exists(static::class, $methodName)) {
|
||||
$methodMap = ['all' => 'select', 'get' => 'find'];
|
||||
if (isset($methodMap[$name])) {
|
||||
$methodName = $methodMap[$name];
|
||||
|
|
|
@ -0,0 +1,494 @@
|
|||
<?php
|
||||
|
||||
namespace top\library\database;
|
||||
|
||||
use top\library\exception\DatabaseException;
|
||||
|
||||
abstract class Base
|
||||
{
|
||||
|
||||
protected $insertSql = 'INSERT INTO [table] ([field]) VALUES ([data])';
|
||||
protected $updateSql = 'UPDATE [table] [join] SET [data] [where] [order] [limit]';
|
||||
protected $selectSql = 'SELECT [field] FROM [table] [join] [where] [order] [limit]';
|
||||
protected $deleteSql = 'DELETE FROM [table] [join] [where] [order] [limit]';
|
||||
|
||||
/**
|
||||
* PDO连接
|
||||
* @var \PDO
|
||||
*/
|
||||
protected $pdo = null;
|
||||
|
||||
private $sql = null;
|
||||
|
||||
/**
|
||||
* 连接数据库
|
||||
* @return $this
|
||||
*/
|
||||
public function connect($config)
|
||||
{
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取主键
|
||||
* @return string
|
||||
*/
|
||||
public function getPk($table, $database)
|
||||
{
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 插入记录
|
||||
* @param $table
|
||||
* @param $data
|
||||
* @return mixed
|
||||
*/
|
||||
public function insert($table, $data)
|
||||
{
|
||||
$this->executeInsert($table, $data);
|
||||
return $this->pdo->lastInsertId();
|
||||
}
|
||||
|
||||
/**
|
||||
* 更新记录
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @param $data
|
||||
* @return int
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function update($table, $alias, $join, $where, $order, $limit, $data)
|
||||
{
|
||||
$stmt = $this->executeUpdate($table, $alias, $join, $where, $order, $limit, $data);
|
||||
return $stmt->rowCount();
|
||||
}
|
||||
|
||||
/**
|
||||
* 查找一条记录
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @return mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function find($table, $alias, $distinct, $field, $join, $where, $order)
|
||||
{
|
||||
$stmt = $this->executeSelect($table, $alias, $distinct, $field, $join, $where, $order, '1');
|
||||
return $stmt->fetch(\PDO::FETCH_ASSOC);
|
||||
}
|
||||
|
||||
/**
|
||||
* 查找全部
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @return array|mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function select($table, $alias, $distinct, $field, $join, $where, $order, $limit)
|
||||
{
|
||||
$stmt = $this->executeSelect($table, $alias, $distinct, $field, $join, $where, $order, $limit);
|
||||
return $stmt->fetchAll(\PDO::FETCH_ASSOC);
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除记录
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @return int|mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function delete($table, $alias, $join, $where, $order, $limit)
|
||||
{
|
||||
$stmt = $this->executeDelete($table, $alias, $join, $where, $order, $limit);
|
||||
return $stmt->rowCount();
|
||||
}
|
||||
|
||||
/**
|
||||
* 执行一条SQL
|
||||
* @param $query
|
||||
* @param $params
|
||||
* @return bool|\PDOStatement
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function query($query, $params)
|
||||
{
|
||||
if (false === ($stmt = $this->pdo->prepare($query))) {
|
||||
throw new DatabaseException($this->pdo->errorInfo()[2]);
|
||||
}
|
||||
$stmt->execute($params);
|
||||
return $stmt;
|
||||
}
|
||||
|
||||
/**
|
||||
* 返回PDO
|
||||
* @return \PDO
|
||||
*/
|
||||
public function getPDO()
|
||||
{
|
||||
return $this->pdo;
|
||||
}
|
||||
|
||||
/**
|
||||
* 事务
|
||||
*/
|
||||
public function begin()
|
||||
{
|
||||
return $this->pdo->beginTransaction();
|
||||
}
|
||||
|
||||
/**
|
||||
* 提交
|
||||
*/
|
||||
public function commit()
|
||||
{
|
||||
return $this->pdo->commit();
|
||||
}
|
||||
|
||||
/**
|
||||
* 回滚
|
||||
*/
|
||||
public function rollback()
|
||||
{
|
||||
return $this->pdo->rollBack();
|
||||
}
|
||||
|
||||
/**
|
||||
* 公共方法
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $type
|
||||
* @return mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function common($table, $alias, $field, $join, $where, $type)
|
||||
{
|
||||
$where = $this->parseWhere($where);
|
||||
$field = $type . '(`' . $this->parseField(false, $field) . '`)';
|
||||
$sql = str_replace(
|
||||
['[field]', '[table]', '[join]', '[where]', '[order]', '[limit]'],
|
||||
[
|
||||
$field,
|
||||
$this->parseTable($table, $alias),
|
||||
$this->parseJoin($join),
|
||||
$where['where'],
|
||||
'',
|
||||
'',
|
||||
], $this->selectSql
|
||||
);
|
||||
|
||||
if (false === ($stmt = $this->pdo->prepare($sql))) {
|
||||
throw new DatabaseException($this->pdo->errorInfo()[2]);
|
||||
}
|
||||
$stmt->execute($where['data']);
|
||||
return $stmt->fetch(\PDO::FETCH_ASSOC)[$field];
|
||||
}
|
||||
|
||||
/**
|
||||
* 预处理并执行插入操作
|
||||
* @param $table
|
||||
* @param $data
|
||||
* @return bool|\PDOStatement
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
private function executeInsert($table, $data)
|
||||
{
|
||||
$insertData = $this->parseInsertData($data);
|
||||
$sql = str_replace(
|
||||
['[table]', '[field]', '[data]'],
|
||||
[
|
||||
$this->parseTable($table, ''),
|
||||
$insertData['field'],
|
||||
$insertData['values'],
|
||||
], $this->insertSql
|
||||
);
|
||||
|
||||
if (false === ($stmt = $this->pdo->prepare($sql))) {
|
||||
throw new DatabaseException($this->pdo->errorInfo()[2]);
|
||||
}
|
||||
$stmt->execute($insertData['data']);
|
||||
return $stmt;
|
||||
}
|
||||
|
||||
/**
|
||||
* 预处理并执行更新操作
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @param $data
|
||||
* @return bool|\PDOStatement
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
private function executeUpdate($table, $alias, $join, $where, $order, $limit, $data)
|
||||
{
|
||||
$where = $this->parseWhere($where);
|
||||
$updateData = $this->parseUpdateData($alias, $data);
|
||||
$sql = str_replace(
|
||||
['[table]', '[data]', '[join]', '[where]', '[order]', '[limit]'],
|
||||
[
|
||||
$this->parseTable($table, $alias),
|
||||
$updateData['field'],
|
||||
$this->parseJoin($join),
|
||||
$where['where'],
|
||||
$this->parseOrder($order),
|
||||
$this->parseLimit($limit),
|
||||
], $this->updateSql
|
||||
);
|
||||
|
||||
if (false === ($stmt = $this->pdo->prepare($sql))) {
|
||||
throw new DatabaseException($this->pdo->errorInfo()[2]);
|
||||
}
|
||||
$bindData = array_merge($updateData['data'], $where['data']);
|
||||
$stmt->execute($bindData);
|
||||
return $stmt;
|
||||
}
|
||||
|
||||
/**
|
||||
* 预处理并执行查询操作
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @return bool|\PDOStatement
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
private function executeSelect($table, $alias, $distinct, $field, $join, $where, $order, $limit)
|
||||
{
|
||||
$where = $this->parseWhere($where);
|
||||
|
||||
$sql = str_replace(
|
||||
['[table]', '[field]', '[join]', '[where]', '[order]', '[limit]'],
|
||||
[
|
||||
$this->parseTable($table, $alias),
|
||||
$this->parseField($distinct, $field),
|
||||
$this->parseJoin($join),
|
||||
$where['where'],
|
||||
$this->parseOrder($order),
|
||||
$this->parseLimit($limit),
|
||||
], $this->selectSql
|
||||
);
|
||||
|
||||
if (false === ($stmt = $this->pdo->prepare($sql))) {
|
||||
throw new DatabaseException($this->pdo->errorInfo()[2]);
|
||||
}
|
||||
$stmt->execute($where['data']);
|
||||
return $stmt;
|
||||
}
|
||||
|
||||
/**
|
||||
* 预处理并执行删除操作
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @param $join
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @return bool|\PDOStatement
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
private function executeDelete($table, $alias, $join, $where, $order, $limit)
|
||||
{
|
||||
$where = $this->parseWhere($where);
|
||||
|
||||
$sql = str_replace(
|
||||
['[table]', '[join]', '[where]', '[order]', '[limit]'],
|
||||
[
|
||||
$this->parseTable($table, $alias),
|
||||
$this->parseJoin($join),
|
||||
$where['where'],
|
||||
$this->parseOrder($order),
|
||||
$this->parseLimit($limit),
|
||||
], $this->deleteSql);
|
||||
|
||||
if (false === ($stmt = $this->pdo->prepare($sql))) {
|
||||
throw new DatabaseException($this->pdo->errorInfo()[2]);
|
||||
}
|
||||
$stmt->execute($where['data']);
|
||||
return $stmt;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析table
|
||||
* @param $table
|
||||
* @param $alias
|
||||
* @return string
|
||||
*/
|
||||
protected function parseTable($table, $alias)
|
||||
{
|
||||
return "`$table`" . ($alias ? ' ' . $alias : '');
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析field
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @return string
|
||||
*/
|
||||
protected function parseField($distinct, $field)
|
||||
{
|
||||
if (is_array($field)) {
|
||||
$field = implode(',', $field);
|
||||
} else if (!$field) {
|
||||
$field = '*';
|
||||
}
|
||||
return ($distinct ? 'DISTINCT ' : '') . $field;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析join
|
||||
* @param $allJoin
|
||||
* @return string
|
||||
*/
|
||||
protected function parseJoin($allJoin)
|
||||
{
|
||||
$joinString = '';
|
||||
foreach ($allJoin as $key => $join) {
|
||||
$joinString .= $join[2] . ' JOIN ' . $join[0] . ' ON ' . $join[1] . ' ';
|
||||
}
|
||||
return mb_substr($joinString, 0, mb_strlen($joinString, 'utf-8') - 1, 'utf-8');
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析where
|
||||
* @param $allWhere
|
||||
* @return array
|
||||
*/
|
||||
protected function parseWhere($allWhere)
|
||||
{
|
||||
$whereArray = [
|
||||
'temp' => [],
|
||||
'where' => '',
|
||||
'data' => [],
|
||||
];
|
||||
foreach ($allWhere as $key => $item) {
|
||||
foreach ($item as $field => $value) {
|
||||
if (is_array($value)) { // 数组
|
||||
$whereArray['temp'][] = $field . ' ' . $value[0] . ' ?';
|
||||
$whereArray['data'][] = $value[1];
|
||||
} else {
|
||||
$whereArray['temp'][] = $field . ' = ?';
|
||||
$whereArray['data'][] = $value;
|
||||
}
|
||||
}
|
||||
}
|
||||
$whereArray['where'] = (!empty($whereArray['temp'])) ? 'WHERE ' . implode(' AND ', $whereArray['temp']) : '';
|
||||
unset($whereArray['temp']);
|
||||
return $whereArray;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析order
|
||||
* @param $order
|
||||
* @return string
|
||||
*/
|
||||
protected function parseOrder($order)
|
||||
{
|
||||
return ($order) ? ' ORDER BY ' . $order : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析limit
|
||||
* @param $limit
|
||||
* @return string
|
||||
*/
|
||||
protected function parseLimit($limit)
|
||||
{
|
||||
if (is_array($limit)) {
|
||||
$limit = implode(',', $limit);
|
||||
}
|
||||
return ($limit) ? 'LIMIT ' . $limit : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析insert参数
|
||||
* @param $data
|
||||
* @return array
|
||||
*/
|
||||
protected function parseInsertData($data)
|
||||
{
|
||||
$insertData = [
|
||||
'field' => [],
|
||||
'values' => [],
|
||||
'data' => [],
|
||||
];
|
||||
foreach ($data as $field => $value) {
|
||||
$insertData['field'][] = '`' . $field . '`';
|
||||
$insertData['values'][] = '?';
|
||||
$insertData['data'][] = $value;
|
||||
}
|
||||
$insertData['field'] = implode(',', $insertData['field']);
|
||||
$insertData['values'] = implode(',', $insertData['values']);
|
||||
return $insertData;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析update参数
|
||||
* @param $alias
|
||||
* @param $data
|
||||
* @return array
|
||||
*/
|
||||
protected function parseUpdateData($alias, $data)
|
||||
{
|
||||
$updateData = [
|
||||
'field' => [],
|
||||
'data' => [],
|
||||
];
|
||||
foreach ($data as $field => $value) {
|
||||
if (strstr($field, '.')) {
|
||||
$field = explode('.', $field);
|
||||
$updateData['field'][] = $field[0] . '.`' . $field[1] . '` = ?';
|
||||
} else {
|
||||
$updateData['field'][] = ($alias ? $alias . '.' : '') . '`' . $field . '` = ?';
|
||||
}
|
||||
$updateData['data'][] = $value;
|
||||
}
|
||||
$updateData['field'] = implode(',', $updateData['field']);
|
||||
return $updateData;
|
||||
}
|
||||
|
||||
/**
|
||||
* 返回最后执行的sql
|
||||
* @return string
|
||||
*/
|
||||
public function sql()
|
||||
{
|
||||
return $this->sql;
|
||||
}
|
||||
|
||||
/**
|
||||
* 关闭数据库连接
|
||||
* @return mixed
|
||||
*/
|
||||
public function close()
|
||||
{
|
||||
// TODO: Implement close() method.
|
||||
}
|
||||
}
|
|
@ -1,532 +0,0 @@
|
|||
<?php
|
||||
|
||||
namespace top\library\database\driver;
|
||||
|
||||
use top\library\database\ifs\DatabaseIfs;
|
||||
use top\library\exception\DatabaseException;
|
||||
use top\traits\Instance;
|
||||
|
||||
/**
|
||||
* MySQLi数据库驱动
|
||||
* @author topnuomi 2018年11月20日
|
||||
*/
|
||||
class MySQLi implements DatabaseIfs
|
||||
{
|
||||
|
||||
use Instance;
|
||||
|
||||
private $link;
|
||||
|
||||
private $sql;
|
||||
|
||||
/**
|
||||
* 连接数据库
|
||||
* @param array $config
|
||||
* @return $this
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function connect($config)
|
||||
{
|
||||
$link = $this->link = @mysqli_connect($config['host'], $config['user'], $config['passwd'], $config['dbname'], $config['port']);
|
||||
if ($link === false) {
|
||||
throw new DatabaseException(mysqli_connect_error());
|
||||
}
|
||||
mysqli_query($link, 'set names ' . $config['charset']);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 插入
|
||||
* @param string $table
|
||||
* @param array $data
|
||||
* @return int|string
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function insert($table, $data)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
if (count($data) == count($data, 1)) { // 一维数组
|
||||
$query = 'insert into ' . $table;
|
||||
$field = ' (' . implode(',', array_keys($data)) . ')';
|
||||
$value = array_values($data);
|
||||
$value = '(' . implode(',', $this->checkNull($value)) . ')';
|
||||
$this->sql = $query .= $field . ' values ' . $value . ';';
|
||||
$this->query($query);
|
||||
} else { // 二维数组
|
||||
foreach ($data as $key => $value) {
|
||||
$query = 'insert into ' . $table;
|
||||
$allField = ' (' . implode(',', array_keys($value)) . ')';
|
||||
$allValue = '(' . implode(',', $this->checkNull($value)) . ')';
|
||||
$this->sql = $query .= $allField . ' values ' . $allValue . ';';
|
||||
$this->query($query);
|
||||
}
|
||||
}
|
||||
return mysqli_insert_id($this->link);
|
||||
}
|
||||
|
||||
/**
|
||||
* 更新
|
||||
* @param string $table
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @param array $data
|
||||
* @return int
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function update($table, $join, $on, $where, $order, $limit, $data)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$limit = $this->parseLimit($limit);
|
||||
$query = "update {$tableInfo['table']}{$join} set ";
|
||||
$updateData = [];
|
||||
foreach ($data as $key => $value) {
|
||||
if (!is_numeric($value) && !$value) {
|
||||
$value = 'NULL';
|
||||
} else {
|
||||
$value = '\'' . mysqli_real_escape_string($this->link, $value) . '\'';
|
||||
}
|
||||
$updateData[] = $key . '=' . $value;
|
||||
}
|
||||
$this->sql = $query .= implode(',', $updateData) . "{$where}{$order}{$limit}";
|
||||
$this->query($query);
|
||||
return mysqli_affected_rows($this->link);
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询一条记录
|
||||
* @param string $table
|
||||
* @param $distinct
|
||||
* @param array|string $field
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @return array|null
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function find($table, $distinct, $field, $join, $on, $where, $order)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
$distinct = $this->parseDistinct($distinct);
|
||||
if ($distinct) {
|
||||
$field = $distinct;
|
||||
} else {
|
||||
$field = $this->parseField($field);
|
||||
}
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$this->sql = "select {$field} from {$tableInfo['table']}{$join}{$where}{$order} limit 1";
|
||||
$result = $this->query($this->sql);
|
||||
return mysqli_fetch_assoc($result);
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询所有记录
|
||||
* @param string $table
|
||||
* @param $distinct
|
||||
* @param array|string $field
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @return array|null
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function select($table, $distinct, $field, $join, $on, $where, $order, $limit)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
$distinct = $this->parseDistinct($distinct);
|
||||
if ($distinct) {
|
||||
$field = $distinct;
|
||||
} else {
|
||||
$field = $this->parseField($field);
|
||||
}
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$limit = $this->parseLimit($limit);
|
||||
$this->sql = "select {$field} from {$tableInfo['table']}{$join}{$where}{$order}{$limit}";
|
||||
$result = $this->query($this->sql);
|
||||
return mysqli_fetch_all($result, MYSQLI_ASSOC);
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除
|
||||
* @param array|string $effect
|
||||
* @param string $table
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @return int
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function delete($effect, $table, $join, $on, $where, $order, $limit)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$effect = $this->parseEffect($effect);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$limit = $this->parseLimit($limit);
|
||||
$this->sql = "delete{$effect} from {$tableInfo['table']}{$join}{$where}{$order}{$limit}";
|
||||
$this->query($this->sql);
|
||||
return mysqli_affected_rows($this->link);
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取表结构
|
||||
* @param $table
|
||||
* @return array|bool|null
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function tableDesc($table)
|
||||
{
|
||||
$sql = 'desc ' . $table;
|
||||
if (!$result = $this->query($sql)) {
|
||||
return false;
|
||||
}
|
||||
$data = mysqli_fetch_all($result, MYSQLI_ASSOC);
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* 公共方法
|
||||
* @param $table
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $on
|
||||
* @param $where
|
||||
* @param $type
|
||||
* @return bool
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function common($table, $distinct, $field, $join, $on, $where, $type)
|
||||
{
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$distinct = $this->parseDistinct($distinct);
|
||||
if ($distinct) {
|
||||
$field = $distinct;
|
||||
} else {
|
||||
$field = $this->parseField($field);
|
||||
}
|
||||
$join = $this->parseJoin($join, $on);
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$this->sql = "select {$type}({$field}) from {$tableInfo['table']}{$join}{$where}";
|
||||
$result = $this->query($this->sql);
|
||||
$data = mysqli_fetch_array($result);
|
||||
if (isset($data[0])) {
|
||||
return $data[0];
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 事务
|
||||
*/
|
||||
public function begin()
|
||||
{
|
||||
mysqli_begin_transaction($this->link);
|
||||
}
|
||||
|
||||
/**
|
||||
* 提交
|
||||
*/
|
||||
public function commit()
|
||||
{
|
||||
mysqli_commit($this->link);
|
||||
}
|
||||
|
||||
/**
|
||||
* 回滚
|
||||
*/
|
||||
public function rollback()
|
||||
{
|
||||
mysqli_rollback($this->link);
|
||||
}
|
||||
|
||||
/**
|
||||
* 执行SQL
|
||||
* @param string $query
|
||||
* @return bool|\mysqli_result
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function query($query)
|
||||
{
|
||||
$result = mysqli_query($this->link, $query);
|
||||
if (!$result) {
|
||||
throw new DatabaseException(mysqli_error($this->link));
|
||||
}
|
||||
$this->writeLogs($result, $query);
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取执行的最后一条SQL
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function sql()
|
||||
{
|
||||
return trim($this->sql, ' ');
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析表信息
|
||||
* @param $table
|
||||
* @return array
|
||||
*/
|
||||
private function parseTable($table)
|
||||
{
|
||||
$info = [];
|
||||
// 如果是多表查询,给当前表名别名this
|
||||
if ($table[1] === true) {
|
||||
$info['table'] = $table[0] . ' as this';
|
||||
$info['where'] = [];
|
||||
// 如果存在主键的条件,给键名加上别名
|
||||
if (!empty($table[2])) {
|
||||
$field = 'this.' . array_keys($table[2])[0];
|
||||
$value = array_values($table[2])[0];
|
||||
$info['where'] = [$field => $value];
|
||||
}
|
||||
} else {
|
||||
$info['table'] = $table[0];
|
||||
$info['where'] = $table[2];
|
||||
}
|
||||
return $info;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析多表的删除
|
||||
* @param $effect
|
||||
* @return string
|
||||
*/
|
||||
public function parseEffect($effect)
|
||||
{
|
||||
if ($effect) {
|
||||
if (is_array($effect)) {
|
||||
$effect = implode(',', $effect);
|
||||
}
|
||||
return ' ' . $effect;
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析数据去重
|
||||
* @param $distinct
|
||||
* @return string
|
||||
*/
|
||||
private function parseDistinct($distinct)
|
||||
{
|
||||
if ($distinct) {
|
||||
if (is_array($distinct)) {
|
||||
$distinct = implode(',', $distinct);
|
||||
}
|
||||
return 'distinct ' . $distinct;
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合字段
|
||||
* @param string|array $field
|
||||
* @return string
|
||||
*/
|
||||
private function parseField($field)
|
||||
{
|
||||
if (!$field) {
|
||||
$field = '*';
|
||||
} else if (is_array($field)) {
|
||||
$field = implode(',', $field);
|
||||
}
|
||||
return $field;
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合where条件
|
||||
* @param array $array
|
||||
* @param string $glue
|
||||
* @return string
|
||||
*/
|
||||
private function parseWhere(array $array, $glue = 'and')
|
||||
{
|
||||
$where = [];
|
||||
foreach ($array as $value) {
|
||||
if (empty($value)) continue;
|
||||
if (is_array($value)) {
|
||||
foreach ($value as $key => $val) {
|
||||
if (is_array($val)) {
|
||||
switch (strtolower($val[0])) {
|
||||
case 'in':
|
||||
$arr_ = explode(',', $val[1]);
|
||||
$str = '';
|
||||
for ($i = 0; $i < count($arr_); $i++) {
|
||||
$str .= (($i != 0) ? ',' : '') . $this->checkNull(trim($arr_[$i]));
|
||||
}
|
||||
$where[] = $key . ' ' . $val[0] . ' (' . $str . ')';
|
||||
break;
|
||||
case 'like':
|
||||
$where[] = $key . ' ' . $val[0] . ' \'%' . $val[1] . '%\'';
|
||||
break;
|
||||
default:
|
||||
$where[] = $key . ' ' . $val[0] . ' ' . $this->checkNull($val[1]);
|
||||
}
|
||||
} else {
|
||||
$val = $this->checkNull($val);
|
||||
$where[] = $key . '=' . $val;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
$where[] = $value;
|
||||
}
|
||||
}
|
||||
if (empty($where)) {
|
||||
return '';
|
||||
} else {
|
||||
return ' where ' . implode(' ' . $glue . ' ', $where);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合order
|
||||
* @param string $order
|
||||
* @return string
|
||||
*/
|
||||
private function parseOrder($order = '')
|
||||
{
|
||||
if ($order) {
|
||||
$order = ' order by ' . $order;
|
||||
}
|
||||
return $order;
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合limit
|
||||
* @param string $limit
|
||||
* @return string
|
||||
*/
|
||||
private function parseLimit($limit = '')
|
||||
{
|
||||
if ($limit) {
|
||||
if (is_array($limit)) {
|
||||
$limit = ' limit ' . implode(',', $limit);
|
||||
} else {
|
||||
$limit = ' limit ' . $limit;
|
||||
}
|
||||
}
|
||||
return $limit;
|
||||
}
|
||||
|
||||
/**
|
||||
* 链接多表(join on)
|
||||
* @param array $data
|
||||
* @param string|array $on
|
||||
* @return string
|
||||
*/
|
||||
private function parseJoin($data, $on)
|
||||
{
|
||||
$join = [];
|
||||
for ($i = 0; $i < count($data); $i++) {
|
||||
$string = null;
|
||||
if (isset($on[$i])) {
|
||||
if (is_array($on[$i])) {
|
||||
$pieces = [];
|
||||
foreach ($on[$i] as $key => $value) {
|
||||
$pieces[] = $key . ' = ' . $value;
|
||||
}
|
||||
$string = ' on ' . implode(' and ', $pieces);
|
||||
} else {
|
||||
$string = ' on ' . $on[$i];
|
||||
}
|
||||
}
|
||||
$join[] = $data[$i][0] . ' join ' . $data[$i][1] . ($data[$i][2] ? ' as ' . $data[$i][2] : '') . $string;
|
||||
}
|
||||
if (!empty($join)) {
|
||||
return ' ' . implode(' ', $join);
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查并处理空值
|
||||
* @param $value
|
||||
* @return array|string
|
||||
*/
|
||||
private function checkNull($value)
|
||||
{
|
||||
if (is_array($value)) {
|
||||
foreach ($value as $k => $v) {
|
||||
if (!is_numeric($v) && !$v) {
|
||||
$value[$k] = 'NULL';
|
||||
} else {
|
||||
$value[$k] = '\'' . mysqli_real_escape_string($this->link, $v) . '\'';
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (!is_numeric($value) && !$value) {
|
||||
$value = 'NULL';
|
||||
} else {
|
||||
$value = '\'' . mysqli_real_escape_string($this->link, $value) . '\'';
|
||||
}
|
||||
}
|
||||
return $value;
|
||||
}
|
||||
|
||||
/**
|
||||
* SQL存文件
|
||||
* @param $result
|
||||
* @param $query
|
||||
*/
|
||||
private function writeLogs($result, $query)
|
||||
{
|
||||
if (DEBUG) {
|
||||
$error = '';
|
||||
if (!$result) {
|
||||
$error = mysqli_error($this->link);
|
||||
}
|
||||
$nowTime = date('Y-m-d H:i:s', time());
|
||||
$content = "[{$nowTime}] SQL: {$query} {$error}" . PHP_EOL;
|
||||
file_put_contents('./runtime/database_logs.txt', $content, FILE_APPEND);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 关闭数据库连接
|
||||
*/
|
||||
public function close()
|
||||
{
|
||||
if ($this->link) {
|
||||
if (mysqli_close($this->link)) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public function __destruct()
|
||||
{
|
||||
$this->close();
|
||||
}
|
||||
}
|
|
@ -0,0 +1,45 @@
|
|||
<?php
|
||||
|
||||
namespace top\library\database\driver;
|
||||
|
||||
use top\library\database\Base;
|
||||
use top\library\database\ifs\DatabaseIfs;
|
||||
use top\traits\Instance;
|
||||
|
||||
class Mysql extends Base implements DatabaseIfs
|
||||
{
|
||||
|
||||
use Instance;
|
||||
|
||||
/**
|
||||
* 连接数据库
|
||||
* @param $config
|
||||
* @return $this
|
||||
*/
|
||||
public function connect($config)
|
||||
{
|
||||
$dsn = "mysql:host={$config['host']};dbname={$config['dbname']}";
|
||||
$this->pdo = new \PDO($dsn, $config['user'], $config['passwd']);
|
||||
$this->pdo->setAttribute(\PDO::ATTR_EMULATE_PREPARES, false);
|
||||
// 设置字符集
|
||||
$this->pdo->exec('SET NAMES ' . $config['charset']);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取主键
|
||||
* @param $table
|
||||
* @param $database
|
||||
* @return string
|
||||
*/
|
||||
public function getPk($table, $database)
|
||||
{
|
||||
$stmt = $this->pdo->query("SELECT COLUMN_NAME,COLUMN_KEY FROM INFORMATION_SCHEMA.COLUMNS WHERE COLUMN_KEY = 'PRI' AND TABLE_NAME='{$table}' AND TABLE_SCHEMA='{$database}'");
|
||||
$stmt->execute();
|
||||
|
||||
$column = $stmt->fetch(\PDO::FETCH_ASSOC);
|
||||
|
||||
return (!empty($column)) ? $column['COLUMN_NAME'] : '';
|
||||
}
|
||||
}
|
|
@ -1,508 +0,0 @@
|
|||
<?php
|
||||
|
||||
namespace top\library\database\driver;
|
||||
|
||||
use top\library\database\ifs\DatabaseIfs;
|
||||
use top\library\exception\DatabaseException;
|
||||
use top\traits\Instance;
|
||||
|
||||
/**
|
||||
* MySQL数据库驱动
|
||||
* @author topnuomi 2018年11月20日
|
||||
*/
|
||||
class ObjectForMySQL implements DatabaseIfs
|
||||
{
|
||||
|
||||
use Instance;
|
||||
|
||||
private $link;
|
||||
|
||||
private $mysqli = null;
|
||||
|
||||
private $sql;
|
||||
|
||||
/**
|
||||
* 连接数据库
|
||||
* @param array $config
|
||||
* @return $this|DatabaseIfs
|
||||
*/
|
||||
public function connect($config)
|
||||
{
|
||||
$this->mysqli = new \mysqli($config['host'], $config['user'], $config['passwd'], $config['dbname'], $config['port']);
|
||||
$this->mysqli->query('set names ' . $config['charset']);
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 插入记录
|
||||
* @param string $table
|
||||
* @param array $data
|
||||
* @return mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function insert($table, $data)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
if (count($data) == count($data, 1)) { // 一维数组
|
||||
$query = 'insert into ' . $table;
|
||||
$field = ' (' . implode(',', array_keys($data)) . ')';
|
||||
$value = array_values($data);
|
||||
$value = '(' . implode(',', $this->checkNull($value)) . ')';
|
||||
$this->sql = $query .= $field . ' values ' . $value . ';';
|
||||
$this->query($query);
|
||||
} else { // 二维数组
|
||||
foreach ($data as $key => $value) {
|
||||
$query = 'insert into ' . $table;
|
||||
$allField = ' (' . implode(',', array_keys($value)) . ')';
|
||||
$allValue = '(' . implode(',', $this->checkNull($value)) . ')';
|
||||
$this->sql = $query .= $allField . ' values ' . $allValue . ';';
|
||||
$this->query($query);
|
||||
}
|
||||
}
|
||||
return $this->mysqli->insert_id;
|
||||
}
|
||||
|
||||
/**
|
||||
* 更新记录
|
||||
* @param string $table
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @param array $data
|
||||
* @return int|mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function update($table, $join, $on, $where, $order, $limit, $data)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$limit = $this->parseLimit($limit);
|
||||
$query = 'update ' . $tableInfo['table'] . "{$join} set ";
|
||||
$updateData = [];
|
||||
foreach ($data as $key => $value) {
|
||||
if (!is_numeric($value) && !$value) {
|
||||
$value = 'NULL';
|
||||
} else {
|
||||
$value = '\'' . mysqli_real_escape_string($this->link, $value) . '\'';
|
||||
}
|
||||
$updateData[] = $key . '=' . $value;
|
||||
}
|
||||
$this->sql = $query .= implode(',', $updateData) . "{$where}{$order}{$limit}";
|
||||
$this->query($query);
|
||||
return $this->mysqli->affected_rows;
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询一条记录
|
||||
* @param string $table
|
||||
* @param $distinct
|
||||
* @param array|string $field
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @return array|mixed|null
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function find($table, $distinct, $field, $join, $on, $where, $order)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
$distinct = $this->parseDistinct($distinct);
|
||||
if ($distinct) {
|
||||
$field = $distinct;
|
||||
} else {
|
||||
$field = $this->parseField($field);
|
||||
}
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$this->sql = "select {$field} from {$tableInfo['table']}{$join}{$where}{$order} limit 1";
|
||||
$result = $this->query($this->sql);
|
||||
return $result->fetch_assoc();
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询所有记录
|
||||
* @param string $table
|
||||
* @param $distinct
|
||||
* @param array|string $field
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @return array|mixed|null
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function select($table, $distinct, $field, $join, $on, $where, $order, $limit)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
$distinct = $this->parseDistinct($distinct);
|
||||
if ($distinct) {
|
||||
$field = $distinct;
|
||||
} else {
|
||||
$field = $this->parseField($field);
|
||||
}
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$limit = $this->parseLimit($limit);
|
||||
$this->sql = "select {$field} from {$tableInfo['table']}{$join}{$where}{$order}{$limit}";
|
||||
$result = $this->query($this->sql);
|
||||
return $result->fetch_all(MYSQLI_ASSOC);
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除记录
|
||||
* @param array|string $effect
|
||||
* @param string $table
|
||||
* @param array $join
|
||||
* @param array|string $on
|
||||
* @param array|string $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @return int|mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function delete($effect, $table, $join, $on, $where, $order, $limit)
|
||||
{
|
||||
// TODO Auto-generated method stub
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$effect = $this->parseEffect($effect);
|
||||
$join = $this->parseJoin($join, $on);
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$order = $this->parseOrder($order);
|
||||
$limit = $this->parseLimit($limit);
|
||||
$this->sql = "delete{$effect} from {$tableInfo['table']}{$join}{$where}{$order}{$limit}";
|
||||
$this->query($this->sql);
|
||||
return $this->mysqli->affected_rows;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取表结构
|
||||
* @param $table
|
||||
* @return bool|mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function tableDesc($table)
|
||||
{
|
||||
$sql = 'desc ' . $table;
|
||||
if (!$result = $this->query($sql)) {
|
||||
return false;
|
||||
}
|
||||
$data = $result->fetch_all(MYSQLI_ASSOC);
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* 公共方法
|
||||
* @param $table
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $on
|
||||
* @param $where
|
||||
* @param $type
|
||||
* @return bool
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function common($table, $distinct, $field, $join, $on, $where, $type)
|
||||
{
|
||||
$tableInfo = $this->parseTable($table);
|
||||
$distinct = $this->parseDistinct($distinct);
|
||||
if ($distinct) {
|
||||
$field = $distinct;
|
||||
} else {
|
||||
$field = $this->parseField($field);
|
||||
}
|
||||
$join = $this->parseJoin($join, $on);
|
||||
array_push($where, $tableInfo['where']);
|
||||
$where = $this->parseWhere($where);
|
||||
$this->sql = "select {$type}({$field}) from {$tableInfo['table']}{$join}{$where}";
|
||||
$result = $this->query($this->sql);
|
||||
$data = $result->fetch_array();
|
||||
if (isset($data[0])) {
|
||||
return $data[0];
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 开启事务
|
||||
*/
|
||||
public function begin()
|
||||
{
|
||||
$this->mysqli->begin_transaction();
|
||||
}
|
||||
|
||||
/**
|
||||
* 提交
|
||||
*/
|
||||
public function commit()
|
||||
{
|
||||
$this->mysqli->commit();
|
||||
}
|
||||
|
||||
/**
|
||||
* 回滚
|
||||
*/
|
||||
public function rollback()
|
||||
{
|
||||
$this->mysqli->rollback();
|
||||
}
|
||||
|
||||
/**
|
||||
* 执行SQL
|
||||
* @param string $query
|
||||
* @return mixed
|
||||
* @throws DatabaseException
|
||||
*/
|
||||
public function query($query)
|
||||
{
|
||||
$result = $this->mysqli->query($query);
|
||||
if (!$result) {
|
||||
throw new DatabaseException($this->mysqli->error);
|
||||
}
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取执行的最后一条SQL
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
public function sql()
|
||||
{
|
||||
return trim($this->sql, ' ');
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析表信息
|
||||
* @param $table
|
||||
* @return array
|
||||
*/
|
||||
private function parseTable($table)
|
||||
{
|
||||
$info = [];
|
||||
// 如果是多表查询,给当前表名别名this
|
||||
if ($table[1] === true) {
|
||||
$info['table'] = $table[0] . ' as this';
|
||||
$info['where'] = [];
|
||||
// 如果存在主键的条件,给键名加上别名
|
||||
if (!empty($table[2])) {
|
||||
$field = 'this.' . array_keys($table[2])[0];
|
||||
$value = array_values($table[2])[0];
|
||||
$info['where'] = [$field => $value];
|
||||
}
|
||||
} else {
|
||||
$info['table'] = $table[0];
|
||||
$info['where'] = $table[2];
|
||||
}
|
||||
return $info;
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析多表的删除
|
||||
* @param $effect
|
||||
* @return string
|
||||
*/
|
||||
public function parseEffect($effect)
|
||||
{
|
||||
if ($effect) {
|
||||
if (is_array($effect)) {
|
||||
$effect = implode(',', $effect);
|
||||
}
|
||||
return ' ' . $effect;
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析数据去重
|
||||
* @param $distinct
|
||||
* @return string
|
||||
*/
|
||||
private function parseDistinct($distinct)
|
||||
{
|
||||
if ($distinct) {
|
||||
if (is_array($distinct)) {
|
||||
$distinct = implode(',', $distinct);
|
||||
}
|
||||
return 'distinct ' . $distinct;
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合字段
|
||||
* @param string|array $field
|
||||
* @return string
|
||||
*/
|
||||
private function parseField($field)
|
||||
{
|
||||
if (!$field) {
|
||||
$field = '*';
|
||||
} else if (is_array($field)) {
|
||||
$field = implode(',', $field);
|
||||
}
|
||||
return $field;
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合where条件
|
||||
* @param array $array
|
||||
* @param string $glue
|
||||
* @return string
|
||||
*/
|
||||
private function parseWhere(array $array, $glue = 'and')
|
||||
{
|
||||
$where = [];
|
||||
foreach ($array as $value) {
|
||||
if (empty($value)) continue;
|
||||
if (is_array($value)) {
|
||||
foreach ($value as $key => $val) {
|
||||
if (is_array($val)) {
|
||||
switch (strtolower($val[0])) {
|
||||
case 'in':
|
||||
$arr_ = (is_array($val[1])) ? $val[1] : explode(',', $val[1]);
|
||||
$str = '';
|
||||
for ($i = 0; $i < count($arr_); $i++) {
|
||||
$str .= (($i != 0) ? ',' : '') . $this->checkNull(trim($arr_[$i]));
|
||||
}
|
||||
$where[] = $key . ' ' . $val[0] . ' (' . $str . ')';
|
||||
break;
|
||||
case 'like':
|
||||
$where[] = $key . ' ' . $val[0] . ' \'%' . $val[1] . '%\'';
|
||||
break;
|
||||
default:
|
||||
$where[] = $key . ' ' . $val[0] . ' ' . $this->checkNull($val[1]);
|
||||
}
|
||||
} else {
|
||||
$val = $this->checkNull($val);
|
||||
$where[] = $key . '=' . $val;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
$where[] = $value;
|
||||
}
|
||||
}
|
||||
if (empty($where)) {
|
||||
return '';
|
||||
} else {
|
||||
return ' where ' . implode(' ' . $glue . ' ', $where);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合order
|
||||
* @param string $order
|
||||
* @return string
|
||||
*/
|
||||
private function parseOrder($order = '')
|
||||
{
|
||||
if ($order) {
|
||||
$order = ' order by ' . $order;
|
||||
}
|
||||
return $order;
|
||||
}
|
||||
|
||||
/**
|
||||
* 组合limit
|
||||
* @param string $limit
|
||||
* @return string
|
||||
*/
|
||||
private function parseLimit($limit = '')
|
||||
{
|
||||
if ($limit) {
|
||||
if (is_array($limit)) {
|
||||
$limit = ' limit ' . implode(',', $limit);
|
||||
} else {
|
||||
$limit = ' limit ' . $limit;
|
||||
}
|
||||
}
|
||||
return $limit;
|
||||
}
|
||||
|
||||
/**
|
||||
* 链接多表(join on)
|
||||
* @param array $data
|
||||
* @param string|array $on
|
||||
* @return string
|
||||
*/
|
||||
private function parseJoin($data, $on)
|
||||
{
|
||||
$join = [];
|
||||
for ($i = 0; $i < count($data); $i++) {
|
||||
$string = null;
|
||||
if (isset($on[$i])) {
|
||||
if (is_array($on[$i])) {
|
||||
$pieces = [];
|
||||
foreach ($on[$i] as $key => $value) {
|
||||
$pieces[] = $key . ' = ' . $value;
|
||||
}
|
||||
$string = ' on ' . implode(' and ', $pieces);
|
||||
} else {
|
||||
$string = ' on ' . $on[$i];
|
||||
}
|
||||
}
|
||||
$join[] = $data[$i][0] . ' join ' . $data[$i][1] . ($data[$i][2] ? ' as ' . $data[$i][2] : '') . $string;
|
||||
}
|
||||
if (!empty($join)) {
|
||||
return ' ' . implode(' ', $join);
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查并处理空值
|
||||
* @param $value
|
||||
* @return array|string
|
||||
*/
|
||||
private function checkNull($value)
|
||||
{
|
||||
if (is_array($value)) {
|
||||
foreach ($value as $k => $v) {
|
||||
if (!is_numeric($v) && !$v) {
|
||||
$value[$k] = 'NULL';
|
||||
} else {
|
||||
$value[$k] = '\'' . $this->mysqli->real_escape_string($v) . '\'';
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (!is_numeric($value) && !$value) {
|
||||
$value = 'NULL';
|
||||
} else {
|
||||
$value = '\'' . $this->mysqli->real_escape_string($value) . '\'';
|
||||
}
|
||||
}
|
||||
return $value;
|
||||
}
|
||||
|
||||
/**
|
||||
* 关闭数据库连接
|
||||
*/
|
||||
public function close()
|
||||
{
|
||||
if ($this->mysqli->close()) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public function __destruct()
|
||||
{
|
||||
$this->close();
|
||||
}
|
||||
}
|
|
@ -0,0 +1,51 @@
|
|||
<?php
|
||||
|
||||
namespace top\library\database\driver;
|
||||
|
||||
use top\library\database\Base;
|
||||
use top\library\database\ifs\DatabaseIfs;
|
||||
use top\traits\Instance;
|
||||
|
||||
class Sqlite extends Base implements DatabaseIfs
|
||||
{
|
||||
|
||||
use Instance;
|
||||
|
||||
/**
|
||||
* 连接数据库
|
||||
* @param $config
|
||||
* @return $this
|
||||
*/
|
||||
public function connect($config)
|
||||
{
|
||||
$dsn = "sqlite:{$config['dbname']}";
|
||||
$this->pdo = new \PDO($dsn);
|
||||
$this->pdo->setAttribute(\PDO::ATTR_EMULATE_PREPARES, false);
|
||||
// 设置字符集
|
||||
$this->pdo->exec('SET NAMES ' . $config['charset']);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取主键
|
||||
* @param $table
|
||||
* @param $database
|
||||
* @return string
|
||||
*/
|
||||
public function getPk($table, $database)
|
||||
{
|
||||
$stmt = $this->pdo->query("PRAGMA TABLE_INFO('$table')");
|
||||
$stmt->execute();
|
||||
|
||||
$columns = $stmt->fetchAll(\PDO::FETCH_ASSOC);
|
||||
$pk = '';
|
||||
foreach ($columns as $column) {
|
||||
if ($column['pk'] == 1) {
|
||||
$pk = $column['name'];
|
||||
break;
|
||||
}
|
||||
}
|
||||
return $pk;
|
||||
}
|
||||
}
|
|
@ -11,76 +11,17 @@ interface DatabaseIfs
|
|||
|
||||
/**
|
||||
* 连接数据库
|
||||
* @param array $config
|
||||
* @param $config
|
||||
* @return mixed
|
||||
*/
|
||||
public function connect($config);
|
||||
|
||||
/**
|
||||
* 插入记录
|
||||
* @param string $table
|
||||
* @param array $data
|
||||
*/
|
||||
public function insert($table, $data);
|
||||
|
||||
/**
|
||||
* 更新记录
|
||||
* @param string $table
|
||||
* @param array $join
|
||||
* @param string|array $on
|
||||
* @param string|array $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
* @param array $data
|
||||
*/
|
||||
public function update($table, $join, $on, $where, $order, $limit, $data);
|
||||
|
||||
/**
|
||||
* 查找一条记录
|
||||
* 获取主键
|
||||
* @param $table
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $on
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $database
|
||||
* @return mixed
|
||||
*/
|
||||
public function find($table, $distinct, $field, $join, $on, $where, $order);
|
||||
public function getPk($table, $database);
|
||||
|
||||
/**
|
||||
* 查找全部
|
||||
* @param $table
|
||||
* @param $distinct
|
||||
* @param $field
|
||||
* @param $join
|
||||
* @param $on
|
||||
* @param $where
|
||||
* @param $order
|
||||
* @param $limit
|
||||
* @return mixed
|
||||
*/
|
||||
public function select($table, $distinct, $field, $join, $on, $where, $order, $limit);
|
||||
|
||||
/**
|
||||
* 删除记录
|
||||
* @param string|array $effect
|
||||
* @param string $table
|
||||
* @param array $join
|
||||
* @param string|array $on
|
||||
* @param string|array $where
|
||||
* @param string $order
|
||||
* @param string $limit
|
||||
*/
|
||||
public function delete($effect, $table, $join, $on, $where, $order, $limit);
|
||||
|
||||
/**
|
||||
* 执行一条SQL
|
||||
* @param string $query
|
||||
*/
|
||||
public function query($query);
|
||||
|
||||
/**
|
||||
* 关闭数据库连接
|
||||
*/
|
||||
public function close();
|
||||
}
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
namespace top\library\error;
|
||||
|
||||
use Throwable;
|
||||
use top\library\exception\BaseException;
|
||||
|
||||
class BaseError extends \Error
|
||||
{
|
||||
|
@ -17,15 +18,11 @@ class BaseError extends \Error
|
|||
* @param $errstr
|
||||
* @param $errfile
|
||||
* @param $errline
|
||||
* @throws BaseException
|
||||
*/
|
||||
public function handler($errno, $errstr, $errfile, $errline)
|
||||
{
|
||||
if (DEBUG) {
|
||||
$content = $errstr . '<br />' . $errfile . ' 第' . $errline . '行';
|
||||
} else {
|
||||
$content = $errstr;
|
||||
}
|
||||
// throw new BaseException($errstr, 0, null, $errfile, $errline);
|
||||
echo '<p style="font-size: 12px; font-weight: 100;">' . $content . '</p>';
|
||||
throw new BaseException($errstr, 0, null, $errfile, $errline);
|
||||
// echo '<p style="font-size: 12px; font-weight: 100;">' . $content . '</p>';
|
||||
}
|
||||
}
|
||||
|
|
|
@ -2,40 +2,15 @@
|
|||
|
||||
namespace top\library\exception;
|
||||
|
||||
use Throwable;
|
||||
|
||||
class DatabaseException extends BaseException
|
||||
{
|
||||
public function __construct($message = "", $code = 0, Throwable $previous = null)
|
||||
public function __construct($message = "", $code = 0, \Throwable $previous = null)
|
||||
{
|
||||
$message = $this->processMessage($message);
|
||||
parent::__construct($message, $code, $previous);
|
||||
parent::__construct('[DatabaseException]' . $message, $code, $previous);
|
||||
}
|
||||
|
||||
public function handler($exception = null)
|
||||
{
|
||||
parent::handler($this); // TODO: Change the autogenerated stub
|
||||
}
|
||||
|
||||
private function processMessage($message)
|
||||
{
|
||||
$message = str_ireplace([
|
||||
'database',
|
||||
'table',
|
||||
'doesn\'t exist',
|
||||
'unknown',
|
||||
'column',
|
||||
'field',
|
||||
'list'
|
||||
], [
|
||||
'数据库: ',
|
||||
'表',
|
||||
'不存在',
|
||||
'未知',
|
||||
'列',
|
||||
'字段',
|
||||
'列表'
|
||||
], $message);
|
||||
return $message;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -4,5 +4,16 @@ namespace top\library\exception;
|
|||
|
||||
class RequestException extends BaseException
|
||||
{
|
||||
public function __construct($message = "", $code = 0, \Throwable $previous = null)
|
||||
{
|
||||
parent::__construct('[RequestException]' . $message, $code, $previous);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param \Exception $exception
|
||||
*/
|
||||
public function handler($exception = null)
|
||||
{
|
||||
parent::handler($this); // TODO: Change the autogenerated stub
|
||||
}
|
||||
}
|
|
@ -2,14 +2,11 @@
|
|||
|
||||
namespace top\library\exception;
|
||||
|
||||
use Throwable;
|
||||
|
||||
class RouteException extends BaseException
|
||||
{
|
||||
public function __construct($message = "", $code = 0, Throwable $previous = null)
|
||||
public function __construct($message = "", $code = 0, \Throwable $previous = null)
|
||||
{
|
||||
$message = $this->processMessage($message);
|
||||
parent::__construct($message, $code, $previous);
|
||||
parent::__construct('[DatabaseException]' . $message, $code, $previous);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -19,20 +16,4 @@ class RouteException extends BaseException
|
|||
{
|
||||
parent::handler($this); // TODO: Change the autogenerated stub
|
||||
}
|
||||
|
||||
private function processMessage($message)
|
||||
{
|
||||
$message = str_replace([
|
||||
'Module',
|
||||
'Controller',
|
||||
'function',
|
||||
'doesn\'t exist',
|
||||
], [
|
||||
'模块',
|
||||
'控制器',
|
||||
'方法',
|
||||
'不存在',
|
||||
], $message);
|
||||
return $message;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,49 +1,5 @@
|
|||
<?php
|
||||
|
||||
/**
|
||||
* 获取headers
|
||||
* @return array|false
|
||||
*/
|
||||
function get_header()
|
||||
{
|
||||
if (PHP_SAPI === 'apache2handler') {
|
||||
$headers = getallheaders();
|
||||
$data = [];
|
||||
foreach ($headers as $key => $value) {
|
||||
$data[strtolower($key)] = $value;
|
||||
}
|
||||
unset($headers);
|
||||
return $data;
|
||||
} else {
|
||||
$server = $_SERVER;
|
||||
$headers = [];
|
||||
foreach ($server as $key => $value) {
|
||||
if ('http_' == strtolower(substr($key, 0, 5))) {
|
||||
$headers[strtolower(substr($key, 5))] = $value;
|
||||
}
|
||||
}
|
||||
unset($server);
|
||||
return $headers;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 过滤数组
|
||||
* @param array $array
|
||||
* @param string $filter
|
||||
* @param array $result
|
||||
*/
|
||||
function filterArray($array = [], $filter = 'filter', &$result = [])
|
||||
{
|
||||
foreach ($array as $key => $value) {
|
||||
if (is_array($value)) {
|
||||
filterArray($value, $result[$key]);
|
||||
} else {
|
||||
$result[$key] = (!$filter) ? $value : $filter($value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取/设置配置
|
||||
* @param $key
|
||||
|
@ -65,11 +21,6 @@ function config($key, $value = '__NULL__VALUE__')
|
|||
*/
|
||||
function request()
|
||||
{
|
||||
/*static $instance;
|
||||
if (!$instance) {
|
||||
$instance = new \top\library\http\Request();
|
||||
}
|
||||
return $instance;*/
|
||||
return \top\library\http\Request::instance();
|
||||
}
|
||||
|
||||
|
@ -79,32 +30,59 @@ function request()
|
|||
*/
|
||||
function response()
|
||||
{
|
||||
/*static $instance;
|
||||
if (!$instance) {
|
||||
$instance = new \top\library\http\Response();
|
||||
}
|
||||
return $instance;*/
|
||||
return \top\library\http\Response::instance();
|
||||
}
|
||||
|
||||
/**
|
||||
* 调用模型
|
||||
* @param $class
|
||||
* @return mixed
|
||||
* @param $name
|
||||
* @return \top\library\Model
|
||||
*/
|
||||
function model($class)
|
||||
function model($name)
|
||||
{
|
||||
static $model = [];
|
||||
$class = class_full_name($name, 'model');
|
||||
if (!isset($model[$class])) {
|
||||
// 模型类存在则直接实例化
|
||||
// 模型类不存在则直接将传入的模型名当作表名处理
|
||||
if (class_exists($class)) {
|
||||
$model[$class] = new $class();
|
||||
} else {
|
||||
$model[$class] = new \top\library\Model($class);
|
||||
$model[$class] = new \top\library\Model($name);
|
||||
}
|
||||
}
|
||||
return $model[$class];
|
||||
}
|
||||
|
||||
/**
|
||||
* 调用逻辑
|
||||
* @param $name
|
||||
* @return mixed
|
||||
*/
|
||||
function logic($name)
|
||||
{
|
||||
static $logic = [];
|
||||
$class = class_full_name($name, 'logic');
|
||||
if (!isset($logic[$class])) {
|
||||
// 实例化逻辑类
|
||||
$logic[$class] = new $class();
|
||||
}
|
||||
return $logic[$class];
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取类全限定名
|
||||
* @param $name
|
||||
* @param string $type
|
||||
* @return string
|
||||
*/
|
||||
function class_full_name($name, $type = 'model')
|
||||
{
|
||||
if (!strstr($name, '\\')) { // 不是类全限定名,则直接拼接全限定名
|
||||
return APP_NS . '\\' . BIND_MODULE . '\\' . $type . '\\' . $name;
|
||||
} else return $name;
|
||||
}
|
||||
|
||||
/**
|
||||
* 拼接链接(暂时先这样
|
||||
* @param string $url
|
||||
|
@ -124,6 +102,16 @@ function url($url, $param = '')
|
|||
return '/' . $url . $param . '.html';
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取当前视图文件的缓存标识
|
||||
* @return string
|
||||
*/
|
||||
function view_cache_ident()
|
||||
{
|
||||
$ident = md5($_SERVER['REQUEST_URI'] . request()->requestMethod());
|
||||
return $ident;
|
||||
}
|
||||
|
||||
/**
|
||||
* 设置视图缓存时间
|
||||
* @param $sec
|
||||
|
@ -155,6 +143,76 @@ function view($file = '', $param = [], $cache = false)
|
|||
return \top\library\View::instance()->fetch($file, $param, $cache);
|
||||
}
|
||||
|
||||
/**
|
||||
* 页面跳转
|
||||
* @param $url
|
||||
* @return false|string
|
||||
*/
|
||||
function redirect($url)
|
||||
{
|
||||
if (request()->is('ajax')) {
|
||||
return json_encode([
|
||||
'redirect' => $url,
|
||||
]);
|
||||
} else {
|
||||
header('location: ' . $url);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 框架session操作
|
||||
* @param $name
|
||||
* @param string $value
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
*/
|
||||
function session($name, $value = '')
|
||||
{
|
||||
$config = \top\library\Config::instance()->get('session');
|
||||
if (empty($config) || !$config['prefix']) {
|
||||
$prefix = request()->module();
|
||||
} else {
|
||||
$prefix = $config['prefix'];
|
||||
}
|
||||
if ($value === '') {
|
||||
if (isset($_SESSION[$prefix][$name])) {
|
||||
return $_SESSION[$prefix][$name];
|
||||
}
|
||||
return false;
|
||||
} else if ($value === false) {
|
||||
unset($_SESSION[$prefix][$name]);
|
||||
} else {
|
||||
$_SESSION[$prefix][$name] = $value;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取headers
|
||||
* @return array|false
|
||||
*/
|
||||
function get_header()
|
||||
{
|
||||
if (PHP_SAPI === 'apache2handler') {
|
||||
$headers = getallheaders();
|
||||
$data = [];
|
||||
foreach ($headers as $key => $value) {
|
||||
$data[strtolower($key)] = $value;
|
||||
}
|
||||
unset($headers);
|
||||
return $data;
|
||||
} else {
|
||||
$server = $_SERVER;
|
||||
$headers = [];
|
||||
foreach ($server as $key => $value) {
|
||||
if ('http_' == strtolower(substr($key, 0, 5))) {
|
||||
$headers[strtolower(substr($key, 5))] = $value;
|
||||
}
|
||||
}
|
||||
unset($server);
|
||||
return $headers;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取表名
|
||||
* @param $classname
|
||||
|
@ -175,37 +233,6 @@ function get_table_name($classname)
|
|||
return strtolower($table);
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建HTTP请求
|
||||
* @param $url
|
||||
* @param array $data
|
||||
* @param array $header
|
||||
* @return bool|mixed
|
||||
*/
|
||||
function create_http_request($url, $data = [], $header = [])
|
||||
{
|
||||
$curl = curl_init();
|
||||
curl_setopt($curl, CURLOPT_URL, $url);
|
||||
if (!empty($data)) {
|
||||
curl_setopt($curl, CURLOPT_POST, true);
|
||||
curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
|
||||
}
|
||||
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($curl, CURLOPT_HEADER, $header);
|
||||
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
|
||||
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
|
||||
$response = curl_exec($curl);
|
||||
if (!empty($header)) {
|
||||
$headerSize = curl_getinfo($curl, CURLINFO_HEADER_SIZE);
|
||||
$response = substr($response, $headerSize);
|
||||
}
|
||||
curl_close($curl);
|
||||
if ($response) {
|
||||
return $response;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取客户端IP
|
||||
* @param int $type
|
||||
|
@ -246,19 +273,34 @@ function get_client_ip($type = 0, $client = true)
|
|||
}
|
||||
|
||||
/**
|
||||
* 页面跳转
|
||||
* 创建HTTP请求
|
||||
* @param $url
|
||||
* @return false|string
|
||||
* @param array $data
|
||||
* @param array $header
|
||||
* @return bool|mixed
|
||||
*/
|
||||
function redirect($url)
|
||||
function create_http_request($url, $data = [], $header = [])
|
||||
{
|
||||
if (request()->is('ajax')) {
|
||||
return json_encode([
|
||||
'redirect' => $url,
|
||||
]);
|
||||
} else {
|
||||
header('location: ' . $url);
|
||||
$curl = curl_init();
|
||||
curl_setopt($curl, CURLOPT_URL, $url);
|
||||
if (!empty($data)) {
|
||||
curl_setopt($curl, CURLOPT_POST, true);
|
||||
curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
|
||||
}
|
||||
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($curl, CURLOPT_HEADER, $header);
|
||||
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
|
||||
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
|
||||
$response = curl_exec($curl);
|
||||
if (!empty($header)) {
|
||||
$headerSize = curl_getinfo($curl, CURLINFO_HEADER_SIZE);
|
||||
$response = substr($response, $headerSize);
|
||||
}
|
||||
curl_close($curl);
|
||||
if ($response) {
|
||||
return $response;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -299,29 +341,19 @@ function filter($str)
|
|||
}
|
||||
|
||||
/**
|
||||
* 框架session操作
|
||||
* @param $name
|
||||
* @param string $value
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
* 过滤数组
|
||||
* @param array $array
|
||||
* @param string $filter
|
||||
* @param array $result
|
||||
*/
|
||||
function session($name, $value = '')
|
||||
function filter_array($array = [], $filter = 'filter', &$result = [])
|
||||
{
|
||||
$config = \top\library\Config::instance()->get('session');
|
||||
if (empty($config) || !$config['prefix']) {
|
||||
$prefix = request()->module();
|
||||
} else {
|
||||
$prefix = $config['prefix'];
|
||||
}
|
||||
if ($value === '') {
|
||||
if (isset($_SESSION[$prefix][$name])) {
|
||||
return $_SESSION[$prefix][$name];
|
||||
foreach ($array as $key => $value) {
|
||||
if (is_array($value)) {
|
||||
filter_array($value, $result[$key]);
|
||||
} else {
|
||||
$result[$key] = (!$filter) ? $value : $filter($value);
|
||||
}
|
||||
return false;
|
||||
} else if ($value === false) {
|
||||
unset($_SESSION[$prefix][$name]);
|
||||
} else {
|
||||
$_SESSION[$prefix][$name] = $value;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -479,16 +511,6 @@ function is_mobile()
|
|||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取当前视图文件的缓存标识
|
||||
* @return string
|
||||
*/
|
||||
function view_cache_ident()
|
||||
{
|
||||
$ident = md5($_SERVER['REQUEST_URI'] . request()->requestMethod());
|
||||
return $ident;
|
||||
}
|
||||
|
||||
// 模型自动验证函数
|
||||
|
||||
/**
|
||||
|
@ -533,7 +555,7 @@ function notEqual($value, $value1)
|
|||
* @param int $max
|
||||
* @return boolean
|
||||
*/
|
||||
function isBetween($value, $min, $max)
|
||||
function length($value, $min, $max)
|
||||
{
|
||||
$length = mb_strlen($value, 'utf8');
|
||||
if ($length < $min || $length > $max) {
|
||||
|
|
|
@ -263,7 +263,7 @@ class Request
|
|||
// 重置except的值
|
||||
$this->except = [];
|
||||
|
||||
filterArray($data, $filter, $data);
|
||||
filter_array($data, $filter, $data);
|
||||
|
||||
if ($name) {
|
||||
if (isset($data[$name])) {
|
||||
|
|
|
@ -2,11 +2,23 @@
|
|||
|
||||
namespace top\traits;
|
||||
|
||||
|
||||
/**
|
||||
* Trait Instance
|
||||
* @package top\traits
|
||||
*/
|
||||
trait Instance
|
||||
{
|
||||
/**
|
||||
* 实例
|
||||
* @var object
|
||||
*/
|
||||
private static $instance;
|
||||
|
||||
/**
|
||||
* 获取类实例
|
||||
* @param null $param
|
||||
* @return static
|
||||
*/
|
||||
public static function instance($param = null)
|
||||
{
|
||||
if (!self::$instance) {
|
||||
|
@ -15,10 +27,17 @@ trait Instance
|
|||
return self::$instance;
|
||||
}
|
||||
|
||||
/**
|
||||
* 私有化构造方法
|
||||
* Instance constructor.
|
||||
*/
|
||||
private function __construct()
|
||||
{
|
||||
}
|
||||
|
||||
/**
|
||||
* 私有化克隆方法
|
||||
*/
|
||||
private function __clone()
|
||||
{
|
||||
// TODO: Implement __clone() method.
|
||||
|
|
|
@ -2,8 +2,19 @@
|
|||
|
||||
namespace top\traits;
|
||||
|
||||
/**
|
||||
* Trait Json
|
||||
* @package top\traits
|
||||
*/
|
||||
trait Json {
|
||||
|
||||
/**
|
||||
* 格式化数据为json
|
||||
* @param $msg
|
||||
* @param int $code
|
||||
* @param array $data
|
||||
* @return false|string
|
||||
*/
|
||||
public function returnJson($msg, $code = 0, $data = [])
|
||||
{
|
||||
if (is_array($msg)) {
|
||||
|
|
|
@ -0,0 +1,47 @@
|
|||
<?php
|
||||
|
||||
namespace top\traits;
|
||||
|
||||
trait Magic
|
||||
{
|
||||
private static $magicParameters = [];
|
||||
|
||||
/**
|
||||
* 使用成员变量调用某些类
|
||||
* @param $name
|
||||
* @return mixed
|
||||
*/
|
||||
public function __get($name)
|
||||
{
|
||||
if (!isset(self::$magicParameters[$name])) {
|
||||
$stringArray = str_split($name);
|
||||
$length = count($stringArray);
|
||||
$start = 0;
|
||||
$prefix = '';
|
||||
for ($i = 0; $i < $length; $i++) {
|
||||
$ord = ord($stringArray[$i]);
|
||||
if ($ord > 64 && $ord < 91 && $i != 0) { // 找大写字母
|
||||
$start = $i;
|
||||
break;
|
||||
}
|
||||
$prefix .= $stringArray[$i];
|
||||
}
|
||||
if ($prefix != '' && $start > 0) { // 存在前缀,并且存在大写字母
|
||||
$value = substr($name, $start, $length);
|
||||
switch ($prefix) {
|
||||
case 'model':
|
||||
self::$magicParameters[$name] = model($value);
|
||||
break;
|
||||
case 'logic':
|
||||
self::$magicParameters[$name] = logic($value);
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
// 无法被处理,抛出异常
|
||||
throw new \Exception('变量' . $name . '不存在');
|
||||
}
|
||||
}
|
||||
|
||||
return self::$magicParameters[$name];
|
||||
}
|
||||
}
|
|
@ -6,5 +6,7 @@ $vendorDir = dirname(dirname(__FILE__));
|
|||
$baseDir = dirname(dirname($vendorDir));
|
||||
|
||||
return array(
|
||||
'top\\' => array($baseDir . '/framework'),
|
||||
'app\\' => array($baseDir . '/application'),
|
||||
'Firebase\\JWT\\' => array($vendorDir . '/firebase/php-jwt/src'),
|
||||
);
|
||||
|
|
|
@ -7,6 +7,14 @@ namespace Composer\Autoload;
|
|||
class ComposerStaticInit7b44678ec2aea793416a22dbbbba76ef
|
||||
{
|
||||
public static $prefixLengthsPsr4 = array (
|
||||
't' =>
|
||||
array (
|
||||
'top\\' => 4,
|
||||
),
|
||||
'a' =>
|
||||
array (
|
||||
'app\\' => 4,
|
||||
),
|
||||
'F' =>
|
||||
array (
|
||||
'Firebase\\JWT\\' => 13,
|
||||
|
@ -14,6 +22,14 @@ class ComposerStaticInit7b44678ec2aea793416a22dbbbba76ef
|
|||
);
|
||||
|
||||
public static $prefixDirsPsr4 = array (
|
||||
'top\\' =>
|
||||
array (
|
||||
0 => __DIR__ . '/../../..' . '/framework',
|
||||
),
|
||||
'app\\' =>
|
||||
array (
|
||||
0 => __DIR__ . '/../../..' . '/application',
|
||||
),
|
||||
'Firebase\\JWT\\' =>
|
||||
array (
|
||||
0 => __DIR__ . '/..' . '/firebase/php-jwt/src',
|
||||
|
|
|
@ -1,26 +1,26 @@
|
|||
[
|
||||
{
|
||||
"name": "firebase/php-jwt",
|
||||
"version": "v5.0.0",
|
||||
"version_normalized": "5.0.0.0",
|
||||
"version": "v5.2.0",
|
||||
"version_normalized": "5.2.0.0",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/firebase/php-jwt.git",
|
||||
"reference": "9984a4d3a32ae7673d6971ea00bae9d0a1abba0e"
|
||||
"reference": "feb0e820b8436873675fd3aca04f3728eb2185cb"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/firebase/php-jwt/zipball/9984a4d3a32ae7673d6971ea00bae9d0a1abba0e",
|
||||
"reference": "9984a4d3a32ae7673d6971ea00bae9d0a1abba0e",
|
||||
"url": "https://api.github.com/repos/firebase/php-jwt/zipball/feb0e820b8436873675fd3aca04f3728eb2185cb",
|
||||
"reference": "feb0e820b8436873675fd3aca04f3728eb2185cb",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=5.3.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": " 4.8.35"
|
||||
"phpunit/phpunit": ">=4.8 <=9"
|
||||
},
|
||||
"time": "2017-06-27T22:17:23+00:00",
|
||||
"time": "2020-03-25T18:49:23+00:00",
|
||||
"type": "library",
|
||||
"installation-source": "dist",
|
||||
"autoload": {
|
||||
|
@ -45,6 +45,10 @@
|
|||
}
|
||||
],
|
||||
"description": "A simple library to encode and decode JSON Web Tokens (JWT) in PHP. Should conform to the current spec.",
|
||||
"homepage": "https://github.com/firebase/php-jwt"
|
||||
"homepage": "https://github.com/firebase/php-jwt",
|
||||
"keywords": [
|
||||
"jwt",
|
||||
"php"
|
||||
]
|
||||
}
|
||||
]
|
||||
|
|
|
@ -23,7 +23,7 @@ Example
|
|||
use \Firebase\JWT\JWT;
|
||||
|
||||
$key = "example_key";
|
||||
$token = array(
|
||||
$payload = array(
|
||||
"iss" => "http://example.org",
|
||||
"aud" => "http://example.com",
|
||||
"iat" => 1356999524,
|
||||
|
@ -36,7 +36,7 @@ $token = array(
|
|||
* https://tools.ietf.org/html/draft-ietf-jose-json-web-algorithms-40
|
||||
* for a list of spec-compliant algorithms.
|
||||
*/
|
||||
$jwt = JWT::encode($token, $key);
|
||||
$jwt = JWT::encode($payload, $key);
|
||||
$decoded = JWT::decode($jwt, $key, array('HS256'));
|
||||
|
||||
print_r($decoded);
|
||||
|
@ -93,14 +93,14 @@ ehde/zUxo6UvS7UrBQIDAQAB
|
|||
-----END PUBLIC KEY-----
|
||||
EOD;
|
||||
|
||||
$token = array(
|
||||
$payload = array(
|
||||
"iss" => "example.org",
|
||||
"aud" => "example.com",
|
||||
"iat" => 1356999524,
|
||||
"nbf" => 1357000000
|
||||
);
|
||||
|
||||
$jwt = JWT::encode($token, $privateKey, 'RS256');
|
||||
$jwt = JWT::encode($payload, $privateKey, 'RS256');
|
||||
echo "Encode:\n" . print_r($jwt, true) . "\n";
|
||||
|
||||
$decoded = JWT::decode($jwt, $publicKey, array('RS256'));
|
||||
|
|
|
@ -2,6 +2,10 @@
|
|||
"name": "firebase/php-jwt",
|
||||
"description": "A simple library to encode and decode JSON Web Tokens (JWT) in PHP. Should conform to the current spec.",
|
||||
"homepage": "https://github.com/firebase/php-jwt",
|
||||
"keywords": [
|
||||
"php",
|
||||
"jwt"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Neuman Vong",
|
||||
|
@ -24,6 +28,6 @@
|
|||
}
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": " 4.8.35"
|
||||
"phpunit/phpunit": ">=4.8 <=9"
|
||||
}
|
||||
}
|
||||
|
|
|
@ -3,5 +3,4 @@ namespace Firebase\JWT;
|
|||
|
||||
class BeforeValidException extends \UnexpectedValueException
|
||||
{
|
||||
|
||||
}
|
||||
|
|
|
@ -3,5 +3,4 @@ namespace Firebase\JWT;
|
|||
|
||||
class ExpiredException extends \UnexpectedValueException
|
||||
{
|
||||
|
||||
}
|
||||
|
|
|
@ -0,0 +1,171 @@
|
|||
<?php
|
||||
|
||||
namespace Firebase\JWT;
|
||||
|
||||
use DomainException;
|
||||
use UnexpectedValueException;
|
||||
|
||||
/**
|
||||
* JSON Web Key implementation, based on this spec:
|
||||
* https://tools.ietf.org/html/draft-ietf-jose-json-web-key-41
|
||||
*
|
||||
* PHP version 5
|
||||
*
|
||||
* @category Authentication
|
||||
* @package Authentication_JWT
|
||||
* @author Bui Sy Nguyen <nguyenbs@gmail.com>
|
||||
* @license http://opensource.org/licenses/BSD-3-Clause 3-clause BSD
|
||||
* @link https://github.com/firebase/php-jwt
|
||||
*/
|
||||
class JWK
|
||||
{
|
||||
/**
|
||||
* Parse a set of JWK keys
|
||||
*
|
||||
* @param array $jwks The JSON Web Key Set as an associative array
|
||||
*
|
||||
* @return array An associative array that represents the set of keys
|
||||
*
|
||||
* @throws InvalidArgumentException Provided JWK Set is empty
|
||||
* @throws UnexpectedValueException Provided JWK Set was invalid
|
||||
* @throws DomainException OpenSSL failure
|
||||
*
|
||||
* @uses parseKey
|
||||
*/
|
||||
public static function parseKeySet(array $jwks)
|
||||
{
|
||||
$keys = array();
|
||||
|
||||
if (!isset($jwks['keys'])) {
|
||||
throw new UnexpectedValueException('"keys" member must exist in the JWK Set');
|
||||
}
|
||||
if (empty($jwks['keys'])) {
|
||||
throw new InvalidArgumentException('JWK Set did not contain any keys');
|
||||
}
|
||||
|
||||
foreach ($jwks['keys'] as $k => $v) {
|
||||
$kid = isset($v['kid']) ? $v['kid'] : $k;
|
||||
if ($key = self::parseKey($v)) {
|
||||
$keys[$kid] = $key;
|
||||
}
|
||||
}
|
||||
|
||||
if (0 === \count($keys)) {
|
||||
throw new UnexpectedValueException('No supported algorithms found in JWK Set');
|
||||
}
|
||||
|
||||
return $keys;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a JWK key
|
||||
*
|
||||
* @param array $jwk An individual JWK
|
||||
*
|
||||
* @return resource|array An associative array that represents the key
|
||||
*
|
||||
* @throws InvalidArgumentException Provided JWK is empty
|
||||
* @throws UnexpectedValueException Provided JWK was invalid
|
||||
* @throws DomainException OpenSSL failure
|
||||
*
|
||||
* @uses createPemFromModulusAndExponent
|
||||
*/
|
||||
private static function parseKey(array $jwk)
|
||||
{
|
||||
if (empty($jwk)) {
|
||||
throw new InvalidArgumentException('JWK must not be empty');
|
||||
}
|
||||
if (!isset($jwk['kty'])) {
|
||||
throw new UnexpectedValueException('JWK must contain a "kty" parameter');
|
||||
}
|
||||
|
||||
switch ($jwk['kty']) {
|
||||
case 'RSA':
|
||||
if (\array_key_exists('d', $jwk)) {
|
||||
throw new UnexpectedValueException('RSA private keys are not supported');
|
||||
}
|
||||
if (!isset($jwk['n']) || !isset($jwk['e'])) {
|
||||
throw new UnexpectedValueException('RSA keys must contain values for both "n" and "e"');
|
||||
}
|
||||
|
||||
$pem = self::createPemFromModulusAndExponent($jwk['n'], $jwk['e']);
|
||||
$publicKey = \openssl_pkey_get_public($pem);
|
||||
if (false === $publicKey) {
|
||||
throw new DomainException(
|
||||
'OpenSSL error: ' . \openssl_error_string()
|
||||
);
|
||||
}
|
||||
return $publicKey;
|
||||
default:
|
||||
// Currently only RSA is supported
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a public key represented in PEM format from RSA modulus and exponent information
|
||||
*
|
||||
* @param string $n The RSA modulus encoded in Base64
|
||||
* @param string $e The RSA exponent encoded in Base64
|
||||
*
|
||||
* @return string The RSA public key represented in PEM format
|
||||
*
|
||||
* @uses encodeLength
|
||||
*/
|
||||
private static function createPemFromModulusAndExponent($n, $e)
|
||||
{
|
||||
$modulus = JWT::urlsafeB64Decode($n);
|
||||
$publicExponent = JWT::urlsafeB64Decode($e);
|
||||
|
||||
$components = array(
|
||||
'modulus' => \pack('Ca*a*', 2, self::encodeLength(\strlen($modulus)), $modulus),
|
||||
'publicExponent' => \pack('Ca*a*', 2, self::encodeLength(\strlen($publicExponent)), $publicExponent)
|
||||
);
|
||||
|
||||
$rsaPublicKey = \pack(
|
||||
'Ca*a*a*',
|
||||
48,
|
||||
self::encodeLength(\strlen($components['modulus']) + \strlen($components['publicExponent'])),
|
||||
$components['modulus'],
|
||||
$components['publicExponent']
|
||||
);
|
||||
|
||||
// sequence(oid(1.2.840.113549.1.1.1), null)) = rsaEncryption.
|
||||
$rsaOID = \pack('H*', '300d06092a864886f70d0101010500'); // hex version of MA0GCSqGSIb3DQEBAQUA
|
||||
$rsaPublicKey = \chr(0) . $rsaPublicKey;
|
||||
$rsaPublicKey = \chr(3) . self::encodeLength(\strlen($rsaPublicKey)) . $rsaPublicKey;
|
||||
|
||||
$rsaPublicKey = \pack(
|
||||
'Ca*a*',
|
||||
48,
|
||||
self::encodeLength(\strlen($rsaOID . $rsaPublicKey)),
|
||||
$rsaOID . $rsaPublicKey
|
||||
);
|
||||
|
||||
$rsaPublicKey = "-----BEGIN PUBLIC KEY-----\r\n" .
|
||||
\chunk_split(\base64_encode($rsaPublicKey), 64) .
|
||||
'-----END PUBLIC KEY-----';
|
||||
|
||||
return $rsaPublicKey;
|
||||
}
|
||||
|
||||
/**
|
||||
* DER-encode the length
|
||||
*
|
||||
* DER supports lengths up to (2**8)**127, however, we'll only support lengths up to (2**8)**4. See
|
||||
* {@link http://itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#p=13 X.690 paragraph 8.1.3} for more information.
|
||||
*
|
||||
* @param int $length
|
||||
* @return string
|
||||
*/
|
||||
private static function encodeLength($length)
|
||||
{
|
||||
if ($length <= 0x7F) {
|
||||
return \chr($length);
|
||||
}
|
||||
|
||||
$temp = \ltrim(\pack('N', $length), \chr(0));
|
||||
|
||||
return \pack('Ca*', 0x80 | \strlen($temp), $temp);
|
||||
}
|
||||
}
|
|
@ -1,6 +1,7 @@
|
|||
<?php
|
||||
|
||||
namespace Firebase\JWT;
|
||||
|
||||
use \DomainException;
|
||||
use \InvalidArgumentException;
|
||||
use \UnexpectedValueException;
|
||||
|
@ -21,6 +22,9 @@ use \DateTime;
|
|||
*/
|
||||
class JWT
|
||||
{
|
||||
const ASN1_INTEGER = 0x02;
|
||||
const ASN1_SEQUENCE = 0x10;
|
||||
const ASN1_BIT_STRING = 0x03;
|
||||
|
||||
/**
|
||||
* When checking nbf, iat or expiration times,
|
||||
|
@ -38,9 +42,10 @@ class JWT
|
|||
public static $timestamp = null;
|
||||
|
||||
public static $supported_algs = array(
|
||||
'ES256' => array('openssl', 'SHA256'),
|
||||
'HS256' => array('hash_hmac', 'SHA256'),
|
||||
'HS512' => array('hash_hmac', 'SHA512'),
|
||||
'HS384' => array('hash_hmac', 'SHA384'),
|
||||
'HS512' => array('hash_hmac', 'SHA512'),
|
||||
'RS256' => array('openssl', 'SHA256'),
|
||||
'RS384' => array('openssl', 'SHA384'),
|
||||
'RS512' => array('openssl', 'SHA512'),
|
||||
|
@ -49,11 +54,11 @@ class JWT
|
|||
/**
|
||||
* Decodes a JWT string into a PHP object.
|
||||
*
|
||||
* @param string $jwt The JWT
|
||||
* @param string|array $key The key, or map of keys.
|
||||
* If the algorithm used is asymmetric, this is the public key
|
||||
* @param array $allowed_algs List of supported verification algorithms
|
||||
* Supported algorithms are 'HS256', 'HS384', 'HS512' and 'RS256'
|
||||
* @param string $jwt The JWT
|
||||
* @param string|array|resource $key The key, or map of keys.
|
||||
* If the algorithm used is asymmetric, this is the public key
|
||||
* @param array $allowed_algs List of supported verification algorithms
|
||||
* Supported algorithms are 'ES256', 'HS256', 'HS384', 'HS512', 'RS256', 'RS384', and 'RS512'
|
||||
*
|
||||
* @return object The JWT's payload as a PHP object
|
||||
*
|
||||
|
@ -68,13 +73,13 @@ class JWT
|
|||
*/
|
||||
public static function decode($jwt, $key, array $allowed_algs = array())
|
||||
{
|
||||
$timestamp = is_null(static::$timestamp) ? time() : static::$timestamp;
|
||||
$timestamp = \is_null(static::$timestamp) ? \time() : static::$timestamp;
|
||||
|
||||
if (empty($key)) {
|
||||
throw new InvalidArgumentException('Key may not be empty');
|
||||
}
|
||||
$tks = explode('.', $jwt);
|
||||
if (count($tks) != 3) {
|
||||
$tks = \explode('.', $jwt);
|
||||
if (\count($tks) != 3) {
|
||||
throw new UnexpectedValueException('Wrong number of segments');
|
||||
}
|
||||
list($headb64, $bodyb64, $cryptob64) = $tks;
|
||||
|
@ -93,10 +98,15 @@ class JWT
|
|||
if (empty(static::$supported_algs[$header->alg])) {
|
||||
throw new UnexpectedValueException('Algorithm not supported');
|
||||
}
|
||||
if (!in_array($header->alg, $allowed_algs)) {
|
||||
if (!\in_array($header->alg, $allowed_algs)) {
|
||||
throw new UnexpectedValueException('Algorithm not allowed');
|
||||
}
|
||||
if (is_array($key) || $key instanceof \ArrayAccess) {
|
||||
if ($header->alg === 'ES256') {
|
||||
// OpenSSL expects an ASN.1 DER sequence for ES256 signatures
|
||||
$sig = self::signatureToDER($sig);
|
||||
}
|
||||
|
||||
if (\is_array($key) || $key instanceof \ArrayAccess) {
|
||||
if (isset($header->kid)) {
|
||||
if (!isset($key[$header->kid])) {
|
||||
throw new UnexpectedValueException('"kid" invalid, unable to lookup correct key');
|
||||
|
@ -112,11 +122,11 @@ class JWT
|
|||
throw new SignatureInvalidException('Signature verification failed');
|
||||
}
|
||||
|
||||
// Check if the nbf if it is defined. This is the time that the
|
||||
// Check the nbf if it is defined. This is the time that the
|
||||
// token can actually be used. If it's not yet that time, abort.
|
||||
if (isset($payload->nbf) && $payload->nbf > ($timestamp + static::$leeway)) {
|
||||
throw new BeforeValidException(
|
||||
'Cannot handle token prior to ' . date(DateTime::ISO8601, $payload->nbf)
|
||||
'Cannot handle token prior to ' . \date(DateTime::ISO8601, $payload->nbf)
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -125,7 +135,7 @@ class JWT
|
|||
// correctly used the nbf claim).
|
||||
if (isset($payload->iat) && $payload->iat > ($timestamp + static::$leeway)) {
|
||||
throw new BeforeValidException(
|
||||
'Cannot handle token prior to ' . date(DateTime::ISO8601, $payload->iat)
|
||||
'Cannot handle token prior to ' . \date(DateTime::ISO8601, $payload->iat)
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -144,7 +154,7 @@ class JWT
|
|||
* @param string $key The secret key.
|
||||
* If the algorithm used is asymmetric, this is the private key
|
||||
* @param string $alg The signing algorithm.
|
||||
* Supported algorithms are 'HS256', 'HS384', 'HS512' and 'RS256'
|
||||
* Supported algorithms are 'ES256', 'HS256', 'HS384', 'HS512', 'RS256', 'RS384', and 'RS512'
|
||||
* @param mixed $keyId
|
||||
* @param array $head An array with header elements to attach
|
||||
*
|
||||
|
@ -159,18 +169,18 @@ class JWT
|
|||
if ($keyId !== null) {
|
||||
$header['kid'] = $keyId;
|
||||
}
|
||||
if ( isset($head) && is_array($head) ) {
|
||||
$header = array_merge($head, $header);
|
||||
if (isset($head) && \is_array($head)) {
|
||||
$header = \array_merge($head, $header);
|
||||
}
|
||||
$segments = array();
|
||||
$segments[] = static::urlsafeB64Encode(static::jsonEncode($header));
|
||||
$segments[] = static::urlsafeB64Encode(static::jsonEncode($payload));
|
||||
$signing_input = implode('.', $segments);
|
||||
$signing_input = \implode('.', $segments);
|
||||
|
||||
$signature = static::sign($signing_input, $key, $alg);
|
||||
$segments[] = static::urlsafeB64Encode($signature);
|
||||
|
||||
return implode('.', $segments);
|
||||
return \implode('.', $segments);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -179,7 +189,7 @@ class JWT
|
|||
* @param string $msg The message to sign
|
||||
* @param string|resource $key The secret key
|
||||
* @param string $alg The signing algorithm.
|
||||
* Supported algorithms are 'HS256', 'HS384', 'HS512' and 'RS256'
|
||||
* Supported algorithms are 'ES256', 'HS256', 'HS384', 'HS512', 'RS256', 'RS384', and 'RS512'
|
||||
*
|
||||
* @return string An encrypted message
|
||||
*
|
||||
|
@ -191,15 +201,18 @@ class JWT
|
|||
throw new DomainException('Algorithm not supported');
|
||||
}
|
||||
list($function, $algorithm) = static::$supported_algs[$alg];
|
||||
switch($function) {
|
||||
switch ($function) {
|
||||
case 'hash_hmac':
|
||||
return hash_hmac($algorithm, $msg, $key, true);
|
||||
return \hash_hmac($algorithm, $msg, $key, true);
|
||||
case 'openssl':
|
||||
$signature = '';
|
||||
$success = openssl_sign($msg, $signature, $key, $algorithm);
|
||||
$success = \openssl_sign($msg, $signature, $key, $algorithm);
|
||||
if (!$success) {
|
||||
throw new DomainException("OpenSSL unable to sign data");
|
||||
} else {
|
||||
if ($alg === 'ES256') {
|
||||
$signature = self::signatureFromDER($signature, 256);
|
||||
}
|
||||
return $signature;
|
||||
}
|
||||
}
|
||||
|
@ -225,9 +238,9 @@ class JWT
|
|||
}
|
||||
|
||||
list($function, $algorithm) = static::$supported_algs[$alg];
|
||||
switch($function) {
|
||||
switch ($function) {
|
||||
case 'openssl':
|
||||
$success = openssl_verify($msg, $signature, $key, $algorithm);
|
||||
$success = \openssl_verify($msg, $signature, $key, $algorithm);
|
||||
if ($success === 1) {
|
||||
return true;
|
||||
} elseif ($success === 0) {
|
||||
|
@ -235,19 +248,19 @@ class JWT
|
|||
}
|
||||
// returns 1 on success, 0 on failure, -1 on error.
|
||||
throw new DomainException(
|
||||
'OpenSSL error: ' . openssl_error_string()
|
||||
'OpenSSL error: ' . \openssl_error_string()
|
||||
);
|
||||
case 'hash_hmac':
|
||||
default:
|
||||
$hash = hash_hmac($algorithm, $msg, $key, true);
|
||||
if (function_exists('hash_equals')) {
|
||||
return hash_equals($signature, $hash);
|
||||
$hash = \hash_hmac($algorithm, $msg, $key, true);
|
||||
if (\function_exists('hash_equals')) {
|
||||
return \hash_equals($signature, $hash);
|
||||
}
|
||||
$len = min(static::safeStrlen($signature), static::safeStrlen($hash));
|
||||
$len = \min(static::safeStrlen($signature), static::safeStrlen($hash));
|
||||
|
||||
$status = 0;
|
||||
for ($i = 0; $i < $len; $i++) {
|
||||
$status |= (ord($signature[$i]) ^ ord($hash[$i]));
|
||||
$status |= (\ord($signature[$i]) ^ \ord($hash[$i]));
|
||||
}
|
||||
$status |= (static::safeStrlen($signature) ^ static::safeStrlen($hash));
|
||||
|
||||
|
@ -266,23 +279,23 @@ class JWT
|
|||
*/
|
||||
public static function jsonDecode($input)
|
||||
{
|
||||
if (version_compare(PHP_VERSION, '5.4.0', '>=') && !(defined('JSON_C_VERSION') && PHP_INT_SIZE > 4)) {
|
||||
if (\version_compare(PHP_VERSION, '5.4.0', '>=') && !(\defined('JSON_C_VERSION') && PHP_INT_SIZE > 4)) {
|
||||
/** In PHP >=5.4.0, json_decode() accepts an options parameter, that allows you
|
||||
* to specify that large ints (like Steam Transaction IDs) should be treated as
|
||||
* strings, rather than the PHP default behaviour of converting them to floats.
|
||||
*/
|
||||
$obj = json_decode($input, false, 512, JSON_BIGINT_AS_STRING);
|
||||
$obj = \json_decode($input, false, 512, JSON_BIGINT_AS_STRING);
|
||||
} else {
|
||||
/** Not all servers will support that, however, so for older versions we must
|
||||
* manually detect large ints in the JSON string and quote them (thus converting
|
||||
*them to strings) before decoding, hence the preg_replace() call.
|
||||
*/
|
||||
$max_int_length = strlen((string) PHP_INT_MAX) - 1;
|
||||
$json_without_bigints = preg_replace('/:\s*(-?\d{'.$max_int_length.',})/', ': "$1"', $input);
|
||||
$obj = json_decode($json_without_bigints);
|
||||
$max_int_length = \strlen((string) PHP_INT_MAX) - 1;
|
||||
$json_without_bigints = \preg_replace('/:\s*(-?\d{'.$max_int_length.',})/', ': "$1"', $input);
|
||||
$obj = \json_decode($json_without_bigints);
|
||||
}
|
||||
|
||||
if (function_exists('json_last_error') && $errno = json_last_error()) {
|
||||
if ($errno = \json_last_error()) {
|
||||
static::handleJsonError($errno);
|
||||
} elseif ($obj === null && $input !== 'null') {
|
||||
throw new DomainException('Null result with non-null input');
|
||||
|
@ -301,8 +314,8 @@ class JWT
|
|||
*/
|
||||
public static function jsonEncode($input)
|
||||
{
|
||||
$json = json_encode($input);
|
||||
if (function_exists('json_last_error') && $errno = json_last_error()) {
|
||||
$json = \json_encode($input);
|
||||
if ($errno = \json_last_error()) {
|
||||
static::handleJsonError($errno);
|
||||
} elseif ($json === 'null' && $input !== null) {
|
||||
throw new DomainException('Null result with non-null input');
|
||||
|
@ -319,12 +332,12 @@ class JWT
|
|||
*/
|
||||
public static function urlsafeB64Decode($input)
|
||||
{
|
||||
$remainder = strlen($input) % 4;
|
||||
$remainder = \strlen($input) % 4;
|
||||
if ($remainder) {
|
||||
$padlen = 4 - $remainder;
|
||||
$input .= str_repeat('=', $padlen);
|
||||
$input .= \str_repeat('=', $padlen);
|
||||
}
|
||||
return base64_decode(strtr($input, '-_', '+/'));
|
||||
return \base64_decode(\strtr($input, '-_', '+/'));
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -336,7 +349,7 @@ class JWT
|
|||
*/
|
||||
public static function urlsafeB64Encode($input)
|
||||
{
|
||||
return str_replace('=', '', strtr(base64_encode($input), '+/', '-_'));
|
||||
return \str_replace('=', '', \strtr(\base64_encode($input), '+/', '-_'));
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -365,15 +378,135 @@ class JWT
|
|||
/**
|
||||
* Get the number of bytes in cryptographic strings.
|
||||
*
|
||||
* @param string
|
||||
* @param string $str
|
||||
*
|
||||
* @return int
|
||||
*/
|
||||
private static function safeStrlen($str)
|
||||
{
|
||||
if (function_exists('mb_strlen')) {
|
||||
return mb_strlen($str, '8bit');
|
||||
if (\function_exists('mb_strlen')) {
|
||||
return \mb_strlen($str, '8bit');
|
||||
}
|
||||
return strlen($str);
|
||||
return \strlen($str);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert an ECDSA signature to an ASN.1 DER sequence
|
||||
*
|
||||
* @param string $sig The ECDSA signature to convert
|
||||
* @return string The encoded DER object
|
||||
*/
|
||||
private static function signatureToDER($sig)
|
||||
{
|
||||
// Separate the signature into r-value and s-value
|
||||
list($r, $s) = \str_split($sig, (int) (\strlen($sig) / 2));
|
||||
|
||||
// Trim leading zeros
|
||||
$r = \ltrim($r, "\x00");
|
||||
$s = \ltrim($s, "\x00");
|
||||
|
||||
// Convert r-value and s-value from unsigned big-endian integers to
|
||||
// signed two's complement
|
||||
if (\ord($r[0]) > 0x7f) {
|
||||
$r = "\x00" . $r;
|
||||
}
|
||||
if (\ord($s[0]) > 0x7f) {
|
||||
$s = "\x00" . $s;
|
||||
}
|
||||
|
||||
return self::encodeDER(
|
||||
self::ASN1_SEQUENCE,
|
||||
self::encodeDER(self::ASN1_INTEGER, $r) .
|
||||
self::encodeDER(self::ASN1_INTEGER, $s)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Encodes a value into a DER object.
|
||||
*
|
||||
* @param int $type DER tag
|
||||
* @param string $value the value to encode
|
||||
* @return string the encoded object
|
||||
*/
|
||||
private static function encodeDER($type, $value)
|
||||
{
|
||||
$tag_header = 0;
|
||||
if ($type === self::ASN1_SEQUENCE) {
|
||||
$tag_header |= 0x20;
|
||||
}
|
||||
|
||||
// Type
|
||||
$der = \chr($tag_header | $type);
|
||||
|
||||
// Length
|
||||
$der .= \chr(\strlen($value));
|
||||
|
||||
return $der . $value;
|
||||
}
|
||||
|
||||
/**
|
||||
* Encodes signature from a DER object.
|
||||
*
|
||||
* @param string $der binary signature in DER format
|
||||
* @param int $keySize the number of bits in the key
|
||||
* @return string the signature
|
||||
*/
|
||||
private static function signatureFromDER($der, $keySize)
|
||||
{
|
||||
// OpenSSL returns the ECDSA signatures as a binary ASN.1 DER SEQUENCE
|
||||
list($offset, $_) = self::readDER($der);
|
||||
list($offset, $r) = self::readDER($der, $offset);
|
||||
list($offset, $s) = self::readDER($der, $offset);
|
||||
|
||||
// Convert r-value and s-value from signed two's compliment to unsigned
|
||||
// big-endian integers
|
||||
$r = \ltrim($r, "\x00");
|
||||
$s = \ltrim($s, "\x00");
|
||||
|
||||
// Pad out r and s so that they are $keySize bits long
|
||||
$r = \str_pad($r, $keySize / 8, "\x00", STR_PAD_LEFT);
|
||||
$s = \str_pad($s, $keySize / 8, "\x00", STR_PAD_LEFT);
|
||||
|
||||
return $r . $s;
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads binary DER-encoded data and decodes into a single object
|
||||
*
|
||||
* @param string $der the binary data in DER format
|
||||
* @param int $offset the offset of the data stream containing the object
|
||||
* to decode
|
||||
* @return array [$offset, $data] the new offset and the decoded object
|
||||
*/
|
||||
private static function readDER($der, $offset = 0)
|
||||
{
|
||||
$pos = $offset;
|
||||
$size = \strlen($der);
|
||||
$constructed = (\ord($der[$pos]) >> 5) & 0x01;
|
||||
$type = \ord($der[$pos++]) & 0x1f;
|
||||
|
||||
// Length
|
||||
$len = \ord($der[$pos++]);
|
||||
if ($len & 0x80) {
|
||||
$n = $len & 0x1f;
|
||||
$len = 0;
|
||||
while ($n-- && $pos < $size) {
|
||||
$len = ($len << 8) | \ord($der[$pos++]);
|
||||
}
|
||||
}
|
||||
|
||||
// Value
|
||||
if ($type == self::ASN1_BIT_STRING) {
|
||||
$pos++; // Skip the first contents octet (padding indicator)
|
||||
$data = \substr($der, $pos, $len - 1);
|
||||
$pos += $len - 1;
|
||||
} elseif (!$constructed) {
|
||||
$data = \substr($der, $pos, $len);
|
||||
$pos += $len;
|
||||
} else {
|
||||
$data = null;
|
||||
}
|
||||
|
||||
return array($pos, $data);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -3,5 +3,4 @@ namespace Firebase\JWT;
|
|||
|
||||
class SignatureInvalidException extends \UnexpectedValueException
|
||||
{
|
||||
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue