取消了命令行支持、强制入口文件绑定模块、支持注解路由、注解中间件、注解前后置操作、支持参数自动注入

This commit is contained in:
TOP糯米 2020-05-13 17:24:21 +08:00
parent e32d3c523c
commit 8da633a984
35 changed files with 1022 additions and 935 deletions

View File

@ -2,7 +2,10 @@
namespace top; namespace top;
use top\library\App; use top\library\Application;
// 定义简写文件分割符号常量
!defined('DS') && define('DS', DIRECTORY_SEPARATOR);
/** /**
* 框架入口 * 框架入口
@ -18,31 +21,32 @@ class Framework
/** /**
* 框架入口 * 框架入口
* @param string $callable * @param string $callable
* @param array $namespaceMap
*/ */
public static function startApp($callable = '') public static function startApp($callable = '', $namespaceMap = [])
{ {
if (is_callable($callable)) { (is_callable($callable)) && $callable(self::class);
$callable(self::class);
}
// 指定时区 // 指定时区
date_default_timezone_set('PRC'); date_default_timezone_set('PRC');
self::debug();
// 强制在入口文件指定应用目录 // 强制在入口文件指定应用目录
if (defined('APP_PATH')) { if (defined('APP_PATH')) {
self::debug();
// self::appPath(); // self::appPath();
self::bindModule();
self::appNameSpace(); self::appNameSpace();
self::resourcePath(); self::resourcePath();
self::frameworkPath(); self::frameworkPath();
self::sessionPath(); self::sessionPath();
require 'library/App.php'; // 配置文件目录
App::run(); !defined('CONFIG_DIR') && define('CONFIG_DIR', APP_PATH . BIND_MODULE . DS . 'config' . DS);
} else {
echo '请使用Framework::appPath()指定应用目录'; require 'library/Application.php';
} Application::run($namespaceMap);
} else echo '请使用Framework::appPath()指定应用目录';
} }
/** /**
@ -52,10 +56,8 @@ class Framework
public static function appPath($path = '') public static function appPath($path = '')
{ {
if (!defined('APP_PATH')) { if (!defined('APP_PATH')) {
if (!$path) { (!$path) && $path = '.' . DS . 'application' . DS;
$path = './application/'; define('APP_PATH', str_replace('/', DS, $path));
}
define('APP_PATH', $path);
} }
} }
@ -65,8 +67,18 @@ class Framework
*/ */
public static function debug($status = false) public static function debug($status = false)
{ {
if (!defined('DEBUG')) { (!defined('DEBUG')) && define('DEBUG', $status);
define('DEBUG', $status); }
/**
* 绑定模块
* @param string $module
*/
public static function bindModule($module = '')
{
if (!defined('BIND_MODULE')) {
(!$module) && $module = 'home';
define('BIND_MODULE', $module);
} }
} }
@ -77,19 +89,19 @@ class Framework
public static function frameworkPath($path = '') public static function frameworkPath($path = '')
{ {
if (!defined('FRAMEWORK_PATH')) { if (!defined('FRAMEWORK_PATH')) {
if (!$path) { (!$path) && $path = __DIR__ . DS;
$path = __DIR__ . '/'; define('FRAMEWORK_PATH', str_replace('/', DS, $path));
}
define('FRAMEWORK_PATH', $path);
} }
} }
/**
* 应用命名空间
* @param string $namespace
*/
public static function appNameSpace($namespace = '') public static function appNameSpace($namespace = '')
{ {
if (!defined('APP_NS')) { if (!defined('APP_NS')) {
if (!$namespace) { (!$namespace) && $namespace = 'app';
$namespace = 'app';
}
define('APP_NS', $namespace); define('APP_NS', $namespace);
} }
} }
@ -105,9 +117,9 @@ class Framework
$scriptName = $_SERVER['SCRIPT_NAME']; $scriptName = $_SERVER['SCRIPT_NAME'];
$pos = strrpos($scriptName, '/'); $pos = strrpos($scriptName, '/');
$root = substr($scriptName, 0, $pos + 1); $root = substr($scriptName, 0, $pos + 1);
$path = $root . 'resource/'; $path = $root . 'resource' . DS;
} }
define('RESOURCE', $path); define('RESOURCE', str_replace('/', DS, $path));
} }
} }
@ -118,13 +130,9 @@ class Framework
public static function sessionPath($path = '') public static function sessionPath($path = '')
{ {
if (!defined('SESSION_PATH')) { if (!defined('SESSION_PATH')) {
if (!$path) { (!$path) && $path = '.' . DS . 'runtime' . DS . 'session' . DS;
$path = './runtime/session/'; (!is_dir($path)) && mkdir($path, 0755, true);
} define('SESSION_PATH', str_replace('/', DS, $path));
if (!is_dir($path)) {
mkdir($path, 0755, true);
}
define('SESSION_PATH', $path);
} }
} }

View File

