diff --git a/.gitignore b/.gitignore
new file mode 100644
index 000000000..c96a04f00
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,2 @@
+*
+!.gitignore
\ No newline at end of file
diff --git a/_composer.bat b/_composer.bat
new file mode 100644
index 000000000..7279376c1
--- /dev/null
+++ b/_composer.bat
@@ -0,0 +1,7 @@
+:: Composer 安装更新脚本
+@title Composer Install
+@rmdir /s/q vendor thinkphp runtime
+@echo ========= 下载并安装插件 =========
+@composer update --profile --prefer-dist --optimize-autoloader
+@echo ========= 压缩并发布插件 =========
+@composer dump-autoload --optimize
diff --git a/application/.htaccess b/application/.htaccess
new file mode 100644
index 000000000..3418e55a6
--- /dev/null
+++ b/application/.htaccess
@@ -0,0 +1 @@
+deny from all
\ No newline at end of file
diff --git a/application/command.php b/application/command.php
new file mode 100644
index 000000000..826bb2b23
--- /dev/null
+++ b/application/command.php
@@ -0,0 +1,12 @@
+
+// +----------------------------------------------------------------------
+
+return [];
diff --git a/application/common.php b/application/common.php
new file mode 100644
index 000000000..55d22f266
--- /dev/null
+++ b/application/common.php
@@ -0,0 +1,12 @@
+
+// +----------------------------------------------------------------------
+
+// 搴旂敤鍏叡鏂囦欢
diff --git a/application/extra/queue.php b/application/extra/queue.php
new file mode 100644
index 000000000..41fd544f4
--- /dev/null
+++ b/application/extra/queue.php
@@ -0,0 +1,14 @@
+
+// +----------------------------------------------------------------------
+
+return [
+ 'connector' => 'Sync'
+];
\ No newline at end of file
diff --git a/application/index/controller/Index.php b/application/index/controller/Index.php
new file mode 100644
index 000000000..2ab0697ba
--- /dev/null
+++ b/application/index/controller/Index.php
@@ -0,0 +1,10 @@
+*{ padding: 0; margin: 0; } .think_default_text{ padding: 4px 48px;} a{color:#2E5CD5;cursor: pointer;text-decoration: none} a:hover{text-decoration:underline; } body{ background: #fff; font-family: "Century Gothic","Microsoft yahei"; color: #333;font-size:18px} h1{ font-size: 100px; font-weight: normal; margin-bottom: 12px; } p{ line-height: 1.6em; font-size: 42px }
:)
ThinkPHP V5
鍗佸勾纾ㄤ竴鍓 - 涓篈PI寮鍙戣璁$殑楂樻ц兘妗嗘灦
[ V5.0 鐗堟湰鐢 涓冪墰浜 鐙璧炲姪鍙戝竷 ] ';
+ }
+}
diff --git a/composer.json b/composer.json
new file mode 100644
index 000000000..f05ba237c
--- /dev/null
+++ b/composer.json
@@ -0,0 +1,36 @@
+{
+ "name": "topthink/think",
+ "description": "the new thinkphp framework",
+ "type": "project",
+ "keywords": [
+ "framework",
+ "thinkphp",
+ "ORM"
+ ],
+ "homepage": "http://thinkphp.cn/",
+ "license": "Apache-2.0",
+ "authors": [
+ {
+ "name": "liu21st",
+ "email": "liu21st@gmail.com"
+ }
+ ],
+ "require": {
+ "php": ">=5.4.0",
+ "pclzip/pclzip": "^2.8",
+ "qiniu/php-sdk": "^7.0",
+ "aferrandini/phpqrcode": "^1.0",
+ "zoujingli/wechat-php-sdk": "dev-master",
+ "topthink/framework": "^5.0",
+ "topthink/think-captcha": "^1.0",
+ "topthink/think-mongo": "^1.1",
+ "topthink/think-worker": "^1.0",
+ "topthink/think-queue": "^1.0"
+ },
+ "extra": {
+ "think-path": "thinkphp"
+ },
+ "config": {
+ "preferred-install": "dist"
+ }
+}
diff --git a/composer.lock b/composer.lock
new file mode 100644
index 000000000..c24ba0df9
--- /dev/null
+++ b/composer.lock
@@ -0,0 +1,515 @@
+{
+ "_readme": [
+ "This file locks the dependencies of your project to a known state",
+ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
+ "This file is @generated automatically"
+ ],
+ "hash": "3abb4855c5aa76f9606affb625fbe434",
+ "content-hash": "7317ea9bc409d341375e07971b55b540",
+ "packages": [
+ {
+ "name": "aferrandini/phpqrcode",
+ "version": "1.0.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/aferrandini/PHPQRCode.git",
+ "reference": "3c1c0454d43710ab5bbe19a51ad4cb41c22e3d46"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/aferrandini/PHPQRCode/3c1c0454d43710ab5bbe19a51ad4cb41c22e3d46.zip",
+ "reference": "3c1c0454d43710ab5bbe19a51ad4cb41c22e3d46",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=5.3.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-0": {
+ "PHPQRCode": "lib/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Ariel Ferrandini",
+ "email": "arielferrandini@gmail.com",
+ "homepage": "http://www.ferrandini.com/",
+ "role": "Developer"
+ }
+ ],
+ "description": "PHPQRCode porting and changed for PHP 5.3 compatibility",
+ "homepage": "https://github.com/aferrandini/PHPQRCode",
+ "keywords": [
+ "barcode",
+ "php",
+ "qrcode"
+ ],
+ "time": "2013-07-08 09:39:08"
+ },
+ {
+ "name": "pclzip/pclzip",
+ "version": "2.8.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/ivanlanin/pclzip.git",
+ "reference": "19dd1de9d3f5fc4d7d70175b4c344dee329f45fd"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/ivanlanin/pclzip/19dd1de9d3f5fc4d7d70175b4c344dee329f45fd.zip",
+ "reference": "19dd1de9d3f5fc4d7d70175b4c344dee329f45fd",
+ "shasum": ""
+ },
+ "type": "library",
+ "autoload": {
+ "classmap": [
+ "pclzip.lib.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "LGPL-2.1"
+ ],
+ "authors": [
+ {
+ "name": "Vincent Blavet"
+ }
+ ],
+ "description": "A PHP library that offers compression and extraction functions for Zip formatted archives",
+ "homepage": "http://www.phpconcept.net/pclzip",
+ "keywords": [
+ "php",
+ "zip"
+ ],
+ "time": "2014-06-05 11:42:24"
+ },
+ {
+ "name": "qiniu/php-sdk",
+ "version": "v7.1.3",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/qiniu/php-sdk.git",
+ "reference": "b91653485e36b4797d7a302cc86c49695e47a642"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/qiniu/php-sdk/b91653485e36b4797d7a302cc86c49695e47a642.zip",
+ "reference": "b91653485e36b4797d7a302cc86c49695e47a642",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=5.3.3"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "~4.0",
+ "squizlabs/php_codesniffer": "~2.3"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Qiniu\\": "src/Qiniu"
+ },
+ "files": [
+ "src/Qiniu/functions.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Qiniu",
+ "email": "sdk@qiniu.com",
+ "homepage": "http://www.qiniu.com"
+ }
+ ],
+ "description": "Qiniu Resource (Cloud) Storage SDK for PHP",
+ "homepage": "http://developer.qiniu.com/",
+ "keywords": [
+ "cloud",
+ "qiniu",
+ "sdk",
+ "storage"
+ ],
+ "time": "2016-11-18 02:57:31"
+ },
+ {
+ "name": "topthink/framework",
+ "version": "v5.0.5",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/framework.git",
+ "reference": "86cc9378a0c46e66dabed6681f8b8de758585ae1"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/framework/86cc9378a0c46e66dabed6681f8b8de758585ae1.zip",
+ "reference": "86cc9378a0c46e66dabed6681f8b8de758585ae1",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=5.4.0",
+ "topthink/think-installer": "~1.0"
+ },
+ "require-dev": {
+ "johnkary/phpunit-speedtrap": "^1.0",
+ "mikey179/vfsstream": "~1.6",
+ "phpdocumentor/reflection-docblock": "^2.0",
+ "phploc/phploc": "2.*",
+ "phpunit/phpunit": "4.8.*",
+ "sebastian/phpcpd": "2.*"
+ },
+ "type": "think-framework",
+ "autoload": {
+ "psr-4": {
+ "think\\": "library/think"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "liu21st",
+ "email": "liu21st@gmail.com"
+ }
+ ],
+ "description": "the new thinkphp framework",
+ "homepage": "http://thinkphp.cn/",
+ "keywords": [
+ "framework",
+ "orm",
+ "thinkphp"
+ ],
+ "time": "2017-01-23 05:59:21"
+ },
+ {
+ "name": "topthink/think-captcha",
+ "version": "v1.0.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/think-captcha.git",
+ "reference": "0c55455df26a1626a60d0dc35d2d89002b741d44"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/think-captcha/0c55455df26a1626a60d0dc35d2d89002b741d44.zip",
+ "reference": "0c55455df26a1626a60d0dc35d2d89002b741d44",
+ "shasum": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "think\\captcha\\": "src/"
+ },
+ "files": [
+ "src/helper.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "yunwuxin",
+ "email": "448901948@qq.com"
+ }
+ ],
+ "description": "captcha package for thinkphp5",
+ "time": "2016-07-06 01:47:11"
+ },
+ {
+ "name": "topthink/think-helper",
+ "version": "v1.0.5",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/think-helper.git",
+ "reference": "ed64408cdc4cdbd390365ba0906d208b987af520"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/think-helper/ed64408cdc4cdbd390365ba0906d208b987af520.zip",
+ "reference": "ed64408cdc4cdbd390365ba0906d208b987af520",
+ "shasum": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "think\\helper\\": "src"
+ },
+ "files": [
+ "src/helper.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "yunwuxin",
+ "email": "448901948@qq.com"
+ }
+ ],
+ "description": "The ThinkPHP5 Helper Package",
+ "time": "2016-12-01 07:08:40"
+ },
+ {
+ "name": "topthink/think-installer",
+ "version": "v1.0.11",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/think-installer.git",
+ "reference": "4c6e1ebecd1afce3f4ccc47e147d61bbe1bf641d"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/think-installer/4c6e1ebecd1afce3f4ccc47e147d61bbe1bf641d.zip",
+ "reference": "4c6e1ebecd1afce3f4ccc47e147d61bbe1bf641d",
+ "shasum": ""
+ },
+ "require": {
+ "composer-plugin-api": "^1.0"
+ },
+ "require-dev": {
+ "composer/composer": "1.0.*@dev"
+ },
+ "type": "composer-plugin",
+ "extra": {
+ "class": "think\\composer\\Plugin"
+ },
+ "autoload": {
+ "psr-4": {
+ "think\\composer\\": "src"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "yunwuxin",
+ "email": "448901948@qq.com"
+ }
+ ],
+ "time": "2016-12-01 09:08:45"
+ },
+ {
+ "name": "topthink/think-mongo",
+ "version": "v1.5",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/think-mongo.git",
+ "reference": "2dd7ecae965cd3a6e5cc99f3db7c63353dae4cf3"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/think-mongo/2dd7ecae965cd3a6e5cc99f3db7c63353dae4cf3.zip",
+ "reference": "2dd7ecae965cd3a6e5cc99f3db7c63353dae4cf3",
+ "shasum": ""
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "think\\mongo\\": "src"
+ },
+ "files": []
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "liu21st",
+ "email": "liu21st@gmail.com"
+ }
+ ],
+ "description": "mongodb driver for thinkphp5",
+ "time": "2017-02-06 06:05:55"
+ },
+ {
+ "name": "topthink/think-queue",
+ "version": "v1.1.2",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/think-queue.git",
+ "reference": "503c5b809585ca60cba9485a233aa8be4b22c990"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/think-queue/503c5b809585ca60cba9485a233aa8be4b22c990.zip",
+ "reference": "503c5b809585ca60cba9485a233aa8be4b22c990",
+ "shasum": ""
+ },
+ "require": {
+ "topthink/think-helper": ">=1.0.4",
+ "topthink/think-installer": ">=1.0.10"
+ },
+ "type": "think-extend",
+ "extra": {
+ "think-config": {
+ "queue": "src/config.php"
+ }
+ },
+ "autoload": {
+ "psr-4": {
+ "think\\": "src"
+ },
+ "files": [
+ "src/common.php"
+ ]
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "yunwuxin",
+ "email": "448901948@qq.com"
+ }
+ ],
+ "description": "The ThinkPHP5 Queue Package",
+ "time": "2016-12-01 04:29:39"
+ },
+ {
+ "name": "topthink/think-worker",
+ "version": "v1.0.1",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/top-think/think-worker.git",
+ "reference": "b609ff5e38dbb7194aab027d2b2c6b31a7ed1bd1"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/top-think/think-worker/b609ff5e38dbb7194aab027d2b2c6b31a7ed1bd1.zip",
+ "reference": "b609ff5e38dbb7194aab027d2b2c6b31a7ed1bd1",
+ "shasum": ""
+ },
+ "require": {
+ "workerman/workerman": "^3.3.0"
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "think\\worker\\": "src"
+ },
+ "files": []
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "Apache-2.0"
+ ],
+ "authors": [
+ {
+ "name": "liu21st",
+ "email": "liu21st@gmail.com"
+ }
+ ],
+ "description": "workerman extend for thinkphp5",
+ "time": "2016-10-08 06:07:03"
+ },
+ {
+ "name": "workerman/workerman",
+ "version": "v3.3.7",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/walkor/Workerman.git",
+ "reference": "d466c0f4b37c6cfb4f27c69b158175c7e7ccc24c"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/walkor/Workerman/d466c0f4b37c6cfb4f27c69b158175c7e7ccc24c.zip",
+ "reference": "d466c0f4b37c6cfb4f27c69b158175c7e7ccc24c",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=5.3"
+ },
+ "suggest": {
+ "ext-event": "For better performance."
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "Workerman\\": "./"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "walkor",
+ "email": "walkor@workerman.net",
+ "homepage": "http://www.workerman.net",
+ "role": "Developer"
+ }
+ ],
+ "description": "An asynchronous event driven PHP framework for easily building fast, scalable network applications.",
+ "homepage": "http://www.workerman.net",
+ "keywords": [
+ "asynchronous",
+ "event-loop"
+ ],
+ "time": "2017-02-02 02:52:58"
+ },
+ {
+ "name": "zoujingli/wechat-php-sdk",
+ "version": "dev-master",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/zoujingli/wechat-php-sdk.git",
+ "reference": "36ad89c0f7fe1e4b29591cdb5c25079e1e0bd0e0"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://packagist.phpcomposer.com/files/zoujingli/wechat-php-sdk/36ad89c0f7fe1e4b29591cdb5c25079e1e0bd0e0.zip",
+ "reference": "36ad89c0f7fe1e4b29591cdb5c25079e1e0bd0e0",
+ "shasum": ""
+ },
+ "require": {
+ "php": ">=5.3.3"
+ },
+ "type": "project",
+ "autoload": {
+ "psr-4": {
+ "Wechat\\": "./Wechat"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "description": "WeChat development of SDK",
+ "homepage": "http://www.kancloud.cn/zoujingli/wechat-php-sdk",
+ "keywords": [
+ "wechat-php-sdk"
+ ],
+ "time": "2017-02-07 02:34:38"
+ }
+ ],
+ "packages-dev": [],
+ "aliases": [],
+ "minimum-stability": "stable",
+ "stability-flags": {
+ "zoujingli/wechat-php-sdk": 20
+ },
+ "prefer-stable": false,
+ "prefer-lowest": false,
+ "platform": {
+ "php": ">=5.4.0"
+ },
+ "platform-dev": []
+}
diff --git a/config/config.php b/config/config.php
new file mode 100644
index 000000000..21c0531f7
--- /dev/null
+++ b/config/config.php
@@ -0,0 +1,239 @@
+
+// +----------------------------------------------------------------------
+
+return [
+ // +----------------------------------------------------------------------
+ // | 搴旂敤璁剧疆
+ // +----------------------------------------------------------------------
+
+ // 搴旂敤鍛藉悕绌洪棿
+ 'app_namespace' => 'app',
+ // 搴旂敤璋冭瘯妯″紡
+ 'app_debug' => true,
+ // 搴旂敤Trace
+ 'app_trace' => false,
+ // 搴旂敤妯″紡鐘舵
+ 'app_status' => '',
+ // 鏄惁鏀寔澶氭ā鍧
+ 'app_multi_module' => true,
+ // 鍏ュ彛鑷姩缁戝畾妯″潡
+ 'auto_bind_module' => false,
+ // 娉ㄥ唽鐨勬牴鍛藉悕绌洪棿
+ 'root_namespace' => [],
+ // 鎵╁睍鍑芥暟鏂囦欢
+ 'extra_file_list' => [THINK_PATH . 'helper' . EXT],
+ // 榛樿杈撳嚭绫诲瀷
+ 'default_return_type' => 'html',
+ // 榛樿AJAX 鏁版嵁杩斿洖鏍煎紡,鍙塲son xml ...
+ 'default_ajax_return' => 'json',
+ // 榛樿JSONP鏍煎紡杩斿洖鐨勫鐞嗘柟娉
+ 'default_jsonp_handler' => 'jsonpReturn',
+ // 榛樿JSONP澶勭悊鏂规硶
+ 'var_jsonp_handler' => 'callback',
+ // 榛樿鏃跺尯
+ 'default_timezone' => 'PRC',
+ // 鏄惁寮鍚璇█
+ 'lang_switch_on' => false,
+ // 榛樿鍏ㄥ眬杩囨护鏂规硶 鐢ㄩ楀彿鍒嗛殧澶氫釜
+ 'default_filter' => '',
+ // 榛樿璇█
+ 'default_lang' => 'zh-cn',
+ // 搴旂敤绫诲簱鍚庣紑
+ 'class_suffix' => false,
+ // 鎺у埗鍣ㄧ被鍚庣紑
+ 'controller_suffix' => false,
+
+ // +----------------------------------------------------------------------
+ // | 妯″潡璁剧疆
+ // +----------------------------------------------------------------------
+
+ // 榛樿妯″潡鍚
+ 'default_module' => 'index',
+ // 绂佹璁块棶妯″潡
+ 'deny_module_list' => ['common'],
+ // 榛樿鎺у埗鍣ㄥ悕
+ 'default_controller' => 'Index',
+ // 榛樿鎿嶄綔鍚
+ 'default_action' => 'index',
+ // 榛樿楠岃瘉鍣
+ 'default_validate' => '',
+ // 榛樿鐨勭┖鎺у埗鍣ㄥ悕
+ 'empty_controller' => 'Error',
+ // 鎿嶄綔鏂规硶鍚庣紑
+ 'action_suffix' => '',
+ // 鑷姩鎼滅储鎺у埗鍣
+ 'controller_auto_search' => false,
+
+ // +----------------------------------------------------------------------
+ // | URL璁剧疆
+ // +----------------------------------------------------------------------
+
+ // PATHINFO鍙橀噺鍚 鐢ㄤ簬鍏煎妯″紡
+ 'var_pathinfo' => 's',
+ // 鍏煎PATH_INFO鑾峰彇
+ 'pathinfo_fetch' => ['ORIG_PATH_INFO', 'REDIRECT_PATH_INFO', 'REDIRECT_URL'],
+ // pathinfo鍒嗛殧绗
+ 'pathinfo_depr' => '/',
+ // URL浼潤鎬佸悗缂
+ 'url_html_suffix' => 'html',
+ // URL鏅氭柟寮忓弬鏁 鐢ㄤ簬鑷姩鐢熸垚
+ 'url_common_param' => false,
+ // URL鍙傛暟鏂瑰紡 0 鎸夊悕绉版垚瀵硅В鏋 1 鎸夐『搴忚В鏋
+ 'url_param_type' => 0,
+ // 鏄惁寮鍚矾鐢
+ 'url_route_on' => true,
+ // 璺敱浣跨敤瀹屾暣鍖归厤
+ 'route_complete_match' => false,
+ // 璺敱閰嶇疆鏂囦欢锛堟敮鎸侀厤缃涓級
+ 'route_config_file' => ['route'],
+ // 鏄惁寮哄埗浣跨敤璺敱
+ 'url_route_must' => false,
+ // 鍩熷悕閮ㄧ讲
+ 'url_domain_deploy' => false,
+ // 鍩熷悕鏍癸紝濡倀hinkphp.cn
+ 'url_domain_root' => '',
+ // 鏄惁鑷姩杞崲URL涓殑鎺у埗鍣ㄥ拰鎿嶄綔鍚
+ 'url_convert' => true,
+ // 榛樿鐨勮闂帶鍒跺櫒灞
+ 'url_controller_layer' => 'controller',
+ // 琛ㄥ崟璇锋眰绫诲瀷浼鍙橀噺
+ 'var_method' => '_method',
+ // 琛ㄥ崟ajax浼鍙橀噺
+ 'var_ajax' => '_ajax',
+ // 琛ㄥ崟pjax浼鍙橀噺
+ 'var_pjax' => '_pjax',
+ // 鏄惁寮鍚姹傜紦瀛 true鑷姩缂撳瓨 鏀寔璁剧疆璇锋眰缂撳瓨瑙勫垯
+ 'request_cache' => false,
+ // 璇锋眰缂撳瓨鏈夋晥鏈
+ 'request_cache_expire' => null,
+
+ // +----------------------------------------------------------------------
+ // | 妯℃澘璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'template' => [
+ // 妯℃澘寮曟搸绫诲瀷 鏀寔 php think 鏀寔鎵╁睍
+ 'type' => 'Think',
+ // 妯℃澘璺緞
+ 'view_path' => '',
+ // 妯℃澘鍚庣紑
+ 'view_suffix' => 'html',
+ // 妯℃澘鏂囦欢鍚嶅垎闅旂
+ 'view_depr' => DS,
+ // 妯℃澘寮曟搸鏅氭爣绛惧紑濮嬫爣璁
+ 'tpl_begin' => '{',
+ // 妯℃澘寮曟搸鏅氭爣绛剧粨鏉熸爣璁
+ 'tpl_end' => '}',
+ // 鏍囩搴撴爣绛惧紑濮嬫爣璁
+ 'taglib_begin' => '{',
+ // 鏍囩搴撴爣绛剧粨鏉熸爣璁
+ 'taglib_end' => '}',
+ ],
+
+ // 瑙嗗浘杈撳嚭瀛楃涓插唴瀹规浛鎹
+ 'view_replace_str' => [],
+ // 榛樿璺宠浆椤甸潰瀵瑰簲鐨勬ā鏉挎枃浠
+ 'dispatch_success_tmpl' => THINK_PATH . 'tpl' . DS . 'dispatch_jump.tpl',
+ 'dispatch_error_tmpl' => THINK_PATH . 'tpl' . DS . 'dispatch_jump.tpl',
+
+ // +----------------------------------------------------------------------
+ // | 寮傚父鍙婇敊璇缃
+ // +----------------------------------------------------------------------
+
+ // 寮傚父椤甸潰鐨勬ā鏉挎枃浠
+ 'exception_tmpl' => THINK_PATH . 'tpl' . DS . 'think_exception.tpl',
+
+ // 閿欒鏄剧ず淇℃伅,闈炶皟璇曟ā寮忔湁鏁
+ 'error_message' => '椤甸潰閿欒锛佽绋嶅悗鍐嶈瘯锝',
+ // 鏄剧ず閿欒淇℃伅
+ 'show_error_msg' => false,
+ // 寮傚父澶勭悊handle绫 鐣欑┖浣跨敤 \think\exception\Handle
+ 'exception_handle' => '',
+
+ // +----------------------------------------------------------------------
+ // | 鏃ュ織璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'log' => [
+ // 鏃ュ織璁板綍鏂瑰紡锛屽唴缃 file socket 鏀寔鎵╁睍
+ 'type' => 'File',
+ // 鏃ュ織淇濆瓨鐩綍
+ 'path' => LOG_PATH,
+ // 鏃ュ織璁板綍绾у埆
+ 'level' => [],
+ ],
+
+ // +----------------------------------------------------------------------
+ // | Trace璁剧疆 寮鍚 app_trace 鍚 鏈夋晥
+ // +----------------------------------------------------------------------
+ 'trace' => [
+ // 鍐呯疆Html Console 鏀寔鎵╁睍
+ 'type' => 'Html',
+ ],
+
+ // +----------------------------------------------------------------------
+ // | 缂撳瓨璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'cache' => [
+ // 椹卞姩鏂瑰紡
+ 'type' => 'File',
+ // 缂撳瓨淇濆瓨鐩綍
+ 'path' => CACHE_PATH,
+ // 缂撳瓨鍓嶇紑
+ 'prefix' => '',
+ // 缂撳瓨鏈夋晥鏈 0琛ㄧず姘镐箙缂撳瓨
+ 'expire' => 0,
+ ],
+
+ // +----------------------------------------------------------------------
+ // | 浼氳瘽璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'session' => [
+ 'id' => '',
+ // SESSION_ID鐨勬彁浜ゅ彉閲,瑙e喅flash涓婁紶璺ㄥ煙
+ 'var_session_id' => '',
+ // SESSION 鍓嶇紑
+ 'prefix' => 'think',
+ // 椹卞姩鏂瑰紡 鏀寔redis memcache memcached
+ 'type' => '',
+ // 鏄惁鑷姩寮鍚 SESSION
+ 'auto_start' => true,
+ ],
+
+ // +----------------------------------------------------------------------
+ // | Cookie璁剧疆
+ // +----------------------------------------------------------------------
+ 'cookie' => [
+ // cookie 鍚嶇О鍓嶇紑
+ 'prefix' => '',
+ // cookie 淇濆瓨鏃堕棿
+ 'expire' => 0,
+ // cookie 淇濆瓨璺緞
+ 'path' => '/',
+ // cookie 鏈夋晥鍩熷悕
+ 'domain' => '',
+ // cookie 鍚敤瀹夊叏浼犺緭
+ 'secure' => false,
+ // httponly璁剧疆
+ 'httponly' => '',
+ // 鏄惁浣跨敤 setcookie
+ 'setcookie' => true,
+ ],
+
+ //鍒嗛〉閰嶇疆
+ 'paginate' => [
+ 'type' => 'bootstrap',
+ 'var_page' => 'page',
+ 'list_rows' => 15,
+ ],
+];
diff --git a/config/database.php b/config/database.php
new file mode 100644
index 000000000..5d3cab15b
--- /dev/null
+++ b/config/database.php
@@ -0,0 +1,57 @@
+
+// +----------------------------------------------------------------------
+
+return [
+ // 鏁版嵁搴撶被鍨
+ 'type' => 'mysql',
+ // 鏈嶅姟鍣ㄥ湴鍧
+ 'hostname' => '127.0.0.1',
+ // 鏁版嵁搴撳悕
+ 'database' => '',
+ // 鐢ㄦ埛鍚
+ 'username' => 'root',
+ // 瀵嗙爜
+ 'password' => '',
+ // 绔彛
+ 'hostport' => '',
+ // 杩炴帴dsn
+ 'dsn' => '',
+ // 鏁版嵁搴撹繛鎺ュ弬鏁
+ 'params' => [],
+ // 鏁版嵁搴撶紪鐮侀粯璁ら噰鐢╱tf8
+ 'charset' => 'utf8',
+ // 鏁版嵁搴撹〃鍓嶇紑
+ 'prefix' => '',
+ // 鏁版嵁搴撹皟璇曟ā寮
+ 'debug' => true,
+ // 鏁版嵁搴撻儴缃叉柟寮:0 闆嗕腑寮(鍗曚竴鏈嶅姟鍣),1 鍒嗗竷寮(涓讳粠鏈嶅姟鍣)
+ 'deploy' => 0,
+ // 鏁版嵁搴撹鍐欐槸鍚﹀垎绂 涓讳粠寮忔湁鏁
+ 'rw_separate' => false,
+ // 璇诲啓鍒嗙鍚 涓绘湇鍔″櫒鏁伴噺
+ 'master_num' => 1,
+ // 鎸囧畾浠庢湇鍔″櫒搴忓彿
+ 'slave_no' => '',
+ // 鏄惁涓ユ牸妫鏌ュ瓧娈垫槸鍚﹀瓨鍦
+ 'fields_strict' => true,
+ // 鏁版嵁闆嗚繑鍥炵被鍨
+ 'resultset_type' => 'array',
+ // 鑷姩鍐欏叆鏃堕棿鎴冲瓧娈
+ 'auto_timestamp' => false,
+ // 鏃堕棿瀛楁鍙栧嚭鍚庣殑榛樿鏃堕棿鏍煎紡
+ 'datetime_format' => 'Y-m-d H:i:s',
+ // 鏄惁闇瑕佽繘琛孲QL鎬ц兘鍒嗘瀽
+ 'sql_explain' => false,
+ // Builder绫
+ 'builder' => '',
+ // Query绫
+ 'query' => '\\think\\db\\Query',
+];
diff --git a/config/route.php b/config/route.php
new file mode 100644
index 000000000..f648d3b4e
--- /dev/null
+++ b/config/route.php
@@ -0,0 +1,21 @@
+
+// +----------------------------------------------------------------------
+
+return [
+ '__pattern__' => [
+ 'name' => '\w+',
+ ],
+ '[hello]' => [
+ ':id' => ['index/hello', ['method' => 'get'], ['id' => '\d+']],
+ ':name' => ['index/hello', ['method' => 'post']],
+ ],
+
+];
diff --git a/config/tags.php b/config/tags.php
new file mode 100644
index 000000000..e213e0aae
--- /dev/null
+++ b/config/tags.php
@@ -0,0 +1,28 @@
+
+// +----------------------------------------------------------------------
+
+// 搴旂敤琛屼负鎵╁睍瀹氫箟鏂囦欢
+return [
+ // 搴旂敤鍒濆鍖
+ 'app_init' => [],
+ // 搴旂敤寮濮
+ 'app_begin' => [],
+ // 妯″潡鍒濆鍖
+ 'module_init' => [],
+ // 鎿嶄綔寮濮嬫墽琛
+ 'action_begin' => [],
+ // 瑙嗗浘鍐呭杩囨护
+ 'view_filter' => [],
+ // 鏃ュ織鍐欏叆
+ 'log_write' => [],
+ // 搴旂敤缁撴潫
+ 'app_end' => [],
+];
diff --git a/extend/.gitignore b/extend/.gitignore
new file mode 100644
index 000000000..c96a04f00
--- /dev/null
+++ b/extend/.gitignore
@@ -0,0 +1,2 @@
+*
+!.gitignore
\ No newline at end of file
diff --git a/public/.htaccess b/public/.htaccess
new file mode 100644
index 000000000..cbc786893
--- /dev/null
+++ b/public/.htaccess
@@ -0,0 +1,8 @@
+
+ Options +FollowSymlinks -Multiviews
+ RewriteEngine On
+
+ RewriteCond %{REQUEST_FILENAME} !-d
+ RewriteCond %{REQUEST_FILENAME} !-f
+ RewriteRule ^(.*)$ index.php/$1 [QSA,PT,L]
+
diff --git a/public/favicon.ico b/public/favicon.ico
new file mode 100644
index 000000000..e71815a66
Binary files /dev/null and b/public/favicon.ico differ
diff --git a/public/index.php b/public/index.php
new file mode 100644
index 000000000..1eb0e75da
--- /dev/null
+++ b/public/index.php
@@ -0,0 +1,10 @@
+
+// +----------------------------------------------------------------------
+// $Id$
+
+if (is_file($_SERVER["DOCUMENT_ROOT"] . $_SERVER["REQUEST_URI"])) {
+ return false;
+} else {
+ require __DIR__ . "/index.php";
+}
diff --git a/public/static/.gitignore b/public/static/.gitignore
new file mode 100644
index 000000000..c96a04f00
--- /dev/null
+++ b/public/static/.gitignore
@@ -0,0 +1,2 @@
+*
+!.gitignore
\ No newline at end of file
diff --git a/thinkphp/.gitignore b/thinkphp/.gitignore
new file mode 100644
index 000000000..7e31ef510
--- /dev/null
+++ b/thinkphp/.gitignore
@@ -0,0 +1,4 @@
+/composer.lock
+/vendor
+.idea
+.DS_Store
diff --git a/thinkphp/.htaccess b/thinkphp/.htaccess
new file mode 100644
index 000000000..3418e55a6
--- /dev/null
+++ b/thinkphp/.htaccess
@@ -0,0 +1 @@
+deny from all
\ No newline at end of file
diff --git a/thinkphp/.travis.yml b/thinkphp/.travis.yml
new file mode 100644
index 000000000..f74ffca11
--- /dev/null
+++ b/thinkphp/.travis.yml
@@ -0,0 +1,47 @@
+sudo: false
+
+language: php
+
+services:
+ - memcached
+ - mongodb
+ - mysql
+ - postgresql
+ - redis-server
+
+matrix:
+ fast_finish: true
+ include:
+ - php: 5.4
+ - php: 5.5
+ - php: 5.6
+ - php: 7.0
+ - php: hhvm
+ allow_failures:
+ - php: hhvm
+
+cache:
+ directories:
+ - $HOME/.composer/cache
+
+before_install:
+ - composer self-update
+ - mysql -e "create database IF NOT EXISTS test;" -uroot
+ - psql -c 'DROP DATABASE IF EXISTS test;' -U postgres
+ - psql -c 'create database test;' -U postgres
+
+install:
+ - ./tests/script/install.sh
+
+script:
+ ## LINT
+ - find . -path ./vendor -prune -o -type f -name \*.php -exec php -l {} \;
+ ## PHP Copy/Paste Detector
+ - vendor/bin/phpcpd --verbose --exclude vendor ./ || true
+ ## PHPLOC
+ - vendor/bin/phploc --exclude vendor ./
+ ## PHPUNIT
+ - vendor/bin/phpunit --coverage-clover=coverage.xml --configuration=phpunit.xml
+
+after_success:
+ - bash <(curl -s https://codecov.io/bash)
diff --git a/thinkphp/CONTRIBUTING.md b/thinkphp/CONTRIBUTING.md
new file mode 100644
index 000000000..6cefcb38c
--- /dev/null
+++ b/thinkphp/CONTRIBUTING.md
@@ -0,0 +1,119 @@
+濡備綍璐$尞鎴戠殑婧愪唬鐮
+===
+
+姝ゆ枃妗d粙缁嶄簡 ThinkPHP 鍥㈤槦鐨勭粍鎴愪互鍙婅繍杞満鍒讹紝鎮ㄦ彁浜ょ殑浠g爜灏嗙粰 ThinkPHP 椤圭洰甯︽潵浠涔堝ソ澶勶紝浠ュ強濡備綍鎵嶈兘鍔犲叆鎴戜滑鐨勮鍒椼
+
+## 閫氳繃 Github 璐$尞浠g爜
+
+ThinkPHP 鐩墠浣跨敤 Git 鏉ユ帶鍒剁▼搴忕増鏈紝濡傛灉浣犳兂涓 ThinkPHP 璐$尞婧愪唬鐮侊紝璇峰厛澶ц嚧浜嗚В Git 鐨勪娇鐢ㄦ柟娉曘傛垜浠洰鍓嶆妸椤圭洰鎵樼鍦 GitHub 涓婏紝浠讳綍 GitHub 鐢ㄦ埛閮藉彲浠ュ悜鎴戜滑璐$尞浠g爜銆
+
+鍙備笌鐨勬柟寮忓緢绠鍗曪紝`fork`涓浠 ThinkPHP 鐨勪唬鐮佸埌浣犵殑浠撳簱涓紝淇敼鍚庢彁浜わ紝骞跺悜鎴戜滑鍙戣捣`pull request`鐢宠锛屾垜浠細鍙婃椂瀵逛唬鐮佽繘琛屽鏌ュ苟澶勭悊浣犵殑鐢宠骞躲傚鏌ラ氳繃鍚庯紝浣犵殑浠g爜灏嗚`merge`杩涙垜浠殑浠撳簱涓紝杩欐牱浣犲氨浼氳嚜鍔ㄥ嚭鐜板湪璐$尞鑰呭悕鍗曢噷浜嗭紝闈炲父鏂逛究銆
+
+鎴戜滑甯屾湜浣犺础鐚殑浠g爜绗﹀悎锛
+
+* ThinkPHP 鐨勭紪鐮佽鑼
+* 閫傚綋鐨勬敞閲婏紝鑳借鍏朵粬浜鸿鎳
+* 閬靛惊 Apache2 寮婧愬崗璁
+
+**濡傛灉鎯宠浜嗚В鏇村缁嗚妭鎴栨湁浠讳綍鐤戦棶锛岃缁х画闃呰涓嬮潰鐨勫唴瀹**
+
+### 娉ㄦ剰浜嬮」
+
+* 鏈」鐩唬鐮佹牸寮忓寲鏍囧噯閫夌敤 [**PSR-2**](http://www.kancloud.cn/thinkphp/php-fig-psr/3141)锛
+* 绫诲悕鍜岀被鏂囦欢鍚嶉伒寰 [**PSR-4**](http://www.kancloud.cn/thinkphp/php-fig-psr/3144)锛
+* 瀵逛簬 Issues 鐨勫鐞嗭紝璇蜂娇鐢ㄨ濡 `fix #xxx(Issue ID)` 鐨 commit title 鐩存帴鍏抽棴 issue銆
+* 绯荤粺浼氳嚜鍔ㄥ湪 PHP 5.4 5.5 5.6 7.0 鍜 HHVM 涓婃祴璇曚慨鏀癸紝鍏朵腑 HHVM 涓嬬殑娴嬭瘯瀹硅鎶ラ敊锛岃纭繚浣犵殑淇敼绗﹀悎 PHP 5.4 ~ 5.6 鍜 PHP 7.0 鐨勮娉曡鑼冿紱
+* 绠$悊鍛樹笉浼氬悎骞堕犳垚 CI faild 鐨勪慨鏀癸紝鑻ュ嚭鐜 CI faild 璇锋鏌ヨ嚜宸辩殑婧愪唬鐮佹垨淇敼鐩稿簲鐨刐鍗曞厓娴嬭瘯鏂囦欢](tests)锛
+
+## GitHub Issue
+
+GitHub 鎻愪緵浜 Issue 鍔熻兘锛岃鍔熻兘鍙互鐢ㄤ簬锛
+
+* 鎻愬嚭 bug
+* 鎻愬嚭鍔熻兘鏀硅繘
+* 鍙嶉浣跨敤浣撻獙
+
+璇ュ姛鑳戒笉搴旇鐢ㄤ簬锛
+
+ * 鎻愬嚭淇敼鎰忚锛堟秹鍙婁唬鐮佺讲鍚嶅拰淇杩芥函闂锛
+ * 涓嶅弸鍠勭殑瑷璁
+
+## 蹇熶慨鏀
+
+**GitHub 鎻愪緵浜嗗揩閫熺紪杈戞枃浠剁殑鍔熻兘**
+
+1. 鐧诲綍 GitHub 甯愬彿锛
+2. 娴忚椤圭洰鏂囦欢锛屾壘鍒拌杩涜淇敼鐨勬枃浠讹紱
+3. 鐐瑰嚮鍙充笂瑙掗搮绗斿浘鏍囪繘琛屼慨鏀癸紱
+4. 濉啓 `Commit changes` 鐩稿叧鍐呭锛圱itle 蹇呭~锛夛紱
+5. 鎻愪氦淇敼锛岀瓑寰 CI 楠岃瘉鍜岀鐞嗗憳鍚堝苟銆
+
+**鑻ユ偍闇瑕佷竴娆℃彁浜ゅぇ閲忎慨鏀癸紝璇风户缁槄璇讳笅闈㈢殑鍐呭**
+
+## 瀹屾暣娴佺▼
+
+1. `fork`鏈」鐩紱
+2. 鍏嬮殕(`clone`)浣 `fork` 鐨勯」鐩埌鏈湴锛
+3. 鏂板缓鍒嗘敮(`branch`)骞舵鍑(`checkout`)鏂板垎鏀紱
+4. 娣诲姞鏈」鐩埌浣犵殑鏈湴 git 浠撳簱浣滀负涓婃父(`upstream`)锛
+5. 杩涜淇敼锛岃嫢浣犵殑淇敼鍖呭惈鏂规硶鎴栧嚱鏁扮殑澧炲噺锛岃璁板緱淇敼[鍗曞厓娴嬭瘯鏂囦欢](tests)锛
+6. 鍙樺熀锛堣鍚 `rebase`锛変綘鐨勫垎鏀埌涓婃父 master 鍒嗘敮锛
+7. `push` 浣犵殑鏈湴浠撳簱鍒 GitHub锛
+8. 鎻愪氦 `pull request`锛
+9. 绛夊緟 CI 楠岃瘉锛堣嫢涓嶉氳繃鍒欓噸澶 5~7锛孏itHub 浼氳嚜鍔ㄦ洿鏂颁綘鐨 `pull request`锛夛紱
+10. 绛夊緟绠$悊鍛樺鐞嗭紝骞跺強鏃 `rebase` 浣犵殑鍒嗘敮鍒颁笂娓 master 鍒嗘敮锛堣嫢涓婃父 master 鍒嗘敮鏈変慨鏀癸級銆
+
+*鑻ユ湁蹇呰锛屽彲浠 `git push -f` 寮鸿鎺ㄩ rebase 鍚庣殑鍒嗘敮鍒拌嚜宸辩殑 `fork`*
+
+*缁濆涓嶅彲浠ヤ娇鐢 `git push -f` 寮鸿鎺ㄩ佷慨鏀瑰埌涓婃父*
+
+### 娉ㄦ剰浜嬮」
+
+* 鑻ュ涓婅堪娴佺▼鏈変换浣曚笉娓呮鐨勫湴鏂癸紝璇锋煡闃 GIT 鏁欑▼锛屽 [杩欎釜](http://backlogtool.com/git-guide/cn/)锛
+* 瀵逛簬浠g爜**涓嶅悓鏂归潰**鐨勪慨鏀癸紝璇峰湪鑷繁 `fork` 鐨勯」鐩腑**鍒涘缓涓嶅悓鐨勫垎鏀**锛堝師鍥犲弬瑙乣瀹屾暣娴佺▼`绗9鏉″娉ㄩ儴鍒嗭級锛
+* 鍙樺熀鍙婁氦浜掑紡鍙樺熀鎿嶄綔鍙傝 [Git 浜や簰寮忓彉鍩篯(http://pakchoi.me/2015/03/17/git-interactive-rebase/)
+
+## 鎺ㄨ崘璧勬簮
+
+### 寮鍙戠幆澧
+
+* XAMPP for Windows 5.5.x
+* WampServer (for Windows)
+* upupw Apache PHP5.4 ( for Windows)
+
+鎴栬嚜琛屽畨瑁
+
+- Apache / Nginx
+- PHP 5.4 ~ 5.6
+- MySQL / MariaDB
+
+*Windows 鐢ㄦ埛鎺ㄨ崘娣诲姞 PHP bin 鐩綍鍒 PATH锛屾柟渚夸娇鐢 composer*
+
+*Linux 鐢ㄦ埛鑷閰嶇疆鐜锛 Mac 鐢ㄦ埛鎺ㄨ崘浣跨敤鍐呯疆 Apache 閰嶅悎 Homebrew 瀹夎 PHP 鍜 MariaDB*
+
+### 缂栬緫鍣
+
+Sublime Text 3 + phpfmt 鎻掍欢
+
+phpfmt 鎻掍欢鍙傛暟
+
+```json
+{
+ "autocomplete": true,
+ "enable_auto_align": true,
+ "format_on_save": true,
+ "indent_with_space": true,
+ "psr1_naming": false,
+ "psr2": true,
+ "version": 4
+}
+```
+
+鎴栧叾浠 缂栬緫鍣 / IDE 閰嶅悎 PSR2 鑷姩鏍煎紡鍖栧伐鍏
+
+### Git GUI
+
+* SourceTree
+* GitHub Desktop
+
+鎴栧叾浠 Git 鍥惧舰鐣岄潰瀹㈡埛绔
diff --git a/thinkphp/LICENSE.txt b/thinkphp/LICENSE.txt
new file mode 100644
index 000000000..574a39c40
--- /dev/null
+++ b/thinkphp/LICENSE.txt
@@ -0,0 +1,32 @@
+
+ThinkPHP閬靛惊Apache2寮婧愬崗璁彂甯冿紝骞舵彁渚涘厤璐逛娇鐢ㄣ
+鐗堟潈鎵鏈塁opyright 漏 2006-2016 by ThinkPHP (http://thinkphp.cn)
+All rights reserved銆
+ThinkPHP庐 鍟嗘爣鍜岃憲浣滄潈鎵鏈夎呬负涓婃捣椤舵兂淇℃伅绉戞妧鏈夐檺鍏徃銆
+
+Apache Licence鏄憲鍚嶇殑闈炵泩鍒╁紑婧愮粍缁嘇pache閲囩敤鐨勫崗璁
+璇ュ崗璁拰BSD绫讳技锛岄紦鍔变唬鐮佸叡浜拰灏婇噸鍘熶綔鑰呯殑钁椾綔鏉冿紝
+鍏佽浠g爜淇敼锛屽啀浣滀负寮婧愭垨鍟嗕笟杞欢鍙戝竷銆傞渶瑕佹弧瓒
+鐨勬潯浠讹細
+1锛 闇瑕佺粰浠g爜鐨勭敤鎴蜂竴浠紸pache Licence 锛
+2锛 濡傛灉浣犱慨鏀逛簡浠g爜锛岄渶瑕佸湪琚慨鏀圭殑鏂囦欢涓鏄庯紱
+3锛 鍦ㄥ欢浼哥殑浠g爜涓紙淇敼鍜屾湁婧愪唬鐮佽鐢熺殑浠g爜涓級闇瑕
+甯︽湁鍘熸潵浠g爜涓殑鍗忚锛屽晢鏍囷紝涓撳埄澹版槑鍜屽叾浠栧師鏉ヤ綔鑰呰
+瀹氶渶瑕佸寘鍚殑璇存槑锛
+4锛 濡傛灉鍐嶅彂甯冪殑浜у搧涓寘鍚竴涓狽otice鏂囦欢锛屽垯鍦∟otice鏂
+浠朵腑闇瑕佸甫鏈夋湰鍗忚鍐呭銆備綘鍙互鍦∟otice涓鍔犺嚜宸辩殑
+璁稿彲锛屼絾涓嶅彲浠ヨ〃鐜颁负瀵笰pache Licence鏋勬垚鏇存敼銆
+鍏蜂綋鐨勫崗璁弬鑰冿細http://www.apache.org/licenses/LICENSE-2.0
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
+BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
+ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGE.
diff --git a/thinkphp/README.md b/thinkphp/README.md
new file mode 100644
index 000000000..d81d68c8a
--- /dev/null
+++ b/thinkphp/README.md
@@ -0,0 +1,113 @@
+ThinkPHP 5.0
+===============
+
+[](https://styleci.io/repos/48530411)
+[](https://travis-ci.org/top-think/framework)
+[](http://codecov.io/github/github/top-think/framework?branch=master)
+[](https://packagist.org/packages/topthink/framework)
+[](https://packagist.org/packages/topthink/framework)
+[](https://packagist.org/packages/topthink/framework)
+[](https://packagist.org/packages/topthink/framework)
+
+ThinkPHP5鍦ㄤ繚鎸佸揩閫熷紑鍙戝拰澶ч亾鑷崇畝鐨勬牳蹇冪悊蹇典笉鍙樼殑鍚屾椂锛孭HP鐗堟湰瑕佹眰鎻愬崌鍒5.4锛屼紭鍖栨牳蹇冿紝鍑忓皯渚濊禆锛屽熀浜庡叏鏂扮殑鏋舵瀯鎬濇兂鍜屽懡鍚嶇┖闂村疄鐜帮紝鏄疶hinkPHP绐佺牬鍘熸湁妗嗘灦鎬濊矾鐨勯瑕嗕箣浣滐紝鍏朵富瑕佺壒鎬у寘鎷細
+
+ + 鍩轰簬鍛藉悕绌洪棿鍜屼紬澶歅HP鏂扮壒鎬
+ + 鏍稿績鍔熻兘缁勪欢鍖
+ + 寮哄寲璺敱鍔熻兘
+ + 鏇寸伒娲荤殑鎺у埗鍣
+ + 閲嶆瀯鐨勬ā鍨嬪拰鏁版嵁搴撶被
+ + 閰嶇疆鏂囦欢鍙垎绂
+ + 閲嶅啓鐨勮嚜鍔ㄩ獙璇佸拰瀹屾垚
+ + 绠鍖栨墿灞曟満鍒
+ + API鏀寔瀹屽杽
+ + 鏀硅繘鐨凩og绫
+ + 鍛戒护琛岃闂敮鎸
+ + REST鏀寔
+ + 寮曞鏂囦欢鏀寔
+ + 鏂逛究鐨勮嚜鍔ㄧ敓鎴愬畾涔
+ + 鐪熸鎯版у姞杞
+ + 鍒嗗竷寮忕幆澧冩敮鎸
+ + 鏀寔Composer
+
+> ThinkPHP5鐨勮繍琛岀幆澧冭姹侾HP5.4浠ヤ笂銆
+
+璇︾粏寮鍙戞枃妗e弬鑰 [ThinkPHP5瀹屽叏寮鍙戞墜鍐宂(http://www.kancloud.cn/manual/thinkphp5)
+
+## 鐩綍缁撴瀯
+
+鍒濆鐨勭洰褰曠粨鏋勫涓嬶細
+
+~~~
+www WEB閮ㄧ讲鐩綍锛堟垨鑰呭瓙鐩綍锛
+鈹溾攢application 搴旂敤鐩綍
+鈹 鈹溾攢common 鍏叡妯″潡鐩綍锛堝彲浠ユ洿鏀癸級
+鈹 鈹溾攢module_name 妯″潡鐩綍
+鈹 鈹 鈹溾攢config.php 妯″潡閰嶇疆鏂囦欢
+鈹 鈹 鈹溾攢common.php 妯″潡鍑芥暟鏂囦欢
+鈹 鈹 鈹溾攢controller 鎺у埗鍣ㄧ洰褰
+鈹 鈹 鈹溾攢model 妯″瀷鐩綍
+鈹 鈹 鈹溾攢view 瑙嗗浘鐩綍
+鈹 鈹 鈹斺攢 ... 鏇村绫诲簱鐩綍
+鈹 鈹
+鈹 鈹溾攢command.php 鍛戒护琛屽伐鍏烽厤缃枃浠
+鈹 鈹溾攢common.php 鍏叡鍑芥暟鏂囦欢
+鈹 鈹溾攢config.php 鍏叡閰嶇疆鏂囦欢
+鈹 鈹溾攢route.php 璺敱閰嶇疆鏂囦欢
+鈹 鈹溾攢tags.php 搴旂敤琛屼负鎵╁睍瀹氫箟鏂囦欢
+鈹 鈹斺攢database.php 鏁版嵁搴撻厤缃枃浠
+鈹
+鈹溾攢public WEB鐩綍锛堝澶栬闂洰褰曪級
+鈹 鈹溾攢index.php 鍏ュ彛鏂囦欢
+鈹 鈹溾攢router.php 蹇熸祴璇曟枃浠
+鈹 鈹斺攢.htaccess 鐢ㄤ簬apache鐨勯噸鍐
+鈹
+鈹溾攢thinkphp 妗嗘灦绯荤粺鐩綍
+鈹 鈹溾攢lang 璇█鏂囦欢鐩綍
+鈹 鈹溾攢library 妗嗘灦绫诲簱鐩綍
+鈹 鈹 鈹溾攢think Think绫诲簱鍖呯洰褰
+鈹 鈹 鈹斺攢traits 绯荤粺Trait鐩綍
+鈹 鈹
+鈹 鈹溾攢tpl 绯荤粺妯℃澘鐩綍
+鈹 鈹溾攢base.php 鍩虹瀹氫箟鏂囦欢
+鈹 鈹溾攢console.php 鎺у埗鍙板叆鍙f枃浠
+鈹 鈹溾攢convention.php 妗嗘灦鎯緥閰嶇疆鏂囦欢
+鈹 鈹溾攢helper.php 鍔╂墜鍑芥暟鏂囦欢
+鈹 鈹溾攢phpunit.xml phpunit閰嶇疆鏂囦欢
+鈹 鈹斺攢start.php 妗嗘灦鍏ュ彛鏂囦欢
+鈹
+鈹溾攢extend 鎵╁睍绫诲簱鐩綍
+鈹溾攢runtime 搴旂敤鐨勮繍琛屾椂鐩綍锛堝彲鍐欙紝鍙畾鍒讹級
+鈹溾攢vendor 绗笁鏂圭被搴撶洰褰曪紙Composer渚濊禆搴擄級
+鈹溾攢build.php 鑷姩鐢熸垚瀹氫箟鏂囦欢锛堝弬鑰冿級
+鈹溾攢composer.json composer 瀹氫箟鏂囦欢
+鈹溾攢LICENSE.txt 鎺堟潈璇存槑鏂囦欢
+鈹溾攢README.md README 鏂囦欢
+鈹溾攢think 鍛戒护琛屽叆鍙f枃浠
+~~~
+
+> router.php鐢ㄤ簬php鑷甫webserver鏀寔锛屽彲鐢ㄤ簬蹇熸祴璇
+> 鍒囨崲鍒皃ublic鐩綍鍚庯紝鍚姩鍛戒护锛歱hp -S localhost:8888 router.php
+> 涓婇潰鐨勭洰褰曠粨鏋勫拰鍚嶇О鏄彲浠ユ敼鍙樼殑锛岃繖鍙栧喅浜庝綘鐨勫叆鍙f枃浠跺拰閰嶇疆鍙傛暟銆
+
+## 鍛藉悕瑙勮寖
+
+ThinkPHP5鐨勫懡鍚嶈鑼冮伒寰狿SR-2瑙勮寖浠ュ強PSR-4鑷姩鍔犺浇瑙勮寖銆
+
+## 鍙備笌寮鍙
+娉ㄥ唽骞剁櫥褰 Github 甯愬彿锛 fork 鏈」鐩苟杩涜鏀瑰姩銆
+
+鏇村缁嗚妭鍙傞槄 [CONTRIBUTING.md](CONTRIBUTING.md)
+
+## 鐗堟潈淇℃伅
+
+ThinkPHP閬靛惊Apache2寮婧愬崗璁彂甯冿紝骞舵彁渚涘厤璐逛娇鐢ㄣ
+
+鏈」鐩寘鍚殑绗笁鏂规簮鐮佸拰浜岃繘鍒舵枃浠朵箣鐗堟潈淇℃伅鍙﹁鏍囨敞銆
+
+鐗堟潈鎵鏈塁opyright 漏 2006-2016 by ThinkPHP (http://thinkphp.cn)
+
+All rights reserved銆
+
+ThinkPHP庐 鍟嗘爣鍜岃憲浣滄潈鎵鏈夎呬负涓婃捣椤舵兂淇℃伅绉戞妧鏈夐檺鍏徃銆
+
+鏇村缁嗚妭鍙傞槄 [LICENSE.txt](LICENSE.txt)
diff --git a/thinkphp/base.php b/thinkphp/base.php
new file mode 100644
index 000000000..cbe288bf3
--- /dev/null
+++ b/thinkphp/base.php
@@ -0,0 +1,63 @@
+
+// +----------------------------------------------------------------------
+
+define('THINK_VERSION', '5.0.5');
+define('THINK_START_TIME', microtime(true));
+define('THINK_START_MEM', memory_get_usage());
+define('EXT', '.php');
+define('DS', DIRECTORY_SEPARATOR);
+defined('THINK_PATH') or define('THINK_PATH', __DIR__ . DS);
+define('LIB_PATH', THINK_PATH . 'library' . DS);
+define('CORE_PATH', LIB_PATH . 'think' . DS);
+define('TRAIT_PATH', LIB_PATH . 'traits' . DS);
+defined('APP_PATH') or define('APP_PATH', dirname($_SERVER['SCRIPT_FILENAME']) . DS);
+defined('ROOT_PATH') or define('ROOT_PATH', dirname(realpath(APP_PATH)) . DS);
+defined('EXTEND_PATH') or define('EXTEND_PATH', ROOT_PATH . 'extend' . DS);
+defined('VENDOR_PATH') or define('VENDOR_PATH', ROOT_PATH . 'vendor' . DS);
+defined('RUNTIME_PATH') or define('RUNTIME_PATH', ROOT_PATH . 'runtime' . DS);
+defined('LOG_PATH') or define('LOG_PATH', RUNTIME_PATH . 'log' . DS);
+defined('CACHE_PATH') or define('CACHE_PATH', RUNTIME_PATH . 'cache' . DS);
+defined('TEMP_PATH') or define('TEMP_PATH', RUNTIME_PATH . 'temp' . DS);
+defined('CONF_PATH') or define('CONF_PATH', APP_PATH); // 閰嶇疆鏂囦欢鐩綍
+defined('CONF_EXT') or define('CONF_EXT', EXT); // 閰嶇疆鏂囦欢鍚庣紑
+defined('ENV_PREFIX') or define('ENV_PREFIX', 'PHP_'); // 鐜鍙橀噺鐨勯厤缃墠缂
+
+// 鐜甯搁噺
+define('IS_CLI', PHP_SAPI == 'cli' ? true : false);
+define('IS_WIN', strpos(PHP_OS, 'WIN') !== false);
+
+// 杞藉叆Loader绫
+require CORE_PATH . 'Loader.php';
+
+// 鍔犺浇鐜鍙橀噺閰嶇疆鏂囦欢
+if (is_file(ROOT_PATH . '.env')) {
+ $env = parse_ini_file(ROOT_PATH . '.env', true);
+ foreach ($env as $key => $val) {
+ $name = ENV_PREFIX . strtoupper($key);
+ if (is_array($val)) {
+ foreach ($val as $k => $v) {
+ $item = $name . '_' . strtoupper($k);
+ putenv("$item=$v");
+ }
+ } else {
+ putenv("$name=$val");
+ }
+ }
+}
+
+// 娉ㄥ唽鑷姩鍔犺浇
+\think\Loader::register();
+
+// 娉ㄥ唽閿欒鍜屽紓甯稿鐞嗘満鍒
+\think\Error::register();
+
+// 鍔犺浇鎯緥閰嶇疆鏂囦欢
+\think\Config::set(include THINK_PATH . 'convention' . EXT);
diff --git a/thinkphp/codecov.yml b/thinkphp/codecov.yml
new file mode 100644
index 000000000..bef9d643a
--- /dev/null
+++ b/thinkphp/codecov.yml
@@ -0,0 +1,12 @@
+comment:
+ layout: header, changes, diff
+coverage:
+ ignore:
+ - base.php
+ - helper.php
+ - convention.php
+ - lang/zh-cn.php
+ - start.php
+ - console.php
+ status:
+ patch: false
diff --git a/thinkphp/composer.json b/thinkphp/composer.json
new file mode 100644
index 000000000..c546e1142
--- /dev/null
+++ b/thinkphp/composer.json
@@ -0,0 +1,35 @@
+{
+ "name": "topthink/framework",
+ "description": "the new thinkphp framework",
+ "type": "think-framework",
+ "keywords": [
+ "framework",
+ "thinkphp",
+ "ORM"
+ ],
+ "homepage": "http://thinkphp.cn/",
+ "license": "Apache-2.0",
+ "authors": [
+ {
+ "name": "liu21st",
+ "email": "liu21st@gmail.com"
+ }
+ ],
+ "require": {
+ "php": ">=5.4.0",
+ "topthink/think-installer": "~1.0"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "4.8.*",
+ "johnkary/phpunit-speedtrap": "^1.0",
+ "mikey179/vfsStream": "~1.6",
+ "phploc/phploc": "2.*",
+ "sebastian/phpcpd": "2.*",
+ "phpdocumentor/reflection-docblock": "^2.0"
+ },
+ "autoload": {
+ "psr-4": {
+ "think\\": "library/think"
+ }
+ }
+}
diff --git a/thinkphp/console.php b/thinkphp/console.php
new file mode 100644
index 000000000..fa7778011
--- /dev/null
+++ b/thinkphp/console.php
@@ -0,0 +1,20 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+// ThinkPHP 寮曞鏂囦欢
+// 鍔犺浇鍩虹鏂囦欢
+require __DIR__ . '/base.php';
+
+// 鎵ц搴旂敤
+App::initCommon();
+Console::init();
diff --git a/thinkphp/convention.php b/thinkphp/convention.php
new file mode 100644
index 000000000..c66ef5832
--- /dev/null
+++ b/thinkphp/convention.php
@@ -0,0 +1,288 @@
+ 'app',
+ // 搴旂敤璋冭瘯妯″紡
+ 'app_debug' => true,
+ // 搴旂敤Trace
+ 'app_trace' => false,
+ // 搴旂敤妯″紡鐘舵
+ 'app_status' => '',
+ // 鏄惁鏀寔澶氭ā鍧
+ 'app_multi_module' => true,
+ // 鍏ュ彛鑷姩缁戝畾妯″潡
+ 'auto_bind_module' => false,
+ // 娉ㄥ唽鐨勬牴鍛藉悕绌洪棿
+ 'root_namespace' => [],
+ // 鎵╁睍鍑芥暟鏂囦欢
+ 'extra_file_list' => [THINK_PATH . 'helper' . EXT],
+ // 榛樿杈撳嚭绫诲瀷
+ 'default_return_type' => 'html',
+ // 榛樿AJAX 鏁版嵁杩斿洖鏍煎紡,鍙塲son xml ...
+ 'default_ajax_return' => 'json',
+ // 榛樿JSONP鏍煎紡杩斿洖鐨勫鐞嗘柟娉
+ 'default_jsonp_handler' => 'jsonpReturn',
+ // 榛樿JSONP澶勭悊鏂规硶
+ 'var_jsonp_handler' => 'callback',
+ // 榛樿鏃跺尯
+ 'default_timezone' => 'PRC',
+ // 鏄惁寮鍚璇█
+ 'lang_switch_on' => false,
+ // 榛樿鍏ㄥ眬杩囨护鏂规硶 鐢ㄩ楀彿鍒嗛殧澶氫釜
+ 'default_filter' => '',
+ // 榛樿璇█
+ 'default_lang' => 'zh-cn',
+ // 搴旂敤绫诲簱鍚庣紑
+ 'class_suffix' => false,
+ // 鎺у埗鍣ㄧ被鍚庣紑
+ 'controller_suffix' => false,
+
+ // +----------------------------------------------------------------------
+ // | 妯″潡璁剧疆
+ // +----------------------------------------------------------------------
+
+ // 榛樿妯″潡鍚
+ 'default_module' => 'index',
+ // 绂佹璁块棶妯″潡
+ 'deny_module_list' => ['common'],
+ // 榛樿鎺у埗鍣ㄥ悕
+ 'default_controller' => 'Index',
+ // 榛樿鎿嶄綔鍚
+ 'default_action' => 'index',
+ // 榛樿楠岃瘉鍣
+ 'default_validate' => '',
+ // 榛樿鐨勭┖鎺у埗鍣ㄥ悕
+ 'empty_controller' => 'Error',
+ // 鎿嶄綔鏂规硶鍓嶇紑
+ 'use_action_prefix' => false,
+ // 鎿嶄綔鏂规硶鍚庣紑
+ 'action_suffix' => '',
+ // 鑷姩鎼滅储鎺у埗鍣
+ 'controller_auto_search' => false,
+
+ // +----------------------------------------------------------------------
+ // | URL璁剧疆
+ // +----------------------------------------------------------------------
+
+ // PATHINFO鍙橀噺鍚 鐢ㄤ簬鍏煎妯″紡
+ 'var_pathinfo' => 's',
+ // 鍏煎PATH_INFO鑾峰彇
+ 'pathinfo_fetch' => ['ORIG_PATH_INFO', 'REDIRECT_PATH_INFO', 'REDIRECT_URL'],
+ // pathinfo鍒嗛殧绗
+ 'pathinfo_depr' => '/',
+ // URL浼潤鎬佸悗缂
+ 'url_html_suffix' => 'html',
+ // URL鏅氭柟寮忓弬鏁 鐢ㄤ簬鑷姩鐢熸垚
+ 'url_common_param' => false,
+ // URL鍙傛暟鏂瑰紡 0 鎸夊悕绉版垚瀵硅В鏋 1 鎸夐『搴忚В鏋
+ 'url_param_type' => 0,
+ // 鏄惁寮鍚矾鐢
+ 'url_route_on' => true,
+ // 璺敱閰嶇疆鏂囦欢锛堟敮鎸侀厤缃涓級
+ 'route_config_file' => ['route'],
+ // 璺敱浣跨敤瀹屾暣鍖归厤
+ 'route_complete_match' => false,
+ // 鏄惁寮哄埗浣跨敤璺敱
+ 'url_route_must' => false,
+ // 鍩熷悕閮ㄧ讲
+ 'url_domain_deploy' => false,
+ // 鍩熷悕鏍癸紝濡倀hinkphp.cn
+ 'url_domain_root' => '',
+ // 鏄惁鑷姩杞崲URL涓殑鎺у埗鍣ㄥ拰鎿嶄綔鍚
+ 'url_convert' => true,
+ // 榛樿鐨勮闂帶鍒跺櫒灞
+ 'url_controller_layer' => 'controller',
+ // 琛ㄥ崟璇锋眰绫诲瀷浼鍙橀噺
+ 'var_method' => '_method',
+ // 琛ㄥ崟ajax浼鍙橀噺
+ 'var_ajax' => '_ajax',
+ // 琛ㄥ崟pjax浼鍙橀噺
+ 'var_pjax' => '_pjax',
+ // 鏄惁寮鍚姹傜紦瀛 true鑷姩缂撳瓨 鏀寔璁剧疆璇锋眰缂撳瓨瑙勫垯
+ 'request_cache' => false,
+ // 璇锋眰缂撳瓨鏈夋晥鏈
+ 'request_cache_expire' => null,
+
+ // +----------------------------------------------------------------------
+ // | 妯℃澘璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'template' => [
+ // 妯℃澘寮曟搸绫诲瀷 鏀寔 php think 鏀寔鎵╁睍
+ 'type' => 'Think',
+ // 瑙嗗浘鍩虹鐩綍锛岄厤缃洰褰曚负鎵鏈夋ā鍧楃殑瑙嗗浘璧峰鐩綍
+ 'view_base' => '',
+ // 褰撳墠妯℃澘鐨勮鍥剧洰褰 鐣欑┖涓鸿嚜鍔ㄨ幏鍙
+ 'view_path' => '',
+ // 妯℃澘鍚庣紑
+ 'view_suffix' => 'html',
+ // 妯℃澘鏂囦欢鍚嶅垎闅旂
+ 'view_depr' => DS,
+ // 妯℃澘寮曟搸鏅氭爣绛惧紑濮嬫爣璁
+ 'tpl_begin' => '{',
+ // 妯℃澘寮曟搸鏅氭爣绛剧粨鏉熸爣璁
+ 'tpl_end' => '}',
+ // 鏍囩搴撴爣绛惧紑濮嬫爣璁
+ 'taglib_begin' => '{',
+ // 鏍囩搴撴爣绛剧粨鏉熸爣璁
+ 'taglib_end' => '}',
+ ],
+
+ // 瑙嗗浘杈撳嚭瀛楃涓插唴瀹规浛鎹
+ 'view_replace_str' => [],
+ // 榛樿璺宠浆椤甸潰瀵瑰簲鐨勬ā鏉挎枃浠
+ 'dispatch_success_tmpl' => THINK_PATH . 'tpl' . DS . 'dispatch_jump.tpl',
+ 'dispatch_error_tmpl' => THINK_PATH . 'tpl' . DS . 'dispatch_jump.tpl',
+
+ // +----------------------------------------------------------------------
+ // | 寮傚父鍙婇敊璇缃
+ // +----------------------------------------------------------------------
+
+ // 寮傚父椤甸潰鐨勬ā鏉挎枃浠
+ 'exception_tmpl' => THINK_PATH . 'tpl' . DS . 'think_exception.tpl',
+
+ // 閿欒鏄剧ず淇℃伅,闈炶皟璇曟ā寮忔湁鏁
+ 'error_message' => '椤甸潰閿欒锛佽绋嶅悗鍐嶈瘯锝',
+ // 鏄剧ず閿欒淇℃伅
+ 'show_error_msg' => false,
+ // 寮傚父澶勭悊handle绫 鐣欑┖浣跨敤 \think\exception\Handle
+ 'exception_handle' => '',
+
+ // +----------------------------------------------------------------------
+ // | 鏃ュ織璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'log' => [
+ // 鏃ュ織璁板綍鏂瑰紡锛屽唴缃 file socket 鏀寔鎵╁睍
+ 'type' => 'File',
+ // 鏃ュ織淇濆瓨鐩綍
+ 'path' => LOG_PATH,
+ // 鏃ュ織璁板綍绾у埆
+ 'level' => [],
+ ],
+
+ // +----------------------------------------------------------------------
+ // | Trace璁剧疆 寮鍚 app_trace 鍚 鏈夋晥
+ // +----------------------------------------------------------------------
+ 'trace' => [
+ // 鍐呯疆Html Console 鏀寔鎵╁睍
+ 'type' => 'Html',
+ ],
+
+ // +----------------------------------------------------------------------
+ // | 缂撳瓨璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'cache' => [
+ // 椹卞姩鏂瑰紡
+ 'type' => 'File',
+ // 缂撳瓨淇濆瓨鐩綍
+ 'path' => CACHE_PATH,
+ // 缂撳瓨鍓嶇紑
+ 'prefix' => '',
+ // 缂撳瓨鏈夋晥鏈 0琛ㄧず姘镐箙缂撳瓨
+ 'expire' => 0,
+ ],
+
+ // +----------------------------------------------------------------------
+ // | 浼氳瘽璁剧疆
+ // +----------------------------------------------------------------------
+
+ 'session' => [
+ 'id' => '',
+ // SESSION_ID鐨勬彁浜ゅ彉閲,瑙e喅flash涓婁紶璺ㄥ煙
+ 'var_session_id' => '',
+ // SESSION 鍓嶇紑
+ 'prefix' => 'think',
+ // 椹卞姩鏂瑰紡 鏀寔redis memcache memcached
+ 'type' => '',
+ // 鏄惁鑷姩寮鍚 SESSION
+ 'auto_start' => true,
+ 'httponly' => true,
+ 'secure' => true,
+ ],
+
+ // +----------------------------------------------------------------------
+ // | Cookie璁剧疆
+ // +----------------------------------------------------------------------
+ 'cookie' => [
+ // cookie 鍚嶇О鍓嶇紑
+ 'prefix' => '',
+ // cookie 淇濆瓨鏃堕棿
+ 'expire' => 0,
+ // cookie 淇濆瓨璺緞
+ 'path' => '/',
+ // cookie 鏈夋晥鍩熷悕
+ 'domain' => '',
+ // cookie 鍚敤瀹夊叏浼犺緭
+ 'secure' => false,
+ // httponly璁剧疆
+ 'httponly' => '',
+ // 鏄惁浣跨敤 setcookie
+ 'setcookie' => true,
+ ],
+
+ // +----------------------------------------------------------------------
+ // | 鏁版嵁搴撹缃
+ // +----------------------------------------------------------------------
+
+ 'database' => [
+ // 鏁版嵁搴撶被鍨
+ 'type' => 'mysql',
+ // 鏁版嵁搴撹繛鎺SN閰嶇疆
+ 'dsn' => '',
+ // 鏈嶅姟鍣ㄥ湴鍧
+ 'hostname' => '127.0.0.1',
+ // 鏁版嵁搴撳悕
+ 'database' => '',
+ // 鏁版嵁搴撶敤鎴峰悕
+ 'username' => 'root',
+ // 鏁版嵁搴撳瘑鐮
+ 'password' => '',
+ // 鏁版嵁搴撹繛鎺ョ鍙
+ 'hostport' => '',
+ // 鏁版嵁搴撹繛鎺ュ弬鏁
+ 'params' => [],
+ // 鏁版嵁搴撶紪鐮侀粯璁ら噰鐢╱tf8
+ 'charset' => 'utf8',
+ // 鏁版嵁搴撹〃鍓嶇紑
+ 'prefix' => '',
+ // 鏁版嵁搴撹皟璇曟ā寮
+ 'debug' => false,
+ // 鏁版嵁搴撻儴缃叉柟寮:0 闆嗕腑寮(鍗曚竴鏈嶅姟鍣),1 鍒嗗竷寮(涓讳粠鏈嶅姟鍣)
+ 'deploy' => 0,
+ // 鏁版嵁搴撹鍐欐槸鍚﹀垎绂 涓讳粠寮忔湁鏁
+ 'rw_separate' => false,
+ // 璇诲啓鍒嗙鍚 涓绘湇鍔″櫒鏁伴噺
+ 'master_num' => 1,
+ // 鎸囧畾浠庢湇鍔″櫒搴忓彿
+ 'slave_no' => '',
+ // 鏄惁涓ユ牸妫鏌ュ瓧娈垫槸鍚﹀瓨鍦
+ 'fields_strict' => true,
+ // 鏁版嵁闆嗚繑鍥炵被鍨
+ 'resultset_type' => 'array',
+ // 鑷姩鍐欏叆鏃堕棿鎴冲瓧娈
+ 'auto_timestamp' => false,
+ // 鏃堕棿瀛楁鍙栧嚭鍚庣殑榛樿鏃堕棿鏍煎紡
+ 'datetime_format' => 'Y-m-d H:i:s',
+ // 鏄惁闇瑕佽繘琛孲QL鎬ц兘鍒嗘瀽
+ 'sql_explain' => false,
+ // Builder绫
+ 'builder' => '',
+ // Query绫
+ 'query' => '\\think\\db\\Query',
+ ],
+
+ //鍒嗛〉閰嶇疆
+ 'paginate' => [
+ 'type' => 'bootstrap',
+ 'var_page' => 'page',
+ 'list_rows' => 15,
+ ],
+
+];
diff --git a/thinkphp/helper.php b/thinkphp/helper.php
new file mode 100644
index 000000000..543942bef
--- /dev/null
+++ b/thinkphp/helper.php
@@ -0,0 +1,585 @@
+
+// +----------------------------------------------------------------------
+
+//------------------------
+// ThinkPHP 鍔╂墜鍑芥暟
+//-------------------------
+
+use think\Cache;
+use think\Config;
+use think\Cookie;
+use think\Db;
+use think\Debug;
+use think\exception\HttpException;
+use think\exception\HttpResponseException;
+use think\Lang;
+use think\Loader;
+use think\Log;
+use think\Model;
+use think\Request;
+use think\Response;
+use think\Session;
+use think\Url;
+use think\View;
+
+if (!function_exists('load_trait')) {
+ /**
+ * 蹇熷鍏raits PHP5.5浠ヤ笂鏃犻渶璋冪敤
+ * @param string $class trait搴
+ * @param string $ext 绫诲簱鍚庣紑
+ * @return boolean
+ */
+ function load_trait($class, $ext = EXT)
+ {
+ return Loader::import($class, TRAIT_PATH, $ext);
+ }
+}
+
+if (!function_exists('exception')) {
+ /**
+ * 鎶涘嚭寮傚父澶勭悊
+ *
+ * @param string $msg 寮傚父娑堟伅
+ * @param integer $code 寮傚父浠g爜 榛樿涓0
+ * @param string $exception 寮傚父绫
+ *
+ * @throws Exception
+ */
+ function exception($msg, $code = 0, $exception = '')
+ {
+ $e = $exception ?: '\think\Exception';
+ throw new $e($msg, $code);
+ }
+}
+
+if (!function_exists('debug')) {
+ /**
+ * 璁板綍鏃堕棿锛堝井绉掞級鍜屽唴瀛樹娇鐢ㄦ儏鍐
+ * @param string $start 寮濮嬫爣绛
+ * @param string $end 缁撴潫鏍囩
+ * @param integer|string $dec 灏忔暟浣 濡傛灉鏄痬 琛ㄧず缁熻鍐呭瓨鍗犵敤
+ * @return mixed
+ */
+ function debug($start, $end = '', $dec = 6)
+ {
+ if ('' == $end) {
+ Debug::remark($start);
+ } else {
+ return 'm' == $dec ? Debug::getRangeMem($start, $end) : Debug::getRangeTime($start, $end, $dec);
+ }
+ }
+}
+
+if (!function_exists('lang')) {
+ /**
+ * 鑾峰彇璇█鍙橀噺鍊
+ * @param string $name 璇█鍙橀噺鍚
+ * @param array $vars 鍔ㄦ佸彉閲忓
+ * @param string $lang 璇█
+ * @return mixed
+ */
+ function lang($name, $vars = [], $lang = '')
+ {
+ return Lang::get($name, $vars, $lang);
+ }
+}
+
+if (!function_exists('config')) {
+ /**
+ * 鑾峰彇鍜岃缃厤缃弬鏁
+ * @param string|array $name 鍙傛暟鍚
+ * @param mixed $value 鍙傛暟鍊
+ * @param string $range 浣滅敤鍩
+ * @return mixed
+ */
+ function config($name = '', $value = null, $range = '')
+ {
+ if (is_null($value) && is_string($name)) {
+ return 0 === strpos($name, '?') ? Config::has(substr($name, 1), $range) : Config::get($name, $range);
+ } else {
+ return Config::set($name, $value, $range);
+ }
+ }
+}
+
+if (!function_exists('input')) {
+ /**
+ * 鑾峰彇杈撳叆鏁版嵁 鏀寔榛樿鍊煎拰杩囨护
+ * @param string $key 鑾峰彇鐨勫彉閲忓悕
+ * @param mixed $default 榛樿鍊
+ * @param string $filter 杩囨护鏂规硶
+ * @return mixed
+ */
+ function input($key = '', $default = null, $filter = '')
+ {
+ if (0 === strpos($key, '?')) {
+ $key = substr($key, 1);
+ $has = true;
+ }
+ if ($pos = strpos($key, '.')) {
+ // 鎸囧畾鍙傛暟鏉ユ簮
+ list($method, $key) = explode('.', $key, 2);
+ if (!in_array($method, ['get', 'post', 'put', 'patch', 'delete', 'param', 'request', 'session', 'cookie', 'server', 'env', 'path', 'file'])) {
+ $key = $method . '.' . $key;
+ $method = 'param';
+ }
+ } else {
+ // 榛樿涓鸿嚜鍔ㄥ垽鏂
+ $method = 'param';
+ }
+ if (isset($has)) {
+ return request()->has($key, $method, $default);
+ } else {
+ return request()->$method($key, $default, $filter);
+ }
+ }
+}
+
+if (!function_exists('widget')) {
+ /**
+ * 娓叉煋杈撳嚭Widget
+ * @param string $name Widget鍚嶇О
+ * @param array $data 浼犲叆鐨勫弬鏁
+ * @return mixed
+ */
+ function widget($name, $data = [])
+ {
+ return Loader::action($name, $data, 'widget');
+ }
+}
+
+if (!function_exists('model')) {
+ /**
+ * 瀹炰緥鍖朚odel
+ * @param string $name Model鍚嶇О
+ * @param string $layer 涓氬姟灞傚悕绉
+ * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑
+ * @return \think\Model
+ */
+ function model($name = '', $layer = 'model', $appendSuffix = false)
+ {
+ return Loader::model($name, $layer, $appendSuffix);
+ }
+}
+
+if (!function_exists('validate')) {
+ /**
+ * 瀹炰緥鍖栭獙璇佸櫒
+ * @param string $name 楠岃瘉鍣ㄥ悕绉
+ * @param string $layer 涓氬姟灞傚悕绉
+ * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑
+ * @return \think\Validate
+ */
+ function validate($name = '', $layer = 'validate', $appendSuffix = false)
+ {
+ return Loader::validate($name, $layer, $appendSuffix);
+ }
+}
+
+if (!function_exists('db')) {
+ /**
+ * 瀹炰緥鍖栨暟鎹簱绫
+ * @param string $name 鎿嶄綔鐨勬暟鎹〃鍚嶇О锛堜笉鍚墠缂锛
+ * @param array|string $config 鏁版嵁搴撻厤缃弬鏁
+ * @param bool $force 鏄惁寮哄埗閲嶆柊杩炴帴
+ * @return \think\db\Query
+ */
+ function db($name = '', $config = [], $force = true)
+ {
+ return Db::connect($config, $force)->name($name);
+ }
+}
+
+if (!function_exists('controller')) {
+ /**
+ * 瀹炰緥鍖栨帶鍒跺櫒 鏍煎紡锛歔妯″潡/]鎺у埗鍣
+ * @param string $name 璧勬簮鍦板潃
+ * @param string $layer 鎺у埗灞傚悕绉
+ * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑
+ * @return \think\Controller
+ */
+ function controller($name, $layer = 'controller', $appendSuffix = false)
+ {
+ return Loader::controller($name, $layer, $appendSuffix);
+ }
+}
+
+if (!function_exists('action')) {
+ /**
+ * 璋冪敤妯″潡鐨勬搷浣滄柟娉 鍙傛暟鏍煎紡 [妯″潡/鎺у埗鍣/]鎿嶄綔
+ * @param string $url 璋冪敤鍦板潃
+ * @param string|array $vars 璋冪敤鍙傛暟 鏀寔瀛楃涓插拰鏁扮粍
+ * @param string $layer 瑕佽皟鐢ㄧ殑鎺у埗灞傚悕绉
+ * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑
+ * @return mixed
+ */
+ function action($url, $vars = [], $layer = 'controller', $appendSuffix = false)
+ {
+ return Loader::action($url, $vars, $layer, $appendSuffix);
+ }
+}
+
+if (!function_exists('import')) {
+ /**
+ * 瀵煎叆鎵闇鐨勭被搴 鍚宩ava鐨処mport 鏈嚱鏁版湁缂撳瓨鍔熻兘
+ * @param string $class 绫诲簱鍛藉悕绌洪棿瀛楃涓
+ * @param string $baseUrl 璧峰璺緞
+ * @param string $ext 瀵煎叆鐨勬枃浠舵墿灞曞悕
+ * @return boolean
+ */
+ function import($class, $baseUrl = '', $ext = EXT)
+ {
+ return Loader::import($class, $baseUrl, $ext);
+ }
+}
+
+if (!function_exists('vendor')) {
+ /**
+ * 蹇熷鍏ョ涓夋柟妗嗘灦绫诲簱 鎵鏈夌涓夋柟妗嗘灦鐨勭被搴撴枃浠剁粺涓鏀惧埌 绯荤粺鐨刅endor鐩綍涓嬮潰
+ * @param string $class 绫诲簱
+ * @param string $ext 绫诲簱鍚庣紑
+ * @return boolean
+ */
+ function vendor($class, $ext = EXT)
+ {
+ return Loader::import($class, VENDOR_PATH, $ext);
+ }
+}
+
+if (!function_exists('dump')) {
+ /**
+ * 娴忚鍣ㄥ弸濂界殑鍙橀噺杈撳嚭
+ * @param mixed $var 鍙橀噺
+ * @param boolean $echo 鏄惁杈撳嚭 榛樿涓簍rue 濡傛灉涓篺alse 鍒欒繑鍥炶緭鍑哄瓧绗︿覆
+ * @param string $label 鏍囩 榛樿涓虹┖
+ * @return void|string
+ */
+ function dump($var, $echo = true, $label = null)
+ {
+ return Debug::dump($var, $echo, $label);
+ }
+}
+
+if (!function_exists('url')) {
+ /**
+ * Url鐢熸垚
+ * @param string $url 璺敱鍦板潃
+ * @param string|array $vars 鍙橀噺
+ * @param bool|string $suffix 鐢熸垚鐨刄RL鍚庣紑
+ * @param bool|string $domain 鍩熷悕
+ * @return string
+ */
+ function url($url = '', $vars = '', $suffix = true, $domain = false)
+ {
+ return Url::build($url, $vars, $suffix, $domain);
+ }
+}
+
+if (!function_exists('session')) {
+ /**
+ * Session绠$悊
+ * @param string|array $name session鍚嶇О锛屽鏋滀负鏁扮粍琛ㄧず杩涜session璁剧疆
+ * @param mixed $value session鍊
+ * @param string $prefix 鍓嶇紑
+ * @return mixed
+ */
+ function session($name, $value = '', $prefix = null)
+ {
+ if (is_array($name)) {
+ // 鍒濆鍖
+ Session::init($name);
+ } elseif (is_null($name)) {
+ // 娓呴櫎
+ Session::clear('' === $value ? null : $value);
+ } elseif ('' === $value) {
+ // 鍒ゆ柇鎴栬幏鍙
+ return 0 === strpos($name, '?') ? Session::has(substr($name, 1), $prefix) : Session::get($name, $prefix);
+ } elseif (is_null($value)) {
+ // 鍒犻櫎
+ return Session::delete($name, $prefix);
+ } else {
+ // 璁剧疆
+ return Session::set($name, $value, $prefix);
+ }
+ }
+}
+
+if (!function_exists('cookie')) {
+ /**
+ * Cookie绠$悊
+ * @param string|array $name cookie鍚嶇О锛屽鏋滀负鏁扮粍琛ㄧず杩涜cookie璁剧疆
+ * @param mixed $value cookie鍊
+ * @param mixed $option 鍙傛暟
+ * @return mixed
+ */
+ function cookie($name, $value = '', $option = null)
+ {
+ if (is_array($name)) {
+ // 鍒濆鍖
+ Cookie::init($name);
+ } elseif (is_null($name)) {
+ // 娓呴櫎
+ Cookie::clear($value);
+ } elseif ('' === $value) {
+ // 鑾峰彇
+ return 0 === strpos($name, '?') ? Cookie::has(substr($name, 1), $option) : Cookie::get($name, $option);
+ } elseif (is_null($value)) {
+ // 鍒犻櫎
+ return Cookie::delete($name);
+ } else {
+ // 璁剧疆
+ return Cookie::set($name, $value, $option);
+ }
+ }
+}
+
+if (!function_exists('cache')) {
+ /**
+ * 缂撳瓨绠$悊
+ * @param mixed $name 缂撳瓨鍚嶇О锛屽鏋滀负鏁扮粍琛ㄧず杩涜缂撳瓨璁剧疆
+ * @param mixed $value 缂撳瓨鍊
+ * @param mixed $options 缂撳瓨鍙傛暟
+ * @param string $tag 缂撳瓨鏍囩
+ * @return mixed
+ */
+ function cache($name, $value = '', $options = null, $tag = null)
+ {
+ if (is_array($options)) {
+ // 缂撳瓨鎿嶄綔鐨勫悓鏃跺垵濮嬪寲
+ Cache::connect($options);
+ } elseif (is_array($name)) {
+ // 缂撳瓨鍒濆鍖
+ return Cache::connect($name);
+ }
+ if (is_null($name)) {
+ return Cache::clear($value);
+ } elseif ('' === $value) {
+ // 鑾峰彇缂撳瓨
+ return 0 === strpos($name, '?') ? Cache::has(substr($name, 1)) : Cache::get($name);
+ } elseif (is_null($value)) {
+ // 鍒犻櫎缂撳瓨
+ return Cache::rm($name);
+ } elseif (0 === strpos($name, '?') && '' !== $value) {
+ $expire = is_numeric($options) ? $options : null;
+ return Cache::remember(substr($name, 1), $value, $expire);
+ } else {
+ // 缂撳瓨鏁版嵁
+ if (is_array($options)) {
+ $expire = isset($options['expire']) ? $options['expire'] : null; //淇鏌ヨ缂撳瓨鏃犳硶璁剧疆杩囨湡鏃堕棿
+ } else {
+ $expire = is_numeric($options) ? $options : null; //榛樿蹇嵎缂撳瓨璁剧疆杩囨湡鏃堕棿
+ }
+ if (is_null($tag)) {
+ return Cache::set($name, $value, $expire);
+ } else {
+ return Cache::tag($tag)->set($name, $value, $expire);
+ }
+ }
+ }
+}
+
+if (!function_exists('trace')) {
+ /**
+ * 璁板綍鏃ュ織淇℃伅
+ * @param mixed $log log淇℃伅 鏀寔瀛楃涓插拰鏁扮粍
+ * @param string $level 鏃ュ織绾у埆
+ * @return void|array
+ */
+ function trace($log = '[think]', $level = 'log')
+ {
+ if ('[think]' === $log) {
+ return Log::getLog();
+ } else {
+ Log::record($log, $level);
+ }
+ }
+}
+
+if (!function_exists('request')) {
+ /**
+ * 鑾峰彇褰撳墠Request瀵硅薄瀹炰緥
+ * @return Request
+ */
+ function request()
+ {
+ return Request::instance();
+ }
+}
+
+if (!function_exists('response')) {
+ /**
+ * 鍒涘缓鏅 Response 瀵硅薄瀹炰緥
+ * @param mixed $data 杈撳嚭鏁版嵁
+ * @param int|string $code 鐘舵佺爜
+ * @param array $header 澶翠俊鎭
+ * @param string $type
+ * @return Response
+ */
+ function response($data = [], $code = 200, $header = [], $type = 'html')
+ {
+ return Response::create($data, $type, $code, $header);
+ }
+}
+
+if (!function_exists('view')) {
+ /**
+ * 娓叉煋妯℃澘杈撳嚭
+ * @param string $template 妯℃澘鏂囦欢
+ * @param array $vars 妯℃澘鍙橀噺
+ * @param array $replace 妯℃澘鏇挎崲
+ * @param integer $code 鐘舵佺爜
+ * @return \think\response\View
+ */
+ function view($template = '', $vars = [], $replace = [], $code = 200)
+ {
+ return Response::create($template, 'view', $code)->replace($replace)->assign($vars);
+ }
+}
+
+if (!function_exists('json')) {
+ /**
+ * 鑾峰彇\think\response\Json瀵硅薄瀹炰緥
+ * @param mixed $data 杩斿洖鐨勬暟鎹
+ * @param integer $code 鐘舵佺爜
+ * @param array $header 澶撮儴
+ * @param array $options 鍙傛暟
+ * @return \think\response\Json
+ */
+ function json($data = [], $code = 200, $header = [], $options = [])
+ {
+ return Response::create($data, 'json', $code, $header, $options);
+ }
+}
+
+if (!function_exists('jsonp')) {
+ /**
+ * 鑾峰彇\think\response\Jsonp瀵硅薄瀹炰緥
+ * @param mixed $data 杩斿洖鐨勬暟鎹
+ * @param integer $code 鐘舵佺爜
+ * @param array $header 澶撮儴
+ * @param array $options 鍙傛暟
+ * @return \think\response\Jsonp
+ */
+ function jsonp($data = [], $code = 200, $header = [], $options = [])
+ {
+ return Response::create($data, 'jsonp', $code, $header, $options);
+ }
+}
+
+if (!function_exists('xml')) {
+ /**
+ * 鑾峰彇\think\response\Xml瀵硅薄瀹炰緥
+ * @param mixed $data 杩斿洖鐨勬暟鎹
+ * @param integer $code 鐘舵佺爜
+ * @param array $header 澶撮儴
+ * @param array $options 鍙傛暟
+ * @return \think\response\Xml
+ */
+ function xml($data = [], $code = 200, $header = [], $options = [])
+ {
+ return Response::create($data, 'xml', $code, $header, $options);
+ }
+}
+
+if (!function_exists('redirect')) {
+ /**
+ * 鑾峰彇\think\response\Redirect瀵硅薄瀹炰緥
+ * @param mixed $url 閲嶅畾鍚戝湴鍧 鏀寔Url::build鏂规硶鐨勫湴鍧
+ * @param array|integer $params 棰濆鍙傛暟
+ * @param integer $code 鐘舵佺爜
+ * @return \think\response\Redirect
+ */
+ function redirect($url = [], $params = [], $code = 302)
+ {
+ if (is_integer($params)) {
+ $code = $params;
+ $params = [];
+ }
+ return Response::create($url, 'redirect', $code)->params($params);
+ }
+}
+
+if (!function_exists('abort')) {
+ /**
+ * 鎶涘嚭HTTP寮傚父
+ * @param integer|Response $code 鐘舵佺爜 鎴栬 Response瀵硅薄瀹炰緥
+ * @param string $message 閿欒淇℃伅
+ * @param array $header 鍙傛暟
+ */
+ function abort($code, $message = null, $header = [])
+ {
+ if ($code instanceof Response) {
+ throw new HttpResponseException($code);
+ } else {
+ throw new HttpException($code, $message, null, $header);
+ }
+ }
+}
+
+if (!function_exists('halt')) {
+ /**
+ * 璋冭瘯鍙橀噺骞朵笖涓柇杈撳嚭
+ * @param mixed $var 璋冭瘯鍙橀噺鎴栬呬俊鎭
+ */
+ function halt($var)
+ {
+ dump($var);
+ throw new HttpResponseException(new Response);
+ }
+}
+
+if (!function_exists('token')) {
+ /**
+ * 鐢熸垚琛ㄥ崟浠ょ墝
+ * @param string $name 浠ょ墝鍚嶇О
+ * @param mixed $type 浠ょ墝鐢熸垚鏂规硶
+ * @return string
+ */
+ function token($name = '__token__', $type = 'md5')
+ {
+ $token = Request::instance()->token($name, $type);
+ return '';
+ }
+}
+
+if (!function_exists('load_relation')) {
+ /**
+ * 寤惰繜棰勮浇鍏ュ叧鑱旀煡璇
+ * @param mixed $resultSet 鏁版嵁闆
+ * @param mixed $relation 鍏宠仈
+ * @return array
+ */
+ function load_relation($resultSet, $relation)
+ {
+ $item = current($resultSet);
+ if ($item instanceof Model) {
+ $item->eagerlyResultSet($resultSet, $relation);
+ }
+ return $resultSet;
+ }
+}
+
+if (!function_exists('collection')) {
+ /**
+ * 鏁扮粍杞崲涓烘暟鎹泦瀵硅薄
+ * @param array $resultSet 鏁版嵁闆嗘暟缁
+ * @return \think\model\Collection|\think\Collection
+ */
+ function collection($resultSet)
+ {
+ $item = current($resultSet);
+ if ($item instanceof Model) {
+ return \think\model\Collection::make($resultSet);
+ } else {
+ return \think\Collection::make($resultSet);
+ }
+ }
+}
diff --git a/thinkphp/lang/zh-cn.php b/thinkphp/lang/zh-cn.php
new file mode 100644
index 000000000..b837bc410
--- /dev/null
+++ b/thinkphp/lang/zh-cn.php
@@ -0,0 +1,67 @@
+
+// +----------------------------------------------------------------------
+
+// 鏍稿績涓枃璇█鍖
+return [
+ // 绯荤粺閿欒鎻愮ず
+ 'Undefined variable' => '鏈畾涔夊彉閲',
+ 'Undefined index' => '鏈畾涔夋暟缁勭储寮',
+ 'Undefined offset' => '鏈畾涔夋暟缁勪笅鏍',
+ 'Parse error' => '璇硶瑙f瀽閿欒',
+ 'Type error' => '绫诲瀷閿欒',
+ 'Fatal error' => '鑷村懡閿欒',
+ 'syntax error' => '璇硶閿欒',
+
+ // 妗嗘灦鏍稿績閿欒鎻愮ず
+ 'dispatch type not support' => '涓嶆敮鎸佺殑璋冨害绫诲瀷',
+ 'method param miss' => '鏂规硶鍙傛暟閿欒',
+ 'method not exists' => '鏂规硶涓嶅瓨鍦',
+ 'module not exists' => '妯″潡涓嶅瓨鍦',
+ 'controller not exists' => '鎺у埗鍣ㄤ笉瀛樺湪',
+ 'class not exists' => '绫讳笉瀛樺湪',
+ 'property not exists' => '绫荤殑灞炴т笉瀛樺湪',
+ 'template not exists' => '妯℃澘鏂囦欢涓嶅瓨鍦',
+ 'illegal controller name' => '闈炴硶鐨勬帶鍒跺櫒鍚嶇О',
+ 'illegal action name' => '闈炴硶鐨勬搷浣滃悕绉',
+ 'url suffix deny' => '绂佹鐨刄RL鍚庣紑璁块棶',
+ 'Route Not Found' => '褰撳墠璁块棶璺敱鏈畾涔',
+ 'Underfined db type' => '鏈畾涔夋暟鎹簱绫诲瀷',
+ 'variable type error' => '鍙橀噺绫诲瀷閿欒',
+ 'PSR-4 error' => 'PSR-4 瑙勮寖閿欒',
+ 'not support total' => '绠娲佹ā寮忎笅涓嶈兘鑾峰彇鏁版嵁鎬绘暟',
+ 'not support last' => '绠娲佹ā寮忎笅涓嶈兘鑾峰彇鏈鍚庝竴椤',
+ 'error session handler' => '閿欒鐨凷ESSION澶勭悊鍣ㄧ被',
+ 'not allow php tag' => '妯℃澘涓嶅厑璁镐娇鐢≒HP璇硶',
+ 'not support' => '涓嶆敮鎸',
+ 'redisd master' => 'Redisd 涓绘湇鍔″櫒閿欒',
+ 'redisd slave' => 'Redisd 浠庢湇鍔″櫒閿欒',
+ 'must run at sae' => '蹇呴』鍦⊿AE杩愯',
+ 'memcache init error' => '鏈紑閫歁emcache鏈嶅姟锛岃鍦⊿AE绠$悊骞冲彴鍒濆鍖朚emcache鏈嶅姟',
+ 'KVDB init error' => '娌℃湁鍒濆鍖朘VDB锛岃鍦⊿AE绠$悊骞冲彴鍒濆鍖朘VDB鏈嶅姟',
+ 'fields not exists' => '鏁版嵁琛ㄥ瓧娈典笉瀛樺湪',
+ 'where express error' => '鏌ヨ琛ㄨ揪寮忛敊璇',
+ 'no data to update' => '娌℃湁浠讳綍鏁版嵁闇瑕佹洿鏂',
+ 'miss data to insert' => '缂哄皯闇瑕佸啓鍏ョ殑鏁版嵁',
+ 'miss complex primary data' => '缂哄皯澶嶅悎涓婚敭鏁版嵁',
+ 'miss update condition' => '缂哄皯鏇存柊鏉′欢',
+ 'model data Not Found' => '妯″瀷鏁版嵁涓嶅瓨鍦',
+ 'table data not Found' => '琛ㄦ暟鎹笉瀛樺湪',
+ 'delete without condition' => '娌℃湁鏉′欢涓嶄細鎵ц鍒犻櫎鎿嶄綔',
+ 'miss relation data' => '缂哄皯鍏宠仈琛ㄦ暟鎹',
+ 'tag attr must' => '妯℃澘鏍囩灞炴у繀椤',
+ 'tag error' => '妯℃澘鏍囩閿欒',
+ 'cache write error' => '缂撳瓨鍐欏叆澶辫触',
+ 'sae mc write error' => 'SAE mc 鍐欏叆閿欒',
+ 'route name not exists' => '璺敱鏍囪瘑涓嶅瓨鍦紙鎴栧弬鏁颁笉澶燂級',
+ 'invalid request' => '闈炴硶璇锋眰',
+ 'bind attr has exists' => '妯″瀷鐨勫睘鎬у凡缁忓瓨鍦',
+ 'relation data not exists' => '鍏宠仈鏁版嵁涓嶅瓨鍦',
+];
diff --git a/thinkphp/library/think/App.php b/thinkphp/library/think/App.php
new file mode 100644
index 000000000..77721112d
--- /dev/null
+++ b/thinkphp/library/think/App.php
@@ -0,0 +1,567 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+use think\exception\HttpException;
+use think\exception\HttpResponseException;
+use think\exception\RouteNotFoundException;
+
+/**
+ * App 搴旂敤绠$悊
+ * @author liu21st
+ */
+class App
+{
+ /**
+ * @var bool 鏄惁鍒濆鍖栬繃
+ */
+ protected static $init = false;
+
+ /**
+ * @var string 褰撳墠妯″潡璺緞
+ */
+ public static $modulePath;
+
+ /**
+ * @var bool 搴旂敤璋冭瘯妯″紡
+ */
+ public static $debug = true;
+
+ /**
+ * @var string 搴旂敤绫诲簱鍛藉悕绌洪棿
+ */
+ public static $namespace = 'app';
+
+ /**
+ * @var bool 搴旂敤绫诲簱鍚庣紑
+ */
+ public static $suffix = false;
+
+ /**
+ * @var bool 搴旂敤璺敱妫娴
+ */
+ protected static $routeCheck;
+
+ /**
+ * @var bool 涓ユ牸璺敱妫娴
+ */
+ protected static $routeMust;
+
+ protected static $dispatch;
+ protected static $file = [];
+
+ /**
+ * 鎵ц搴旂敤绋嬪簭
+ * @access public
+ * @param Request $request Request瀵硅薄
+ * @return Response
+ * @throws Exception
+ */
+ public static function run(Request $request = null)
+ {
+ is_null($request) && $request = Request::instance();
+
+ try {
+ $config = self::initCommon();
+ if (defined('BIND_MODULE')) {
+ // 妯″潡/鎺у埗鍣ㄧ粦瀹
+ BIND_MODULE && Route::bind(BIND_MODULE);
+ } elseif ($config['auto_bind_module']) {
+ // 鍏ュ彛鑷姩缁戝畾
+ $name = pathinfo($request->baseFile(), PATHINFO_FILENAME);
+ if ($name && 'index' != $name && is_dir(APP_PATH . $name)) {
+ Route::bind($name);
+ }
+ }
+
+ $request->filter($config['default_filter']);
+
+ if ($config['lang_switch_on']) {
+ // 寮鍚璇█鏈哄埗 妫娴嬪綋鍓嶈瑷
+ Lang::detect();
+ } else {
+ // 璇诲彇榛樿璇█
+ Lang::range($config['default_lang']);
+ }
+ $request->langset(Lang::range());
+ // 鍔犺浇绯荤粺璇█鍖
+ Lang::load([
+ THINK_PATH . 'lang' . DS . $request->langset() . EXT,
+ APP_PATH . 'lang' . DS . $request->langset() . EXT,
+ ]);
+
+ // 鑾峰彇搴旂敤璋冨害淇℃伅
+ $dispatch = self::$dispatch;
+ if (empty($dispatch)) {
+ // 杩涜URL璺敱妫娴
+ $dispatch = self::routeCheck($request, $config);
+ }
+ // 璁板綍褰撳墠璋冨害淇℃伅
+ $request->dispatch($dispatch);
+
+ // 璁板綍璺敱鍜岃姹備俊鎭
+ if (self::$debug) {
+ Log::record('[ ROUTE ] ' . var_export($dispatch, true), 'info');
+ Log::record('[ HEADER ] ' . var_export($request->header(), true), 'info');
+ Log::record('[ PARAM ] ' . var_export($request->param(), true), 'info');
+ }
+
+ // 鐩戝惉app_begin
+ Hook::listen('app_begin', $dispatch);
+ // 璇锋眰缂撳瓨妫鏌
+ $request->cache($config['request_cache'], $config['request_cache_expire']);
+
+ switch ($dispatch['type']) {
+ case 'redirect':
+ // 鎵ц閲嶅畾鍚戣烦杞
+ $data = Response::create($dispatch['url'], 'redirect')->code($dispatch['status']);
+ break;
+ case 'module':
+ // 妯″潡/鎺у埗鍣/鎿嶄綔
+ $data = self::module($dispatch['module'], $config, isset($dispatch['convert']) ? $dispatch['convert'] : null);
+ break;
+ case 'controller':
+ // 鎵ц鎺у埗鍣ㄦ搷浣
+ $vars = array_merge(Request::instance()->param(), $dispatch['var']);
+ $data = Loader::action($dispatch['controller'], $vars, $config['url_controller_layer'], $config['controller_suffix']);
+ break;
+ case 'method':
+ // 鎵ц鍥炶皟鏂规硶
+ $vars = array_merge(Request::instance()->param(), $dispatch['var']);
+ $data = self::invokeMethod($dispatch['method'], $vars);
+ break;
+ case 'function':
+ // 鎵ц闂寘
+ $data = self::invokeFunction($dispatch['function']);
+ break;
+ case 'response':
+ $data = $dispatch['response'];
+ break;
+ default:
+ throw new \InvalidArgumentException('dispatch type not support');
+ }
+ } catch (HttpResponseException $exception) {
+ $data = $exception->getResponse();
+ }
+
+ // 娓呯┖绫荤殑瀹炰緥鍖
+ Loader::clearInstance();
+
+ // 杈撳嚭鏁版嵁鍒板鎴风
+ if ($data instanceof Response) {
+ $response = $data;
+ } elseif (!is_null($data)) {
+ // 榛樿鑷姩璇嗗埆鍝嶅簲杈撳嚭绫诲瀷
+ $isAjax = $request->isAjax();
+ $type = $isAjax ? Config::get('default_ajax_return') : Config::get('default_return_type');
+ $response = Response::create($data, $type);
+ } else {
+ $response = Response::create();
+ }
+
+ // 鐩戝惉app_end
+ Hook::listen('app_end', $response);
+
+ return $response;
+ }
+
+ /**
+ * 璁剧疆褰撳墠璇锋眰鐨勮皟搴︿俊鎭
+ * @access public
+ * @param array|string $dispatch 璋冨害淇℃伅
+ * @param string $type 璋冨害绫诲瀷
+ * @return void
+ */
+ public static function dispatch($dispatch, $type = 'module')
+ {
+ self::$dispatch = ['type' => $type, $type => $dispatch];
+ }
+
+ /**
+ * 鎵ц鍑芥暟鎴栬呴棴鍖呮柟娉 鏀寔鍙傛暟璋冪敤
+ * @access public
+ * @param string|array|\Closure $function 鍑芥暟鎴栬呴棴鍖
+ * @param array $vars 鍙橀噺
+ * @return mixed
+ */
+ public static function invokeFunction($function, $vars = [])
+ {
+ $reflect = new \ReflectionFunction($function);
+ $args = self::bindParams($reflect, $vars);
+ // 璁板綍鎵ц淇℃伅
+ self::$debug && Log::record('[ RUN ] ' . $reflect->__toString(), 'info');
+ return $reflect->invokeArgs($args);
+ }
+
+ /**
+ * 璋冪敤鍙嶅皠鎵ц绫荤殑鏂规硶 鏀寔鍙傛暟缁戝畾
+ * @access public
+ * @param string|array $method 鏂规硶
+ * @param array $vars 鍙橀噺
+ * @return mixed
+ */
+ public static function invokeMethod($method, $vars = [])
+ {
+ if (is_array($method)) {
+ $class = is_object($method[0]) ? $method[0] : self::invokeClass($method[0]);
+ $reflect = new \ReflectionMethod($class, $method[1]);
+ } else {
+ // 闈欐佹柟娉
+ $reflect = new \ReflectionMethod($method);
+ }
+ $args = self::bindParams($reflect, $vars);
+
+ self::$debug && Log::record('[ RUN ] ' . $reflect->class . '->' . $reflect->name . '[ ' . $reflect->getFileName() . ' ]', 'info');
+ return $reflect->invokeArgs(isset($class) ? $class : null, $args);
+ }
+
+ /**
+ * 璋冪敤鍙嶅皠鎵ц绫荤殑瀹炰緥鍖 鏀寔渚濊禆娉ㄥ叆
+ * @access public
+ * @param string $class 绫诲悕
+ * @param array $vars 鍙橀噺
+ * @return mixed
+ */
+ public static function invokeClass($class, $vars = [])
+ {
+ $reflect = new \ReflectionClass($class);
+ $constructor = $reflect->getConstructor();
+ if ($constructor) {
+ $args = self::bindParams($constructor, $vars);
+ } else {
+ $args = [];
+ }
+ return $reflect->newInstanceArgs($args);
+ }
+
+ /**
+ * 缁戝畾鍙傛暟
+ * @access public
+ * @param \ReflectionMethod|\ReflectionFunction $reflect 鍙嶅皠绫
+ * @param array $vars 鍙橀噺
+ * @return array
+ */
+ private static function bindParams($reflect, $vars = [])
+ {
+ if (empty($vars)) {
+ // 鑷姩鑾峰彇璇锋眰鍙橀噺
+ if (Config::get('url_param_type')) {
+ $vars = Request::instance()->route();
+ } else {
+ $vars = Request::instance()->param();
+ }
+ }
+ $args = [];
+ // 鍒ゆ柇鏁扮粍绫诲瀷 鏁板瓧鏁扮粍鏃舵寜椤哄簭缁戝畾鍙傛暟
+ reset($vars);
+ $type = key($vars) === 0 ? 1 : 0;
+ if ($reflect->getNumberOfParameters() > 0) {
+ $params = $reflect->getParameters();
+ foreach ($params as $param) {
+ $name = $param->getName();
+ $class = $param->getClass();
+ if ($class) {
+ $className = $class->getName();
+ $bind = Request::instance()->$name;
+ if ($bind instanceof $className) {
+ $args[] = $bind;
+ } else {
+ if (method_exists($className, 'invoke')) {
+ $method = new \ReflectionMethod($className, 'invoke');
+ if ($method->isPublic() && $method->isStatic()) {
+ $args[] = $className::invoke(Request::instance());
+ continue;
+ }
+ }
+ $args[] = method_exists($className, 'instance') ? $className::instance() : new $className;
+ }
+ } elseif (1 == $type && !empty($vars)) {
+ $args[] = array_shift($vars);
+ } elseif (0 == $type && isset($vars[$name])) {
+ $args[] = $vars[$name];
+ } elseif ($param->isDefaultValueAvailable()) {
+ $args[] = $param->getDefaultValue();
+ } else {
+ throw new \InvalidArgumentException('method param miss:' . $name);
+ }
+ }
+ }
+ return $args;
+ }
+
+ /**
+ * 鎵ц妯″潡
+ * @access public
+ * @param array $result 妯″潡/鎺у埗鍣/鎿嶄綔
+ * @param array $config 閰嶇疆鍙傛暟
+ * @param bool $convert 鏄惁鑷姩杞崲鎺у埗鍣ㄥ拰鎿嶄綔鍚
+ * @return mixed
+ */
+ public static function module($result, $config, $convert = null)
+ {
+ if (is_string($result)) {
+ $result = explode('/', $result);
+ }
+ $request = Request::instance();
+ if ($config['app_multi_module']) {
+ // 澶氭ā鍧楅儴缃
+ $module = strip_tags(strtolower($result[0] ?: $config['default_module']));
+ $bind = Route::getBind('module');
+ $available = false;
+ if ($bind) {
+ // 缁戝畾妯″潡
+ list($bindModule) = explode('/', $bind);
+ if (empty($result[0])) {
+ $module = $bindModule;
+ $available = true;
+ } elseif ($module == $bindModule) {
+ $available = true;
+ }
+ } elseif (!in_array($module, $config['deny_module_list']) && is_dir(APP_PATH . $module)) {
+ $available = true;
+ }
+
+ // 妯″潡鍒濆鍖
+ if ($module && $available) {
+ // 鍒濆鍖栨ā鍧
+ $request->module($module);
+ $config = self::init($module);
+ // 妯″潡璇锋眰缂撳瓨妫鏌
+ $request->cache($config['request_cache'], $config['request_cache_expire']);
+ } else {
+ throw new HttpException(404, 'module not exists:' . $module);
+ }
+ } else {
+ // 鍗曚竴妯″潡閮ㄧ讲
+ $module = '';
+ $request->module($module);
+ }
+ // 褰撳墠妯″潡璺緞
+ App::$modulePath = APP_PATH . ($module ? $module . DS : '');
+
+ // 鏄惁鑷姩杞崲鎺у埗鍣ㄥ拰鎿嶄綔鍚
+ $convert = is_bool($convert) ? $convert : $config['url_convert'];
+ // 鑾峰彇鎺у埗鍣ㄥ悕
+ $controller = strip_tags($result[1] ?: $config['default_controller']);
+ $controller = $convert ? strtolower($controller) : $controller;
+
+ // 鑾峰彇鎿嶄綔鍚
+ $actionName = strip_tags($result[2] ?: $config['default_action']);
+ $actionName = $convert ? strtolower($actionName) : $actionName;
+
+ // 璁剧疆褰撳墠璇锋眰鐨勬帶鍒跺櫒銆佹搷浣
+ $request->controller(Loader::parseName($controller, 1))->action($actionName);
+
+ // 鐩戝惉module_init
+ Hook::listen('module_init', $request);
+
+ $instance = Loader::controller($controller, $config['url_controller_layer'], $config['controller_suffix'], $config['empty_controller']);
+ if (is_null($instance)) {
+ throw new HttpException(404, 'controller not exists:' . Loader::parseName($controller, 1));
+ }
+ // 鑾峰彇褰撳墠鎿嶄綔鍚
+ $action = $actionName . $config['action_suffix'];
+
+ $vars = [];
+ if (is_callable([$instance, $action])) {
+ // 鎵ц鎿嶄綔鏂规硶
+ $call = [$instance, $action];
+ } elseif (is_callable([$instance, '_empty'])) {
+ // 绌烘搷浣
+ $call = [$instance, '_empty'];
+ $vars = [$actionName];
+ } else {
+ // 鎿嶄綔涓嶅瓨鍦
+ throw new HttpException(404, 'method not exists:' . get_class($instance) . '->' . $action . '()');
+ }
+
+ Hook::listen('action_begin', $call);
+
+ return self::invokeMethod($call, $vars);
+ }
+
+ /**
+ * 鍒濆鍖栧簲鐢
+ */
+ public static function initCommon()
+ {
+ if (empty(self::$init)) {
+ // 鍒濆鍖栧簲鐢
+ $config = self::init();
+ self::$suffix = $config['class_suffix'];
+
+ // 搴旂敤璋冭瘯妯″紡
+ self::$debug = Env::get('app_debug', Config::get('app_debug'));
+ if (!self::$debug) {
+ ini_set('display_errors', 'Off');
+ } elseif (!IS_CLI) {
+ //閲嶆柊鐢宠涓鍧楁瘮杈冨ぇ鐨刡uffer
+ if (ob_get_level() > 0) {
+ $output = ob_get_clean();
+ }
+ ob_start();
+ if (!empty($output)) {
+ echo $output;
+ }
+ }
+
+ // 娉ㄥ唽搴旂敤鍛藉悕绌洪棿
+ self::$namespace = $config['app_namespace'];
+ Loader::addNamespace($config['app_namespace'], APP_PATH);
+ if (!empty($config['root_namespace'])) {
+ Loader::addNamespace($config['root_namespace']);
+ }
+
+ // 鍔犺浇棰濆鏂囦欢
+ if (!empty($config['extra_file_list'])) {
+ foreach ($config['extra_file_list'] as $file) {
+ $file = strpos($file, '.') ? $file : APP_PATH . $file . EXT;
+ if (is_file($file) && !isset(self::$file[$file])) {
+ include $file;
+ self::$file[$file] = true;
+ }
+ }
+ }
+
+ // 璁剧疆绯荤粺鏃跺尯
+ date_default_timezone_set($config['default_timezone']);
+
+ // 鐩戝惉app_init
+ Hook::listen('app_init');
+
+ self::$init = true;
+ }
+ return Config::get();
+ }
+
+ /**
+ * 鍒濆鍖栧簲鐢ㄦ垨妯″潡
+ * @access public
+ * @param string $module 妯″潡鍚
+ * @return array
+ */
+ private static function init($module = '')
+ {
+ // 瀹氫綅妯″潡鐩綍
+ $module = $module ? $module . DS : '';
+
+ // 鍔犺浇鍒濆鍖栨枃浠
+ if (is_file(APP_PATH . $module . 'init' . EXT)) {
+ include APP_PATH . $module . 'init' . EXT;
+ } elseif (is_file(RUNTIME_PATH . $module . 'init' . EXT)) {
+ include RUNTIME_PATH . $module . 'init' . EXT;
+ } else {
+ $path = APP_PATH . $module;
+ // 鍔犺浇妯″潡閰嶇疆
+ $config = Config::load(CONF_PATH . $module . 'config' . CONF_EXT);
+ // 璇诲彇鏁版嵁搴撻厤缃枃浠
+ $filename = CONF_PATH . $module . 'database' . CONF_EXT;
+ Config::load($filename, 'database');
+ // 璇诲彇鎵╁睍閰嶇疆鏂囦欢
+ if (is_dir(CONF_PATH . $module . 'extra')) {
+ $dir = CONF_PATH . $module . 'extra';
+ $files = scandir($dir);
+ foreach ($files as $file) {
+ if (strpos($file, CONF_EXT)) {
+ $filename = $dir . DS . $file;
+ Config::load($filename, pathinfo($file, PATHINFO_FILENAME));
+ }
+ }
+ }
+
+ // 鍔犺浇搴旂敤鐘舵侀厤缃
+ if ($config['app_status']) {
+ $config = Config::load(CONF_PATH . $module . $config['app_status'] . CONF_EXT);
+ }
+
+ // 鍔犺浇琛屼负鎵╁睍鏂囦欢
+ if (is_file(CONF_PATH . $module . 'tags' . EXT)) {
+ Hook::import(include CONF_PATH . $module . 'tags' . EXT);
+ }
+
+ // 鍔犺浇鍏叡鏂囦欢
+ if (is_file($path . 'common' . EXT)) {
+ include $path . 'common' . EXT;
+ }
+
+ // 鍔犺浇褰撳墠妯″潡璇█鍖
+ if ($module) {
+ Lang::load($path . 'lang' . DS . Request::instance()->langset() . EXT);
+ }
+ }
+ return Config::get();
+ }
+
+ /**
+ * URL璺敱妫娴嬶紙鏍规嵁PATH_INFO)
+ * @access public
+ * @param \think\Request $request
+ * @param array $config
+ * @return array
+ * @throws \think\Exception
+ */
+ public static function routeCheck($request, array $config)
+ {
+ $path = $request->path();
+ $depr = $config['pathinfo_depr'];
+ $result = false;
+ // 璺敱妫娴
+ $check = !is_null(self::$routeCheck) ? self::$routeCheck : $config['url_route_on'];
+ if ($check) {
+ // 寮鍚矾鐢
+ if (is_file(RUNTIME_PATH . 'route.php')) {
+ // 璇诲彇璺敱缂撳瓨
+ $rules = include RUNTIME_PATH . 'route.php';
+ if (is_array($rules)) {
+ Route::rules($rules);
+ }
+ } else {
+ $files = $config['route_config_file'];
+ foreach ($files as $file) {
+ if (is_file(CONF_PATH . $file . CONF_EXT)) {
+ // 瀵煎叆璺敱閰嶇疆
+ $rules = include CONF_PATH . $file . CONF_EXT;
+ if (is_array($rules)) {
+ Route::import($rules);
+ }
+ }
+ }
+ }
+
+ // 璺敱妫娴嬶紙鏍规嵁璺敱瀹氫箟杩斿洖涓嶅悓鐨刄RL璋冨害锛
+ $result = Route::check($request, $path, $depr, $config['url_domain_deploy']);
+ $must = !is_null(self::$routeMust) ? self::$routeMust : $config['url_route_must'];
+ if ($must && false === $result) {
+ // 璺敱鏃犳晥
+ throw new RouteNotFoundException();
+ }
+ }
+ if (false === $result) {
+ // 璺敱鏃犳晥 瑙f瀽妯″潡/鎺у埗鍣/鎿嶄綔/鍙傛暟... 鏀寔鎺у埗鍣ㄨ嚜鍔ㄦ悳绱
+ $result = Route::parseUrl($path, $depr, $config['controller_auto_search']);
+ }
+ return $result;
+ }
+
+ /**
+ * 璁剧疆搴旂敤鐨勮矾鐢辨娴嬫満鍒
+ * @access public
+ * @param bool $route 鏄惁闇瑕佹娴嬭矾鐢
+ * @param bool $must 鏄惁寮哄埗妫娴嬭矾鐢
+ * @return void
+ */
+ public static function route($route, $must = false)
+ {
+ self::$routeCheck = $route;
+ self::$routeMust = $must;
+ }
+}
diff --git a/thinkphp/library/think/Build.php b/thinkphp/library/think/Build.php
new file mode 100644
index 000000000..13b7bfdc0
--- /dev/null
+++ b/thinkphp/library/think/Build.php
@@ -0,0 +1,204 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+class Build
+{
+ /**
+ * 鏍规嵁浼犲叆鐨刡uild璧勬枡鍒涘缓鐩綍鍜屾枃浠
+ * @access protected
+ * @param array $build build鍒楄〃
+ * @param string $namespace 搴旂敤绫诲簱鍛藉悕绌洪棿
+ * @param bool $suffix 绫诲簱鍚庣紑
+ * @return void
+ */
+ public static function run(array $build = [], $namespace = 'app', $suffix = false)
+ {
+ // 閿佸畾
+ $lockfile = APP_PATH . 'build.lock';
+ if (is_writable($lockfile)) {
+ return;
+ } elseif (!touch($lockfile)) {
+ throw new Exception('搴旂敤鐩綍[' . APP_PATH . ']涓嶅彲鍐欙紝鐩綍鏃犳硶鑷姩鐢熸垚锛
璇锋墜鍔ㄧ敓鎴愰」鐩洰褰晘', 10006);
+ }
+ foreach ($build as $module => $list) {
+ if ('__dir__' == $module) {
+ // 鍒涘缓鐩綍鍒楄〃
+ self::buildDir($list);
+ } elseif ('__file__' == $module) {
+ // 鍒涘缓鏂囦欢鍒楄〃
+ self::buildFile($list);
+ } else {
+ // 鍒涘缓妯″潡
+ self::module($module, $list, $namespace, $suffix);
+ }
+ }
+ // 瑙i櫎閿佸畾
+ unlink($lockfile);
+ }
+
+ /**
+ * 鍒涘缓鐩綍
+ * @access protected
+ * @param array $list 鐩綍鍒楄〃
+ * @return void
+ */
+ protected static function buildDir($list)
+ {
+ foreach ($list as $dir) {
+ if (!is_dir(APP_PATH . $dir)) {
+ // 鍒涘缓鐩綍
+ mkdir(APP_PATH . $dir, 0755, true);
+ }
+ }
+ }
+
+ /**
+ * 鍒涘缓鏂囦欢
+ * @access protected
+ * @param array $list 鏂囦欢鍒楄〃
+ * @return void
+ */
+ protected static function buildFile($list)
+ {
+ foreach ($list as $file) {
+ if (!is_dir(APP_PATH . dirname($file))) {
+ // 鍒涘缓鐩綍
+ mkdir(APP_PATH . dirname($file), 0755, true);
+ }
+ if (!is_file(APP_PATH . $file)) {
+ file_put_contents(APP_PATH . $file, 'php' == pathinfo($file, PATHINFO_EXTENSION) ? " ['config.php', 'common.php'],
+ '__dir__' => ['controller', 'model', 'view'],
+ ];
+ }
+ // 鍒涘缓瀛愮洰褰曞拰鏂囦欢
+ foreach ($list as $path => $file) {
+ $modulePath = APP_PATH . $module . DS;
+ if ('__dir__' == $path) {
+ // 鐢熸垚瀛愮洰褰
+ foreach ($file as $dir) {
+ if (!is_dir($modulePath . $dir)) {
+ // 鍒涘缓鐩綍
+ mkdir($modulePath . $dir, 0755, true);
+ }
+ }
+ } elseif ('__file__' == $path) {
+ // 鐢熸垚锛堢┖鐧斤級鏂囦欢
+ foreach ($file as $name) {
+ if (!is_file($modulePath . $name)) {
+ file_put_contents($modulePath . $name, 'php' == pathinfo($name, PATHINFO_EXTENSION) ? "
+// +----------------------------------------------------------------------
+
+namespace think;
+
+use think\cache\Driver;
+
+class Cache
+{
+ protected static $instance = [];
+ public static $readTimes = 0;
+ public static $writeTimes = 0;
+
+ /**
+ * 鎿嶄綔鍙ユ焺
+ * @var object
+ * @access protected
+ */
+ protected static $handler;
+
+ /**
+ * 杩炴帴缂撳瓨
+ * @access public
+ * @param array $options 閰嶇疆鏁扮粍
+ * @param bool|string $name 缂撳瓨杩炴帴鏍囪瘑 true 寮哄埗閲嶆柊杩炴帴
+ * @return Driver
+ */
+ public static function connect(array $options = [], $name = false)
+ {
+ $type = !empty($options['type']) ? $options['type'] : 'File';
+ if (false === $name) {
+ $name = md5(serialize($options));
+ }
+
+ if (true === $name || !isset(self::$instance[$name])) {
+ $class = false !== strpos($type, '\\') ? $type : '\\think\\cache\\driver\\' . ucwords($type);
+
+ // 璁板綍鍒濆鍖栦俊鎭
+ App::$debug && Log::record('[ CACHE ] INIT ' . $type, 'info');
+ if (true === $name) {
+ return new $class($options);
+ } else {
+ self::$instance[$name] = new $class($options);
+ }
+ }
+ self::$handler = self::$instance[$name];
+ return self::$handler;
+ }
+
+ /**
+ * 鑷姩鍒濆鍖栫紦瀛
+ * @access public
+ * @param array $options 閰嶇疆鏁扮粍
+ * @return void
+ */
+ public static function init(array $options = [])
+ {
+ if (is_null(self::$handler)) {
+ // 鑷姩鍒濆鍖栫紦瀛
+ if (!empty($options)) {
+ self::connect($options);
+ } elseif ('complex' == Config::get('cache.type')) {
+ self::connect(Config::get('cache.default'));
+ } else {
+ self::connect(Config::get('cache'));
+ }
+ }
+ }
+
+ /**
+ * 鍒囨崲缂撳瓨绫诲瀷 闇瑕侀厤缃 cache.type 涓 complex
+ * @access public
+ * @param string $name 缂撳瓨鏍囪瘑
+ * @return Driver
+ */
+ public static function store($name)
+ {
+ if ('complex' == Config::get('cache.type')) {
+ self::connect(Config::get('cache.' . $name), strtolower($name));
+ }
+ return self::$handler;
+ }
+
+ /**
+ * 鍒ゆ柇缂撳瓨鏄惁瀛樺湪
+ * @access public
+ * @param string $name 缂撳瓨鍙橀噺鍚
+ * @return bool
+ */
+ public static function has($name)
+ {
+ self::init();
+ self::$readTimes++;
+ return self::$handler->has($name);
+ }
+
+ /**
+ * 璇诲彇缂撳瓨
+ * @access public
+ * @param string $name 缂撳瓨鏍囪瘑
+ * @param mixed $default 榛樿鍊
+ * @return mixed
+ */
+ public static function get($name, $default = false)
+ {
+ self::init();
+ self::$readTimes++;
+ return self::$handler->get($name, $default);
+ }
+
+ /**
+ * 鍐欏叆缂撳瓨
+ * @access public
+ * @param string $name 缂撳瓨鏍囪瘑
+ * @param mixed $value 瀛樺偍鏁版嵁
+ * @param int|null $expire 鏈夋晥鏃堕棿 0涓烘案涔
+ * @return boolean
+ */
+ public static function set($name, $value, $expire = null)
+ {
+ self::init();
+ self::$writeTimes++;
+ return self::$handler->set($name, $value, $expire);
+ }
+
+ /**
+ * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級
+ * @access public
+ * @param string $name 缂撳瓨鍙橀噺鍚
+ * @param int $step 姝ラ暱
+ * @return false|int
+ */
+ public static function inc($name, $step = 1)
+ {
+ self::init();
+ self::$writeTimes++;
+ return self::$handler->inc($name, $step);
+ }
+
+ /**
+ * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級
+ * @access public
+ * @param string $name 缂撳瓨鍙橀噺鍚
+ * @param int $step 姝ラ暱
+ * @return false|int
+ */
+ public static function dec($name, $step = 1)
+ {
+ self::init();
+ self::$writeTimes++;
+ return self::$handler->dec($name, $step);
+ }
+
+ /**
+ * 鍒犻櫎缂撳瓨
+ * @access public
+ * @param string $name 缂撳瓨鏍囪瘑
+ * @return boolean
+ */
+ public static function rm($name)
+ {
+ self::init();
+ self::$writeTimes++;
+ return self::$handler->rm($name);
+ }
+
+ /**
+ * 娓呴櫎缂撳瓨
+ * @access public
+ * @param string $tag 鏍囩鍚
+ * @return boolean
+ */
+ public static function clear($tag = null)
+ {
+ self::init();
+ self::$writeTimes++;
+ return self::$handler->clear($tag);
+ }
+
+ /**
+ * 璇诲彇缂撳瓨骞跺垹闄
+ * @access public
+ * @param string $name 缂撳瓨鍙橀噺鍚
+ * @return mixed
+ */
+ public static function pull($name)
+ {
+ self::init();
+ self::$readTimes++;
+ self::$writeTimes++;
+ return self::$handler->pull($name);
+ }
+
+ /**
+ * 濡傛灉涓嶅瓨鍦ㄥ垯鍐欏叆缂撳瓨
+ * @access public
+ * @param string $name 缂撳瓨鍙橀噺鍚
+ * @param mixed $value 瀛樺偍鏁版嵁
+ * @param int $expire 鏈夋晥鏃堕棿 0涓烘案涔
+ * @return mixed
+ */
+ public static function remember($name, $value, $expire = null)
+ {
+ self::init();
+ self::$readTimes++;
+ return self::$handler->remember($name, $value, $expire);
+ }
+
+ /**
+ * 缂撳瓨鏍囩
+ * @access public
+ * @param string $name 鏍囩鍚
+ * @param string|array $keys 缂撳瓨鏍囪瘑
+ * @param bool $overlay 鏄惁瑕嗙洊
+ * @return Driver
+ */
+ public static function tag($name, $keys = null, $overlay = false)
+ {
+ self::init();
+ return self::$handler->tag($name, $keys, $overlay);
+ }
+
+}
diff --git a/thinkphp/library/think/Collection.php b/thinkphp/library/think/Collection.php
new file mode 100644
index 000000000..41b427598
--- /dev/null
+++ b/thinkphp/library/think/Collection.php
@@ -0,0 +1,373 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+use ArrayAccess;
+use ArrayIterator;
+use Countable;
+use IteratorAggregate;
+use JsonSerializable;
+
+class Collection implements ArrayAccess, Countable, IteratorAggregate, JsonSerializable
+{
+ protected $items = [];
+
+ public function __construct($items = [])
+ {
+ $this->items = $this->convertToArray($items);
+ }
+
+ public static function make($items = [])
+ {
+ return new static($items);
+ }
+
+ /**
+ * 鏄惁涓虹┖
+ * @return bool
+ */
+ public function isEmpty()
+ {
+ return empty($this->items);
+ }
+
+ public function toArray()
+ {
+ return array_map(function ($value) {
+ return ($value instanceof Model || $value instanceof self) ? $value->toArray() : $value;
+ }, $this->items);
+ }
+
+ public function all()
+ {
+ return $this->items;
+ }
+
+ /**
+ * 鍚堝苟鏁扮粍
+ *
+ * @param mixed $items
+ * @return static
+ */
+ public function merge($items)
+ {
+ return new static(array_merge($this->items, $this->convertToArray($items)));
+ }
+
+ /**
+ * 姣旇緝鏁扮粍锛岃繑鍥炲樊闆
+ *
+ * @param mixed $items
+ * @return static
+ */
+ public function diff($items)
+ {
+ return new static(array_diff($this->items, $this->convertToArray($items)));
+ }
+
+ /**
+ * 浜ゆ崲鏁扮粍涓殑閿拰鍊
+ *
+ * @return static
+ */
+ public function flip()
+ {
+ return new static(array_flip($this->items));
+ }
+
+ /**
+ * 姣旇緝鏁扮粍锛岃繑鍥炰氦闆
+ *
+ * @param mixed $items
+ * @return static
+ */
+ public function intersect($items)
+ {
+ return new static(array_intersect($this->items, $this->convertToArray($items)));
+ }
+
+ /**
+ * 杩斿洖鏁扮粍涓墍鏈夌殑閿悕
+ *
+ * @return static
+ */
+ public function keys()
+ {
+ return new static(array_keys($this->items));
+ }
+
+ /**
+ * 鍒犻櫎鏁扮粍鐨勬渶鍚庝竴涓厓绱狅紙鍑烘爤锛
+ *
+ * @return mixed
+ */
+ public function pop()
+ {
+ return array_pop($this->items);
+ }
+
+ /**
+ * 閫氳繃浣跨敤鐢ㄦ埛鑷畾涔夊嚱鏁帮紝浠ュ瓧绗︿覆杩斿洖鏁扮粍
+ *
+ * @param callable $callback
+ * @param mixed $initial
+ * @return mixed
+ */
+ public function reduce(callable $callback, $initial = null)
+ {
+ return array_reduce($this->items, $callback, $initial);
+ }
+
+ /**
+ * 浠ョ浉鍙嶇殑椤哄簭杩斿洖鏁扮粍銆
+ *
+ * @return static
+ */
+ public function reverse()
+ {
+ return new static(array_reverse($this->items));
+ }
+
+ /**
+ * 鍒犻櫎鏁扮粍涓涓厓绱狅紝骞惰繑鍥炶鍒犻櫎鍏冪礌鐨勫
+ *
+ * @return mixed
+ */
+ public function shift()
+ {
+ return array_shift($this->items);
+ }
+
+ /**
+ * 鎶婁竴涓暟缁勫垎鍓蹭负鏂扮殑鏁扮粍鍧.
+ *
+ * @param int $size
+ * @param bool $preserveKeys
+ * @return static
+ */
+ public function chunk($size, $preserveKeys = false)
+ {
+ $chunks = [];
+
+ foreach (array_chunk($this->items, $size, $preserveKeys) as $chunk) {
+ $chunks[] = new static($chunk);
+ }
+
+ return new static($chunks);
+ }
+
+ /**
+ * 鍦ㄦ暟缁勫紑澶存彃鍏ヤ竴涓厓绱
+ * @param mixed $value
+ * @param null $key
+ * @return int
+ */
+ public function unshift($value, $key = null)
+ {
+ if (is_null($key)) {
+ array_unshift($this->items, $value);
+ } else {
+ $this->items = [$key => $value] + $this->items;
+ }
+ }
+
+ /**
+ * 缁欐瘡涓厓绱犳墽琛屼釜鍥炶皟
+ *
+ * @param callable $callback
+ * @return $this
+ */
+ public function each(callable $callback)
+ {
+ foreach ($this->items as $key => $item) {
+ if ($callback($item, $key) === false) {
+ break;
+ }
+ }
+
+ return $this;
+ }
+
+ /**
+ * 鐢ㄥ洖璋冨嚱鏁拌繃婊ゆ暟缁勪腑鐨勫厓绱
+ * @param callable|null $callback
+ * @return static
+ */
+ public function filter(callable $callback = null)
+ {
+ if ($callback) {
+ return new static(array_filter($this->items, $callback));
+ }
+
+ return new static(array_filter($this->items));
+ }
+
+ /**
+ * 杩斿洖鏁扮粍涓寚瀹氱殑涓鍒
+ * @param $column_key
+ * @param null $index_key
+ * @return array
+ */
+ public function column($column_key, $index_key = null)
+ {
+ if (function_exists('array_column')) {
+ return array_column($this->items, $column_key, $index_key);
+ }
+
+ $result = [];
+ foreach ($this->items as $row) {
+ $key = $value = null;
+ $keySet = $valueSet = false;
+ if (null !== $index_key && array_key_exists($index_key, $row)) {
+ $keySet = true;
+ $key = (string) $row[$index_key];
+ }
+ if (null === $column_key) {
+ $valueSet = true;
+ $value = $row;
+ } elseif (is_array($row) && array_key_exists($column_key, $row)) {
+ $valueSet = true;
+ $value = $row[$column_key];
+ }
+ if ($valueSet) {
+ if ($keySet) {
+ $result[$key] = $value;
+ } else {
+ $result[] = $value;
+ }
+ }
+ }
+ return $result;
+ }
+
+ /**
+ * 瀵规暟缁勬帓搴
+ *
+ * @param callable|null $callback
+ * @return static
+ */
+ public function sort(callable $callback = null)
+ {
+ $items = $this->items;
+
+ $callback ? uasort($items, $callback) : uasort($items, function ($a, $b) {
+
+ if ($a == $b) {
+ return 0;
+ }
+
+ return ($a < $b) ? -1 : 1;
+ });
+
+ return new static($items);
+ }
+
+ /**
+ * 灏嗘暟缁勬墦涔
+ *
+ * @return static
+ */
+ public function shuffle()
+ {
+ $items = $this->items;
+
+ shuffle($items);
+
+ return new static($items);
+ }
+
+ /**
+ * 鎴彇鏁扮粍
+ *
+ * @param int $offset
+ * @param int $length
+ * @param bool $preserveKeys
+ * @return static
+ */
+ public function slice($offset, $length = null, $preserveKeys = false)
+ {
+ return new static(array_slice($this->items, $offset, $length, $preserveKeys));
+ }
+
+ // ArrayAccess
+ public function offsetExists($offset)
+ {
+ return array_key_exists($offset, $this->items);
+ }
+
+ public function offsetGet($offset)
+ {
+ return $this->items[$offset];
+ }
+
+ public function offsetSet($offset, $value)
+ {
+ if (is_null($offset)) {
+ $this->items[] = $value;
+ } else {
+ $this->items[$offset] = $value;
+ }
+ }
+
+ public function offsetUnset($offset)
+ {
+ unset($this->items[$offset]);
+ }
+
+ //Countable
+ public function count()
+ {
+ return count($this->items);
+ }
+
+ //IteratorAggregate
+ public function getIterator()
+ {
+ return new ArrayIterator($this->items);
+ }
+
+ //JsonSerializable
+ public function jsonSerialize()
+ {
+ return $this->toArray();
+ }
+
+ /**
+ * 杞崲褰撳墠鏁版嵁闆嗕负JSON瀛楃涓
+ * @access public
+ * @param integer $options json鍙傛暟
+ * @return string
+ */
+ public function toJson($options = JSON_UNESCAPED_UNICODE)
+ {
+ return json_encode($this->toArray(), $options);
+ }
+
+ public function __toString()
+ {
+ return $this->toJson();
+ }
+
+ /**
+ * 杞崲鎴愭暟缁
+ *
+ * @param mixed $items
+ * @return array
+ */
+ protected function convertToArray($items)
+ {
+ if ($items instanceof self) {
+ return $items->all();
+ }
+ return (array) $items;
+ }
+}
diff --git a/thinkphp/library/think/Config.php b/thinkphp/library/think/Config.php
new file mode 100644
index 000000000..fc6c50ac8
--- /dev/null
+++ b/thinkphp/library/think/Config.php
@@ -0,0 +1,170 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+class Config
+{
+ // 閰嶇疆鍙傛暟
+ private static $config = [];
+ // 鍙傛暟浣滅敤鍩
+ private static $range = '_sys_';
+
+ // 璁惧畾閰嶇疆鍙傛暟鐨勪綔鐢ㄥ煙
+ public static function range($range)
+ {
+ self::$range = $range;
+ if (!isset(self::$config[$range])) {
+ self::$config[$range] = [];
+ }
+ }
+
+ /**
+ * 瑙f瀽閰嶇疆鏂囦欢鎴栧唴瀹
+ * @param string $config 閰嶇疆鏂囦欢璺緞鎴栧唴瀹
+ * @param string $type 閰嶇疆瑙f瀽绫诲瀷
+ * @param string $name 閰嶇疆鍚嶏紙濡傝缃嵆琛ㄧず浜岀骇閰嶇疆锛
+ * @param string $range 浣滅敤鍩
+ * @return mixed
+ */
+ public static function parse($config, $type = '', $name = '', $range = '')
+ {
+ $range = $range ?: self::$range;
+ if (empty($type)) {
+ $type = pathinfo($config, PATHINFO_EXTENSION);
+ }
+ $class = false !== strpos($type, '\\') ? $type : '\\think\\config\\driver\\' . ucwords($type);
+ return self::set((new $class())->parse($config), $name, $range);
+ }
+
+ /**
+ * 鍔犺浇閰嶇疆鏂囦欢锛圥HP鏍煎紡锛
+ * @param string $file 閰嶇疆鏂囦欢鍚
+ * @param string $name 閰嶇疆鍚嶏紙濡傝缃嵆琛ㄧず浜岀骇閰嶇疆锛
+ * @param string $range 浣滅敤鍩
+ * @return mixed
+ */
+ public static function load($file, $name = '', $range = '')
+ {
+ $range = $range ?: self::$range;
+ if (!isset(self::$config[$range])) {
+ self::$config[$range] = [];
+ }
+ if (is_file($file)) {
+ $name = strtolower($name);
+ $type = pathinfo($file, PATHINFO_EXTENSION);
+ if ('php' == $type) {
+ return self::set(include $file, $name, $range);
+ } elseif ('yaml' == $type && function_exists('yaml_parse_file')) {
+ return self::set(yaml_parse_file($file), $name, $range);
+ } else {
+ return self::parse($file, $type, $name, $range);
+ }
+ } else {
+ return self::$config[$range];
+ }
+ }
+
+ /**
+ * 妫娴嬮厤缃槸鍚﹀瓨鍦
+ * @param string $name 閰嶇疆鍙傛暟鍚嶏紙鏀寔浜岀骇閰嶇疆 .鍙峰垎鍓诧級
+ * @param string $range 浣滅敤鍩
+ * @return bool
+ */
+ public static function has($name, $range = '')
+ {
+ $range = $range ?: self::$range;
+
+ if (!strpos($name, '.')) {
+ return isset(self::$config[$range][strtolower($name)]);
+ } else {
+ // 浜岀淮鏁扮粍璁剧疆鍜岃幏鍙栨敮鎸
+ $name = explode('.', $name);
+ return isset(self::$config[$range][strtolower($name[0])][$name[1]]);
+ }
+ }
+
+ /**
+ * 鑾峰彇閰嶇疆鍙傛暟 涓虹┖鍒欒幏鍙栨墍鏈夐厤缃
+ * @param string $name 閰嶇疆鍙傛暟鍚嶏紙鏀寔浜岀骇閰嶇疆 .鍙峰垎鍓诧級
+ * @param string $range 浣滅敤鍩
+ * @return mixed
+ */
+ public static function get($name = null, $range = '')
+ {
+ $range = $range ?: self::$range;
+ // 鏃犲弬鏁版椂鑾峰彇鎵鏈
+ if (empty($name) && isset(self::$config[$range])) {
+ return self::$config[$range];
+ }
+
+ if (!strpos($name, '.')) {
+ $name = strtolower($name);
+ return isset(self::$config[$range][$name]) ? self::$config[$range][$name] : null;
+ } else {
+ // 浜岀淮鏁扮粍璁剧疆鍜岃幏鍙栨敮鎸
+ $name = explode('.', $name);
+ $name[0] = strtolower($name[0]);
+ return isset(self::$config[$range][$name[0]][$name[1]]) ? self::$config[$range][$name[0]][$name[1]] : null;
+ }
+ }
+
+ /**
+ * 璁剧疆閰嶇疆鍙傛暟 name涓烘暟缁勫垯涓烘壒閲忚缃
+ * @param string|array $name 閰嶇疆鍙傛暟鍚嶏紙鏀寔浜岀骇閰嶇疆 .鍙峰垎鍓诧級
+ * @param mixed $value 閰嶇疆鍊
+ * @param string $range 浣滅敤鍩
+ * @return mixed
+ */
+ public static function set($name, $value = null, $range = '')
+ {
+ $range = $range ?: self::$range;
+ if (!isset(self::$config[$range])) {
+ self::$config[$range] = [];
+ }
+ if (is_string($name)) {
+ if (!strpos($name, '.')) {
+ self::$config[$range][strtolower($name)] = $value;
+ } else {
+ // 浜岀淮鏁扮粍璁剧疆鍜岃幏鍙栨敮鎸
+ $name = explode('.', $name);
+ self::$config[$range][strtolower($name[0])][$name[1]] = $value;
+ }
+ return;
+ } elseif (is_array($name)) {
+ // 鎵归噺璁剧疆
+ if (!empty($value)) {
+ self::$config[$range][$value] = isset(self::$config[$range][$value]) ?
+ array_merge(self::$config[$range][$value], $name) :
+ self::$config[$range][$value] = $name;
+ return self::$config[$range][$value];
+ } else {
+ return self::$config[$range] = array_merge(self::$config[$range], array_change_key_case($name));
+ }
+ } else {
+ // 涓虹┖鐩存帴杩斿洖 宸叉湁閰嶇疆
+ return self::$config[$range];
+ }
+ }
+
+ /**
+ * 閲嶇疆閰嶇疆鍙傛暟
+ */
+ public static function reset($range = '')
+ {
+ $range = $range ?: self::$range;
+ if (true === $range) {
+ self::$config = [];
+ } else {
+ self::$config[$range] = [];
+ }
+ }
+}
diff --git a/thinkphp/library/think/Console.php b/thinkphp/library/think/Console.php
new file mode 100644
index 000000000..1d97ab2b7
--- /dev/null
+++ b/thinkphp/library/think/Console.php
@@ -0,0 +1,719 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+use think\console\Command;
+use think\console\command\Help as HelpCommand;
+use think\console\Input;
+use think\console\input\Argument as InputArgument;
+use think\console\input\Definition as InputDefinition;
+use think\console\input\Option as InputOption;
+use think\console\Output;
+use think\console\output\driver\Buffer;
+
+class Console
+{
+
+ private $name;
+ private $version;
+
+ /** @var Command[] */
+ private $commands = [];
+
+ private $wantHelps = false;
+
+ private $catchExceptions = true;
+ private $autoExit = true;
+ private $definition;
+ private $defaultCommand;
+
+ private static $defaultCommands = [
+ "think\\console\\command\\Help",
+ "think\\console\\command\\Lists",
+ "think\\console\\command\\Build",
+ "think\\console\\command\\Clear",
+ "think\\console\\command\\make\\Controller",
+ "think\\console\\command\\make\\Model",
+ "think\\console\\command\\optimize\\Autoload",
+ "think\\console\\command\\optimize\\Config",
+ "think\\console\\command\\optimize\\Route",
+ "think\\console\\command\\optimize\\Schema",
+ ];
+
+ public function __construct($name = 'UNKNOWN', $version = 'UNKNOWN')
+ {
+ $this->name = $name;
+ $this->version = $version;
+
+ $this->defaultCommand = 'list';
+ $this->definition = $this->getDefaultInputDefinition();
+
+ foreach ($this->getDefaultCommands() as $command) {
+ $this->add($command);
+ }
+ }
+
+ public static function init($run = true)
+ {
+ static $console;
+ if (!$console) {
+ // 瀹炰緥鍖朿onsole
+ $console = new self('Think Console', '0.1');
+ // 璇诲彇鎸囦护闆
+ if (is_file(CONF_PATH . 'command' . EXT)) {
+ $commands = include CONF_PATH . 'command' . EXT;
+ if (is_array($commands)) {
+ foreach ($commands as $command) {
+ if (class_exists($command) && is_subclass_of($command, "\\think\\console\\Command")) {
+ // 娉ㄥ唽鎸囦护
+ $console->add(new $command());
+ }
+ }
+ }
+ }
+ }
+ if ($run) {
+ // 杩愯
+ return $console->run();
+ } else {
+ return $console;
+ }
+ }
+
+ /**
+ * @param $command
+ * @param array $parameters
+ * @param string $driver
+ * @return Output|Buffer
+ */
+ public static function call($command, array $parameters = [], $driver = 'buffer')
+ {
+ $console = self::init(false);
+
+ array_unshift($parameters, $command);
+
+ $input = new Input($parameters);
+ $output = new Output($driver);
+
+ $console->setCatchExceptions(false);
+ $console->find($command)->run($input, $output);
+
+ return $output;
+ }
+
+ /**
+ * 鎵ц褰撳墠鐨勬寚浠
+ * @return int
+ * @throws \Exception
+ * @api
+ */
+ public function run()
+ {
+ $input = new Input();
+ $output = new Output();
+
+ $this->configureIO($input, $output);
+
+ try {
+ $exitCode = $this->doRun($input, $output);
+ } catch (\Exception $e) {
+ if (!$this->catchExceptions) {
+ throw $e;
+ }
+
+ $output->renderException($e);
+
+ $exitCode = $e->getCode();
+ if (is_numeric($exitCode)) {
+ $exitCode = (int) $exitCode;
+ if (0 === $exitCode) {
+ $exitCode = 1;
+ }
+ } else {
+ $exitCode = 1;
+ }
+ }
+
+ if ($this->autoExit) {
+ if ($exitCode > 255) {
+ $exitCode = 255;
+ }
+
+ exit($exitCode);
+ }
+
+ return $exitCode;
+ }
+
+ /**
+ * 鎵ц鎸囦护
+ * @param Input $input
+ * @param Output $output
+ * @return int
+ */
+ public function doRun(Input $input, Output $output)
+ {
+ if (true === $input->hasParameterOption(['--version', '-V'])) {
+ $output->writeln($this->getLongVersion());
+
+ return 0;
+ }
+
+ $name = $this->getCommandName($input);
+
+ if (true === $input->hasParameterOption(['--help', '-h'])) {
+ if (!$name) {
+ $name = 'help';
+ $input = new Input(['help']);
+ } else {
+ $this->wantHelps = true;
+ }
+ }
+
+ if (!$name) {
+ $name = $this->defaultCommand;
+ $input = new Input([$this->defaultCommand]);
+ }
+
+ $command = $this->find($name);
+
+ $exitCode = $this->doRunCommand($command, $input, $output);
+
+ return $exitCode;
+ }
+
+ /**
+ * 璁剧疆杈撳叆鍙傛暟瀹氫箟
+ * @param InputDefinition $definition
+ */
+ public function setDefinition(InputDefinition $definition)
+ {
+ $this->definition = $definition;
+ }
+
+ /**
+ * 鑾峰彇杈撳叆鍙傛暟瀹氫箟
+ * @return InputDefinition The InputDefinition instance
+ */
+ public function getDefinition()
+ {
+ return $this->definition;
+ }
+
+ /**
+ * Gets the help message.
+ * @return string A help message.
+ */
+ public function getHelp()
+ {
+ return $this->getLongVersion();
+ }
+
+ /**
+ * 鏄惁鎹曡幏寮傚父
+ * @param bool $boolean
+ * @api
+ */
+ public function setCatchExceptions($boolean)
+ {
+ $this->catchExceptions = (bool) $boolean;
+ }
+
+ /**
+ * 鏄惁鑷姩閫鍑
+ * @param bool $boolean
+ * @api
+ */
+ public function setAutoExit($boolean)
+ {
+ $this->autoExit = (bool) $boolean;
+ }
+
+ /**
+ * 鑾峰彇鍚嶇О
+ * @return string
+ */
+ public function getName()
+ {
+ return $this->name;
+ }
+
+ /**
+ * 璁剧疆鍚嶇О
+ * @param string $name
+ */
+ public function setName($name)
+ {
+ $this->name = $name;
+ }
+
+ /**
+ * 鑾峰彇鐗堟湰
+ * @return string
+ * @api
+ */
+ public function getVersion()
+ {
+ return $this->version;
+ }
+
+ /**
+ * 璁剧疆鐗堟湰
+ * @param string $version
+ */
+ public function setVersion($version)
+ {
+ $this->version = $version;
+ }
+
+ /**
+ * 鑾峰彇瀹屾暣鐨勭増鏈彿
+ * @return string
+ */
+ public function getLongVersion()
+ {
+ if ('UNKNOWN' !== $this->getName() && 'UNKNOWN' !== $this->getVersion()) {
+ return sprintf('%s version %s', $this->getName(), $this->getVersion());
+ }
+
+ return 'Console Tool';
+ }
+
+ /**
+ * 娉ㄥ唽涓涓寚浠
+ * @param string $name
+ * @return Command
+ */
+ public function register($name)
+ {
+ return $this->add(new Command($name));
+ }
+
+ /**
+ * 娣诲姞鎸囦护
+ * @param Command[] $commands
+ */
+ public function addCommands(array $commands)
+ {
+ foreach ($commands as $command) {
+ $this->add($command);
+ }
+ }
+
+ /**
+ * 娣诲姞涓涓寚浠
+ * @param Command $command
+ * @return Command
+ */
+ public function add(Command $command)
+ {
+ $command->setConsole($this);
+
+ if (!$command->isEnabled()) {
+ $command->setConsole(null);
+ return;
+ }
+
+ if (null === $command->getDefinition()) {
+ throw new \LogicException(sprintf('Command class "%s" is not correctly initialized. You probably forgot to call the parent constructor.', get_class($command)));
+ }
+
+ $this->commands[$command->getName()] = $command;
+
+ foreach ($command->getAliases() as $alias) {
+ $this->commands[$alias] = $command;
+ }
+
+ return $command;
+ }
+
+ /**
+ * 鑾峰彇鎸囦护
+ * @param string $name 鎸囦护鍚嶇О
+ * @return Command
+ * @throws \InvalidArgumentException
+ */
+ public function get($name)
+ {
+ if (!isset($this->commands[$name])) {
+ throw new \InvalidArgumentException(sprintf('The command "%s" does not exist.', $name));
+ }
+
+ $command = $this->commands[$name];
+
+ if ($this->wantHelps) {
+ $this->wantHelps = false;
+
+ /** @var HelpCommand $helpCommand */
+ $helpCommand = $this->get('help');
+ $helpCommand->setCommand($command);
+
+ return $helpCommand;
+ }
+
+ return $command;
+ }
+
+ /**
+ * 鏌愪釜鎸囦护鏄惁瀛樺湪
+ * @param string $name 鎸囦护鍚嶇О
+ * @return bool
+ */
+ public function has($name)
+ {
+ return isset($this->commands[$name]);
+ }
+
+ /**
+ * 鑾峰彇鎵鏈夌殑鍛藉悕绌洪棿
+ * @return array
+ */
+ public function getNamespaces()
+ {
+ $namespaces = [];
+ foreach ($this->commands as $command) {
+ $namespaces = array_merge($namespaces, $this->extractAllNamespaces($command->getName()));
+
+ foreach ($command->getAliases() as $alias) {
+ $namespaces = array_merge($namespaces, $this->extractAllNamespaces($alias));
+ }
+ }
+
+ return array_values(array_unique(array_filter($namespaces)));
+ }
+
+ /**
+ * 鏌ユ壘娉ㄥ唽鍛藉悕绌洪棿涓殑鍚嶇О鎴栫缉鍐欍
+ * @param string $namespace
+ * @return string
+ * @throws \InvalidArgumentException
+ */
+ public function findNamespace($namespace)
+ {
+ $allNamespaces = $this->getNamespaces();
+ $expr = preg_replace_callback('{([^:]+|)}', function ($matches) {
+ return preg_quote($matches[1]) . '[^:]*';
+ }, $namespace);
+ $namespaces = preg_grep('{^' . $expr . '}', $allNamespaces);
+
+ if (empty($namespaces)) {
+ $message = sprintf('There are no commands defined in the "%s" namespace.', $namespace);
+
+ if ($alternatives = $this->findAlternatives($namespace, $allNamespaces)) {
+ if (1 == count($alternatives)) {
+ $message .= "\n\nDid you mean this?\n ";
+ } else {
+ $message .= "\n\nDid you mean one of these?\n ";
+ }
+
+ $message .= implode("\n ", $alternatives);
+ }
+
+ throw new \InvalidArgumentException($message);
+ }
+
+ $exact = in_array($namespace, $namespaces, true);
+ if (count($namespaces) > 1 && !$exact) {
+ throw new \InvalidArgumentException(sprintf('The namespace "%s" is ambiguous (%s).', $namespace, $this->getAbbreviationSuggestions(array_values($namespaces))));
+ }
+
+ return $exact ? $namespace : reset($namespaces);
+ }
+
+ /**
+ * 鏌ユ壘鎸囦护
+ * @param string $name 鍚嶇О鎴栬呭埆鍚
+ * @return Command
+ * @throws \InvalidArgumentException
+ */
+ public function find($name)
+ {
+ $allCommands = array_keys($this->commands);
+ $expr = preg_replace_callback('{([^:]+|)}', function ($matches) {
+ return preg_quote($matches[1]) . '[^:]*';
+ }, $name);
+ $commands = preg_grep('{^' . $expr . '}', $allCommands);
+
+ if (empty($commands) || count(preg_grep('{^' . $expr . '$}', $commands)) < 1) {
+ if (false !== $pos = strrpos($name, ':')) {
+ $this->findNamespace(substr($name, 0, $pos));
+ }
+
+ $message = sprintf('Command "%s" is not defined.', $name);
+
+ if ($alternatives = $this->findAlternatives($name, $allCommands)) {
+ if (1 == count($alternatives)) {
+ $message .= "\n\nDid you mean this?\n ";
+ } else {
+ $message .= "\n\nDid you mean one of these?\n ";
+ }
+ $message .= implode("\n ", $alternatives);
+ }
+
+ throw new \InvalidArgumentException($message);
+ }
+
+ if (count($commands) > 1) {
+ $commandList = $this->commands;
+ $commands = array_filter($commands, function ($nameOrAlias) use ($commandList, $commands) {
+ $commandName = $commandList[$nameOrAlias]->getName();
+
+ return $commandName === $nameOrAlias || !in_array($commandName, $commands);
+ });
+ }
+
+ $exact = in_array($name, $commands, true);
+ if (count($commands) > 1 && !$exact) {
+ $suggestions = $this->getAbbreviationSuggestions(array_values($commands));
+
+ throw new \InvalidArgumentException(sprintf('Command "%s" is ambiguous (%s).', $name, $suggestions));
+ }
+
+ return $this->get($exact ? $name : reset($commands));
+ }
+
+ /**
+ * 鑾峰彇鎵鏈夌殑鎸囦护
+ * @param string $namespace 鍛藉悕绌洪棿
+ * @return Command[]
+ * @api
+ */
+ public function all($namespace = null)
+ {
+ if (null === $namespace) {
+ return $this->commands;
+ }
+
+ $commands = [];
+ foreach ($this->commands as $name => $command) {
+ if ($this->extractNamespace($name, substr_count($namespace, ':') + 1) === $namespace) {
+ $commands[$name] = $command;
+ }
+ }
+
+ return $commands;
+ }
+
+ /**
+ * 鑾峰彇鍙兘鐨勬寚浠ゅ悕
+ * @param array $names
+ * @return array
+ */
+ public static function getAbbreviations($names)
+ {
+ $abbrevs = [];
+ foreach ($names as $name) {
+ for ($len = strlen($name); $len > 0; --$len) {
+ $abbrev = substr($name, 0, $len);
+ $abbrevs[$abbrev][] = $name;
+ }
+ }
+
+ return $abbrevs;
+ }
+
+ /**
+ * 閰嶇疆鍩轰簬鐢ㄦ埛鐨勫弬鏁板拰閫夐」鐨勮緭鍏ュ拰杈撳嚭瀹炰緥銆
+ * @param Input $input 杈撳叆瀹炰緥
+ * @param Output $output 杈撳嚭瀹炰緥
+ */
+ protected function configureIO(Input $input, Output $output)
+ {
+ if (true === $input->hasParameterOption(['--ansi'])) {
+ $output->setDecorated(true);
+ } elseif (true === $input->hasParameterOption(['--no-ansi'])) {
+ $output->setDecorated(false);
+ }
+
+ if (true === $input->hasParameterOption(['--no-interaction', '-n'])) {
+ $input->setInteractive(false);
+ }
+
+ if (true === $input->hasParameterOption(['--quiet', '-q'])) {
+ $output->setVerbosity(Output::VERBOSITY_QUIET);
+ } else {
+ if ($input->hasParameterOption('-vvv') || $input->hasParameterOption('--verbose=3') || $input->getParameterOption('--verbose') === 3) {
+ $output->setVerbosity(Output::VERBOSITY_DEBUG);
+ } elseif ($input->hasParameterOption('-vv') || $input->hasParameterOption('--verbose=2') || $input->getParameterOption('--verbose') === 2) {
+ $output->setVerbosity(Output::VERBOSITY_VERY_VERBOSE);
+ } elseif ($input->hasParameterOption('-v') || $input->hasParameterOption('--verbose=1') || $input->hasParameterOption('--verbose') || $input->getParameterOption('--verbose')) {
+ $output->setVerbosity(Output::VERBOSITY_VERBOSE);
+ }
+ }
+ }
+
+ /**
+ * 鎵ц鎸囦护
+ * @param Command $command 鎸囦护瀹炰緥
+ * @param Input $input 杈撳叆瀹炰緥
+ * @param Output $output 杈撳嚭瀹炰緥
+ * @return int
+ * @throws \Exception
+ */
+ protected function doRunCommand(Command $command, Input $input, Output $output)
+ {
+ return $command->run($input, $output);
+ }
+
+ /**
+ * 鑾峰彇鎸囦护鐨勫熀纭鍚嶇О
+ * @param Input $input
+ * @return string
+ */
+ protected function getCommandName(Input $input)
+ {
+ return $input->getFirstArgument();
+ }
+
+ /**
+ * 鑾峰彇榛樿杈撳叆瀹氫箟
+ * @return InputDefinition
+ */
+ protected function getDefaultInputDefinition()
+ {
+ return new InputDefinition([
+ new InputArgument('command', InputArgument::REQUIRED, 'The command to execute'),
+ new InputOption('--help', '-h', InputOption::VALUE_NONE, 'Display this help message'),
+ new InputOption('--version', '-V', InputOption::VALUE_NONE, 'Display this console version'),
+ new InputOption('--quiet', '-q', InputOption::VALUE_NONE, 'Do not output any message'),
+ new InputOption('--verbose', '-v|vv|vvv', InputOption::VALUE_NONE, 'Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug'),
+ new InputOption('--ansi', '', InputOption::VALUE_NONE, 'Force ANSI output'),
+ new InputOption('--no-ansi', '', InputOption::VALUE_NONE, 'Disable ANSI output'),
+ new InputOption('--no-interaction', '-n', InputOption::VALUE_NONE, 'Do not ask any interactive question'),
+ ]);
+ }
+
+ /**
+ * 璁剧疆榛樿鍛戒护
+ * @return Command[] An array of default Command instances
+ */
+ protected function getDefaultCommands()
+ {
+ $defaultCommands = [];
+
+ foreach (self::$defaultCommands as $classname) {
+ if (class_exists($classname) && is_subclass_of($classname, "think\\console\\Command")) {
+ $defaultCommands[] = new $classname();
+ }
+ }
+
+ return $defaultCommands;
+ }
+
+ public static function addDefaultCommands(array $classnames)
+ {
+ self::$defaultCommands = array_merge(self::$defaultCommands, $classnames);
+ }
+
+ /**
+ * 鑾峰彇鍙兘鐨勫缓璁
+ * @param array $abbrevs
+ * @return string
+ */
+ private function getAbbreviationSuggestions($abbrevs)
+ {
+ return sprintf('%s, %s%s', $abbrevs[0], $abbrevs[1], count($abbrevs) > 2 ? sprintf(' and %d more', count($abbrevs) - 2) : '');
+ }
+
+ /**
+ * 杩斿洖鍛藉悕绌洪棿閮ㄥ垎
+ * @param string $name 鎸囦护
+ * @param string $limit 閮ㄥ垎鐨勫懡鍚嶇┖闂寸殑鏈澶ф暟閲
+ * @return string
+ */
+ public function extractNamespace($name, $limit = null)
+ {
+ $parts = explode(':', $name);
+ array_pop($parts);
+
+ return implode(':', null === $limit ? $parts : array_slice($parts, 0, $limit));
+ }
+
+ /**
+ * 鏌ユ壘鍙浛浠g殑寤鸿
+ * @param string $name
+ * @param array|\Traversable $collection
+ * @return array
+ */
+ private function findAlternatives($name, $collection)
+ {
+ $threshold = 1e3;
+ $alternatives = [];
+
+ $collectionParts = [];
+ foreach ($collection as $item) {
+ $collectionParts[$item] = explode(':', $item);
+ }
+
+ foreach (explode(':', $name) as $i => $subname) {
+ foreach ($collectionParts as $collectionName => $parts) {
+ $exists = isset($alternatives[$collectionName]);
+ if (!isset($parts[$i]) && $exists) {
+ $alternatives[$collectionName] += $threshold;
+ continue;
+ } elseif (!isset($parts[$i])) {
+ continue;
+ }
+
+ $lev = levenshtein($subname, $parts[$i]);
+ if ($lev <= strlen($subname) / 3 || '' !== $subname && false !== strpos($parts[$i], $subname)) {
+ $alternatives[$collectionName] = $exists ? $alternatives[$collectionName] + $lev : $lev;
+ } elseif ($exists) {
+ $alternatives[$collectionName] += $threshold;
+ }
+ }
+ }
+
+ foreach ($collection as $item) {
+ $lev = levenshtein($name, $item);
+ if ($lev <= strlen($name) / 3 || false !== strpos($item, $name)) {
+ $alternatives[$item] = isset($alternatives[$item]) ? $alternatives[$item] - $lev : $lev;
+ }
+ }
+
+ $alternatives = array_filter($alternatives, function ($lev) use ($threshold) {
+ return $lev < 2 * $threshold;
+ });
+ asort($alternatives);
+
+ return array_keys($alternatives);
+ }
+
+ /**
+ * 璁剧疆榛樿鐨勬寚浠
+ * @param string $commandName The Command name
+ */
+ public function setDefaultCommand($commandName)
+ {
+ $this->defaultCommand = $commandName;
+ }
+
+ /**
+ * 杩斿洖鎵鏈夌殑鍛藉悕绌洪棿
+ * @param string $name
+ * @return array
+ */
+ private function extractAllNamespaces($name)
+ {
+ $parts = explode(':', $name, -1);
+ $namespaces = [];
+
+ foreach ($parts as $part) {
+ if (count($namespaces)) {
+ $namespaces[] = end($namespaces) . ':' . $part;
+ } else {
+ $namespaces[] = $part;
+ }
+ }
+
+ return $namespaces;
+ }
+
+}
diff --git a/thinkphp/library/think/Controller.php b/thinkphp/library/think/Controller.php
new file mode 100644
index 000000000..69db0a1d1
--- /dev/null
+++ b/thinkphp/library/think/Controller.php
@@ -0,0 +1,212 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+\think\Loader::import('controller/Jump', TRAIT_PATH, EXT);
+
+use think\exception\ValidateException;
+
+class Controller
+{
+ use \traits\controller\Jump;
+
+ /**
+ * @var \think\View 瑙嗗浘绫诲疄渚
+ */
+ protected $view;
+ /**
+ * @var \think\Request Request瀹炰緥
+ */
+ protected $request;
+ // 楠岃瘉澶辫触鏄惁鎶涘嚭寮傚父
+ protected $failException = false;
+ // 鏄惁鎵归噺楠岃瘉
+ protected $batchValidate = false;
+
+ /**
+ * 鍓嶇疆鎿嶄綔鏂规硶鍒楄〃
+ * @var array $beforeActionList
+ * @access protected
+ */
+ protected $beforeActionList = [];
+
+ /**
+ * 鏋舵瀯鍑芥暟
+ * @param Request $request Request瀵硅薄
+ * @access public
+ */
+ public function __construct(Request $request = null)
+ {
+ if (is_null($request)) {
+ $request = Request::instance();
+ }
+ $this->view = View::instance(Config::get('template'), Config::get('view_replace_str'));
+ $this->request = $request;
+
+ // 鎺у埗鍣ㄥ垵濮嬪寲
+ $this->_initialize();
+
+ // 鍓嶇疆鎿嶄綔鏂规硶
+ if ($this->beforeActionList) {
+ foreach ($this->beforeActionList as $method => $options) {
+ is_numeric($method) ?
+ $this->beforeAction($options) :
+ $this->beforeAction($method, $options);
+ }
+ }
+ }
+
+ // 鍒濆鍖
+ protected function _initialize()
+ {
+ }
+
+ /**
+ * 鍓嶇疆鎿嶄綔
+ * @access protected
+ * @param string $method 鍓嶇疆鎿嶄綔鏂规硶鍚
+ * @param array $options 璋冪敤鍙傛暟 ['only'=>[...]] 鎴栬匸'except'=>[...]]
+ */
+ protected function beforeAction($method, $options = [])
+ {
+ if (isset($options['only'])) {
+ if (is_string($options['only'])) {
+ $options['only'] = explode(',', $options['only']);
+ }
+ if (!in_array($this->request->action(), $options['only'])) {
+ return;
+ }
+ } elseif (isset($options['except'])) {
+ if (is_string($options['except'])) {
+ $options['except'] = explode(',', $options['except']);
+ }
+ if (in_array($this->request->action(), $options['except'])) {
+ return;
+ }
+ }
+
+ call_user_func([$this, $method]);
+ }
+
+ /**
+ * 鍔犺浇妯℃澘杈撳嚭
+ * @access protected
+ * @param string $template 妯℃澘鏂囦欢鍚
+ * @param array $vars 妯℃澘杈撳嚭鍙橀噺
+ * @param array $replace 妯℃澘鏇挎崲
+ * @param array $config 妯℃澘鍙傛暟
+ * @return mixed
+ */
+ protected function fetch($template = '', $vars = [], $replace = [], $config = [])
+ {
+ return $this->view->fetch($template, $vars, $replace, $config);
+ }
+
+ /**
+ * 娓叉煋鍐呭杈撳嚭
+ * @access protected
+ * @param string $content 妯℃澘鍐呭
+ * @param array $vars 妯℃澘杈撳嚭鍙橀噺
+ * @param array $replace 鏇挎崲鍐呭
+ * @param array $config 妯℃澘鍙傛暟
+ * @return mixed
+ */
+ protected function display($content = '', $vars = [], $replace = [], $config = [])
+ {
+ return $this->view->display($content, $vars, $replace, $config);
+ }
+
+ /**
+ * 妯℃澘鍙橀噺璧嬪
+ * @access protected
+ * @param mixed $name 瑕佹樉绀虹殑妯℃澘鍙橀噺
+ * @param mixed $value 鍙橀噺鐨勫
+ * @return void
+ */
+ protected function assign($name, $value = '')
+ {
+ $this->view->assign($name, $value);
+ }
+
+ /**
+ * 鍒濆鍖栨ā鏉垮紩鎿
+ * @access protected
+ * @param array|string $engine 寮曟搸鍙傛暟
+ * @return void
+ */
+ protected function engine($engine)
+ {
+ $this->view->engine($engine);
+ }
+
+ /**
+ * 璁剧疆楠岃瘉澶辫触鍚庢槸鍚︽姏鍑哄紓甯
+ * @access protected
+ * @param bool $fail 鏄惁鎶涘嚭寮傚父
+ * @return $this
+ */
+ protected function validateFailException($fail = true)
+ {
+ $this->failException = $fail;
+ return $this;
+ }
+
+ /**
+ * 楠岃瘉鏁版嵁
+ * @access protected
+ * @param array $data 鏁版嵁
+ * @param string|array $validate 楠岃瘉鍣ㄥ悕鎴栬呴獙璇佽鍒欐暟缁
+ * @param array $message 鎻愮ず淇℃伅
+ * @param bool $batch 鏄惁鎵归噺楠岃瘉
+ * @param mixed $callback 鍥炶皟鏂规硶锛堥棴鍖咃級
+ * @return array|string|true
+ * @throws ValidateException
+ */
+ protected function validate($data, $validate, $message = [], $batch = false, $callback = null)
+ {
+ if (is_array($validate)) {
+ $v = Loader::validate();
+ $v->rule($validate);
+ } else {
+ if (strpos($validate, '.')) {
+ // 鏀寔鍦烘櫙
+ list($validate, $scene) = explode('.', $validate);
+ }
+ $v = Loader::validate($validate);
+ if (!empty($scene)) {
+ $v->scene($scene);
+ }
+ }
+ // 鏄惁鎵归噺楠岃瘉
+ if ($batch || $this->batchValidate) {
+ $v->batch(true);
+ }
+
+ if (is_array($message)) {
+ $v->message($message);
+ }
+
+ if ($callback && is_callable($callback)) {
+ call_user_func_array($callback, [$v, &$data]);
+ }
+
+ if (!$v->check($data)) {
+ if ($this->failException) {
+ throw new ValidateException($v->getError());
+ } else {
+ return $v->getError();
+ }
+ } else {
+ return true;
+ }
+ }
+}
diff --git a/thinkphp/library/think/Cookie.php b/thinkphp/library/think/Cookie.php
new file mode 100644
index 000000000..93edb8d7b
--- /dev/null
+++ b/thinkphp/library/think/Cookie.php
@@ -0,0 +1,211 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+class Cookie
+{
+ protected static $config = [
+ // cookie 鍚嶇О鍓嶇紑
+ 'prefix' => '',
+ // cookie 淇濆瓨鏃堕棿
+ 'expire' => 0,
+ // cookie 淇濆瓨璺緞
+ 'path' => '/',
+ // cookie 鏈夋晥鍩熷悕
+ 'domain' => '',
+ // cookie 鍚敤瀹夊叏浼犺緭
+ 'secure' => false,
+ // httponly璁剧疆
+ 'httponly' => '',
+ // 鏄惁浣跨敤 setcookie
+ 'setcookie' => true,
+ ];
+
+ protected static $init;
+
+ /**
+ * Cookie鍒濆鍖
+ * @param array $config
+ * @return void
+ */
+ public static function init(array $config = [])
+ {
+ if (empty($config)) {
+ $config = Config::get('cookie');
+ }
+ self::$config = array_merge(self::$config, array_change_key_case($config));
+ if (!empty(self::$config['httponly'])) {
+ ini_set('session.cookie_httponly', 1);
+ }
+ self::$init = true;
+ }
+
+ /**
+ * 璁剧疆鎴栬呰幏鍙朿ookie浣滅敤鍩燂紙鍓嶇紑锛
+ * @param string $prefix
+ * @return string|void
+ */
+ public static function prefix($prefix = '')
+ {
+ if (empty($prefix)) {
+ return self::$config['prefix'];
+ }
+ self::$config['prefix'] = $prefix;
+ }
+
+ /**
+ * Cookie 璁剧疆銆佽幏鍙栥佸垹闄
+ *
+ * @param string $name cookie鍚嶇О
+ * @param mixed $value cookie鍊
+ * @param mixed $option 鍙夊弬鏁 鍙兘浼氭槸 null|integer|string
+ *
+ * @return mixed
+ * @internal param mixed $options cookie鍙傛暟
+ */
+ public static function set($name, $value = '', $option = null)
+ {
+ !isset(self::$init) && self::init();
+ // 鍙傛暟璁剧疆(浼氳鐩栭粰璁よ缃)
+ if (!is_null($option)) {
+ if (is_numeric($option)) {
+ $option = ['expire' => $option];
+ } elseif (is_string($option)) {
+ parse_str($option, $option);
+ }
+ $config = array_merge(self::$config, array_change_key_case($option));
+ } else {
+ $config = self::$config;
+ }
+ $name = $config['prefix'] . $name;
+ // 璁剧疆cookie
+ if (is_array($value)) {
+ array_walk_recursive($value, 'self::jsonFormatProtect', 'encode');
+ $value = 'think:' . json_encode($value);
+ }
+ $expire = !empty($config['expire']) ? $_SERVER['REQUEST_TIME'] + intval($config['expire']) : 0;
+ if ($config['setcookie']) {
+ setcookie($name, $value, $expire, $config['path'], $config['domain'], $config['secure'], $config['httponly']);
+ }
+ $_COOKIE[$name] = $value;
+ }
+
+ /**
+ * 姘镐箙淇濆瓨Cookie鏁版嵁
+ * @param string $name cookie鍚嶇О
+ * @param mixed $value cookie鍊
+ * @param mixed $option 鍙夊弬鏁 鍙兘浼氭槸 null|integer|string
+ * @return void
+ */
+ public static function forever($name, $value = '', $option = null)
+ {
+ if (is_null($option) || is_numeric($option)) {
+ $option = [];
+ }
+ $option['expire'] = 315360000;
+ self::set($name, $value, $option);
+ }
+
+ /**
+ * 鍒ゆ柇Cookie鏁版嵁
+ * @param string $name cookie鍚嶇О
+ * @param string|null $prefix cookie鍓嶇紑
+ * @return bool
+ */
+ public static function has($name, $prefix = null)
+ {
+ !isset(self::$init) && self::init();
+ $prefix = !is_null($prefix) ? $prefix : self::$config['prefix'];
+ $name = $prefix . $name;
+ return isset($_COOKIE[$name]);
+ }
+
+ /**
+ * Cookie鑾峰彇
+ * @param string $name cookie鍚嶇О
+ * @param string|null $prefix cookie鍓嶇紑
+ * @return mixed
+ */
+ public static function get($name, $prefix = null)
+ {
+ !isset(self::$init) && self::init();
+ $prefix = !is_null($prefix) ? $prefix : self::$config['prefix'];
+ $name = $prefix . $name;
+ if (isset($_COOKIE[$name])) {
+ $value = $_COOKIE[$name];
+ if (0 === strpos($value, 'think:')) {
+ $value = substr($value, 6);
+ $value = json_decode($value, true);
+ array_walk_recursive($value, 'self::jsonFormatProtect', 'decode');
+ }
+ return $value;
+ } else {
+ return;
+ }
+ }
+
+ /**
+ * Cookie鍒犻櫎
+ * @param string $name cookie鍚嶇О
+ * @param string|null $prefix cookie鍓嶇紑
+ * @return mixed
+ */
+ public static function delete($name, $prefix = null)
+ {
+ !isset(self::$init) && self::init();
+ $config = self::$config;
+ $prefix = !is_null($prefix) ? $prefix : $config['prefix'];
+ $name = $prefix . $name;
+ if ($config['setcookie']) {
+ setcookie($name, '', $_SERVER['REQUEST_TIME'] - 3600, $config['path'], $config['domain'], $config['secure'], $config['httponly']);
+ }
+ // 鍒犻櫎鎸囧畾cookie
+ unset($_COOKIE[$name]);
+ }
+
+ /**
+ * Cookie娓呯┖
+ * @param string|null $prefix cookie鍓嶇紑
+ * @return mixed
+ */
+ public static function clear($prefix = null)
+ {
+ // 娓呴櫎鎸囧畾鍓嶇紑鐨勬墍鏈塩ookie
+ if (empty($_COOKIE)) {
+ return;
+ }
+ !isset(self::$init) && self::init();
+ // 瑕佸垹闄ょ殑cookie鍓嶇紑锛屼笉鎸囧畾鍒欏垹闄onfig璁剧疆鐨勬寚瀹氬墠缂
+ $config = self::$config;
+ $prefix = !is_null($prefix) ? $prefix : $config['prefix'];
+ if ($prefix) {
+ // 濡傛灉鍓嶇紑涓虹┖瀛楃涓插皢涓嶄綔澶勭悊鐩存帴杩斿洖
+ foreach ($_COOKIE as $key => $val) {
+ if (0 === strpos($key, $prefix)) {
+ if ($config['setcookie']) {
+ setcookie($key, '', $_SERVER['REQUEST_TIME'] - 3600, $config['path'], $config['domain'], $config['secure'], $config['httponly']);
+ }
+ unset($_COOKIE[$key]);
+ }
+ }
+ }
+ return;
+ }
+
+ private static function jsonFormatProtect(&$val, $key, $type = 'encode')
+ {
+ if (!empty($val) && true !== $val) {
+ $val = 'decode' == $type ? urldecode($val) : urlencode($val);
+ }
+ }
+
+}
diff --git a/thinkphp/library/think/Db.php b/thinkphp/library/think/Db.php
new file mode 100644
index 000000000..00f719e04
--- /dev/null
+++ b/thinkphp/library/think/Db.php
@@ -0,0 +1,151 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+use think\db\Connection;
+use think\db\Query;
+
+/**
+ * Class Db
+ * @package think
+ * @method Query table(string $table) static 鎸囧畾鏁版嵁琛紙鍚墠缂锛
+ * @method Query name(string $name) static 鎸囧畾鏁版嵁琛紙涓嶅惈鍓嶇紑锛
+ * @method Query where(mixed $field, string $op = null, mixed $condition = null) static 鏌ヨ鏉′欢
+ * @method Query join(mixed $join, mixed $condition = null, string $type = 'INNER') static JOIN鏌ヨ
+ * @method Query union(mixed $union, boolean $all = false) static UNION鏌ヨ
+ * @method Query limit(mixed $offset, integer $length = null) static 鏌ヨLIMIT
+ * @method Query order(mixed $field, string $order = null) static 鏌ヨORDER
+ * @method Query cache(mixed $key = null , integer $expire = null) static 璁剧疆鏌ヨ缂撳瓨
+ * @method mixed value(string $field) static 鑾峰彇鏌愪釜瀛楁鐨勫
+ * @method array column(string $field, string $key = '') static 鑾峰彇鏌愪釜鍒楃殑鍊
+ * @method Query view(mixed $join, mixed $field = null, mixed $on = null, string $type = 'INNER') static 瑙嗗浘鏌ヨ
+ * @method mixed find(mixed $data = null) static 鏌ヨ鍗曚釜璁板綍
+ * @method mixed select(mixed $data = null) static 鏌ヨ澶氫釜璁板綍
+ * @method integer insert(array $data, boolean $replace = false, boolean $getLastInsID = false, string $sequence = null) static 鎻掑叆涓鏉¤褰
+ * @method integer insertGetId(array $data, boolean $replace = false, string $sequence = null) static 鎻掑叆涓鏉¤褰曞苟杩斿洖鑷ID
+ * @method integer insertAll(array $dataSet) static 鎻掑叆澶氭潯璁板綍
+ * @method integer update(array $data) static 鏇存柊璁板綍
+ * @method integer delete(mixed $data = null) static 鍒犻櫎璁板綍
+ * @method boolean chunk(integer $count, callable $callback, string $column = null) static 鍒嗗潡鑾峰彇鏁版嵁
+ * @method mixed query(string $sql, array $bind = [], boolean $fetch = false, boolean $master = false, mixed $class = null) static SQL鏌ヨ
+ * @method integer execute(string $sql, array $bind = [], boolean $fetch = false, boolean $getLastInsID = false, string $sequence = null) static SQL鎵ц
+ * @method Paginator paginate(integer $listRows = 15, mixed $simple = null, array $config = []) static 鍒嗛〉鏌ヨ
+ * @method mixed transaction(callable $callback) static 鎵ц鏁版嵁搴撲簨鍔
+ * @method void startTrans() static 鍚姩浜嬪姟
+ * @method void commit() static 鐢ㄤ簬闈炶嚜鍔ㄦ彁浜ょ姸鎬佷笅闈㈢殑鏌ヨ鎻愪氦
+ * @method void rollback() static 浜嬪姟鍥炴粴
+ * @method boolean batchQuery(array $sqlArray) static 鎵瑰鐞嗘墽琛孲QL璇彞
+ */
+class Db
+{
+ // 鏁版嵁搴撹繛鎺ュ疄渚
+ private static $instance = [];
+ // 鏌ヨ娆℃暟
+ public static $queryTimes = 0;
+ // 鎵ц娆℃暟
+ public static $executeTimes = 0;
+
+ /**
+ * 鏁版嵁搴撳垵濮嬪寲 骞跺彇寰楁暟鎹簱绫诲疄渚
+ * @static
+ * @access public
+ * @param mixed $config 杩炴帴閰嶇疆
+ * @param bool|string $name 杩炴帴鏍囪瘑 true 寮哄埗閲嶆柊杩炴帴
+ * @return Connection
+ * @throws Exception
+ */
+ public static function connect($config = [], $name = false)
+ {
+ if (false === $name) {
+ $name = md5(serialize($config));
+ }
+ if (true === $name || !isset(self::$instance[$name])) {
+ // 瑙f瀽杩炴帴鍙傛暟 鏀寔鏁扮粍鍜屽瓧绗︿覆
+ $options = self::parseConfig($config);
+ if (empty($options['type'])) {
+ throw new \InvalidArgumentException('Underfined db type');
+ }
+ $class = false !== strpos($options['type'], '\\') ? $options['type'] : '\\think\\db\\connector\\' . ucwords($options['type']);
+ // 璁板綍鍒濆鍖栦俊鎭
+ if (App::$debug) {
+ Log::record('[ DB ] INIT ' . $options['type'], 'info');
+ }
+ if (true === $name) {
+ return new $class($options);
+ } else {
+ self::$instance[$name] = new $class($options);
+ }
+ }
+ return self::$instance[$name];
+ }
+
+ /**
+ * 鏁版嵁搴撹繛鎺ュ弬鏁拌В鏋
+ * @static
+ * @access private
+ * @param mixed $config
+ * @return array
+ */
+ private static function parseConfig($config)
+ {
+ if (empty($config)) {
+ $config = Config::get('database');
+ } elseif (is_string($config) && false === strpos($config, '/')) {
+ // 鏀寔璇诲彇閰嶇疆鍙傛暟
+ $config = Config::get($config);
+ }
+ if (is_string($config)) {
+ return self::parseDsn($config);
+ } else {
+ return $config;
+ }
+ }
+
+ /**
+ * DSN瑙f瀽
+ * 鏍煎紡锛 mysql://username:passwd@localhost:3306/DbName?param1=val1¶m2=val2#utf8
+ * @static
+ * @access private
+ * @param string $dsnStr
+ * @return array
+ */
+ private static function parseDsn($dsnStr)
+ {
+ $info = parse_url($dsnStr);
+ if (!$info) {
+ return [];
+ }
+ $dsn = [
+ 'type' => $info['scheme'],
+ 'username' => isset($info['user']) ? $info['user'] : '',
+ 'password' => isset($info['pass']) ? $info['pass'] : '',
+ 'hostname' => isset($info['host']) ? $info['host'] : '',
+ 'hostport' => isset($info['port']) ? $info['port'] : '',
+ 'database' => !empty($info['path']) ? ltrim($info['path'], '/') : '',
+ 'charset' => isset($info['fragment']) ? $info['fragment'] : 'utf8',
+ ];
+
+ if (isset($info['query'])) {
+ parse_str($info['query'], $dsn['params']);
+ } else {
+ $dsn['params'] = [];
+ }
+ return $dsn;
+ }
+
+ // 璋冪敤椹卞姩绫荤殑鏂规硶
+ public static function __callStatic($method, $params)
+ {
+ // 鑷姩鍒濆鍖栨暟鎹簱
+ return call_user_func_array([self::connect(), $method], $params);
+ }
+}
diff --git a/thinkphp/library/think/Debug.php b/thinkphp/library/think/Debug.php
new file mode 100644
index 000000000..844dfb89f
--- /dev/null
+++ b/thinkphp/library/think/Debug.php
@@ -0,0 +1,212 @@
+
+// +----------------------------------------------------------------------
+
+namespace think;
+
+use think\exception\ClassNotFoundException;
+use think\response\Redirect;
+
+class Debug
+{
+ // 鍖洪棿鏃堕棿淇℃伅
+ protected static $info = [];
+ // 鍖洪棿鍐呭瓨淇℃伅
+ protected static $mem = [];
+
+ /**
+ * 璁板綍鏃堕棿锛堝井绉掞級鍜屽唴瀛樹娇鐢ㄦ儏鍐
+ * @param string $name 鏍囪浣嶇疆
+ * @param mixed $value 鏍囪鍊 鐣欑┖鍒欏彇褰撳墠 time 琛ㄧず浠呰褰曟椂闂 鍚﹀垯鍚屾椂璁板綍鏃堕棿鍜屽唴瀛
+ * @return mixed
+ */
+ public static function remark($name, $value = '')
+ {
+ // 璁板綍鏃堕棿鍜屽唴瀛樹娇鐢
+ self::$info[$name] = is_float($value) ? $value : microtime(true);
+ if ('time' != $value) {
+ self::$mem['mem'][$name] = is_float($value) ? $value : memory_get_usage();
+ self::$mem['peak'][$name] = memory_get_peak_usage();
+ }
+ }
+
+ /**
+ * 缁熻鏌愪釜鍖洪棿鐨勬椂闂达紙寰锛変娇鐢ㄦ儏鍐
+ * @param string $start 寮濮嬫爣绛
+ * @param string $end 缁撴潫鏍囩
+ * @param integer|string $dec 灏忔暟浣
+ * @return integer
+ */
+ public static function getRangeTime($start, $end, $dec = 6)
+ {
+ if (!isset(self::$info[$end])) {
+ self::$info[$end] = microtime(true);
+ }
+ return number_format((self::$info[$end] - self::$info[$start]), $dec);
+ }
+
+ /**
+ * 缁熻浠庡紑濮嬪埌缁熻鏃剁殑鏃堕棿锛堝井绉掞級浣跨敤鎯呭喌
+ * @param integer|string $dec 灏忔暟浣
+ * @return integer
+ */
+ public static function getUseTime($dec = 6)
+ {
+ return number_format((microtime(true) - THINK_START_TIME), $dec);
+ }
+
+ /**
+ * 鑾峰彇褰撳墠璁块棶鐨勫悶鍚愮巼鎯呭喌
+ * @return string
+ */
+ public static function getThroughputRate()
+ {
+ return number_format(1 / self::getUseTime(), 2) . 'req/s';
+ }
+
+ /**
+ * 璁板綍鍖洪棿鐨勫唴瀛樹娇鐢ㄦ儏鍐
+ * @param string $start 寮濮嬫爣绛
+ * @param string $end 缁撴潫鏍囩
+ * @param integer|string $dec 灏忔暟浣
+ * @return string
+ */
+ public static function getRangeMem($start, $end, $dec = 2)
+ {
+ if (!isset(self::$mem['mem'][$end])) {
+ self::$mem['mem'][$end] = memory_get_usage();
+ }
+ $size = self::$mem['mem'][$end] - self::$mem['mem'][$start];
+ $a = ['B', 'KB', 'MB', 'GB', 'TB'];
+ $pos = 0;
+ while ($size >= 1024) {
+ $size /= 1024;
+ $pos++;
+ }
+ return round($size, $dec) . " " . $a[$pos];
+ }
+
+ /**
+ * 缁熻浠庡紑濮嬪埌缁熻鏃剁殑鍐呭瓨浣跨敤鎯呭喌
+ * @param integer|string $dec 灏忔暟浣
+ * @return string
+ */
+ public static function getUseMem($dec = 2)
+ {
+ $size = memory_get_usage() - THINK_START_MEM;
+ $a = ['B', 'KB', 'MB', 'GB', 'TB'];
+ $pos = 0;
+ while ($size >= 1024) {
+ $size /= 1024;
+ $pos++;
+ }
+ return round($size, $dec) . " " . $a[$pos];
+ }
+
+ /**
+ * 缁熻鍖洪棿鐨勫唴瀛樺嘲鍊兼儏鍐
+ * @param string $start 寮濮嬫爣绛
+ * @param string $end 缁撴潫鏍囩
+ * @param integer|string $dec 灏忔暟浣
+ * @return mixed
+ */
+ public static function getMemPeak($start, $end, $dec = 2)
+ {
+ if (!isset(self::$mem['peak'][$end])) {
+ self::$mem['peak'][$end] = memory_get_peak_usage();
+ }
+ $size = self::$mem['peak'][$end] - self::$mem['peak'][$start];
+ $a = ['B', 'KB', 'MB', 'GB', 'TB'];
+ $pos = 0;
+ while ($size >= 1024) {
+ $size /= 1024;
+ $pos++;
+ }
+ return round($size, $dec) . " " . $a[$pos];
+ }
+
+ /**
+ * 鑾峰彇鏂囦欢鍔犺浇淇℃伅
+ * @param bool $detail 鏄惁鏄剧ず璇︾粏
+ * @return integer|array
+ */
+ public static function getFile($detail = false)
+ {
+ if ($detail) {
+ $files = get_included_files();
+ $info = [];
+ foreach ($files as $key => $file) {
+ $info[] = $file . ' ( ' . number_format(filesize($file) / 1024, 2) . ' KB )';
+ }
+ return $info;
+ }
+ return count(get_included_files());
+ }
+
+ /**
+ * 娴忚鍣ㄥ弸濂界殑鍙橀噺杈撳嚭
+ * @param mixed $var 鍙橀噺
+ * @param boolean $echo 鏄惁杈撳嚭 榛樿涓簍rue 濡傛灉涓篺alse 鍒欒繑鍥炶緭鍑哄瓧绗︿覆
+ * @param string $label 鏍囩 榛樿涓虹┖
+ * @param integer $flags htmlspecialchars flags
+ * @return void|string
+ */
+ public static function dump($var, $echo = true, $label = null, $flags = ENT_SUBSTITUTE)
+ {
+ $label = (null === $label) ? '' : rtrim($label) . ':';
+ ob_start();
+ var_dump($var);
+ $output = ob_get_clean();
+ $output = preg_replace('/\]\=\>\n(\s+)/m', '] => ', $output);
+ if (IS_CLI) {
+ $output = PHP_EOL . $label . $output . PHP_EOL;
+ } else {
+ if (!extension_loaded('xdebug')) {
+ $output = htmlspecialchars($output, $flags);
+ }
+ $output = '' . $label . $output . '
';
+ }
+ if ($echo) {
+ echo($output);
+ return;
+ } else {
+ return $output;
+ }
+ }
+
+ public static function inject(Response $response, &$content)
+ {
+ $config = Config::get('trace');
+ $type = isset($config['type']) ? $config['type'] : 'Html';
+ $request = Request::instance();
+ $class = false !== strpos($type, '\\') ? $type : '\\think\\debug\\' . ucwords($type);
+ unset($config['type']);
+ if (class_exists($class)) {
+ $trace = new $class($config);
+ } else {
+ throw new ClassNotFoundException('class not exists:' . $class, $class);
+ }
+
+ if ($response instanceof Redirect) {
+ //TODO 璁板綍
+ } else {
+ $output = $trace->output($response, Log::getLog());
+ if (is_string($output)) {
+ // trace璋冭瘯淇℃伅娉ㄥ叆
+ $pos = strripos($content, '