@ -2,11 +2,15 @@
// 默认配置 // 默认配置
return [ return [
'default_controller' => 'Index',
'default_method' => 'index',
'compel_route' => false,
'complete_parameter' => true,
'register' => [ 'register' => [
'Top' => \top\library\template\driver\Top::class, 'Top' => \top\library\template\driver\Top::class,
], ],
'middleware' => [ 'middleware' => [
\top\middleware\Init::class, \top\middleware\Action::class,
\top\middleware\View::class, \top\middleware\View::class,
], ],
'session' => [ 'session' => [

View File

@ -14,10 +14,10 @@ class Create
private $name = ''; private $name = '';
/** /**
* 命名空间 * 应用目录
* @var string * @var string
*/ */
private $namespace = ''; private $path = '';
/** /**
* 入口文件名 * 入口文件名
@ -43,15 +43,15 @@ class Create
*/ */
private $projectPath; private $projectPath;
public function __construct($start, $namespace, $name) public function __construct($start, $path, $name)
{ {
$this->name = $name; $this->name = $name;
$this->dir = __DIR__ . '/'; $this->dir = __DIR__ . DIRECTORY_SEPARATOR;
$this->namespace = $namespace; $this->path = $path;
$this->base = $this->dir . '../../'; $this->base = $this->dir . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR;
if ($start) if ($start)
$this->start = $this->base . $start . '.php'; $this->start = $this->base . $start;
$this->projectPath = $this->base . $this->namespace . '/' . $this->name . '/'; $this->projectPath = $this->base . $this->path . DIRECTORY_SEPARATOR . $this->name . DIRECTORY_SEPARATOR;
$this->create(); $this->create();
} }
@ -63,10 +63,8 @@ class Create
public function replaceContent($content) public function replaceContent($content)
{ {
return str_replace([ return str_replace([
'{namespace}',
'{name}' '{name}'
], [ ], [
$this->namespace,
$this->name $this->name
], $content); ], $content);
} }
@ -78,7 +76,7 @@ class Create
public function createStartFile() public function createStartFile()
{ {
if ($this->start && !is_file($this->start)) { if ($this->start && !is_file($this->start)) {
$content = file_get_contents($this->dir . 'tpl/index.tpl'); $content = file_get_contents($this->dir . 'tpl' . DIRECTORY_SEPARATOR . 'index.tpl');
$content = $this->replaceContent($content); $content = $this->replaceContent($content);
if (file_put_contents($this->start, $content)) { if (file_put_contents($this->start, $content)) {
return true; return true;
@ -93,15 +91,14 @@ class Create
*/ */
public function createConfig() public function createConfig()
{ {
$configPath = $this->projectPath . 'config/'; $configPath = $this->projectPath . 'config' . DIRECTORY_SEPARATOR;
$configFile = $configPath . 'config.php'; $configFile = $configPath . 'config.php';
if (!is_dir($configPath)) { if (!is_dir($configPath)) {
mkdir($configPath, 0755, true); mkdir($configPath, 0755, true);
} }
if (!is_file($configFile)) { if (!is_file($configFile)) {
$content = file_get_contents($this->dir . 'tpl/config/config.tpl'); $content = file_get_contents($this->dir . 'tpl' . DIRECTORY_SEPARATOR . 'config' . DIRECTORY_SEPARATOR . 'config.tpl');
$content = $this->replaceContent($content); $content = $this->replaceContent($content);
$realConfigFile = $this->base . '/' . $this->namespace . '/' . $this->name . '/config/config.php';
if (!file_put_contents($configPath . 'config.php', $content)) { if (!file_put_contents($configPath . 'config.php', $content)) {
exit('error -2'); exit('error -2');
} }
@ -111,7 +108,7 @@ class Create
/** /**
* 创建MVC目录及文件 * 创建MVC目录及文件
*/ */
public function createMVC() public function createControllerAndView()
{ {
$dirArray = [ $dirArray = [
'controller', 'controller',
@ -119,33 +116,25 @@ class Create
'view' 'view'
]; ];
for ($i = 0; $i < count($dirArray); $i++) { for ($i = 0; $i < count($dirArray); $i++) {
if (!is_dir($this->projectPath . $dirArray[$i] . '/')) { if (!is_dir($this->projectPath . $dirArray[$i] . DIRECTORY_SEPARATOR)) {
mkdir($this->projectPath . $dirArray[$i] . '/', 0755, true); mkdir($this->projectPath . $dirArray[$i] . DIRECTORY_SEPARATOR, 0755, true);
} }
} }
$controllerFile = $this->projectPath . 'controller/index.php'; $controllerFile = $this->projectPath . 'controller' . DIRECTORY_SEPARATOR . 'index.php';
if (!is_file($controllerFile)) { if (!is_file($controllerFile)) {
$content = file_get_contents($this->dir . 'tpl/controller/index.tpl'); $content = file_get_contents($this->dir . 'tpl' . DIRECTORY_SEPARATOR . 'controller' . DIRECTORY_SEPARATOR . 'index.tpl');
$content = $this->replaceContent($content); $content = $this->replaceContent($content);
if (!file_put_contents($this->projectPath . 'controller/Index.php', $content)) { if (!file_put_contents($this->projectPath . 'controller' . DIRECTORY_SEPARATOR . 'Index.php', $content)) {
exit('error -4'); exit('error -4');
} }
} }
$modelFile = $this->projectPath . 'model/demo.php'; $viewFile = $this->projectPath . 'view' . DIRECTORY_SEPARATOR . 'index' . DIRECTORY_SEPARATOR . 'index.html';
if (!is_file($modelFile)) {
$content = file_get_contents($this->dir . 'tpl/model/demo.tpl');
$content = $this->replaceContent($content);
if (!file_put_contents($this->projectPath . 'model/Demo.php', $content)) {
exit('error -5');
}
}
$viewFile = $this->projectPath . 'view/index/index.html';
if (!is_file($viewFile)) { if (!is_file($viewFile)) {
$content = file_get_contents($this->dir . 'tpl/view/index.tpl'); $content = file_get_contents($this->dir . 'tpl' . DIRECTORY_SEPARATOR . 'view' . DIRECTORY_SEPARATOR . 'index.tpl');
if (!is_dir($this->projectPath . 'view/Index/')) { if (!is_dir($this->projectPath . 'view' . DIRECTORY_SEPARATOR . 'Index' . DIRECTORY_SEPARATOR)) {
mkdir($this->projectPath . 'view/Index/', 0755, true); mkdir($this->projectPath . 'view' . DIRECTORY_SEPARATOR . 'Index' . DIRECTORY_SEPARATOR, 0755, true);
} }
if (!file_put_contents($this->projectPath . 'view/Index/index.html', $content)) { if (!file_put_contents($this->projectPath . 'view' . DIRECTORY_SEPARATOR . 'Index' . DIRECTORY_SEPARATOR . 'index.html', $content)) {
exit('error -6'); exit('error -6');
} }
} }
@ -164,19 +153,6 @@ class Create
} }
} }
/**
* 创建路由文件
*/
public function createRoute()
{
$file = $this->projectPath . '../route.php';
if (!is_file($file)) {
if (!file_put_contents($file, file_get_contents($this->dir . 'tpl/route.tpl'))) {
exit('-8');
}
}
}
/** /**
* 执行创建操作 * 执行创建操作
*/ */
@ -184,14 +160,13 @@ class Create
{ {
$this->createStartFile(); $this->createStartFile();
$this->createConfig(); $this->createConfig();
$this->createMVC(); $this->createControllerAndView();
$this->createFunctions(); $this->createFunctions();
$this->createRoute();
} }
} }
// 准备创建项目 // 准备创建项目
$namespace = (isset($argv[1]) && $argv[1]) ? $argv[1] : exit('please type namespace~'); $path = (isset($argv[1]) && $argv[1]) ? $argv[1] : exit('please type path~');
$projectName = (isset($argv[2]) && $argv[2]) ? $argv[2] : exit('please type project name~'); $projectName = (isset($argv[2]) && $argv[2]) ? $argv[2] : exit('please type project name~');
$startFile = (isset($argv[3]) && $argv[3]) ? $argv[3] : false; $startFile = (isset($argv[3]) && $argv[3]) ? $argv[3] : false;
new Create($startFile, $namespace, $projectName); new Create($startFile, $path, $projectName);

View File

@ -2,16 +2,29 @@
namespace app\{name}\controller; namespace app\{name}\controller;
use app\{name}\model\Demo; use top\library\Controller;
use top\library\http\Request;
class Index class Index extends Controller
{ {
public function index()
/**
* 首页
* @route /
*
* @param Request $request
* @return array
*/
public function index(Request $request)
{ {
$model = model(Demo::class); $uri = $request->uri();
(!$uri) && $this->redirect('index');
return [ return [
'hello' => $model->get() 'uri' => $uri,
'controller' => $request->controllerFullName(),
'method' => $request->method(),
]; ];
} }
} }

View File

@ -30,13 +30,6 @@ require '../framework/Framework.php';
// Framework::resourcePath('/resource/'); // Framework::resourcePath('/resource/');
// 可使用常量RESOURCE取得该值 // 可使用常量RESOURCE取得该值
// 当前入口文件默认模块缺省值home
// Framework::defaultModule('home');
// 路由模式缺省值1pathinfo和兼容模式
// Framework::runType(1);
Framework::appPath('../application/'); Framework::appPath('../application/');
Framework::bindModule('{name}');
// 执行程序
Framework::startApp(); Framework::startApp();

View File

@ -1,11 +0,0 @@
<?php
namespace app\{name}\model;
class Demo
{
public function get()
{
return '模块{name}正在运行...';
}
}

View File

@ -1,17 +0,0 @@
<?php
// 自定义路由示例
return [
'login' => [
null,
'home/example/login'
],
'example-detail' => [
'[id]',
'home/example/detail'
],
'example' => [
'[:type]',
'home/example/index'
],
];

View File

@ -9,23 +9,16 @@
</head> </head>
<body> <body>
<style> <style>
h3 { * {
color: #666666; color: #666666;
font-weight: 300;
}
.version {
width: 100%;
height: 22px;
color: #b9b9b9;
font-size: 12px;
text-align: center;
line-height: 22px;
margin: 0 auto;
padding-bottom: 20px;
} }
</style> </style>
<h3>{$hello}</h3> <h2>Welcome</h2>
<p>URI: {$uri}</p>
<p>Controller: {$controller}</p>
<p>Method: {$method}</p>
<h5>{:top\library\Loader::getFileNumber()} files loaded.</h5>
<p class="version">TOP-Framework</p>
</body> </body>
</html> </html>

View File

@ -0,0 +1,86 @@
<?php
namespace top\library;
use top\traits\Instance;
/**
* 获取类注解
* Class Annotation
* @package top\library
*/
class Annotation
{
use Instance;
/**
* 获取到的类注解
* @var array
*/
private static $annotations = [];
/**
* 获取方法注解
* @param $className
* @param $methodName
* @param null $annotation
* @return array
*/
public static function getMethodAnnotation($className, $methodName, $annotation = null)
{
return self::getAnnotation($className, $methodName, $annotation);
}
/**
* 获取类注解
* @param $className
* @param null $annotation
* @return array
*/
public static function getClassAnnotation($className, $annotation = null)
{
return self::getAnnotation($className, null, $annotation);
}
/**
* 获取注解
* @param $className
* @param null $methodName
* @param null $annotation
* @return mixed
*/
private static function getAnnotation($className, $methodName = null, $annotation = null)
{
$ident = md5($className . $methodName);
if (!isset(self::$annotations[$ident])) {
// echo '获取' . $className . '::' . $methodName . PHP_EOL;
$self = self::instance();
$reflectionClass = Application::getReflectionClass($className);
if ($methodName) {
$doc = $reflectionClass->getMethod($methodName)->getDocComment();
} else {
$doc = $reflectionClass->getDocComment();
}
self::$annotations[$ident] = $self->parseAnnotation($doc);
}
return ($annotation) ? self::$annotations[$ident][$annotation] : self::$annotations[$ident];
}
/**
* 解析出注解
* @param $doc
* @return array
*/
private function parseAnnotation($doc)
{
$result = [];
preg_match_all('/@([a-zA-Z]+)\s(.*)/', $doc, $matches);
if (!empty($matches)) {
for ($i = 0; $i < count($matches[0]); $i++)
$result[$matches[1][$i]] = trim($matches[2][$i]);
}
$matches = null;
return $result;
}
}

View File

@ -1,117 +0,0 @@
<?php
namespace top\library;
use top\library\http\Request;
use top\library\error\BaseError;
use top\library\exception\BaseException;
use top\library\http\Response;
use top\library\route\driver\Command;
use top\library\route\driver\Compatible;
class App
{
/**
* 开始执行程序
* @param int $type
*/
public static function run()
{
// 注册框架自动加载
require 'Loader.php';
$loader = new Loader();
$loader->set('top', FRAMEWORK_PATH);
$loader->set(APP_NS, APP_PATH);
$loader->register();
// composer自动加载
$composerLoadFile = FRAMEWORK_PATH . 'vendor/autoload.php';
if (file_exists($composerLoadFile)) {
require $composerLoadFile;
}
// 使用whoops美化异常输出
// $whoops = new \Whoops\Run;
// $whoops->pushHandler(new \Whoops\Handler\PrettyPageHandler);
// $whoops->register();
if (PHP_VERSION > 5.6) {
set_error_handler([new BaseError(), 'handler']);
}
set_exception_handler([new BaseException(), 'handler']);
$router = self::initRoute();
$request = Request::instance()->setRoute($router);
$response = Response::instance();
// 加载必要文件
self::loadFiles();
// 处理请求并得到数据
$responseData = $response->header([
'X-Powered-By: TOP-Framework'
])->dispatch($request->execute());
// 输出内容
echo $responseData->content;
}
/**
* 初始化路由
* @return Router
*/
private static function initRoute()
{
$driver = null;
if (Request::instance()->isCLI()) {
// 命令行运行程序
$driver = new Command();
} else {
$driver = new Compatible();
}
return (new Router($driver))->handler();
}
/**
* 加载必要文件
*/
private static function loadFiles()
{
// 加载系统函数库
require FRAMEWORK_PATH . 'library/functions/functions.php';
// 加载用户函数库
$funcFile = APP_PATH . request()->module() . '/functions.php';
if (file_exists($funcFile)) {
require $funcFile;
}
$configInstance = Config::instance();
$sessionConfig = $configInstance->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');
if (!empty($initRegister)) {
foreach ($initRegister as $key => $value) {
Register::set($key, function () use ($value) {
return $value::instance();
});
}
}
}
}

View File

@ -0,0 +1,178 @@
<?php
namespace top\library;
use top\library\error\BaseError;
use top\library\exception\BaseException;
use top\library\http\Request;
use top\library\http\Response;
class Application
{
private static $reflectionClass = [];
private static $reflectionMethod = [];
/**
* 开始执行程序
* @param array $namespaceMap
*/
public static function run($namespaceMap = [])
{
// 注册框架自动加载
require 'Loader.php';
$loader = new Loader();
$loader->set('top', FRAMEWORK_PATH);
$loader->set(APP_NS, APP_PATH);
foreach ($namespaceMap as $prefix => $path) {
$loader->set($prefix, $path);
}
$loader->register();
// composer自动加载
$composerLoadFile = FRAMEWORK_PATH . 'vendor/autoload.php';
(is_file($composerLoadFile)) && require $composerLoadFile;
// 使用whoops美化异常输出
// $whoops = new \Whoops\Run;
// $whoops->pushHandler(new \Whoops\Handler\PrettyPageHandler);
// $whoops->register();
(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');
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');
if (!empty($initRegister)) {
foreach ($initRegister as $key => $value) {
Register::set($key, function () use ($value) {
return $value::instance();
});
}
}
}
/**
* 获取一个类反射
* @param $className
* @return mixed
*/
public static function getReflectionClass($className)
{
if (!isset(self::$reflectionClass[$className])) {
try {
self::$reflectionClass[$className] = new \ReflectionClass($className);
} catch (\ReflectionException $exception) {
}
}
return self::$reflectionClass[$className];
}
/**
* 获取一个类方法反射
* @param $className
* @param $methodName
* @return mixed
*/
public static function getReflectionMethod($className, $methodName)
{
$ident = md5($className . $methodName);
if (!isset(self::$reflectionMethod[$ident])) {
try {
self::$reflectionMethod[$ident] = new \ReflectionMethod($className, $methodName);
} catch (\ReflectionException $exception) {
}
}
return self::$reflectionMethod[$ident];
}
/**
* 获取一个类实例
* @param $className
* @return mixed
* @throws \ReflectionException
*/
public static function getInstance($className)
{
$classRef = self::getReflectionClass($className);
$isInstantiable = $classRef->isInstantiable();
if (!$isInstantiable) { // 不可被实例化
if ($classRef->hasMethod('instance')) {
$instance = $classRef->getMethod('instance');
} else throw new \Exception('不可实例化的类:' . $className);
} else {
$instance = $classRef->getConstructor();
}
if (!is_null($instance)) {
$instanceParams = $instance->getParameters();
if (empty($instanceParams)) { // 构造函数没有参数直接返回当前类实例
if (!$isInstantiable) return $className::instance();
return new $className;
}
} else { // 没有构造方法直接返回实例
if (!$isInstantiable) return $className::instance();
return new $className;
}
// 构造函数存在参数则去递归实例化类
$actualParams = [];
foreach ($instanceParams as $param) {
$actualClass = $param->getClass();
if (!is_null($actualClass)) { // 参数是一个类
$actualParams[$param->name] = self::getInstance($actualClass->name);
}
}
if ($isInstantiable) {
return $classRef->newInstanceArgs($actualParams);
} else {
$reflectionMethod = new \ReflectionMethod($className, 'instance');
return $reflectionMethod->invokeArgs(null, $actualParams);
}
}
}

View File

@ -2,7 +2,6 @@
namespace top\library; namespace top\library;
use top\library\http\Request;
use top\traits\Instance; use top\traits\Instance;
/** /**
@ -20,6 +19,9 @@ class Config
// 保存配置的变量 // 保存配置的变量
private $config = []; private $config = [];
/**
* Config constructor.
*/
private function __construct() private function __construct()
{ {
// 加载默认配置文件 // 加载默认配置文件
@ -52,16 +54,15 @@ class Config
public function get($name = '') public function get($name = '')
{ {
// 加载用户配置文件 // 加载用户配置文件
$module = Request::instance()->module(); $file = CONFIG_DIR . 'config.php';
$file = APP_PATH . $module . '/config/config.php';
if (!isset(self::$files[$file])) { if (!isset(self::$files[$file])) {
if (file_exists($file)) { if (file_exists($file)) {
$config = require $file; $config = require $file;
if (is_array($config) && !empty($config)) { if (is_array($config) && !empty($config)) {
// 合并配置项 // 合并配置项
foreach ($config as $key => $value) { foreach ($config as $key => $value) {
if (array_key_exists($key, $this->config)) { if (array_key_exists($key, $this->config) && is_array($value)) {
$this->config[$key] = array_merge($this->config[$key], $config[$key]); $this->config[$key] = array_merge($this->config[$key], $value);
} else { } else {
$this->config[$key] = $value; $this->config[$key] = $value;
} }
@ -87,8 +88,9 @@ class Config
* 从配置中删除某项 * 从配置中删除某项
* @param string $name * @param string $name
*/ */
public function _unset($name) public function rm($name)
{ {
$this->config[$name] = null;
unset($this->config[$name]); unset($this->config[$name]);
} }
} }

View File

@ -2,6 +2,7 @@
namespace top\library; namespace top\library;
use top\library\http\Request;
use top\traits\Json; use top\traits\Json;
/** /**
@ -13,8 +14,47 @@ abstract class Controller
use Json; use Json;
public function __construct() /**
* 请求类实例
* @var Request
*/
protected $request = null;
/**
* URI
* @var string
*/
protected $uri = null;
/**
* 控制器
* @var string
*/
protected $controller = null;
/**
* 控制器全限定名
* @var string
*/
protected $controllerFullName = null;
/**
* 方法
* @var string
*/
protected $method = null;
/**
* 控制器基类构造方法
* @param Request $request
*/
public function __construct(Request $request)
{ {
$this->request = $request;
$this->uri = $request->uri();
$this->controller = $request->controller();
$this->controllerFullName = $request->controllerFullName();
$this->method = $request->method();
} }
/** /**

View File

@ -7,6 +7,8 @@ class Loader
private $prefixes = []; private $prefixes = [];
private static $number = 0;
/** /**
* 注册自动加载 * 注册自动加载
*/ */
@ -74,12 +76,22 @@ class Loader
/*echo '<br>'; /*echo '<br>';
echo $file . '<br>';*/ echo $file . '<br>';*/
// 如果文件存在则加载文件 // 如果文件存在则加载文件
if (file_exists($file)) { if (is_file($file)) {
require $file; require $file;
self::$number++;
return true; return true;
} }
} }
} }
return false; return false;
} }
/**
* 内置自动加载器加载的文件数
* @return int
*/
public static function getFileNumber()
{
return self::$number;
}
} }

View File

@ -2,6 +2,7 @@
namespace top\library; namespace top\library;
use Exception;
use top\library\exception\DatabaseException; use top\library\exception\DatabaseException;
/** /**
@ -98,12 +99,14 @@ class Model
return Database::table($this->table, $this->pk, $this->prefix); return Database::table($this->table, $this->pk, $this->prefix);
} }
// 可以静态调用的方法--开始
/** /**
* 影响的表仅多表delete * 影响的表仅多表delete
* @param $effect * @param $effect
* @return $this * @return $this
*/ */
public function effect($effect) private function effect($effect)
{ {
$this->getDb()->effect($effect); $this->getDb()->effect($effect);
return $this; return $this;
@ -114,7 +117,7 @@ class Model
* @param $field * @param $field
* @return $this * @return $this
*/ */
public function distinct($field) private function distinct($field)
{ {
$this->getDb()->distinct($field); $this->getDb()->distinct($field);
return $this; return $this;
@ -125,7 +128,7 @@ class Model
* @param $field * @param $field
* @return $this * @return $this
*/ */
public function field($field) private function field($field)
{ {
$this->getDb()->field($field); $this->getDb()->field($field);
return $this; return $this;
@ -135,7 +138,7 @@ class Model
* 查询条件 * 查询条件
* @return $this * @return $this
*/ */
public function where() private function where()
{ {
call_user_func_array([ call_user_func_array([
$this->getDb(), $this->getDb(),
@ -148,7 +151,7 @@ class Model
* 排序 * 排序
* @return $this * @return $this
*/ */
public function order() private function order()
{ {
call_user_func_array([ call_user_func_array([
$this->getDb(), $this->getDb(),
@ -161,7 +164,7 @@ class Model
* 限制 * 限制
* @return $this * @return $this
*/ */
public function limit() private function limit()
{ {
call_user_func_array([ call_user_func_array([
$this->getDb(), $this->getDb(),
@ -177,7 +180,7 @@ class Model
* @param $name * @param $name
* @return $this * @return $this
*/ */
public function join($type, $table, $name) private function join($type, $table, $name)
{ {
$this->getDb()->join($type, $table, $name); $this->getDb()->join($type, $table, $name);
return $this; return $this;
@ -188,12 +191,14 @@ class Model
* @param $on * @param $on
* @return $this * @return $this
*/ */
public function on($on) private function on($on)
{ {
$this->getDb()->on($on); $this->getDb()->on($on);
return $this; return $this;
} }
// 可静态调用的方法--结束
/** /**
* 插入记录 * 插入记录
* @param array $data * @param array $data
@ -630,4 +635,37 @@ class Model
return $data; return $data;
} }
/**
* 非静态调用连贯操作
* @param string $name
* @param array $arguments
* @return mixed
*/
public function __call($name, $arguments)
{
if (method_exists($this, $name)) {
return $this->{$name}($arguments);
} else throw new Exception('不存在的方法:' . $name);
}
/**
* 静态调用连贯操作
* @param string $name
* @param array $arguments
* @return mixed
*/
public static function __callStatic($name, $arguments)
{
$methodName = null;
if (method_exists(static::class, $name)) {
$methodName = $name;
} else {
$methodMap = ['all' => 'select', 'get' => 'find'];
if (isset($methodMap[$name])) {
$methodName = $methodMap[$name];
} else throw new Exception('不存在的方法:' . $name);
}
return call_user_func_array([new static, $methodName], $arguments);
}
} }

View File

@ -3,8 +3,9 @@
namespace top\library; namespace top\library;
use top\library\exception\RouteException; use top\library\exception\RouteException;
use top\library\route\ifs\RouteIfs; use top\library\http\Request;
use top\middleware\ifs\MiddlewareIfs; use top\middleware\ifs\MiddlewareIfs;
use top\traits\Instance;
/** /**
* 路由类 * 路由类
@ -12,32 +13,58 @@ use top\middleware\ifs\MiddlewareIfs;
*/ */
class Router class Router
{ {
/**
* 路由实现 use Instance;
* @var RouteIfs
*/
private $driver;
/** /**
* 自定义路由标识 * 路由配置
* @var null
*/
private $ident = null;
/**
* 自定义路由规则
* @var array * @var array
*/ */
private $rule = []; private $config = [];
/** /**
* 实例化时注入具体路由实现和默认位置 * 请求类
* Router constructor. * @var null
* @param RouteIfs $driver
*/ */
public function __construct(RouteIfs $driver) private $request = null;
/**
* 类全限定名
* @var null
*/
private $controllerFullName = null;
/**
* 类名
* @var null
*/
private $controller = null;
/**
* 方法
* @var null
*/
private $method = null;
/**
* 参数
* @var array
*/
private $params = [];
/**
* 当前加载的路由
* @var null
*/
private $loadRuleParameters = null;
/**
* Router constructor.
* @param Request $request
*/
private function __construct(Request $request)
{ {
$this->driver = $driver; $this->request = $request;
} }
/** /**
@ -46,7 +73,7 @@ class Router
*/ */
public function controllerFullName() public function controllerFullName()
{ {
return $this->driver->controllerFullName(); return $this->controllerFullName;
} }
/** /**
@ -55,7 +82,7 @@ class Router
*/ */
public function controller() public function controller()
{ {
return $this->driver->controller(); return $this->controller;
} }
/** /**
@ -64,7 +91,7 @@ class Router
*/ */
public function module() public function module()
{ {
return $this->driver->module(); return BIND_MODULE;
} }
/** /**
@ -73,7 +100,7 @@ class Router
*/ */
public function method() public function method()
{ {
return $this->driver->method(); return $this->method;
} }
/** /**
@ -82,117 +109,327 @@ class Router
*/ */
public function params() public function params()
{ {
return $this->driver->params(); return $this->params;
} }
/** /**
* 获取当前的URI * 查找最适合的路由匹配
* @param $rules
* @param $uri
* @return mixed
*/
private function findRule($rules, $uri)
{
$result = [];
if ($uri == '/' && isset($rules['/'])) {
$result[] = '/';
} else {
// 如果不是首页则unset掉首页的规则避免参与计算导致出错
unset($rules['/']);
$keys = array_keys($rules);
foreach ($keys as $key) {
$pos = strpos($uri, $key);
if ($pos !== false) {
$endPos = $pos + strlen($key);
$result[$endPos] = $key;
}
}
}
if (!empty($result)) {
$max = max(array_keys($result));
$rest = str_replace($result[$max], '', $uri);
if (($result[$max] == '/' && $uri != '/') || ($rest != '' && substr($rest, 0, 1) != '/')) {
return false;
} else {
$this->loadRuleParameters = $rules[$result[$max]];
return [
'rule' => $result[$max],
'parameters' => $this->loadRuleParameters,
];
}
}
return false;
}
/**
* 解析路由规则
* @param $requestMethod
* @param $uri
* @return array|bool|mixed
*/
private function parseRouteRule($requestMethod, $uri)
{
// 获取所有路由配置(可能从缓存文件读取)
$routeConfig = $this->getRouteConfig();
$rule = [];
if (isset($routeConfig[$requestMethod])) { // 第一次去当前请求方法中查找
$rule = $this->findRule($routeConfig[$requestMethod], $uri);
}
if (empty($rule) && isset($routeConfig['any'])) { // 全部中查找
$rule = $this->findRule($routeConfig['any'], $uri);
}
return (!empty($rule)) ? $rule : false;
}
/**
* 普通路由处理
* @param $uri
* @return array * @return array
*/ */
public function uri() private function parseRoute($uri)
{ {
$uri = null; // 普通处理
if (isset($_SERVER['PATH_INFO'])) { $uriArray = explode('/', trim($uri, '/'));
$uri = $_SERVER['PATH_INFO']; $uriArray[0] = (isset($uriArray[0]) && $uriArray[0]) ? $uriArray[0] : config('default_controller');
} elseif (isset($_GET['s']) && $_GET['s']) { $uriArray[1] = (isset($uriArray[1]) && $uriArray[1]) ? $uriArray[1] : config('default_method');
$uri = $_GET['s'];
} $controller = ucfirst($uriArray[0]);
$uri = str_replace('.html', '', trim($uri, '/')); $rule['class'] = APP_NS . '\\' . BIND_MODULE . '\\controller\\' . $controller;
return $uri; $rule['method'] = $uriArray[1];
return [
'rule' => $uriArray[0] . '/' . $uriArray[1],
'parameters' => $rule,
];
} }
/** /**
* 处理用户自定义路由规则 * 解析请求参数
* @param $class
* @param $method
* @param $prefix
* @param $uri * @param $uri
* @return string * @return array
* @throws RouteException
*/ */
private function customRule($uri) private function parseParameters($class, $method, $prefix, $uri)
{ {
// 准备检查自定义路由 $paramsString = ltrim(substr_replace($uri, '', 0, strlen($prefix)), '/');
$file = APP_PATH . 'route.php'; $paramsArray = explode('/', $paramsString);
if (is_file($file)) $this->rule = require $file;
$uriArray = explode('/', $uri); $params = [];
$ident = $this->ident = $uriArray[0]; if (config('complete_parameter')) { // 如果开启了完全参数名
// 如果标识存在则准备替换URI for ($i = 0; $i < count($paramsArray); $i += 2) {
if (isset($this->rule[$ident])) { if (isset($paramsArray[$i + 1])) {
$uri = $this->rule[$ident][0]; $_GET[$paramsArray[$i]] = $params[$paramsArray[$i]] = $paramsArray[$i + 1];
$paramString = null; }
// 如果存在参数 }
if (isset($this->rule[$ident][1]) && $this->rule[$ident][1]) { } else { // 未开启完全参数名,则利用反射得到参数名做映射
$param = (count($uriArray) > 1) ? array_slice($uriArray, 1) : []; if (!empty($paramsArray)) {
$paramNames = explode(',', $this->rule[$ident][1]); $reflectionMethod = Application::getReflectionMethod($class, $method);
for ($i = 0; $i < count($paramNames); $i++) { $index = 0;
if (substr($paramNames[$i], 0, 1) == '?') { // 可选参数 foreach ($reflectionMethod->getParameters() as $parameter) {
if (isset($param[$i]) && $param[$i]) { // 如果按顺序存在参数,且值有效 $className = $parameter->getClass();
$paramString .= substr($paramNames[$i], 1) . '/' . $param[$i] . '/'; if (is_null($className)) {
} $_GET[$parameter->name] = $params[$parameter->name] = $paramsArray[$index];
} else { $index++;
if (isset($param[$i]) && $param[$i]) {
$paramString .= $paramNames[$i] . '/' . $param[$i] . '/';
} else {
throw new RouteException('链接中缺少必须参数' . $paramNames[$i]);
}
} }
} }
} }
$uri .= '/' . rtrim($paramString, '/');
} }
return $uri; return $params;
} }
/** /**
* 路由中间件 * 执行应用
* @return mixed
* @throws RouteException
*/
public function execute()
{
try {
// 处理路由
$this->handler($this->request->uri());
} catch (RouteException $exception) {
if (!DEBUG) { // 非调试模式直接404
return \response()->notFound();
} else throw $exception;
}
// 将执行应用打包为一个$application
$application = function () {
// 执行控制器方法
$object = Application::getInstance($this->controllerFullName);
$reflectionMethod = Application::getReflectionMethod($this->controllerFullName, $this->method);
$invokeParams = [];
foreach ($reflectionMethod->getParameters() as $parameter) {
$className = $parameter->getClass();
if (!is_null($className)) {
$invokeParams[$parameter->name] = Application::getInstance($className->name);
} else {
if (isset($this->params[$parameter->name])) {
$invokeParams[$parameter->name] = $this->params[$parameter->name];
}
}
}
return $reflectionMethod->invokeArgs($object, $invokeParams);
};
// 路由中间件处理application
return $this->middleware($application);
}
/**
* 中间件处理
* @param \Closure $application * @param \Closure $application
* @return mixed * @return mixed
*/ */
public function middleware(\Closure $application) public function middleware(\Closure $application)
{ {
// 不执行的中间件 // 加载全局配置文件中配置的中间件
$middleware = array_reverse(config('middleware'));
// 配置中不执行的中间件
$exceptMiddlewareArray = []; $exceptMiddlewareArray = [];
// 加载配置文件中配置的中间件 if (isset($this->loadRuleParameters['except_middleware'])
$middlewareArray = Config::instance()->get('middleware'); && $this->loadRuleParameters['except_middleware'] != ''
// 合并路由中配置的中间件、不执行的中间件 ) {
if (isset($this->rule[$this->ident])) { $exceptMiddlewareArray = $this->loadRuleParameters['except_middleware'];
if (isset($this->rule[$this->ident][2]) && !empty(isset($this->rule[$this->ident][2]))) { }
$middlewareArray = array_merge($middlewareArray, $this->rule[$this->ident][2]);
} // 配置中新增的中间件
if (isset($this->rule[$this->ident][3]) && !empty(isset($this->rule[$this->ident][3]))) { if (isset($this->loadRuleParameters['accept_middleware'])
$exceptMiddlewareArray = $this->rule[$this->ident][3]; && $this->loadRuleParameters['accept_middleware'] != ''
) {
$acceptMiddlewareArray = $this->loadRuleParameters['accept_middleware'];
foreach ($acceptMiddlewareArray as $acceptMiddleware) {
if (!in_array($acceptMiddleware, $middleware)) {
$middleware[] = $acceptMiddleware;
} else continue;
} }
} }
$middleware = array_reverse($middlewareArray);
// 应用打包在在洋葱圈最里层
$next = $application; $next = $application;
foreach ($middleware as $value) { foreach ($middleware as $value) {
if (!in_array($value, $exceptMiddlewareArray)) { if (!in_array($value, $exceptMiddlewareArray)) {
$next = function () use ($next, $value) { $next = function () use ($next, $value) {
$middleware = new $value; $middleware = new $value;
if ($middleware instanceof MiddlewareIfs) { if ($middleware instanceof MiddlewareIfs) {
return $middleware->handler($next); return $middleware->handler($this->request, $next);
} else { } else throw new RouteException('无效的中间件:' . $value);
throw new RouteException('中间件' . $value . '不属于MiddlewareIfs类型实例');
}
}; };
} }
} }
return $next(); return $next();
} }
/** /**
* 处理URI * 处理URI
* @return $this * @param $uri
* @throws RouteException
*/ */
public function handler() public function handler($uri)
{ {
$uri = $this->uri(); $uri = $uri ? $uri : '/';
if ($uri) { $defaultMethod = config('default_method');
// 自定义路由规则 $requestMethod = strtolower($this->request->requestMethod());
$uri = $this->customRule($uri); // 第一次用原始uri去做匹配第二次带上默认方法去做匹配
if (false === ($rule = $this->parseRouteRule($requestMethod, $uri))
&& false === ($rule = $this->parseRouteRule($requestMethod, $uri . '/' . $defaultMethod))
) {
// 如果开启强制路由,则抛异常
if (config('compel_route') === true) {
throw new RouteException('不支持的路由规则:' . strtoupper($requestMethod) . ' ' . $uri);
} else {
// 进行普通处理
$rule = $this->parseRoute($uri);
}
}
$ruleParameters = $rule['parameters'];
$this->controllerFullName = $ruleParameters['class'];
$this->controller = substr($this->controllerFullName, strrpos($ruleParameters['class'], '\\') + 1);
$this->method = $ruleParameters['method'];
// 此处还需要检查控制器和方法是否存在
if (!class_exists($this->controllerFullName)) {
throw new RouteException('不存在的控制器:' . $this->controllerFullName);
}
if (!method_exists($this->controllerFullName, $this->method)) {
throw new RouteException('不存在的方法:' . $this->method);
}
$this->params = $this->parseParameters($ruleParameters['class'], $ruleParameters['method'], $rule['rule'], $uri);
}
/**
* 创建路由配置缓存文件
* @return array|mixed
*/
public function getRouteConfig()
{
$fileName = './runtime/' . BIND_MODULE . '_route_cache.php';
if (!DEBUG && is_file($fileName)) {
return require $fileName;
} else {
$result = [];
$controllerPath = APP_PATH . BIND_MODULE . '/controller/';
$namespace = APP_NS . '\\' . BIND_MODULE . '\\controller';
$files = scandir($controllerPath);
for ($i = 2; $i < count($files); $i++) {
$className = $namespace . '\\' . pathinfo($files[$i])['filename'];
$reflectionClass = Application::getReflectionClass($className);
foreach ($reflectionClass->getMethods() as $method) {
if ($method->class == $className && substr($method->name, 0, 1) != '_') {
$annotation = Annotation::getMethodAnnotation($className, $method->name);
$requestMethod = (isset($annotation['requestMethod'])) ? $annotation['requestMethod'] : 'any';
if (isset($annotation['route'])) {
$requestUri = $annotation['route'];
} else continue;
$requestMethod = strtolower($requestMethod);
$rule = ($requestUri == '/') ? $requestUri : trim($requestUri, '/');
$result[$requestMethod][$rule] = [
'class' => $className,
'method' => $method->name,
'except_middleware' => [],
'accept_middleware' => [],
];
if (isset($annotation['exceptMiddleware']) && $annotation['exceptMiddleware'] != '') {
foreach (explode('|', $annotation['exceptMiddleware']) as $exceptMiddleware) {
$result[$requestMethod][$rule]['except_middleware'][] = $exceptMiddleware;
}
}
if (isset($annotation['acceptMiddleware']) && $annotation['acceptMiddleware'] != '') {
foreach (explode('|', $annotation['acceptMiddleware']) as $acceptMiddleware) {
$result[$requestMethod][$rule]['accept_middleware'][] = $acceptMiddleware;
}
}
}
}
}
// 加载配置文件中的路由配置
$routeConfigFile = CONFIG_DIR . 'route.php';
if (is_file($routeConfigFile)) {
$routeConfig = require $routeConfigFile;
foreach ($routeConfig as $key => $value) {
if (isset($result[$key])) { // 存在当前请求方法的配置就检查含有的路由配置
foreach ($value as $uri => $config) {
$uri = ($uri == '/') ? $uri : trim($uri, '/');
if (isset($result[$key][$uri])) { // 如果已经存在这个路由配置,可能不完全,直接合并覆盖已有项
$result[$key][$uri] = array_merge($result[$key][$uri], $config);
} else {
$result[$key][$uri] = $config;
}
}
} else {
$result[$key] = $value;
}
}
}
// 写入文件
ob_start();
var_export($result);
$content = ob_get_contents();
ob_clean();
file_put_contents($fileName, "<?php\nreturn " . $content . ';');
return $result;
} }
// 初始化路由驱动
$this->driver->init(urldecode($uri));
return $this;
} }
} }

View File

@ -40,7 +40,7 @@ class BaseException extends \Exception
{ {
$file = file($filename); $file = file($filename);
$totalLine = count($file); $totalLine = count($file);
$offset = 10; $offset = 6;
$offsetStart = $line - $offset; $offsetStart = $line - $offset;
$offsetEnd = $line + $offset; $offsetEnd = $line + $offset;
$start = ($offsetStart <= 0) ? 2 : $offsetStart; $start = ($offsetStart <= 0) ? 2 : $offsetStart;
@ -163,7 +163,7 @@ class BaseException extends \Exception
overflow-x: scroll; overflow-x: scroll;
} }
.detail .codeblock .errLine { .detail .codeblock .errLine {
background: #f7e6e6 !important; background: #ffc7ad !important;
} }
.detail .codeblock a { .detail .codeblock a {
display: block; display: block;

View File

@ -45,12 +45,46 @@ function filterArray($array = [], $filter = 'filter', &$result = [])
} }
/** /**
* 调用请求类 * 获取/设置配置
* @param $key
* @param string $value
* @return mixed
*/
function config($key, $value = '__NULL__VALUE__')
{
if ($value != '__NULL__VALUE__') {
return \top\library\Config::instance()->set($key, $value);
} else {
return \top\library\Config::instance()->get($key);
}
}
/**
* 请求类
* @return \top\traits\Instance
*/ */
function request() function request()
{ {
$request = \top\library\http\Request::instance(); /*static $instance;
return $request; if (!$instance) {
$instance = new \top\library\http\Request();
}
return $instance;*/
return \top\library\http\Request::instance();
}
/**
* 响应类
* @return \top\traits\Instance
*/
function response()
{
/*static $instance;
if (!$instance) {
$instance = new \top\library\http\Response();
}
return $instance;*/
return \top\library\http\Response::instance();
} }
/** /**
@ -218,7 +252,7 @@ function get_client_ip($type = 0, $client = true)
*/ */
function redirect($url) function redirect($url)
{ {
if (request()->isAjax()) { if (request()->is('ajax')) {
return json_encode([ return json_encode([
'redirect' => $url, 'redirect' => $url,
]); ]);
@ -449,13 +483,9 @@ function is_mobile()
* 获取当前视图文件的缓存标识 * 获取当前视图文件的缓存标识
* @return string * @return string
*/ */
function viewCacheIdent() function view_cache_ident()
{ {
if (isset($_SERVER['REQUEST_URI'])) { $ident = md5($_SERVER['REQUEST_URI'] . request()->requestMethod());
$ident = md5($_SERVER['REQUEST_URI']);
} else {
$ident = request()->module() . request()->controller() . request()->method();
}
return $ident; return $ident;
} }

View File

@ -2,7 +2,8 @@
namespace top\library\http; namespace top\library\http;
use top\library\exception\RouteException; use top\library\Application;
use top\library\Router;
use top\traits\Instance; use top\traits\Instance;
/** /**
@ -21,16 +22,10 @@ class Request
private $server = []; private $server = [];
/** /**
* 中间件 * 当前URI
* @var array * @var mixed|null
*/ */
private $middleware = []; private $uri = null;
/**
* 路由实例
* @var null
*/
private $router = null;
/** /**
* post、get数据删除的值 * post、get数据删除的值
@ -38,99 +33,40 @@ class Request
*/ */
private $except = []; private $except = [];
/**
* Request constructor.
*/
private function __construct() private function __construct()
{ {
$this->server = (!empty($_SERVER)) ? $_SERVER : []; $this->server = (!empty($_SERVER)) ? $_SERVER : [];
// 当前uri
$this->uri = $this->getUri();
} }
/** /**
* 当前请求方式 * 当前请求方式
* @return mixed|string * @return mixed|string
*/ */
private function requestMethod() public function requestMethod()
{ {
return (isset($this->server['REQUEST_METHOD']) && $this->server['REQUEST_METHOD'] != '') ? $this->server['REQUEST_METHOD'] : ''; return (isset($this->server['REQUEST_METHOD']) && $this->server['REQUEST_METHOD'] != '') ? $this->server['REQUEST_METHOD'] : '';
} }
/** /**
* POST * 判断请求方式
* @return boolean * @param $method
* @return bool
*/ */
public function isPost() public function is($method)
{ {
return $this->requestMethod() == 'POST'; $method = strtolower($method);
} if ($method == 'ajax') {
return isset($this->server['HTTP_X_REQUESTED_WITH']) && ($this->server['HTTP_X_REQUESTED_WITH'] == 'XMLHttpRequest');
/** } elseif ($method == 'cli') {
* GET return (php_sapi_name() == 'cli');
* @return boolean } else {
*/ return strtolower($this->requestMethod()) == $method;
public function isGet() }
{
return $this->requestMethod() == 'GET';
}
/**
* PUT
* @return boolean
*/
public function isPut()
{
return $this->requestMethod() == 'PUT';
}
/**
* DELETE
* @return boolean
*/
public function isDelete()
{
return $this->requestMethod() == 'DELETE';
}
/**
* HEAD
* @return boolean
*/
public function isHead()
{
return $this->requestMethod() == 'HEAD';
}
/**
* HEAD
* @return boolean
*/
public function isPatch()
{
return $this->requestMethod() == 'PATCH';
}
/**
* HEAD
* @return boolean
*/
public function isOptions()
{
return $this->requestMethod() == 'OPTIONS';
}
/**
* AJAX
* @return boolean
*/
public function isAjax()
{
return isset($_SERVER['HTTP_X_REQUESTED_WITH']) && ($_SERVER['HTTP_X_REQUESTED_WITH'] == 'XMLHttpRequest');
}
/**
* CLI
* @return boolean
*/
public function isCLI()
{
return (php_sapi_name() == 'cli');
} }
/** /**
@ -156,13 +92,22 @@ class Request
return get_client_ip($type, $client); return get_client_ip($type, $client);
} }
/**
* 主机名
* @return mixed
*/
public function host()
{
return $this->server['HTTP_HOST'];
}
/** /**
* 当前请求的URI * 当前请求的URI
* @return mixed * @return mixed
*/ */
public function uri() public function uri()
{ {
return $this->router->uri(); return $this->uri;
} }
/** /**
@ -171,7 +116,7 @@ class Request
*/ */
public function module() public function module()
{ {
return $this->router->module(); return Router::instance($this)->module();
} }
/** /**
@ -180,7 +125,7 @@ class Request
*/ */
public function controllerFullName() public function controllerFullName()
{ {
return $this->router->controllerFullName(); return Router::instance($this)->controllerFullName();
} }
/** /**
@ -189,7 +134,7 @@ class Request
*/ */
public function controller() public function controller()
{ {
return $this->router->controller(); return Router::instance($this)->controller();
} }
/** /**
@ -198,7 +143,7 @@ class Request
*/ */
public function method() public function method()
{ {
return $this->router->method(); return Router::instance($this)->method();
} }
/** /**
@ -207,7 +152,16 @@ class Request
*/ */
public function params() public function params()
{ {
return $this->router->params(); return Router::instance($this)->params();
}
/**
* 当前加载的路由规则
* @return null
*/
public function routeParameters()
{
return Router::instance($this)->loadRuleParameters();
} }
/** /**
@ -266,6 +220,21 @@ class Request
return $this->requestData('post', $name, $except, $filter); return $this->requestData('post', $name, $except, $filter);
} }
/**
* 得到当前的URI
* @return mixed|null
*/
private function getUri()
{
$uri = null;
if (isset($this->server['PATH_INFO']) && $this->server['PATH_INFO']) {
$uri = $this->server['PATH_INFO'];
} elseif (isset($_GET['s']) && $_GET['s']) {
$uri = $_GET['s'];
}
return str_replace('.html', '', trim($uri, '/'));
}
/** /**
* GET POST公共方法 * GET POST公共方法
* @param $type * @param $type
@ -307,89 +276,4 @@ class Request
} }
} }
/**
* 指定路由
* @param $router
* @return $this
*/
public function setRoute($router)
{
$this->router = $router;
return $this;
}
/**
* 设置路由并执行程序
* @return mixed
*/
public function execute()
{
$this->check();
// 将执行应用打包为$application
$application = function () {
$controllerFullName = $this->controllerFullName();
$method = $this->method();
$params = $this->params();
$data = null;
$object = new $controllerFullName();
$reflectionClass = new \ReflectionClass($controllerFullName);
if ($reflectionClass->hasMethod('_init')) {
$data = $object->_init();
}
if ($data === null || $data === '') {
// 前置方法
$beforeReturnData = null;
$beforeMethod = 'before_' . $method;
if ($reflectionClass->hasMethod($beforeMethod)) {
$beforeReturnData = $object->{$beforeMethod}();
}
if ($beforeReturnData === null || $beforeReturnData === '' || $beforeReturnData === true) {
$reflectionMethod = new \ReflectionMethod($controllerFullName, $method);
$data = $reflectionMethod->invokeArgs($object, $params);
// 后置方法
$afterMethod = 'after_' . $method;
if ($reflectionClass->hasMethod($afterMethod)) {
$object->{$afterMethod}();
}
} else {
$data = $beforeReturnData;
}
}
return $data;
};
// 由路由中间件去处理application并返回结果
return $this->router->middleware($application);
}
/**
* 执行必要检查
* @throws RouteException
*/
private function check()
{
// 检查模块是否存在
if (!is_dir(APP_PATH . $this->module())) {
throw new RouteException('模块' . $this->module() . '不存在');
}
// 检查控制器是否存在
if (!class_exists($this->controllerFullName())) {
throw new RouteException('控制器' . $this->controllerFullName() . '不存在');
}
// 检查方法在控制器中是否存在
if (!in_array($this->method(), get_class_methods($this->controllerFullName()))) {
throw new RouteException('方法' . $this->method() . '在控制器' . $this->controller() . '中不存在');
}
}
public function __destruct()
{
}
} }

View File

@ -2,8 +2,9 @@
namespace top\library\http; namespace top\library\http;
use top\library\http\response\ResponseData; use top\library\View;
use top\traits\Instance; use top\traits\Instance;
use top\traits\Json;
/** /**
* 响应类 * 响应类
@ -15,11 +16,13 @@ class Response
use Instance; use Instance;
use Json;
/** /**
* 响应内容 * 响应内容
* @var null * @var string
*/ */
public $content = null; public $content = '';
/** /**
* 响应头 * 响应头
@ -50,17 +53,62 @@ class Response
* @param $data * @param $data
* @return false|int|null|string * @return false|int|null|string
*/ */
public function dispatch($data) public function send($data)
{ {
if ($data instanceof Response) { if ($data instanceof Response) {
return $data; return $data;
} else { } else {
// 处理响应数据,并返回 // 处理响应数据,并返回
$responseData = new ResponseData($data); $this->content = $this->getContent($data);
$this->content = $responseData->dispatch();
return $this; return $this;
} }
} }
public function notFound()
{
$this->header([
'HTTP/1.1 404 Not Found',
'Status: 404 Not Found'
]);
return <<<EOF
页面找不到了
EOF;
}
/**
* 处理数据
* @param $data
* @return false|int|null|string
*/
private function getContent($data)
{
if (is_array($data)) {
$request = request();
if ($request->is('ajax')) {
$this->header('Content-Type: application/json');
return $this->returnJson($data);
} else {
$this->header('Content-Type: text/html; charset=utf-8');
$filename = $request->controller() . '/' . $request->method();
return View::instance()->fetch($filename, $data);
}
} elseif (is_bool($data)) {
return ($data) ? 'true' : 'false';
} else if (is_object($data)) {
return '[OBJECT]';
}
return $data;
}
/**
* 直接echo处理
* @return string
*/
public function __toString()
{
return $this->content;
}
} }

View File

@ -1,71 +0,0 @@
<?php
namespace top\library\http\response;
use top\library\View;
use top\traits\Json;
/**
* 处理响应数据
* Class FormatResponse
* @package top\library\http
*/
class ResponseData
{
use Json;
/**
* 执行程序后返回的实际数据
* @var false|int|null|string
*/
public $data = null;
public function __construct($data)
{
if (DEBUG === false && php_sapi_name() != 'cli') {
ob_clean();
}
$this->data = $this->checkData($data);
}
/**
* 检查并处理数据
* @param $data
* @return false|int|null|string
*/
private function checkData($data)
{
$responseData = null;
if (is_array($data)) {
if (request()->isAjax()) {
$responseData = $this->returnJson($data);
} else {
$view = View::instance();
$filename = request()->controller() . '/' . request()->method();
$responseData = $view->fetch($filename, $data);
unset($filename);
}
} elseif (is_bool($data)) {
if ($data) {
$responseData = 1;
} else {
$responseData = 0;
}
} else if (is_object($data)) {
$responseData = '[OBJECT]';
} else {
// 否则数据作为字符串处理
$responseData = $data;
}
return $responseData;
}
/**
* 返回处理后的数据
* @return false|int|null|string
*/
public function dispatch()
{
return $this->data;
}
}

View File

@ -1,116 +0,0 @@
<?php
namespace top\library\route\driver;
use top\library\route\ifs\RouteIfs;
/**
* 命令行模式
* Class Command
* @package top\library\route\driver
*/
class Command implements RouteIfs
{
/**
* 解析后的URI信息
* @var array
*/
private $uriArray = [];
/**
* 模块名
* @return mixed|string
*/
public function module()
{
if (isset($this->uriArray[0])) {
return $this->uriArray[0];
}
return 'home';
}
/**
* 完整控制器名
* @return mixed|string
*/
public function controllerFullName()
{
$className = '\\' . APP_NS . '\\' . $this->module() . '\\controller\\' . $this->controller();
return $className;
}
/**
* 控制器名
* @return string
*/
public function controller()
{
if (isset($this->uriArray[1])) {
return ucfirst($this->uriArray[1]);
}
return 'Index';
}
/**
* 方法名
* @return mixed|string
*/
public function method()
{
if (isset($this->uriArray[2])) {
return $this->uriArray[2];
}
return 'index';
}
/**
* 请求参数
* @return array
* @throws \ReflectionException
*/
public function params()
{
return $this->parseParam();
}
/**
* 解析请求参数
* @return array
* @throws \ReflectionException
*/
private function parseParam()
{
$array = array_slice($this->uriArray, 3);
// 查找当前方法存在的参数
$paramName = (new \ReflectionMethod($this->controllerFullName(), $this->method()))->getParameters();
$paramNameArray = [];
foreach ($paramName as $value) {
$paramNameArray[] = $value->name;
}
$param = [];
for ($i = 0; $i < count($array); $i++) {
if (isset($array[$i + 1]) && in_array($array[$i], $paramNameArray)) {
$_GET[$array[$i]] = $param[$array[$i]] = $array[$i + 1];
}
}
return $param;
}
/**
* 执行初始化解析URI为数组并返回当前对象
* @param $uri
* @return $this
*/
public function init($uri)
{
$options = getopt('u:');
if (isset($options['u']) && $options['u']) {
$this->uriArray = $options['u'] ? explode('/', $options['u']) : [];
} else {
$this->uriArray = [];
}
return $this;
}
}

View File

@ -1,110 +0,0 @@
<?php
namespace top\library\route\driver;
use top\library\route\ifs\RouteIfs;
/**
* 兼容模式
* @author topnuomi 2018年11月19日
*/
class Compatible implements RouteIfs
{
/**
* 解析后的URI信息
* @var array
*/
private $uriArray = [];
/**
* 模块名
* @return mixed|string
*/
public function module()
{
if (isset($this->uriArray[0])) {
return $this->uriArray[0];
}
return 'home';
}
/**
* 完整控制器名
* @return mixed|string
*/
public function controllerFullName()
{
$className = '\\' . APP_NS . '\\' . $this->module() . '\\controller\\' . $this->controller();
return $className;
}
/**
* 控制器名
* @return string
*/
public function controller()
{
if (isset($this->uriArray[1])) {
return ucfirst($this->uriArray[1]);
}
return 'Index';
}
/**
* 方法名
* @return mixed|string
*/
public function method()
{
if (isset($this->uriArray[2])) {
return $this->uriArray[2];
}
return 'index';
}
/**
* 请求参数
* @return array
* @throws \ReflectionException
*/
public function params()
{
return $this->parseParam();
}
/**
* 解析请求参数
* @return array
* @throws \ReflectionException
*/
private function parseParam()
{
$array = array_slice($this->uriArray, 3);
// 查找当前方法存在的参数
$paramName = (new \ReflectionMethod($this->controllerFullName(), $this->method()))->getParameters();
$paramNameArray = [];
foreach ($paramName as $value) {
$paramNameArray[] = $value->name;
}
$param = [];
for ($i = 0; $i < count($array); $i++) {
if (isset($array[$i + 1]) && in_array($array[$i], $paramNameArray)) {
$_GET[$array[$i]] = $param[$array[$i]] = $array[$i + 1];
}
}
return $param;
}
/**
* 执行初始化解析URI为数组并返回当前对象
* @param $uri
* @return $this
*/
public function init($uri)
{
$this->uriArray = $uri ? explode('/', $uri) : [];
return $this;
}
}

View File

@ -1,48 +0,0 @@
<?php
namespace top\library\route\ifs;
/**
* 路由接口
* @author topnuomi 2018年11月19日
*/
interface RouteIfs
{
/**
* 模块名
* @return mixed
*/
public function module();
/**
* 完整控制器名
* @return mixed
*/
public function controllerFullName();
/**
* 控制器名
* @return mixed
*/
public function controller();
/**
* 方法名
* @return mixed
*/
public function method();
/**
* 参数
* @return mixed
*/
public function params();
/**
* 初始化路由
* @param $uri
* @return mixed
*/
public function init($uri);
}

View File

@ -90,7 +90,7 @@ class Top implements TemplateIfs
$content = ob_get_contents(); $content = ob_get_contents();
ob_clean(); ob_clean();
// 写入文件缓存 // 写入文件缓存
$ident = viewCacheIdent(); $ident = view_cache_ident();
if ($cache->set($ident, $content, $cacheTime)) { if ($cache->set($ident, $content, $cacheTime)) {
return $cache->get($ident); return $cache->get($ident);
} else { } else {

View File

@ -0,0 +1,49 @@
<?php
namespace top\middleware;
use top\library\Annotation;
use top\library\Application;
use top\library\http\Request;
use top\middleware\ifs\MiddlewareIfs;
/**
* 执行前置后置方法(前置后置操作中应尽量避免过多的数据库操作)
*
* @author topnuomi 2018年11月20日
*/
class Action implements MiddlewareIfs
{
public function handler(Request $request, \Closure $next)
{
$className = $request->controllerFullName();
$methodAnnotations = Annotation::getMethodAnnotation($className, $request->method());
$reflectionClass = Application::getReflectionClass($className);
// 前置操作
if (isset($methodAnnotations['beforeAction'])) {
$functions = explode('|', $methodAnnotations['beforeAction']);
foreach ($functions as $function) {
if (substr($function, 0, 1) == '_' && $reflectionClass->hasMethod($function)) {
$beforeData = (Application::getReflectionMethod($request->controllerFullName(), $function))->invoke(Application::getInstance($request->controllerFullName()));
if (!empty($beforeData)) return $beforeData;
}
}
$functions = null;
}
$closure = $next();
// 后置操作
if (isset($methodAnnotations['afterAction'])) {
$functions = explode('|', $methodAnnotations['afterAction']);
foreach ($functions as $function) {
if (substr($function, 0, 1) == '_' && $reflectionClass->hasMethod($function)) {
(Application::getReflectionMethod($request->controllerFullName(), $function))->invoke(Application::getInstance($request->controllerFullName()));
}
}
}
return $closure;
}
}

View File

@ -1,21 +0,0 @@
<?php
namespace top\middleware;
use top\middleware\ifs\MiddlewareIfs;
/**
* 默认中间件
*
* @author topnuomi 2018年11月20日
*/
class Init implements MiddlewareIfs
{
public function handler(\Closure $next)
{
// echo '应用开始';
$closure = $next();
// echo '应用结束';
return $closure;
}
}

View File

@ -4,24 +4,29 @@ namespace top\middleware;
use top\library\cache\driver\File; use top\library\cache\driver\File;
use top\library\Config; use top\library\Config;
use top\library\http\Request;
use top\library\http\Response; use top\library\http\Response;
use top\middleware\ifs\MiddlewareIfs; use top\middleware\ifs\MiddlewareIfs;
/**
* 检查是否存在静态缓存
* Class View
* @package top\middleware
*/
class View implements MiddlewareIfs class View implements MiddlewareIfs
{ {
public function handler(\Closure $next) public function handler(Request $request, \Closure $next)
{ {
// 非调试模式则直接返回静态缓存
if (!DEBUG) { if (!DEBUG) {
$ident = viewCacheIdent(); $ident = view_cache_ident();
$config = Config::instance()->get('view'); $config = Config::instance()->get('view');
(!$config['cacheDir']) && $config['cacheDir'] = './runtime/cache/application/' . request()->module() . '/'; (!$config['cacheDir']) && $config['cacheDir'] = './runtime/cache/application/' . request()->module() . '/';
$cache = File::instance($config['cacheDir']); $cache = File::instance($config['cacheDir']);
if ($cache->exists($ident)) { if ($cache->exists($ident)) return Response::instance()->send($cache->get($ident));
$content = $cache->get($ident);
return Response::instance()->dispatch($content);
}
} }
return $next(); return $next();
} }

View File

@ -2,6 +2,8 @@
namespace top\middleware\ifs; namespace top\middleware\ifs;
use top\library\http\Request;
/** /**
* 中间件接口 * 中间件接口
* *
@ -9,5 +11,5 @@ namespace top\middleware\ifs;
*/ */
interface MiddlewareIfs interface MiddlewareIfs
{ {
public function handler(\Closure $next); public function handler(Request $request, \Closure $next);
} }

View File

@ -11,7 +11,7 @@ require '../framework/Framework.php';
// 可使用常量DEBUG取得该值 // 可使用常量DEBUG取得该值
// 项目目录,缺省值:./application/ // 项目目录,缺省值:./application/
// Framework::appPath('../application/'); Framework::appPath('../application/');
// 可使用常量APP_PATH取得该值 // 可使用常量APP_PATH取得该值
// 项目命名空间缺省值app // 项目命名空间缺省值app
@ -30,8 +30,9 @@ require '../framework/Framework.php';
// Framework::resourcePath('/resource/'); // Framework::resourcePath('/resource/');
// 可使用常量RESOURCE取得该值 // 可使用常量RESOURCE取得该值
// 路由模式缺省值1pathinfo和兼容模式 // 绑定模块缺省值home
// Framework::runType(1); // Framework::bindModule('index');
// 可使用常量BIND_MODULE取得该值
Framework::appPath('../application/'); // 启动应用
Framework::startApp(); Framework::startApp();