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 +=============== + +[![StyleCI](https://styleci.io/repos/48530411/shield?style=flat&branch=master)](https://styleci.io/repos/48530411) +[![Build Status](https://travis-ci.org/top-think/framework.svg?branch=master)](https://travis-ci.org/top-think/framework) +[![codecov.io](http://codecov.io/github/top-think/framework/coverage.svg?branch=master)](http://codecov.io/github/github/top-think/framework?branch=master) +[![Total Downloads](https://poser.pugx.org/topthink/framework/downloads)](https://packagist.org/packages/topthink/framework) +[![Latest Stable Version](https://poser.pugx.org/topthink/framework/v/stable)](https://packagist.org/packages/topthink/framework) +[![Latest Unstable Version](https://poser.pugx.org/topthink/framework/v/unstable)](https://packagist.org/packages/topthink/framework) +[![License](https://poser.pugx.org/topthink/framework/license)](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, ''); + if (false !== $pos) { + $content = substr($content, 0, $pos) . $output . substr($content, $pos); + } else { + $content = $content . $output; + } + } + } + } +} diff --git a/thinkphp/library/think/Env.php b/thinkphp/library/think/Env.php new file mode 100644 index 000000000..6f31841d5 --- /dev/null +++ b/thinkphp/library/think/Env.php @@ -0,0 +1,31 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class Env +{ + /** + * 鑾峰彇鐜鍙橀噺鍊 + * @param string $name 鐜鍙橀噺鍚嶏紙鏀寔浜岀骇 .鍙峰垎鍓诧級 + * @param string $default 榛樿鍊 + * @return mixed + */ + public static function get($name, $default = null) + { + $result = getenv(ENV_PREFIX . strtoupper(str_replace('.', '_', $name))); + if (false !== $result) { + return $result; + } else { + return $default; + } + } +} diff --git a/thinkphp/library/think/Error.php b/thinkphp/library/think/Error.php new file mode 100644 index 000000000..c17292bf5 --- /dev/null +++ b/thinkphp/library/think/Error.php @@ -0,0 +1,117 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\console\Output as ConsoleOutput; +use think\exception\ErrorException; +use think\exception\Handle; +use think\exception\ThrowableError; + +class Error +{ + /** + * 娉ㄥ唽寮傚父澶勭悊 + * @return void + */ + public static function register() + { + error_reporting(E_ALL); + set_error_handler([__CLASS__, 'appError']); + set_exception_handler([__CLASS__, 'appException']); + register_shutdown_function([__CLASS__, 'appShutdown']); + } + + /** + * Exception Handler + * @param \Exception|\Throwable $e + */ + public static function appException($e) + { + if (!$e instanceof \Exception) { + $e = new ThrowableError($e); + } + + self::getExceptionHandler()->report($e); + if (IS_CLI) { + self::getExceptionHandler()->renderForConsole(new ConsoleOutput, $e); + } else { + self::getExceptionHandler()->render($e)->send(); + } + } + + /** + * Error Handler + * @param integer $errno 閿欒缂栧彿 + * @param integer $errstr 璇︾粏閿欒淇℃伅 + * @param string $errfile 鍑洪敊鐨勬枃浠 + * @param integer $errline 鍑洪敊琛屽彿 + * @param array $errcontext + * @throws ErrorException + */ + public static function appError($errno, $errstr, $errfile = '', $errline = 0, $errcontext = []) + { + $exception = new ErrorException($errno, $errstr, $errfile, $errline, $errcontext); + if (error_reporting() & $errno) { + // 灏嗛敊璇俊鎭墭绠¤嚦 think\exception\ErrorException + throw $exception; + } else { + self::getExceptionHandler()->report($exception); + } + } + + /** + * Shutdown Handler + */ + public static function appShutdown() + { + if (!is_null($error = error_get_last()) && self::isFatal($error['type'])) { + // 灏嗛敊璇俊鎭墭绠¤嚦think\ErrorException + $exception = new ErrorException($error['type'], $error['message'], $error['file'], $error['line']); + + self::appException($exception); + } + + // 鍐欏叆鏃ュ織 + Log::save(); + } + + /** + * 纭畾閿欒绫诲瀷鏄惁鑷村懡 + * + * @param int $type + * @return bool + */ + protected static function isFatal($type) + { + return in_array($type, [E_ERROR, E_CORE_ERROR, E_COMPILE_ERROR, E_PARSE]); + } + + /** + * Get an instance of the exception handler. + * + * @return Handle + */ + public static function getExceptionHandler() + { + static $handle; + if (!$handle) { + // 寮傚父澶勭悊handle + $class = Config::get('exception_handle'); + if ($class && class_exists($class) && is_subclass_of($class, "\\think\\exception\\Handle")) { + $handle = new $class; + } else { + $handle = new Handle; + } + } + return $handle; + } +} diff --git a/thinkphp/library/think/Exception.php b/thinkphp/library/think/Exception.php new file mode 100644 index 000000000..034c85b64 --- /dev/null +++ b/thinkphp/library/think/Exception.php @@ -0,0 +1,54 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class Exception extends \Exception +{ + + /** + * 淇濆瓨寮傚父椤甸潰鏄剧ず鐨勯澶朌ebug鏁版嵁 + * @var array + */ + protected $data = []; + + /** + * 璁剧疆寮傚父棰濆鐨凞ebug鏁版嵁 + * 鏁版嵁灏嗕細鏄剧ず涓轰笅闈㈢殑鏍煎紡 + * + * Exception Data + * -------------------------------------------------- + * Label 1 + * key1 value1 + * key2 value2 + * Label 2 + * key1 value1 + * key2 value2 + * + * @param string $label 鏁版嵁鍒嗙被锛岀敤浜庡紓甯搁〉闈㈡樉绀 + * @param array $data 闇瑕佹樉绀虹殑鏁版嵁锛屽繀椤讳负鍏宠仈鏁扮粍 + */ + final protected function setData($label, array $data) + { + $this->data[$label] = $data; + } + + /** + * 鑾峰彇寮傚父棰濆Debug鏁版嵁 + * 涓昏鐢ㄤ簬杈撳嚭鍒板紓甯搁〉闈究浜庤皟璇 + * @return array 鐢眘etData璁剧疆鐨凞ebug鏁版嵁 + */ + final public function getData() + { + return $this->data; + } + +} diff --git a/thinkphp/library/think/File.php b/thinkphp/library/think/File.php new file mode 100644 index 000000000..ba59273f1 --- /dev/null +++ b/thinkphp/library/think/File.php @@ -0,0 +1,411 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use SplFileInfo; +use SplFileObject; + +class File extends SplFileObject +{ + /** + * 閿欒淇℃伅 + * @var string + */ + private $error = ''; + // 褰撳墠瀹屾暣鏂囦欢鍚 + protected $filename; + // 涓婁紶鏂囦欢鍚 + protected $saveName; + // 鏂囦欢涓婁紶鍛藉悕瑙勫垯 + protected $rule = 'date'; + // 鏂囦欢涓婁紶楠岃瘉瑙勫垯 + protected $validate = []; + // 鍗曞厓娴嬭瘯 + protected $isTest; + // 涓婁紶鏂囦欢淇℃伅 + protected $info; + // 鏂囦欢hash淇℃伅 + protected $hash = []; + + public function __construct($filename, $mode = 'r') + { + parent::__construct($filename, $mode); + $this->filename = $this->getRealPath() ?: $this->getPathname(); + } + + /** + * 鏄惁娴嬭瘯 + * @param bool $test 鏄惁娴嬭瘯 + * @return $this + */ + public function isTest($test = false) + { + $this->isTest = $test; + return $this; + } + + /** + * 璁剧疆涓婁紶淇℃伅 + * @param array $info 涓婁紶鏂囦欢淇℃伅 + * @return $this + */ + public function setUploadInfo($info) + { + $this->info = $info; + return $this; + } + + /** + * 鑾峰彇涓婁紶鏂囦欢鐨勪俊鎭 + * @param string $name + * @return array|string + */ + public function getInfo($name = '') + { + return isset($this->info[$name]) ? $this->info[$name] : $this->info; + } + + /** + * 鑾峰彇涓婁紶鏂囦欢鐨勬枃浠跺悕 + * @return string + */ + public function getSaveName() + { + return $this->saveName; + } + + /** + * 璁剧疆涓婁紶鏂囦欢鐨勪繚瀛樻枃浠跺悕 + * @param string $saveName + * @return $this + */ + public function setSaveName($saveName) + { + $this->saveName = $saveName; + return $this; + } + + /** + * 鑾峰彇鏂囦欢鐨勫搱甯屾暎鍒楀 + * @return $string + */ + public function hash($type = 'sha1') + { + if (!isset($this->hash[$type])) { + $this->hash[$type] = hash_file($type, $this->filename); + } + return $this->hash[$type]; + } + + /** + * 妫鏌ョ洰褰曟槸鍚﹀彲鍐 + * @param string $path 鐩綍 + * @return boolean + */ + protected function checkPath($path) + { + if (is_dir($path)) { + return true; + } + + if (mkdir($path, 0755, true)) { + return true; + } else { + $this->error = "鐩綍 {$path} 鍒涘缓澶辫触锛"; + return false; + } + } + + /** + * 鑾峰彇鏂囦欢绫诲瀷淇℃伅 + * @return string + */ + public function getMime() + { + $finfo = finfo_open(FILEINFO_MIME_TYPE); + return finfo_file($finfo, $this->filename); + } + + /** + * 璁剧疆鏂囦欢鐨勫懡鍚嶈鍒 + * @param string $rule 鏂囦欢鍛藉悕瑙勫垯 + * @return $this + */ + public function rule($rule) + { + $this->rule = $rule; + return $this; + } + + /** + * 璁剧疆涓婁紶鏂囦欢鐨勯獙璇佽鍒 + * @param array $rule 楠岃瘉瑙勫垯 + * @return $this + */ + public function validate($rule = []) + { + $this->validate = $rule; + return $this; + } + + /** + * 妫娴嬫槸鍚﹀悎娉曠殑涓婁紶鏂囦欢 + * @return bool + */ + public function isValid() + { + if ($this->isTest) { + return is_file($this->filename); + } + return is_uploaded_file($this->filename); + } + + /** + * 妫娴嬩笂浼犳枃浠 + * @param array $rule 楠岃瘉瑙勫垯 + * @return bool + */ + public function check($rule = []) + { + $rule = $rule ?: $this->validate; + + /* 妫鏌ユ枃浠跺ぇ灏 */ + if (isset($rule['size']) && !$this->checkSize($rule['size'])) { + $this->error = '涓婁紶鏂囦欢澶у皬涓嶇锛'; + return false; + } + + /* 妫鏌ユ枃浠禡ime绫诲瀷 */ + if (isset($rule['type']) && !$this->checkMime($rule['type'])) { + $this->error = '涓婁紶鏂囦欢MIME绫诲瀷涓嶅厑璁革紒'; + return false; + } + + /* 妫鏌ユ枃浠跺悗缂 */ + if (isset($rule['ext']) && !$this->checkExt($rule['ext'])) { + $this->error = '涓婁紶鏂囦欢鍚庣紑涓嶅厑璁'; + return false; + } + + /* 妫鏌ュ浘鍍忔枃浠 */ + if (!$this->checkImg()) { + $this->error = '闈炴硶鍥惧儚鏂囦欢锛'; + return false; + } + + return true; + } + + /** + * 妫娴嬩笂浼犳枃浠跺悗缂 + * @param array|string $ext 鍏佽鍚庣紑 + * @return bool + */ + public function checkExt($ext) + { + if (is_string($ext)) { + $ext = explode(',', $ext); + } + $extension = strtolower(pathinfo($this->getInfo('name'), PATHINFO_EXTENSION)); + if (!in_array($extension, $ext)) { + return false; + } + return true; + } + + /** + * 妫娴嬪浘鍍忔枃浠 + * @return bool + */ + public function checkImg() + { + $extension = strtolower(pathinfo($this->getInfo('name'), PATHINFO_EXTENSION)); + /* 瀵瑰浘鍍忔枃浠惰繘琛屼弗鏍兼娴 */ + if (in_array($extension, ['gif', 'jpg', 'jpeg', 'bmp', 'png', 'swf']) && !in_array($this->getImageType($this->filename), [1, 2, 3, 4, 6])) { + return false; + } + return true; + } + + // 鍒ゆ柇鍥惧儚绫诲瀷 + protected function getImageType($image) + { + if (function_exists('exif_imagetype')) { + return exif_imagetype($image); + } else { + $info = getimagesize($image); + return $info[2]; + } + } + + /** + * 妫娴嬩笂浼犳枃浠跺ぇ灏 + * @param integer $size 鏈澶уぇ灏 + * @return bool + */ + public function checkSize($size) + { + if ($this->getSize() > $size) { + return false; + } + return true; + } + + /** + * 妫娴嬩笂浼犳枃浠剁被鍨 + * @param array|string $mime 鍏佽绫诲瀷 + * @return bool + */ + public function checkMime($mime) + { + if (is_string($mime)) { + $mime = explode(',', $mime); + } + if (!in_array(strtolower($this->getMime()), $mime)) { + return false; + } + return true; + } + + /** + * 绉诲姩鏂囦欢 + * @param string $path 淇濆瓨璺緞 + * @param string|bool $savename 淇濆瓨鐨勬枃浠跺悕 榛樿鑷姩鐢熸垚 + * @param boolean $replace 鍚屽悕鏂囦欢鏄惁瑕嗙洊 + * @return false|SplFileInfo false-澶辫触 鍚﹀垯杩斿洖SplFileInfo瀹炰緥 + */ + public function move($path, $savename = true, $replace = true) + { + // 鏂囦欢涓婁紶澶辫触锛屾崟鑾烽敊璇唬鐮 + if (!empty($this->info['error'])) { + $this->error($this->info['error']); + return false; + } + + // 妫娴嬪悎娉曟 + if (!$this->isValid()) { + $this->error = '闈炴硶涓婁紶鏂囦欢'; + return false; + } + + // 楠岃瘉涓婁紶 + if (!$this->check()) { + return false; + } + $path = rtrim($path, DS) . DS; + // 鏂囦欢淇濆瓨鍛藉悕瑙勫垯 + $saveName = $this->buildSaveName($savename); + $filename = $path . $saveName; + + // 妫娴嬬洰褰 + if (false === $this->checkPath(dirname($filename))) { + return false; + } + + /* 涓嶈鐩栧悓鍚嶆枃浠 */ + if (!$replace && is_file($filename)) { + $this->error = '瀛樺湪鍚屽悕鏂囦欢' . $filename; + return false; + } + + /* 绉诲姩鏂囦欢 */ + if ($this->isTest) { + rename($this->filename, $filename); + } elseif (!move_uploaded_file($this->filename, $filename)) { + $this->error = '鏂囦欢涓婁紶淇濆瓨閿欒锛'; + return false; + } + // 杩斿洖 File瀵硅薄瀹炰緥 + $file = new self($filename); + $file->setSaveName($saveName); + $file->setUploadInfo($this->info); + return $file; + } + + /** + * 鑾峰彇淇濆瓨鏂囦欢鍚 + * @param string|bool $savename 淇濆瓨鐨勬枃浠跺悕 榛樿鑷姩鐢熸垚 + * @return string + */ + protected function buildSaveName($savename) + { + if (true === $savename) { + // 鑷姩鐢熸垚鏂囦欢鍚 + if ($this->rule instanceof \Closure) { + $savename = call_user_func_array($this->rule, [$this]); + } else { + switch ($this->rule) { + case 'date': + $savename = date('Ymd') . DS . md5(microtime(true)); + break; + default: + if (in_array($this->rule, hash_algos())) { + $hash = $this->hash($this->rule); + $savename = substr($hash, 0, 2) . DS . substr($hash, 2); + } elseif (is_callable($this->rule)) { + $savename = call_user_func($this->rule); + } else { + $savename = date('Ymd') . DS . md5(microtime(true)); + } + } + } + } elseif ('' === $savename) { + $savename = $this->getInfo('name'); + } + if (!strpos($savename, '.')) { + $savename .= '.' . pathinfo($this->getInfo('name'), PATHINFO_EXTENSION); + } + return $savename; + } + + /** + * 鑾峰彇閿欒浠g爜淇℃伅 + * @param int $errorNo 閿欒鍙 + */ + private function error($errorNo) + { + switch ($errorNo) { + case 1: + case 2: + $this->error = '涓婁紶鏂囦欢澶у皬瓒呰繃浜嗘渶澶у硷紒'; + break; + case 3: + $this->error = '鏂囦欢鍙湁閮ㄥ垎琚笂浼狅紒'; + break; + case 4: + $this->error = '娌℃湁鏂囦欢琚笂浼狅紒'; + break; + case 6: + $this->error = '鎵句笉鍒颁复鏃舵枃浠跺す锛'; + break; + case 7: + $this->error = '鏂囦欢鍐欏叆澶辫触锛'; + break; + default: + $this->error = '鏈煡涓婁紶閿欒锛'; + } + } + + /** + * 鑾峰彇閿欒淇℃伅 + * @return mixed + */ + public function getError() + { + return $this->error; + } + + public function __call($method, $args) + { + return $this->hash($method); + } +} diff --git a/thinkphp/library/think/Hook.php b/thinkphp/library/think/Hook.php new file mode 100644 index 000000000..f06196e4d --- /dev/null +++ b/thinkphp/library/think/Hook.php @@ -0,0 +1,136 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class Hook +{ + + private static $tags = []; + + /** + * 鍔ㄦ佹坊鍔犺涓烘墿灞曞埌鏌愪釜鏍囩 + * @param string $tag 鏍囩鍚嶇О + * @param mixed $behavior 琛屼负鍚嶇О + * @param bool $first 鏄惁鏀惧埌寮澶存墽琛 + * @return void + */ + public static function add($tag, $behavior, $first = false) + { + isset(self::$tags[$tag]) || self::$tags[$tag] = []; + if (is_array($behavior) && !is_callable($behavior)) { + if (!array_key_exists('_overlay', $behavior) || !$behavior['_overlay']) { + unset($behavior['_overlay']); + self::$tags[$tag] = array_merge(self::$tags[$tag], $behavior); + } else { + unset($behavior['_overlay']); + self::$tags[$tag] = $behavior; + } + } elseif ($first) { + array_unshift(self::$tags[$tag], $behavior); + } else { + self::$tags[$tag][] = $behavior; + } + } + + /** + * 鎵归噺瀵煎叆鎻掍欢 + * @param array $tags 鎻掍欢淇℃伅 + * @param boolean $recursive 鏄惁閫掑綊鍚堝苟 + */ + public static function import(array $tags, $recursive = true) + { + if ($recursive) { + foreach ($tags as $tag => $behavior) { + self::add($tag, $behavior); + } + } else { + self::$tags = $tags + self::$tags; + } + } + + /** + * 鑾峰彇鎻掍欢淇℃伅 + * @param string $tag 鎻掍欢浣嶇疆 鐣欑┖鑾峰彇鍏ㄩ儴 + * @return array + */ + public static function get($tag = '') + { + if (empty($tag)) { + //鑾峰彇鍏ㄩ儴鐨勬彃浠朵俊鎭 + return self::$tags; + } else { + return array_key_exists($tag, self::$tags) ? self::$tags[$tag] : []; + } + } + + /** + * 鐩戝惉鏍囩鐨勮涓 + * @param string $tag 鏍囩鍚嶇О + * @param mixed $params 浼犲叆鍙傛暟 + * @param mixed $extra 棰濆鍙傛暟 + * @param bool $once 鍙幏鍙栦竴涓湁鏁堣繑鍥炲 + * @return mixed + */ + public static function listen($tag, &$params = null, $extra = null, $once = false) + { + $results = []; + $tags = static::get($tag); + foreach ($tags as $key => $name) { + $results[$key] = self::exec($name, $tag, $params, $extra); + if (false === $results[$key]) { + // 濡傛灉杩斿洖false 鍒欎腑鏂涓烘墽琛 + break; + } elseif (!is_null($results[$key]) && $once) { + break; + } + } + return $once ? end($results) : $results; + } + + /** + * 鎵ц鏌愪釜琛屼负 + * @param mixed $class 瑕佹墽琛岀殑琛屼负 + * @param string $tag 鏂规硶鍚嶏紙鏍囩鍚嶏級 + * @param Mixed $params 浼犱汉鐨勫弬鏁 + * @param mixed $extra 棰濆鍙傛暟 + * @return mixed + */ + public static function exec($class, $tag = '', &$params = null, $extra = null) + { + App::$debug && Debug::remark('behavior_start', 'time'); + $method = Loader::parseName($tag, 1, false); + if ($class instanceof \Closure) { + $result = call_user_func_array($class, [ & $params, $extra]); + $class = 'Closure'; + } elseif (is_array($class)) { + list($class, $method) = $class; + + $result = (new $class())->$method($params, $extra); + $class = $class . '->' . $method; + } elseif (is_object($class)) { + $result = $class->$method($params, $extra); + $class = get_class($class); + } elseif (strpos($class, '::')) { + $result = call_user_func_array($class, [ & $params, $extra]); + } else { + $obj = new $class(); + $method = ($tag && is_callable([$obj, $method])) ? $method : 'run'; + $result = $obj->$method($params, $extra); + } + if (App::$debug) { + Debug::remark('behavior_end', 'time'); + Log::record('[ BEHAVIOR ] Run ' . $class . ' @' . $tag . ' [ RunTime:' . Debug::getRangeTime('behavior_start', 'behavior_end') . 's ]', 'info'); + } + return $result; + } + +} diff --git a/thinkphp/library/think/Lang.php b/thinkphp/library/think/Lang.php new file mode 100644 index 000000000..d52f1947d --- /dev/null +++ b/thinkphp/library/think/Lang.php @@ -0,0 +1,217 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class Lang +{ + // 璇█鏁版嵁 + private static $lang = []; + // 璇█浣滅敤鍩 + private static $range = 'zh-cn'; + // 璇█鑷姩渚︽祴鐨勫彉閲 + protected static $langDetectVar = 'lang'; + // 璇█Cookie鍙橀噺 + protected static $langCookieVar = 'think_var'; + // 璇█Cookie鐨勮繃鏈熸椂闂 + protected static $langCookieExpire = 3600; + // 鍏佽璇█鍒楄〃 + protected static $allowLangList = []; + + // 璁惧畾褰撳墠鐨勮瑷 + public static function range($range = '') + { + if ('' == $range) { + return self::$range; + } else { + self::$range = $range; + } + } + + /** + * 璁剧疆璇█瀹氫箟(涓嶅尯鍒嗗ぇ灏忓啓) + * @param string|array $name 璇█鍙橀噺 + * @param string $value 璇█鍊 + * @param string $range 璇█浣滅敤鍩 + * @return mixed + */ + public static function set($name, $value = null, $range = '') + { + $range = $range ?: self::$range; + // 鎵归噺瀹氫箟 + if (!isset(self::$lang[$range])) { + self::$lang[$range] = []; + } + if (is_array($name)) { + return self::$lang[$range] = array_change_key_case($name) + self::$lang[$range]; + } else { + return self::$lang[$range][strtolower($name)] = $value; + } + } + + /** + * 鍔犺浇璇█瀹氫箟(涓嶅尯鍒嗗ぇ灏忓啓) + * @param string $file 璇█鏂囦欢 + * @param string $range 璇█浣滅敤鍩 + * @return mixed + */ + public static function load($file, $range = '') + { + $range = $range ?: self::$range; + if (!isset(self::$lang[$range])) { + self::$lang[$range] = []; + } + // 鎵归噺瀹氫箟 + if (is_string($file)) { + $file = [$file]; + } + $lang = []; + foreach ($file as $_file) { + if (is_file($_file)) { + // 璁板綍鍔犺浇淇℃伅 + App::$debug && Log::record('[ LANG ] ' . $_file, 'info'); + $_lang = include $_file; + if (is_array($_lang)) { + $lang = array_change_key_case($_lang) + $lang; + } + } + } + if (!empty($lang)) { + self::$lang[$range] = $lang + self::$lang[$range]; + } + return self::$lang[$range]; + } + + /** + * 鑾峰彇璇█瀹氫箟(涓嶅尯鍒嗗ぇ灏忓啓) + * @param string|null $name 璇█鍙橀噺 + * @param array $vars 鍙橀噺鏇挎崲 + * @param string $range 璇█浣滅敤鍩 + * @return mixed + */ + public static function has($name, $range = '') + { + $range = $range ?: self::$range; + return isset(self::$lang[$range][strtolower($name)]); + } + + /** + * 鑾峰彇璇█瀹氫箟(涓嶅尯鍒嗗ぇ灏忓啓) + * @param string|null $name 璇█鍙橀噺 + * @param array $vars 鍙橀噺鏇挎崲 + * @param string $range 璇█浣滅敤鍩 + * @return mixed + */ + public static function get($name = null, $vars = [], $range = '') + { + $range = $range ?: self::$range; + // 绌哄弬鏁拌繑鍥炴墍鏈夊畾涔 + if (empty($name)) { + return self::$lang[$range]; + } + $key = strtolower($name); + $value = isset(self::$lang[$range][$key]) ? self::$lang[$range][$key] : $name; + + // 鍙橀噺瑙f瀽 + if (!empty($vars) && is_array($vars)) { + /** + * Notes: + * 涓轰簡妫娴嬬殑鏂逛究锛屾暟瀛楃储寮曠殑鍒ゆ柇浠呬粎鏄弬鏁版暟缁勭殑绗竴涓厓绱犵殑key涓烘暟瀛0 + * 鏁板瓧绱㈠紩閲囩敤鐨勬槸绯荤粺鐨 sprintf 鍑芥暟鏇挎崲锛岀敤娉曡鍙傝 sprintf 鍑芥暟 + */ + if (key($vars) === 0) { + // 鏁板瓧绱㈠紩瑙f瀽 + array_unshift($vars, $value); + $value = call_user_func_array('sprintf', $vars); + } else { + // 鍏宠仈绱㈠紩瑙f瀽 + $replace = array_keys($vars); + foreach ($replace as &$v) { + $v = "{:{$v}}"; + } + $value = str_replace($replace, $vars, $value); + } + + } + return $value; + } + + /** + * 鑷姩渚︽祴璁剧疆鑾峰彇璇█閫夋嫨 + * @return string + */ + public static function detect() + { + // 鑷姩渚︽祴璁剧疆鑾峰彇璇█閫夋嫨 + $langSet = ''; + if (isset($_GET[self::$langDetectVar])) { + // url涓缃簡璇█鍙橀噺 + $langSet = strtolower($_GET[self::$langDetectVar]); + Cookie::set(self::$langCookieVar, $langSet, self::$langCookieExpire); + } elseif (Cookie::get(self::$langCookieVar)) { + // 鑾峰彇涓婃鐢ㄦ埛鐨勯夋嫨 + $langSet = strtolower(Cookie::get(self::$langCookieVar)); + } elseif (isset($_SERVER['HTTP_ACCEPT_LANGUAGE'])) { + // 鑷姩渚︽祴娴忚鍣ㄨ瑷 + preg_match('/^([a-z\d\-]+)/i', $_SERVER['HTTP_ACCEPT_LANGUAGE'], $matches); + $langSet = strtolower($matches[1]); + Cookie::set(self::$langCookieVar, $langSet, self::$langCookieExpire); + } + if (empty(self::$allowLangList) || in_array($langSet, self::$allowLangList)) { + // 鍚堟硶鐨勮瑷 + self::$range = $langSet ?: self::$range; + } + if ('zh-hans-cn' == self::$range) { + self::$range = 'zh-cn'; + } + return self::$range; + } + + /** + * 璁剧疆璇█鑷姩渚︽祴鐨勫彉閲 + * @param string $var 鍙橀噺鍚嶇О + * @return void + */ + public static function setLangDetectVar($var) + { + self::$langDetectVar = $var; + } + + /** + * 璁剧疆璇█鐨刢ookie淇濆瓨鍙橀噺 + * @param string $var 鍙橀噺鍚嶇О + * @return void + */ + public static function setLangCookieVar($var) + { + self::$langCookieVar = $var; + } + + /** + * 璁剧疆璇█鐨刢ookie鐨勮繃鏈熸椂闂 + * @param string $expire 杩囨湡鏃堕棿 + * @return void + */ + public static function setLangCookieExpire($expire) + { + self::$langCookieExpire = $expire; + } + + /** + * 璁剧疆鍏佽鐨勮瑷鍒楄〃 + * @param array $list 璇█鍒楄〃 + * @return void + */ + public static function setAllowLangList($list) + { + self::$allowLangList = $list; + } +} diff --git a/thinkphp/library/think/Loader.php b/thinkphp/library/think/Loader.php new file mode 100644 index 000000000..70de31753 --- /dev/null +++ b/thinkphp/library/think/Loader.php @@ -0,0 +1,565 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\exception\ClassNotFoundException; + +class Loader +{ + protected static $instance = []; + // 绫诲悕鏄犲皠 + protected static $map = []; + + // 鍛藉悕绌洪棿鍒悕 + protected static $namespaceAlias = []; + + // PSR-4 + private static $prefixLengthsPsr4 = []; + private static $prefixDirsPsr4 = []; + private static $fallbackDirsPsr4 = []; + + // PSR-0 + private static $prefixesPsr0 = []; + private static $fallbackDirsPsr0 = []; + + // 鑷姩鍔犺浇鐨勬枃浠 + private static $autoloadFiles = []; + + // 鑷姩鍔犺浇 + public static function autoload($class) + { + // 妫娴嬪懡鍚嶇┖闂村埆鍚 + if (!empty(self::$namespaceAlias)) { + $namespace = dirname($class); + if (isset(self::$namespaceAlias[$namespace])) { + $original = self::$namespaceAlias[$namespace] . '\\' . basename($class); + if (class_exists($original)) { + return class_alias($original, $class, false); + } + } + } + + if ($file = self::findFile($class)) { + + // Win鐜涓ユ牸鍖哄垎澶у皬鍐 + if (IS_WIN && pathinfo($file, PATHINFO_FILENAME) != pathinfo(realpath($file), PATHINFO_FILENAME)) { + return false; + } + + __include_file($file); + return true; + } + } + + /** + * 鏌ユ壘鏂囦欢 + * @param $class + * @return bool + */ + private static function findFile($class) + { + if (!empty(self::$map[$class])) { + // 绫诲簱鏄犲皠 + return self::$map[$class]; + } + + // 鏌ユ壘 PSR-4 + $logicalPathPsr4 = strtr($class, '\\', DS) . EXT; + + $first = $class[0]; + if (isset(self::$prefixLengthsPsr4[$first])) { + foreach (self::$prefixLengthsPsr4[$first] as $prefix => $length) { + if (0 === strpos($class, $prefix)) { + foreach (self::$prefixDirsPsr4[$prefix] as $dir) { + if (is_file($file = $dir . DS . substr($logicalPathPsr4, $length))) { + return $file; + } + } + } + } + } + + // 鏌ユ壘 PSR-4 fallback dirs + foreach (self::$fallbackDirsPsr4 as $dir) { + if (is_file($file = $dir . DS . $logicalPathPsr4)) { + return $file; + } + } + + // 鏌ユ壘 PSR-0 + if (false !== $pos = strrpos($class, '\\')) { + // namespaced class name + $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1) + . strtr(substr($logicalPathPsr4, $pos + 1), '_', DS); + } else { + // PEAR-like class name + $logicalPathPsr0 = strtr($class, '_', DS) . EXT; + } + + if (isset(self::$prefixesPsr0[$first])) { + foreach (self::$prefixesPsr0[$first] as $prefix => $dirs) { + if (0 === strpos($class, $prefix)) { + foreach ($dirs as $dir) { + if (is_file($file = $dir . DS . $logicalPathPsr0)) { + return $file; + } + } + } + } + } + + // 鏌ユ壘 PSR-0 fallback dirs + foreach (self::$fallbackDirsPsr0 as $dir) { + if (is_file($file = $dir . DS . $logicalPathPsr0)) { + return $file; + } + } + + return self::$map[$class] = false; + } + + // 娉ㄥ唽classmap + public static function addClassMap($class, $map = '') + { + if (is_array($class)) { + self::$map = array_merge(self::$map, $class); + } else { + self::$map[$class] = $map; + } + } + + // 娉ㄥ唽鍛藉悕绌洪棿 + public static function addNamespace($namespace, $path = '') + { + if (is_array($namespace)) { + foreach ($namespace as $prefix => $paths) { + self::addPsr4($prefix . '\\', rtrim($paths, DS), true); + } + } else { + self::addPsr4($namespace . '\\', rtrim($path, DS), true); + } + } + + // 娣诲姞Ps0绌洪棿 + private static function addPsr0($prefix, $paths, $prepend = false) + { + if (!$prefix) { + if ($prepend) { + self::$fallbackDirsPsr0 = array_merge( + (array) $paths, + self::$fallbackDirsPsr0 + ); + } else { + self::$fallbackDirsPsr0 = array_merge( + self::$fallbackDirsPsr0, + (array) $paths + ); + } + + return; + } + + $first = $prefix[0]; + if (!isset(self::$prefixesPsr0[$first][$prefix])) { + self::$prefixesPsr0[$first][$prefix] = (array) $paths; + + return; + } + if ($prepend) { + self::$prefixesPsr0[$first][$prefix] = array_merge( + (array) $paths, + self::$prefixesPsr0[$first][$prefix] + ); + } else { + self::$prefixesPsr0[$first][$prefix] = array_merge( + self::$prefixesPsr0[$first][$prefix], + (array) $paths + ); + } + } + + // 娣诲姞Psr4绌洪棿 + private static function addPsr4($prefix, $paths, $prepend = false) + { + if (!$prefix) { + // Register directories for the root namespace. + if ($prepend) { + self::$fallbackDirsPsr4 = array_merge( + (array) $paths, + self::$fallbackDirsPsr4 + ); + } else { + self::$fallbackDirsPsr4 = array_merge( + self::$fallbackDirsPsr4, + (array) $paths + ); + } + } elseif (!isset(self::$prefixDirsPsr4[$prefix])) { + // Register directories for a new namespace. + $length = strlen($prefix); + if ('\\' !== $prefix[$length - 1]) { + throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator."); + } + self::$prefixLengthsPsr4[$prefix[0]][$prefix] = $length; + self::$prefixDirsPsr4[$prefix] = (array) $paths; + } elseif ($prepend) { + // Prepend directories for an already registered namespace. + self::$prefixDirsPsr4[$prefix] = array_merge( + (array) $paths, + self::$prefixDirsPsr4[$prefix] + ); + } else { + // Append directories for an already registered namespace. + self::$prefixDirsPsr4[$prefix] = array_merge( + self::$prefixDirsPsr4[$prefix], + (array) $paths + ); + } + } + + // 娉ㄥ唽鍛藉悕绌洪棿鍒悕 + public static function addNamespaceAlias($namespace, $original = '') + { + if (is_array($namespace)) { + self::$namespaceAlias = array_merge(self::$namespaceAlias, $namespace); + } else { + self::$namespaceAlias[$namespace] = $original; + } + } + + // 娉ㄥ唽鑷姩鍔犺浇鏈哄埗 + public static function register($autoload = '') + { + // 娉ㄥ唽绯荤粺鑷姩鍔犺浇 + spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true); + // 娉ㄥ唽鍛藉悕绌洪棿瀹氫箟 + self::addNamespace([ + 'think' => LIB_PATH . 'think' . DS, + 'behavior' => LIB_PATH . 'behavior' . DS, + 'traits' => LIB_PATH . 'traits' . DS, + ]); + // 鍔犺浇绫诲簱鏄犲皠鏂囦欢 + if (is_file(RUNTIME_PATH . 'classmap' . EXT)) { + self::addClassMap(__include_file(RUNTIME_PATH . 'classmap' . EXT)); + } + + // Composer鑷姩鍔犺浇鏀寔 + if (is_dir(VENDOR_PATH . 'composer')) { + self::registerComposerLoader(); + } + + // 鑷姩鍔犺浇extend鐩綍 + self::$fallbackDirsPsr4[] = rtrim(EXTEND_PATH, DS); + } + + // 娉ㄥ唽composer鑷姩鍔犺浇 + private static function registerComposerLoader() + { + if (is_file(VENDOR_PATH . 'composer/autoload_namespaces.php')) { + $map = require VENDOR_PATH . 'composer/autoload_namespaces.php'; + foreach ($map as $namespace => $path) { + self::addPsr0($namespace, $path); + } + } + + if (is_file(VENDOR_PATH . 'composer/autoload_psr4.php')) { + $map = require VENDOR_PATH . 'composer/autoload_psr4.php'; + foreach ($map as $namespace => $path) { + self::addPsr4($namespace, $path); + } + } + + if (is_file(VENDOR_PATH . 'composer/autoload_classmap.php')) { + $classMap = require VENDOR_PATH . 'composer/autoload_classmap.php'; + if ($classMap) { + self::addClassMap($classMap); + } + } + + if (is_file(VENDOR_PATH . 'composer/autoload_files.php')) { + $includeFiles = require VENDOR_PATH . 'composer/autoload_files.php'; + foreach ($includeFiles as $fileIdentifier => $file) { + if (empty(self::$autoloadFiles[$fileIdentifier])) { + __require_file($file); + self::$autoloadFiles[$fileIdentifier] = true; + } + } + } + } + + /** + * 瀵煎叆鎵闇鐨勭被搴 鍚宩ava鐨処mport 鏈嚱鏁版湁缂撳瓨鍔熻兘 + * @param string $class 绫诲簱鍛藉悕绌洪棿瀛楃涓 + * @param string $baseUrl 璧峰璺緞 + * @param string $ext 瀵煎叆鐨勬枃浠舵墿灞曞悕 + * @return boolean + */ + public static function import($class, $baseUrl = '', $ext = EXT) + { + static $_file = []; + $key = $class . $baseUrl; + $class = str_replace(['.', '#'], [DS, '.'], $class); + if (isset($_file[$key])) { + return true; + } + + if (empty($baseUrl)) { + list($name, $class) = explode(DS, $class, 2); + + if (isset(self::$prefixDirsPsr4[$name . '\\'])) { + // 娉ㄥ唽鐨勫懡鍚嶇┖闂 + $baseUrl = self::$prefixDirsPsr4[$name . '\\']; + } elseif ('@' == $name) { + //鍔犺浇褰撳墠妯″潡搴旂敤绫诲簱 + $baseUrl = App::$modulePath; + } elseif (is_dir(EXTEND_PATH . $name)) { + $baseUrl = EXTEND_PATH . $name . DS; + } else { + // 鍔犺浇鍏跺畠妯″潡鐨勭被搴 + $baseUrl = APP_PATH . $name . DS; + } + } elseif (substr($baseUrl, -1) != DS) { + $baseUrl .= DS; + } + // 濡傛灉绫诲瓨鍦 鍒欏鍏ョ被搴撴枃浠 + if (is_array($baseUrl)) { + foreach ($baseUrl as $path) { + $filename = $path . DS . $class . $ext; + if (is_file($filename)) { + break; + } + } + } else { + $filename = $baseUrl . $class . $ext; + } + + if (!empty($filename) && is_file($filename)) { + // 寮鍚皟璇曟ā寮廤in鐜涓ユ牸鍖哄垎澶у皬鍐 + if (IS_WIN && pathinfo($filename, PATHINFO_FILENAME) != pathinfo(realpath($filename), PATHINFO_FILENAME)) { + return false; + } + __include_file($filename); + $_file[$key] = true; + return true; + } + return false; + } + + /** + * 瀹炰緥鍖栵紙鍒嗗眰锛夋ā鍨 + * @param string $name Model鍚嶇О + * @param string $layer 涓氬姟灞傚悕绉 + * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑 + * @param string $common 鍏叡妯″潡鍚 + * @return Object + * @throws ClassNotFoundException + */ + public static function model($name = '', $layer = 'model', $appendSuffix = false, $common = 'common') + { + $guid = $name . $layer; + if (isset(self::$instance[$guid])) { + return self::$instance[$guid]; + } + if (strpos($name, '\\')) { + $class = $name; + } else { + if (strpos($name, '/')) { + list($module, $name) = explode('/', $name, 2); + } else { + $module = Request::instance()->module(); + } + $class = self::parseClass($module, $layer, $name, $appendSuffix); + } + if (class_exists($class)) { + $model = new $class(); + } else { + $class = str_replace('\\' . $module . '\\', '\\' . $common . '\\', $class); + if (class_exists($class)) { + $model = new $class(); + } else { + throw new ClassNotFoundException('class not exists:' . $class, $class); + } + } + self::$instance[$guid] = $model; + return $model; + } + + /** + * 瀹炰緥鍖栵紙鍒嗗眰锛夋帶鍒跺櫒 鏍煎紡锛歔妯″潡鍚/]鎺у埗鍣ㄥ悕 + * @param string $name 璧勬簮鍦板潃 + * @param string $layer 鎺у埗灞傚悕绉 + * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑 + * @param string $empty 绌烘帶鍒跺櫒鍚嶇О + * @return Object|false + * @throws ClassNotFoundException + */ + public static function controller($name, $layer = 'controller', $appendSuffix = false, $empty = '') + { + if (strpos($name, '\\')) { + $class = $name; + } else { + if (strpos($name, '/')) { + list($module, $name) = explode('/', $name); + } else { + $module = Request::instance()->module(); + } + $class = self::parseClass($module, $layer, $name, $appendSuffix); + } + if (class_exists($class)) { + return App::invokeClass($class); + } elseif ($empty && class_exists($emptyClass = self::parseClass($module, $layer, $empty, $appendSuffix))) { + return new $emptyClass(Request::instance()); + } + } + + /** + * 瀹炰緥鍖栭獙璇佺被 鏍煎紡锛歔妯″潡鍚/]楠岃瘉鍣ㄥ悕 + * @param string $name 璧勬簮鍦板潃 + * @param string $layer 楠岃瘉灞傚悕绉 + * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑 + * @param string $common 鍏叡妯″潡鍚 + * @return Object|false + * @throws ClassNotFoundException + */ + public static function validate($name = '', $layer = 'validate', $appendSuffix = false, $common = 'common') + { + $name = $name ?: Config::get('default_validate'); + if (empty($name)) { + return new Validate; + } + $guid = $name . $layer; + if (isset(self::$instance[$guid])) { + return self::$instance[$guid]; + } + if (strpos($name, '\\')) { + $class = $name; + } else { + if (strpos($name, '/')) { + list($module, $name) = explode('/', $name); + } else { + $module = Request::instance()->module(); + } + $class = self::parseClass($module, $layer, $name, $appendSuffix); + } + if (class_exists($class)) { + $validate = new $class; + } else { + $class = str_replace('\\' . $module . '\\', '\\' . $common . '\\', $class); + if (class_exists($class)) { + $validate = new $class; + } else { + throw new ClassNotFoundException('class not exists:' . $class, $class); + } + } + self::$instance[$guid] = $validate; + return $validate; + } + + /** + * 鏁版嵁搴撳垵濮嬪寲 骞跺彇寰楁暟鎹簱绫诲疄渚 + * @param mixed $config 鏁版嵁搴撻厤缃 + * @param bool|string $name 杩炴帴鏍囪瘑 true 寮哄埗閲嶆柊杩炴帴 + * @return \think\db\Connection + */ + public static function db($config = [], $name = false) + { + return Db::connect($config, $name); + } + + /** + * 杩滅▼璋冪敤妯″潡鐨勬搷浣滄柟娉 鍙傛暟鏍煎紡 [妯″潡/鎺у埗鍣/]鎿嶄綔 + * @param string $url 璋冪敤鍦板潃 + * @param string|array $vars 璋冪敤鍙傛暟 鏀寔瀛楃涓插拰鏁扮粍 + * @param string $layer 瑕佽皟鐢ㄧ殑鎺у埗灞傚悕绉 + * @param bool $appendSuffix 鏄惁娣诲姞绫诲悕鍚庣紑 + * @return mixed + */ + public static function action($url, $vars = [], $layer = 'controller', $appendSuffix = false) + { + $info = pathinfo($url); + $action = $info['basename']; + $module = '.' != $info['dirname'] ? $info['dirname'] : Request::instance()->controller(); + $class = self::controller($module, $layer, $appendSuffix); + if ($class) { + if (is_scalar($vars)) { + if (strpos($vars, '=')) { + parse_str($vars, $vars); + } else { + $vars = [$vars]; + } + } + return App::invokeMethod([$class, $action . Config::get('action_suffix')], $vars); + } + } + + /** + * 瀛楃涓插懡鍚嶉鏍艰浆鎹 + * type 0 灏咼ava椋庢牸杞崲涓篊鐨勯鏍 1 灏咰椋庢牸杞崲涓篔ava鐨勯鏍 + * @param string $name 瀛楃涓 + * @param integer $type 杞崲绫诲瀷 + * @param bool $ucfirst 棣栧瓧姣嶆槸鍚﹀ぇ鍐欙紙椹煎嘲瑙勫垯锛 + * @return string + */ + public static function parseName($name, $type = 0, $ucfirst = true) + { + if ($type) { + $name = preg_replace_callback('/_([a-zA-Z])/', function ($match) { + return strtoupper($match[1]); + }, $name); + return $ucfirst ? ucfirst($name) : lcfirst($name); + } else { + return strtolower(trim(preg_replace("/[A-Z]/", "_\\0", $name), "_")); + } + } + + /** + * 瑙f瀽搴旂敤绫荤殑绫诲悕 + * @param string $module 妯″潡鍚 + * @param string $layer 灞傚悕 controller model ... + * @param string $name 绫诲悕 + * @param bool $appendSuffix + * @return string + */ + public static function parseClass($module, $layer, $name, $appendSuffix = false) + { + $name = str_replace(['/', '.'], '\\', $name); + $array = explode('\\', $name); + $class = self::parseName(array_pop($array), 1) . (App::$suffix || $appendSuffix ? ucfirst($layer) : ''); + $path = $array ? implode('\\', $array) . '\\' : ''; + return App::$namespace . '\\' . ($module ? $module . '\\' : '') . $layer . '\\' . $path . $class; + } + + /** + * 鍒濆鍖栫被鐨勫疄渚 + * @return void + */ + public static function clearInstance() + { + self::$instance = []; + } +} + +/** + * 浣滅敤鑼冨洿闅旂 + * + * @param $file + * @return mixed + */ +function __include_file($file) +{ + return include $file; +} + +function __require_file($file) +{ + return require $file; +} diff --git a/thinkphp/library/think/Log.php b/thinkphp/library/think/Log.php new file mode 100644 index 000000000..4a8b0b40b --- /dev/null +++ b/thinkphp/library/think/Log.php @@ -0,0 +1,208 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\exception\ClassNotFoundException; + +/** + * Class Log + * @package think + * + * @method void log($msg) static + * @method void error($msg) static + * @method void info($msg) static + * @method void sql($msg) static + * @method void notice($msg) static + * @method void alert($msg) static + */ +class Log +{ + const LOG = 'log'; + const ERROR = 'error'; + const INFO = 'info'; + const SQL = 'sql'; + const NOTICE = 'notice'; + const ALERT = 'alert'; + const DEBUG = 'debug'; + + // 鏃ュ織淇℃伅 + protected static $log = []; + // 閰嶇疆鍙傛暟 + protected static $config = []; + // 鏃ュ織绫诲瀷 + protected static $type = ['log', 'error', 'info', 'sql', 'notice', 'alert', 'debug']; + // 鏃ュ織鍐欏叆椹卞姩 + protected static $driver; + + // 褰撳墠鏃ュ織鎺堟潈key + protected static $key; + + /** + * 鏃ュ織鍒濆鍖 + * @param array $config + */ + public static function init($config = []) + { + $type = isset($config['type']) ? $config['type'] : 'File'; + $class = false !== strpos($type, '\\') ? $type : '\\think\\log\\driver\\' . ucwords($type); + self::$config = $config; + unset($config['type']); + if (class_exists($class)) { + self::$driver = new $class($config); + } else { + throw new ClassNotFoundException('class not exists:' . $class, $class); + } + // 璁板綍鍒濆鍖栦俊鎭 + App::$debug && Log::record('[ LOG ] INIT ' . $type, 'info'); + } + + /** + * 鑾峰彇鏃ュ織淇℃伅 + * @param string $type 淇℃伅绫诲瀷 + * @return array + */ + public static function getLog($type = '') + { + return $type ? self::$log[$type] : self::$log; + } + + /** + * 璁板綍璋冭瘯淇℃伅 + * @param mixed $msg 璋冭瘯淇℃伅 + * @param string $type 淇℃伅绫诲瀷 + * @return void + */ + public static function record($msg, $type = 'log') + { + self::$log[$type][] = $msg; + if (IS_CLI && count(self::$log[$type]) > 100) { + // 鍛戒护琛屼笅闈㈡棩蹇楀啓鍏ユ敼杩 + self::save(); + } + } + + /** + * 娓呯┖鏃ュ織淇℃伅 + * @return void + */ + public static function clear() + { + self::$log = []; + } + + /** + * 褰撳墠鏃ュ織璁板綍鐨勬巿鏉僰ey + * @param string $key 鎺堟潈key + * @return void + */ + public static function key($key) + { + self::$key = $key; + } + + /** + * 妫鏌ユ棩蹇楀啓鍏ユ潈闄 + * @param array $config 褰撳墠鏃ュ織閰嶇疆鍙傛暟 + * @return bool + */ + public static function check($config) + { + if (self::$key && !empty($config['allow_key']) && !in_array(self::$key, $config['allow_key'])) { + return false; + } + return true; + } + + /** + * 淇濆瓨璋冭瘯淇℃伅 + * @return bool + */ + public static function save() + { + if (!empty(self::$log)) { + if (is_null(self::$driver)) { + self::init(Config::get('log')); + } + + if (!self::check(self::$config)) { + // 妫娴嬫棩蹇楀啓鍏ユ潈闄 + return false; + } + + if (empty(self::$config['level'])) { + // 鑾峰彇鍏ㄩ儴鏃ュ織 + $log = self::$log; + if (!App::$debug && isset($log['debug'])) { + unset($log['debug']); + } + } else { + // 璁板綍鍏佽绾у埆 + $log = []; + foreach (self::$config['level'] as $level) { + if (isset(self::$log[$level])) { + $log[$level] = self::$log[$level]; + } + } + } + + $result = self::$driver->save($log); + if ($result) { + self::$log = []; + } + + return $result; + } + return true; + } + + /** + * 瀹炴椂鍐欏叆鏃ュ織淇℃伅 骞舵敮鎸佽涓 + * @param mixed $msg 璋冭瘯淇℃伅 + * @param string $type 淇℃伅绫诲瀷 + * @param bool $force 鏄惁寮哄埗鍐欏叆 + * @return bool + */ + public static function write($msg, $type = 'log', $force = false) + { + // 灏佽鏃ュ織淇℃伅 + if (true === $force || empty(self::$config['level'])) { + $log[$type][] = $msg; + } elseif (in_array($type, self::$config['level'])) { + $log[$type][] = $msg; + } else { + return false; + } + + // 鐩戝惉log_write + Hook::listen('log_write', $log); + if (is_null(self::$driver)) { + self::init(Config::get('log')); + } + // 鍐欏叆鏃ュ織 + return self::$driver->save($log, false); + } + + /** + * 闈欐佽皟鐢 + * @param $method + * @param $args + * @return mixed + */ + public static function __callStatic($method, $args) + { + if (in_array($method, self::$type)) { + array_push($args, $method); + return call_user_func_array('\\think\\Log::record', $args); + } + } + +} diff --git a/thinkphp/library/think/Model.php b/thinkphp/library/think/Model.php new file mode 100644 index 000000000..085ca053d --- /dev/null +++ b/thinkphp/library/think/Model.php @@ -0,0 +1,1877 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use InvalidArgumentException; +use think\db\Query; +use think\Exception\ValidateException; +use think\model\Collection; +use think\model\Relation; +use think\model\relation\BelongsTo; +use think\model\relation\BelongsToMany; +use think\model\relation\HasMany; +use think\model\relation\HasManyThrough; +use think\model\relation\HasOne; +use think\model\relation\MorphMany; +use think\model\relation\MorphTo; + +/** + * Class Model + * @package think + * @mixin Query + */ +abstract class Model implements \JsonSerializable, \ArrayAccess +{ + // 鏁版嵁搴撳璞℃睜 + protected static $links = []; + // 鏁版嵁搴撻厤缃 + protected $connection = []; + // 鏁版嵁搴撴煡璇㈠璞 + protected $query; + // 褰撳墠妯″瀷鍚嶇О + protected $name; + // 鏁版嵁琛ㄥ悕绉 + protected $table; + // 褰撳墠绫诲悕绉 + protected $class; + // 鍥炶皟浜嬩欢 + private static $event = []; + // 閿欒淇℃伅 + protected $error; + // 瀛楁楠岃瘉瑙勫垯 + protected $validate; + // 鏁版嵁琛ㄤ富閿 澶嶅悎涓婚敭浣跨敤鏁扮粍瀹氫箟 涓嶈缃垯鑷姩鑾峰彇 + protected $pk; + // 鏁版嵁琛ㄥ瓧娈典俊鎭 鐣欑┖鍒欒嚜鍔ㄨ幏鍙 + protected $field = []; + // 鍙瀛楁 + protected $readonly = []; + // 鏄剧ず灞炴 + protected $visible = []; + // 闅愯棌灞炴 + protected $hidden = []; + // 杩藉姞灞炴 + protected $append = []; + // 鏁版嵁淇℃伅 + protected $data = []; + // 璁板綍鏀瑰彉瀛楁 + protected $change = []; + + // 淇濆瓨鑷姩瀹屾垚鍒楄〃 + protected $auto = []; + // 鏂板鑷姩瀹屾垚鍒楄〃 + protected $insert = []; + // 鏇存柊鑷姩瀹屾垚鍒楄〃 + protected $update = []; + // 鏄惁闇瑕佽嚜鍔ㄥ啓鍏ユ椂闂存埑 濡傛灉璁剧疆涓哄瓧绗︿覆 鍒欒〃绀烘椂闂村瓧娈电殑绫诲瀷 + protected $autoWriteTimestamp; + // 鍒涘缓鏃堕棿瀛楁 + protected $createTime = 'create_time'; + // 鏇存柊鏃堕棿瀛楁 + protected $updateTime = 'update_time'; + // 鏃堕棿瀛楁鍙栧嚭鍚庣殑榛樿鏃堕棿鏍煎紡 + protected $dateFormat; + // 瀛楁绫诲瀷鎴栬呮牸寮忚浆鎹 + protected $type = []; + // 鏄惁涓烘洿鏂版暟鎹 + protected $isUpdate = false; + // 鏇存柊鏉′欢 + protected $updateWhere; + // 褰撳墠鎵ц鐨勫叧鑱斿璞 + protected $relation; + // 楠岃瘉澶辫触鏄惁鎶涘嚭寮傚父 + protected $failException = false; + // 鍏ㄥ眬鏌ヨ鑼冨洿 + protected $useGlobalScope = true; + // 鏄惁閲囩敤鎵归噺楠岃瘉 + protected $batchValidate = false; + // 鏌ヨ鏁版嵁闆嗗璞 + protected $resultSetType; + // 鍏宠仈鑷姩鍐欏叆 + protected $relationWrite; + // + protected static $db; + + /** + * 鍒濆鍖栬繃鐨勬ā鍨. + * + * @var array + */ + protected static $initialized = []; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param array|object $data 鏁版嵁 + */ + public function __construct($data = []) + { + if (is_object($data)) { + $this->data = get_object_vars($data); + } else { + $this->data = $data; + } + + // 褰撳墠绫诲悕 + $this->class = get_class($this); + + if (empty($this->name)) { + // 褰撳墠妯″瀷鍚 + $name = str_replace('\\', '/', $this->class); + $this->name = basename($name); + if (Config::get('class_suffix')) { + $suffix = basename(dirname($name)); + $this->name = substr($this->name, 0, -strlen($suffix)); + } + } + + if (is_null($this->autoWriteTimestamp)) { + // 鑷姩鍐欏叆鏃堕棿鎴 + $this->autoWriteTimestamp = $this->db(false)->getConfig('auto_timestamp'); + } + + if (is_null($this->dateFormat)) { + // 璁剧疆鏃堕棿鎴虫牸寮 + $this->dateFormat = $this->db(false)->getConfig('datetime_format'); + } + + if (is_null($this->resultSetType)) { + $this->resultSetType = $this->db(false)->getConfig('resultset_type'); + } + // 鎵ц鍒濆鍖栨搷浣 + $this->initialize(); + } + + /** + * 鑾峰彇褰撳墠妯″瀷鐨勬暟鎹簱鏌ヨ瀵硅薄 + * @access public + * @param bool $baseQuery 鏄惁璋冪敤鍏ㄥ眬鏌ヨ鑼冨洿 + * @return Query + */ + public function db($baseQuery = true) + { + $model = $this->class; + if (!isset(self::$links[$model])) { + // 鍚堝苟鏁版嵁搴撻厤缃 + if (!empty($this->connection)) { + if (is_array($this->connection)) { + $connection = array_merge(Config::get('database'), $this->connection); + } else { + $connection = $this->connection; + } + } else { + $connection = []; + } + // 璁剧疆褰撳墠妯″瀷 纭繚鏌ヨ杩斿洖妯″瀷瀵硅薄 + $query = Db::connect($connection)->getQuery($model, $this->query); + + // 璁剧疆褰撳墠鏁版嵁琛ㄥ拰妯″瀷鍚 + if (!empty($this->table)) { + $query->setTable($this->table); + } else { + $query->name($this->name); + } + + if (!empty($this->pk)) { + $query->pk($this->pk); + } + + self::$links[$model] = $query; + } + // 鍏ㄥ眬浣滅敤鍩 + if ($baseQuery && method_exists($this, 'base')) { + call_user_func_array([$this, 'base'], [& self::$links[$model]]); + } + // 杩斿洖褰撳墠妯″瀷鐨勬暟鎹簱鏌ヨ瀵硅薄 + return self::$links[$model]; + } + + /** + * 鍒濆鍖栨ā鍨 + * @access protected + * @return void + */ + protected function initialize() + { + $class = get_class($this); + if (!isset(static::$initialized[$class])) { + static::$initialized[$class] = true; + static::init(); + } + } + + /** + * 鍒濆鍖栧鐞 + * @access protected + * @return void + */ + protected static function init() + { + } + + /** + * 璁剧疆鏁版嵁瀵硅薄鍊 + * @access public + * @param mixed $data 鏁版嵁鎴栬呭睘鎬у悕 + * @param mixed $value 鍊 + * @return $this + */ + public function data($data, $value = null) + { + if (is_string($data)) { + $this->data[$data] = $value; + } else { + // 娓呯┖鏁版嵁 + $this->data = []; + if (is_object($data)) { + $data = get_object_vars($data); + } + if (true === $value) { + // 鏁版嵁瀵硅薄璧嬪 + foreach ($data as $key => $value) { + $this->setAttr($key, $value, $data); + } + } else { + $this->data = $data; + } + } + return $this; + } + + /** + * 鑾峰彇瀵硅薄鍘熷鏁版嵁 濡傛灉涓嶅瓨鍦ㄦ寚瀹氬瓧娈佃繑鍥瀎alse + * @access public + * @param string $name 瀛楁鍚 鐣欑┖鑾峰彇鍏ㄩ儴 + * @return mixed + * @throws InvalidArgumentException + */ + public function getData($name = null) + { + if (is_null($name)) { + return $this->data; + } elseif (array_key_exists($name, $this->data)) { + return $this->data[$name]; + } else { + throw new InvalidArgumentException('property not exists:' . $this->class . '->' . $name); + } + } + + /** + * 淇敼鍣 璁剧疆鏁版嵁瀵硅薄鍊 + * @access public + * @param string $name 灞炴у悕 + * @param mixed $value 灞炴у + * @param array $data 鏁版嵁 + * @return $this + */ + public function setAttr($name, $value, $data = []) + { + if (is_null($value) && $this->autoWriteTimestamp && in_array($name, [$this->createTime, $this->updateTime])) { + // 鑷姩鍐欏叆鐨勬椂闂存埑瀛楁 + $value = $this->autoWriteTimestamp($name); + } else { + // 妫娴嬩慨鏀瑰櫒 + $method = 'set' . Loader::parseName($name, 1) . 'Attr'; + if (method_exists($this, $method)) { + $value = $this->$method($value, array_merge($data, $this->data)); + } elseif (isset($this->type[$name])) { + // 绫诲瀷杞崲 + $value = $this->writeTransform($value, $this->type[$name]); + } + } + + // 鏍囪瀛楁鏇存敼 + if (isset($this->data[$name]) && is_scalar($this->data[$name]) && is_scalar($value) && 0 !== strcmp($this->data[$name], $value)) { + $this->change[] = $name; + } elseif (!isset($this->data[$name]) || $value != $this->data[$name]) { + $this->change[] = $name; + } + // 璁剧疆鏁版嵁瀵硅薄灞炴 + $this->data[$name] = $value; + return $this; + } + + /** + * 鑷姩鍐欏叆鏃堕棿鎴 + * @access public + * @param string $name 鏃堕棿鎴冲瓧娈 + * @return mixed + */ + protected function autoWriteTimestamp($name) + { + if (isset($this->type[$name])) { + $type = $this->type[$name]; + if (strpos($type, ':')) { + list($type, $param) = explode(':', $type, 2); + } + switch ($type) { + case 'datetime': + case 'date': + $format = !empty($param) ? $param : $this->dateFormat; + $value = $this->formatDateTime($_SERVER['REQUEST_TIME'], $format); + break; + case 'timestamp': + case 'integer': + default: + $value = $_SERVER['REQUEST_TIME']; + break; + } + } elseif (is_string($this->autoWriteTimestamp) && in_array(strtolower($this->autoWriteTimestamp), [ + 'datetime', + 'date', + 'timestamp' + ]) + ) { + $value = $this->formatDateTime($_SERVER['REQUEST_TIME'], $this->dateFormat); + } else { + $value = $this->formatDateTime($_SERVER['REQUEST_TIME'], $this->dateFormat, true); + } + return $value; + } + + /** + * 鏃堕棿鏃ユ湡瀛楁鏍煎紡鍖栧鐞 + * @access public + * @param mixed $time 鏃堕棿鏃ユ湡琛ㄨ揪寮 + * @param mixed $format 鏃ユ湡鏍煎紡 + * @param bool $timestamp 鏄惁杩涜鏃堕棿鎴宠浆鎹 + * @return mixed + */ + protected function formatDateTime($time, $format, $timestamp = false) + { + if (false !== strpos($format, '\\')) { + $time = new $format($time); + } elseif (!$timestamp) { + $time = date($format, $time); + } + return $time; + } + + /** + * 鏁版嵁鍐欏叆 绫诲瀷杞崲 + * @access public + * @param mixed $value 鍊 + * @param string|array $type 瑕佽浆鎹㈢殑绫诲瀷 + * @return mixed + */ + protected function writeTransform($value, $type) + { + if (is_array($type)) { + list($type, $param) = $type; + } elseif (strpos($type, ':')) { + list($type, $param) = explode(':', $type, 2); + } + switch ($type) { + case 'integer': + $value = (int) $value; + break; + case 'float': + if (empty($param)) { + $value = (float) $value; + } else { + $value = (float) number_format($value, $param); + } + break; + case 'boolean': + $value = (bool) $value; + break; + case 'timestamp': + if (!is_numeric($value)) { + $value = strtotime($value); + } + break; + case 'datetime': + $format = !empty($param) ? $param : $this->dateFormat; + $value = is_numeric($value) ? $value : strtotime($value); + $value = $this->formatDateTime($value, $format); + break; + case 'object': + if (is_object($value)) { + $value = json_encode($value, JSON_FORCE_OBJECT); + } + break; + case 'array': + $value = (array) $value; + case 'json': + $option = !empty($param) ? (int) $param : JSON_UNESCAPED_UNICODE; + $value = json_encode($value, $option); + break; + case 'serialize': + $value = serialize($value); + break; + + } + return $value; + } + + /** + * 鑾峰彇鍣 鑾峰彇鏁版嵁瀵硅薄鐨勫 + * @access public + * @param string $name 鍚嶇О + * @return mixed + * @throws InvalidArgumentException + */ + public function getAttr($name) + { + try { + $notFound = false; + $value = $this->getData($name); + } catch (InvalidArgumentException $e) { + $notFound = true; + $value = null; + } + + // 妫娴嬪睘鎬ц幏鍙栧櫒 + $method = 'get' . Loader::parseName($name, 1) . 'Attr'; + if (method_exists($this, $method)) { + $value = $this->$method($value, $this->data); + } elseif (isset($this->type[$name])) { + // 绫诲瀷杞崲 + $value = $this->readTransform($value, $this->type[$name]); + } elseif (in_array($name, [$this->createTime, $this->updateTime])) { + if (is_string($this->autoWriteTimestamp) && in_array(strtolower($this->autoWriteTimestamp), [ + 'datetime', + 'date', + 'timestamp' + ]) + ) { + $value = $this->formatDateTime(strtotime($value), $this->dateFormat); + } else { + $value = $this->formatDateTime($value, $this->dateFormat); + } + } elseif ($notFound) { + $method = Loader::parseName($name, 1, false); + if (method_exists($this, $method) && $this->$method() instanceof Relation) { + // 娓呯┖涔嬪墠鐨勬煡璇㈠弬鏁 + $this->$method()->removeOption(); + // 涓嶅瓨鍦ㄨ瀛楁 鑾峰彇鍏宠仈鏁版嵁 + $value = $this->$method()->getRelation(); + // 淇濆瓨鍏宠仈瀵硅薄鍊 + $this->data[$name] = $value; + } else { + throw new InvalidArgumentException('property not exists:' . $this->class . '->' . $name); + } + } + return $value; + } + + /** + * 鏁版嵁璇诲彇 绫诲瀷杞崲 + * @access public + * @param mixed $value 鍊 + * @param string|array $type 瑕佽浆鎹㈢殑绫诲瀷 + * @return mixed + */ + protected function readTransform($value, $type) + { + if (is_array($type)) { + list($type, $param) = $type; + } elseif (strpos($type, ':')) { + list($type, $param) = explode(':', $type, 2); + } + switch ($type) { + case 'integer': + $value = (int) $value; + break; + case 'float': + if (empty($param)) { + $value = (float) $value; + } else { + $value = (float) number_format($value, $param); + } + break; + case 'boolean': + $value = (bool) $value; + break; + case 'timestamp': + if (!is_null($value)) { + $format = !empty($param) ? $param : $this->dateFormat; + $value = $this->formatDateTime($value, $format); + } + break; + case 'datetime': + if (!is_null($value)) { + $format = !empty($param) ? $param : $this->dateFormat; + $value = $this->formatDateTime(strtotime($value), $format); + } + break; + case 'json': + $value = json_decode($value, true); + break; + case 'array': + $value = is_null($value) ? [] : json_decode($value, true); + break; + case 'object': + $value = empty($value) ? new \stdClass() : json_decode($value); + break; + case 'serialize': + $value = unserialize($value); + break; + default: + if (false !== strpos($type, '\\')) { + // 瀵硅薄绫诲瀷 + $value = new $type($value); + } + } + return $value; + } + + /** + * 璁剧疆闇瑕佽拷鍔犵殑杈撳嚭灞炴 + * @access public + * @param array $append 灞炴у垪琛 + * @param bool $override 鏄惁瑕嗙洊 + * @return $this + */ + public function append($append = [], $override = false) + { + $this->append = $override ? $append : array_merge($this->append, $append); + return $this; + } + + /** + * 璁剧疆闄勫姞鍏宠仈瀵硅薄鐨勫睘鎬 + * @access public + * @param string $relation 鍏宠仈鏂规硶 + * @param string|array $append 杩藉姞灞炴у悕 + * @return $this + * @throws Exception + */ + public function appendRelationAttr($relation, $append) + { + if (is_string($append)) { + $append = explode(',', $append); + } + $model = $this->getAttr($relation); + if ($model instanceof Model) { + foreach ($append as $key => $attr) { + $key = is_numeric($key) ? $attr : $key; + if ($this->__isset($key)) { + throw new Exception('bind attr has exists:' . $key); + } else { + $this->setAttr($key, $model->$attr); + } + } + } + return $this; + } + + /** + * 璁剧疆闇瑕侀殣钘忕殑杈撳嚭灞炴 + * @access public + * @param array $hidden 灞炴у垪琛 + * @param bool $override 鏄惁瑕嗙洊 + * @return $this + */ + public function hidden($hidden = [], $override = false) + { + $this->hidden = $override ? $hidden : array_merge($this->hidden, $hidden); + return $this; + } + + /** + * 璁剧疆闇瑕佽緭鍑虹殑灞炴 + * @access public + * @param array $visible + * @param bool $override 鏄惁瑕嗙洊 + * @return $this + */ + public function visible($visible = [], $override = false) + { + $this->visible = $override ? $visible : array_merge($this->visible, $visible); + return $this; + } + + /** + * 瑙f瀽闅愯棌鍙婃樉绀哄睘鎬 + * @access protected + * @param array $attrs 灞炴 + * @param array $result 缁撴灉闆 + * @param bool $visible + * @return array + */ + protected function parseAttr($attrs, &$result, $visible = true) + { + $array = []; + foreach ($attrs as $key => $val) { + if (is_array($val)) { + if ($visible) { + $array[] = $key; + } + $result[$key] = $val; + } elseif (strpos($val, '.')) { + list($key, $name) = explode('.', $val); + if ($visible) { + $array[] = $key; + } + $result[$key][] = $name; + } else { + $array[] = $val; + } + } + return $array; + } + + /** + * 杞崲瀛愭ā鍨嬪璞 + * @access protected + * @param Model|Collection $model + * @param $visible + * @param $hidden + * @param $key + * @return array + */ + protected function subToArray($model, $visible, $hidden, $key) + { + // 鍏宠仈妯″瀷瀵硅薄 + if (isset($visible[$key])) { + $model->visible($visible[$key]); + } elseif (isset($hidden[$key])) { + $model->hidden($hidden[$key]); + } + return $model->toArray(); + } + + /** + * 杞崲褰撳墠妯″瀷瀵硅薄涓烘暟缁 + * @access public + * @return array + */ + public function toArray() + { + $item = []; + $visible = []; + $hidden = []; + // 杩囨护灞炴 + if (!empty($this->visible)) { + $array = $this->parseAttr($this->visible, $visible); + $data = array_intersect_key($this->data, array_flip($array)); + } elseif (!empty($this->hidden)) { + $array = $this->parseAttr($this->hidden, $hidden, false); + $data = array_diff_key($this->data, array_flip($array)); + } else { + $data = $this->data; + } + + foreach ($data as $key => $val) { + if ($val instanceof Model || $val instanceof Collection) { + // 鍏宠仈妯″瀷瀵硅薄 + $item[$key] = $this->subToArray($val, $visible, $hidden, $key); + } elseif (is_array($val) && reset($val) instanceof Model) { + // 鍏宠仈妯″瀷鏁版嵁闆 + $arr = []; + foreach ($val as $k => $value) { + $arr[$k] = $this->subToArray($value, $visible, $hidden, $k); + } + $item[$key] = $arr; + } else { + // 妯″瀷灞炴 + $item[$key] = $this->getAttr($key); + } + } + // 杩藉姞灞炴э紙蹇呴』瀹氫箟鑾峰彇鍣級 + if (!empty($this->append)) { + foreach ($this->append as $key => $name) { + if (is_array($name)) { + // 杩藉姞鍏宠仈瀵硅薄灞炴 + $relation = $this->getAttr($key); + $item[$key] = $relation->append($name)->toArray(); + } elseif (strpos($name, '.')) { + list($key, $attr) = explode('.', $name); + // 杩藉姞鍏宠仈瀵硅薄灞炴 + $relation = $this->getAttr($key); + $item[$key] = $relation->append([$attr])->toArray(); + } else { + $item[$name] = $this->getAttr($name); + } + } + } + return !empty($item) ? $item : []; + } + + /** + * 杞崲褰撳墠妯″瀷瀵硅薄涓篔SON瀛楃涓 + * @access public + * @param integer $options json鍙傛暟 + * @return string + */ + public function toJson($options = JSON_UNESCAPED_UNICODE) + { + return json_encode($this->toArray(), $options); + } + + /** + * 杞崲褰撳墠妯″瀷鏁版嵁闆嗕负鏁版嵁闆嗗璞 + * @access public + * @param array|Collection $collection 鏁版嵁闆 + * @return Collection + */ + public function toCollection($collection) + { + if ($this->resultSetType) { + if ('collection' == $this->resultSetType) { + $collection = new Collection($collection); + } elseif (false !== strpos($this->resultSetType, '\\')) { + $class = $this->resultSetType; + $collection = new $class($collection); + } + } + return $collection; + } + + /** + * 鍏宠仈鏁版嵁涓璧锋洿鏂 + * @access public + * @param mixed $relation 鍏宠仈 + * @return $this + */ + public function together($relation) + { + if (is_string($relation)) { + $relation = explode(',', $relation); + } + $this->relationWrite = $relation; + return $this; + } + + /** + * 鑾峰彇妯″瀷瀵硅薄鐨勪富閿 + * @access public + * @param string $name 妯″瀷鍚 + * @return mixed + */ + public function getPk($name = '') + { + if (!empty($name)) { + $table = $this->db(false)->getTable($name); + return $this->db(false)->getPk($table); + } elseif (empty($this->pk)) { + $this->pk = $this->db(false)->getPk(); + } + return $this->pk; + } + + /** + * 鍒ゆ柇涓涓瓧娈靛悕鏄惁涓轰富閿瓧娈 + * @access public + * @param string $key 鍚嶇О + * @return bool + */ + protected function isPk($key) + { + $pk = $this->getPk(); + if (is_string($pk) && $pk == $key) { + return true; + } elseif (is_array($pk) && in_array($key, $pk)) { + return true; + } + return false; + } + + /** + * 淇濆瓨褰撳墠鏁版嵁瀵硅薄 + * @access public + * @param array $data 鏁版嵁 + * @param array $where 鏇存柊鏉′欢 + * @param string $sequence 鑷搴忓垪鍚 + * @return integer|false + */ + public function save($data = [], $where = [], $sequence = null) + { + if (!empty($data)) { + // 鏁版嵁鑷姩楠岃瘉 + if (!$this->validateData($data)) { + return false; + } + // 鏁版嵁瀵硅薄璧嬪 + foreach ($data as $key => $value) { + $this->setAttr($key, $value, $data); + } + if (!empty($where)) { + $this->isUpdate = true; + } + } + + // 鑷姩鍏宠仈鍐欏叆 + if (!empty($this->relationWrite)) { + $relation = []; + foreach ($this->relationWrite as $key => $name) { + if (!is_numeric($key)) { + $relation[$key] = []; + foreach ($name as $val) { + if (isset($this->data[$val])) { + $relation[$key][$val] = $this->data[$val]; + unset($this->data[$val]); + } + } + } elseif (isset($this->data[$name])) { + $relation[$name] = $this->data[$name]; + if (!$this->isUpdate) { + unset($this->data[$name]); + } + } + } + } + + // 妫娴嬪瓧娈 + if (!empty($this->field)) { + if (true === $this->field) { + $this->field = $this->db(false)->getTableInfo('', 'fields'); + } + foreach ($this->data as $key => $val) { + if (!in_array($key, $this->field)) { + unset($this->data[$key]); + } + } + } + + // 鏁版嵁鑷姩瀹屾垚 + $this->autoCompleteData($this->auto); + + // 鑷姩鍐欏叆鏇存柊鏃堕棿 + if ($this->autoWriteTimestamp && $this->updateTime && (empty($this->change) || !in_array($this->updateTime, $this->change))) { + $this->setAttr($this->updateTime, null); + } + + // 浜嬩欢鍥炶皟 + if (false === $this->trigger('before_write', $this)) { + return false; + } + $pk = $this->getPk(); + if ($this->isUpdate) { + // 鑷姩鏇存柊 + $this->autoCompleteData($this->update); + + // 浜嬩欢鍥炶皟 + if (false === $this->trigger('before_update', $this)) { + return false; + } + + // 鍘婚櫎娌℃湁鏇存柊鐨勫瓧娈 + $data = []; + foreach ($this->data as $key => $val) { + if (in_array($key, $this->change) || $this->isPk($key)) { + $data[$key] = $val; + } + } + + if (!empty($this->readonly)) { + // 鍙瀛楁涓嶅厑璁告洿鏂 + foreach ($this->readonly as $key => $field) { + if (isset($data[$field])) { + unset($data[$field]); + } + } + } + + if (empty($where) && !empty($this->updateWhere)) { + $where = $this->updateWhere; + } + + if (is_string($pk) && isset($data[$pk])) { + if (!isset($where[$pk])) { + unset($where); + $where[$pk] = $data[$pk]; + } + unset($data[$pk]); + } + + // 鍏宠仈鏇存柊 + if (isset($relation)) { + foreach ($relation as $name => $val) { + if (isset($data[$name])) { + unset($data[$name]); + } + } + } + + // 妯″瀷鏇存柊 + $result = $this->db()->where($where)->update($data); + + // 鍏宠仈鏇存柊 + if (isset($relation)) { + foreach ($relation as $name => $val) { + if ($val instanceof Model) { + $val->save(); + } else { + unset($this->data[$name]); + $model = $this->getAttr($name); + if ($model instanceof Model) { + $model->save($val); + } + } + } + } + + // 娓呯┖change + $this->change = []; + // 鏇存柊鍥炶皟 + $this->trigger('after_update', $this); + } else { + // 鑷姩鍐欏叆 + $this->autoCompleteData($this->insert); + + // 鑷姩鍐欏叆鍒涘缓鏃堕棿 + if ($this->autoWriteTimestamp && $this->createTime && (empty($this->change) || !in_array($this->createTime, $this->change))) { + $this->setAttr($this->createTime, null); + } + + if (false === $this->trigger('before_insert', $this)) { + return false; + } + + $result = $this->db()->insert($this->data); + + // 鑾峰彇鑷姩澧為暱涓婚敭 + if ($result && is_string($pk) && (!isset($this->data[$pk]) || '' == $this->data[$pk])) { + $insertId = $this->db()->getLastInsID($sequence); + if ($insertId) { + $this->data[$pk] = $insertId; + } + } + + // 鍏宠仈鍐欏叆 + if (isset($relation)) { + foreach ($relation as $name => $val) { + $method = Loader::parseName($name, 1, false); + $this->$method()->save($val); + } + } + + // 鏍囪涓烘洿鏂 + $this->isUpdate = true; + // 娓呯┖change + $this->change = []; + // 鏂板鍥炶皟 + $this->trigger('after_insert', $this); + } + // 鍐欏叆鍥炶皟 + $this->trigger('after_write', $this); + + return $result; + } + + /** + * 淇濆瓨澶氫釜鏁版嵁鍒板綋鍓嶆暟鎹璞 + * @access public + * @param array $dataSet 鏁版嵁 + * @param boolean $replace 鏄惁鑷姩璇嗗埆鏇存柊鍜屽啓鍏 + * @return array|false + * @throws \Exception + */ + public function saveAll($dataSet, $replace = true) + { + if ($this->validate) { + // 鏁版嵁鎵归噺楠岃瘉 + $validate = $this->validate; + foreach ($dataSet as $data) { + if (!$this->validateData($data, $validate)) { + return false; + } + } + } + + $result = []; + $db = $this->db(); + $db->startTrans(); + try { + $pk = $this->getPk(); + if (is_string($pk) && $replace) { + $auto = true; + } + foreach ($dataSet as $key => $data) { + if (!empty($auto) && isset($data[$pk])) { + $result[$key] = self::update($data, [], $this->field); + } else { + $result[$key] = self::create($data, $this->field); + } + } + $db->commit(); + return $result; + } catch (\Exception $e) { + $db->rollback(); + throw $e; + } + } + + /** + * 璁剧疆鍏佽鍐欏叆鐨勫瓧娈 + * @access public + * @param mixed $field 鍏佽鍐欏叆鐨勫瓧娈 濡傛灉涓簍rue鍙厑璁稿啓鍏ユ暟鎹〃瀛楁 + * @return $this + */ + public function allowField($field) + { + if (is_string($field)) { + $field = explode(',', $field); + } + $this->field = $field; + return $this; + } + + /** + * 鏄惁涓烘洿鏂版暟鎹 + * @access public + * @param bool $update + * @param mixed $where + * @return $this + */ + public function isUpdate($update = true, $where = null) + { + $this->isUpdate = $update; + if (!empty($where)) { + $this->updateWhere = $where; + } + return $this; + } + + /** + * 鏁版嵁鑷姩瀹屾垚 + * @access public + * @param array $auto 瑕佽嚜鍔ㄦ洿鏂扮殑瀛楁鍒楄〃 + * @return void + */ + protected function autoCompleteData($auto = []) + { + foreach ($auto as $field => $value) { + if (is_integer($field)) { + $field = $value; + $value = null; + } + if (!in_array($field, $this->change)) { + $this->setAttr($field, !is_null($value) ? $value : (isset($this->data[$field]) ? $this->data[$field] : $value)); + } + } + } + + /** + * 鍒犻櫎褰撳墠鐨勮褰 + * @access public + * @return integer + */ + public function delete() + { + if (false === $this->trigger('before_delete', $this)) { + return false; + } + + // 鍒犻櫎鏉′欢 + $pk = $this->getPk(); + if (isset($this->data[$pk])) { + $where = [$pk => $this->data[$pk]]; + } elseif (!empty($this->updateWhere)) { + $where = $this->updateWhere; + } else { + $where = null; + } + + // 鍒犻櫎褰撳墠妯″瀷鏁版嵁 + $result = $this->db()->where($where)->delete(); + + // 鍏宠仈鍒犻櫎 + if (!empty($this->relationWrite)) { + foreach ($this->relationWrite as $key => $name) { + $name = is_numeric($key) ? $name : $key; + $model = $this->getAttr($name); + if ($model instanceof Model) { + $model->delete(); + } + } + } + + $this->trigger('after_delete', $this); + return $result; + } + + /** + * 璁剧疆鑷姩瀹屾垚鐨勫瓧娈碉紙 瑙勫垯閫氳繃淇敼鍣ㄥ畾涔夛級 + * @access public + * @param array $fields 闇瑕佽嚜鍔ㄥ畬鎴愮殑瀛楁 + * @return $this + */ + public function auto($fields) + { + $this->auto = $fields; + return $this; + } + + /** + * 璁剧疆瀛楁楠岃瘉 + * @access public + * @param array|string|bool $rule 楠岃瘉瑙勫垯 true琛ㄧず鑷姩璇诲彇楠岃瘉鍣ㄧ被 + * @param array $msg 鎻愮ず淇℃伅 + * @param bool $batch 鎵归噺楠岃瘉 + * @return $this + */ + public function validate($rule = true, $msg = [], $batch = false) + { + if (is_array($rule)) { + $this->validate = [ + 'rule' => $rule, + 'msg' => $msg, + ]; + } else { + $this->validate = true === $rule ? $this->name : $rule; + } + $this->batchValidate = $batch; + return $this; + } + + /** + * 璁剧疆楠岃瘉澶辫触鍚庢槸鍚︽姏鍑哄紓甯 + * @access public + * @param bool $fail 鏄惁鎶涘嚭寮傚父 + * @return $this + */ + public function validateFailException($fail = true) + { + $this->failException = $fail; + return $this; + } + + /** + * 鑷姩楠岃瘉鏁版嵁 + * @access protected + * @param array $data 楠岃瘉鏁版嵁 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param bool $batch 鎵归噺楠岃瘉 + * @return bool + */ + protected function validateData($data, $rule = null, $batch = null) + { + $info = is_null($rule) ? $this->validate : $rule; + + if (!empty($info)) { + if (is_array($info)) { + $validate = Loader::validate(); + $validate->rule($info['rule']); + $validate->message($info['msg']); + } else { + $name = is_string($info) ? $info : $this->name; + if (strpos($name, '.')) { + list($name, $scene) = explode('.', $name); + } + $validate = Loader::validate($name); + if (!empty($scene)) { + $validate->scene($scene); + } + } + $batch = is_null($batch) ? $this->batchValidate : $batch; + + if (!$validate->batch($batch)->check($data)) { + $this->error = $validate->getError(); + if ($this->failException) { + throw new ValidateException($this->error); + } else { + return false; + } + } + $this->validate = null; + } + return true; + } + + /** + * 杩斿洖妯″瀷鐨勯敊璇俊鎭 + * @access public + * @return string + */ + public function getError() + { + return $this->error; + } + + /** + * 娉ㄥ唽鍥炶皟鏂规硶 + * @access public + * @param string $event 浜嬩欢鍚 + * @param callable $callback 鍥炶皟鏂规硶 + * @param bool $override 鏄惁瑕嗙洊 + * @return void + */ + public static function event($event, $callback, $override = false) + { + $class = get_called_class(); + if ($override) { + self::$event[$class][$event] = []; + } + self::$event[$class][$event][] = $callback; + } + + /** + * 瑙﹀彂浜嬩欢 + * @access protected + * @param string $event 浜嬩欢鍚 + * @param mixed $params 浼犲叆鍙傛暟锛堝紩鐢級 + * @return bool + */ + protected function trigger($event, &$params) + { + if (isset(self::$event[$this->class][$event])) { + foreach (self::$event[$this->class][$event] as $callback) { + if (is_callable($callback)) { + $result = call_user_func_array($callback, [& $params]); + if (false === $result) { + return false; + } + } + } + } + return true; + } + + /** + * 鍐欏叆鏁版嵁 + * @access public + * @param array $data 鏁版嵁鏁扮粍 + * @param array|true $field 鍏佽瀛楁 + * @return $this + */ + public static function create($data = [], $field = null) + { + $model = new static(); + if (!empty($field)) { + $model->allowField($field); + } + $model->isUpdate(false)->save($data, []); + return $model; + } + + /** + * 鏇存柊鏁版嵁 + * @access public + * @param array $data 鏁版嵁鏁扮粍 + * @param array $where 鏇存柊鏉′欢 + * @param array|true $field 鍏佽瀛楁 + * @return $this + */ + public static function update($data = [], $where = [], $field = null) + { + $model = new static(); + if (!empty($field)) { + $model->allowField($field); + } + $result = $model->isUpdate(true)->save($data, $where); + return $model; + } + + /** + * 鏌ユ壘鍗曟潯璁板綍 + * @access public + * @param mixed $data 涓婚敭鍊兼垨鑰呮煡璇㈡潯浠讹紙闂寘锛 + * @param array|string $with 鍏宠仈棰勬煡璇 + * @param bool $cache 鏄惁缂撳瓨 + * @return static + * @throws exception\DbException + */ + public static function get($data = null, $with = [], $cache = false) + { + $query = static::parseQuery($data, $with, $cache); + return $query->find($data); + } + + /** + * 鏌ユ壘鎵鏈夎褰 + * @access public + * @param mixed $data 涓婚敭鍒楄〃鎴栬呮煡璇㈡潯浠讹紙闂寘锛 + * @param array|string $with 鍏宠仈棰勬煡璇 + * @param bool $cache 鏄惁缂撳瓨 + * @return static[]|false + * @throws exception\DbException + */ + public static function all($data = null, $with = [], $cache = false) + { + $query = static::parseQuery($data, $with, $cache); + return $query->select($data); + } + + /** + * 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + * @access public + * @param mixed $data 涓婚敭鍒楄〃鎴栬呮煡璇㈡潯浠讹紙闂寘锛 + * @param string $with 鍏宠仈棰勬煡璇 + * @param bool $cache 鏄惁缂撳瓨 + * @return Query + */ + protected static function parseQuery(&$data, $with, $cache) + { + $result = self::with($with)->cache($cache); + if (is_array($data) && key($data) !== 0) { + $result = $result->where($data); + $data = null; + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [& $result]); + $data = null; + } elseif ($data instanceof Query) { + $result = $data->with($with)->cache($cache); + $data = null; + } + return $result; + } + + /** + * 鍒犻櫎璁板綍 + * @access public + * @param mixed $data 涓婚敭鍒楄〃 鏀寔闂寘鏌ヨ鏉′欢 + * @return integer 鎴愬姛鍒犻櫎鐨勮褰曟暟 + */ + public static function destroy($data) + { + $model = new static(); + $query = $model->db(); + if (is_array($data) && key($data) !== 0) { + $query->where($data); + $data = null; + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [& $query]); + $data = null; + } elseif (is_null($data)) { + return 0; + } + $resultSet = $query->select($data); + $count = 0; + if ($resultSet) { + foreach ($resultSet as $data) { + $result = $data->delete(); + $count += $result; + } + } + return $count; + } + + /** + * 鍛藉悕鑼冨洿 + * @access public + * @param string|array|\Closure $name 鍛藉悕鑼冨洿鍚嶇О 閫楀彿鍒嗛殧 + * @internal mixed ...$params 鍙傛暟璋冪敤 + * @return Model|Query + */ + public static function scope($name) + { + if ($name instanceof Query) { + return $name; + } + $model = new static(); + $params = func_get_args(); + $params[0] = $model->db(); + if ($name instanceof \Closure) { + call_user_func_array($name, $params); + } elseif (is_string($name)) { + $name = explode(',', $name); + } + if (is_array($name)) { + foreach ($name as $scope) { + $method = 'scope' . trim($scope); + if (method_exists($model, $method)) { + call_user_func_array([$model, $method], $params); + } + } + } + return $model; + } + + /** + * 璁剧疆鏄惁浣跨敤鍏ㄥ眬鏌ヨ鑼冨洿 + * @param bool $use 鏄惁鍚敤鍏ㄥ眬鏌ヨ鑼冨洿 + * @access public + * @return Model + */ + public static function useGlobalScope($use) + { + $model = new static(); + static::$db = $model->db($use); + return $model; + } + + /** + * 鏍规嵁鍏宠仈鏉′欢鏌ヨ褰撳墠妯″瀷 + * @access public + * @param string $relation 鍏宠仈鏂规硶鍚 + * @param mixed $operator 姣旇緝鎿嶄綔绗 + * @param integer $count 涓暟 + * @param string $id 鍏宠仈琛ㄧ殑缁熻瀛楁 + * @return Model + */ + public static function has($relation, $operator = '>=', $count = 1, $id = '*') + { + $model = new static(); + if (is_array($operator) || $operator instanceof \Closure) { + return $model->$relation()->hasWhere($operator); + } + return $model->$relation()->has($operator, $count, $id); + } + + /** + * 鏍规嵁鍏宠仈鏉′欢鏌ヨ褰撳墠妯″瀷 + * @access public + * @param string $relation 鍏宠仈鏂规硶鍚 + * @param mixed $where 鏌ヨ鏉′欢锛堟暟缁勬垨鑰呴棴鍖咃級 + * @return Model + */ + public static function hasWhere($relation, $where = []) + { + $model = new static(); + return $model->$relation()->hasWhere($where); + } + + /** + * 瑙f瀽妯″瀷鐨勫畬鏁村懡鍚嶇┖闂 + * @access public + * @param string $model 妯″瀷鍚嶏紙鎴栬呭畬鏁寸被鍚嶏級 + * @return string + */ + protected function parseModel($model) + { + if (false === strpos($model, '\\')) { + $path = explode('\\', get_called_class()); + array_pop($path); + array_push($path, Loader::parseName($model, 1)); + $model = implode('\\', $path); + } + return $model; + } + + /** + * 鏌ヨ褰撳墠妯″瀷鐨勫叧鑱旀暟鎹 + * @access public + * @param string|array $relations 鍏宠仈鍚 + * @return $this + */ + public function relationQuery($relations) + { + if (is_string($relations)) { + $relations = explode(',', $relations); + } + + foreach ($relations as $key => $relation) { + $subRelation = ''; + $closure = null; + if ($relation instanceof \Closure) { + // 鏀寔闂寘鏌ヨ杩囨护鍏宠仈鏉′欢 + $closure = $relation; + $relation = $key; + } + if (strpos($relation, '.')) { + list($relation, $subRelation) = explode('.', $relation, 2); + } + $method = Loader::parseName($relation, 1, false); + $this->data[$relation] = $this->$method()->getRelation($subRelation, $closure); + } + return $this; + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 杩斿洖鏁版嵁闆 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 鍏宠仈鍚 + * @return array + */ + public function eagerlyResultSet(&$resultSet, $relation) + { + $relations = is_string($relation) ? explode(',', $relation) : $relation; + foreach ($relations as $key => $relation) { + $subRelation = ''; + $closure = false; + if ($relation instanceof \Closure) { + $closure = $relation; + $relation = $key; + } + if (strpos($relation, '.')) { + list($relation, $subRelation) = explode('.', $relation, 2); + } + $relation = Loader::parseName($relation, 1, false); + $this->$relation()->eagerlyResultSet($resultSet, $relation, $subRelation, $closure); + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 杩斿洖妯″瀷瀵硅薄 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 鍏宠仈鍚 + * @return Model + */ + public function eagerlyResult(&$result, $relation) + { + $relations = is_string($relation) ? explode(',', $relation) : $relation; + + foreach ($relations as $key => $relation) { + $subRelation = ''; + $closure = false; + if ($relation instanceof \Closure) { + $closure = $relation; + $relation = $key; + } + if (strpos($relation, '.')) { + list($relation, $subRelation) = explode('.', $relation, 2); + } + $relation = Loader::parseName($relation, 1, false); + $this->$relation()->eagerlyResult($result, $relation, $subRelation, $closure); + } + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string|array $relation 鍏宠仈鍚 + * @return void + */ + public function relationCount(&$result, $relation) + { + $relations = is_string($relation) ? explode(',', $relation) : $relation; + + foreach ($relations as $key => $relation) { + $closure = false; + if ($relation instanceof \Closure) { + $closure = $relation; + $relation = $key; + } + $relation = Loader::parseName($relation, 1, false); + $count = $this->$relation()->relationCount($result, $closure); + $result->setAttr(Loader::parseName($relation) . '_count', $count); + } + } + + /** + * 鑾峰彇妯″瀷鐨勯粯璁ゅ閿悕 + * @access public + * @param string $name 妯″瀷鍚 + * @return string + */ + protected function getForeignKey($name) + { + if (strpos($name, '\\')) { + $name = basename(str_replace('\\', '/', $name)); + } + return Loader::parseName($name) . '_id'; + } + + /** + * HAS ONE 鍏宠仈瀹氫箟 + * @access public + * @param string $model 妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + * @param array $alias 鍒悕瀹氫箟锛堝凡缁忓簾寮冿級 + * @param string $joinType JOIN绫诲瀷 + * @return HasOne + */ + public function hasOne($model, $foreignKey = '', $localKey = '', $alias = [], $joinType = 'INNER') + { + // 璁板綍褰撳墠鍏宠仈淇℃伅 + $model = $this->parseModel($model); + $localKey = $localKey ?: $this->getPk(); + $foreignKey = $foreignKey ?: $this->getForeignKey($this->name); + return new HasOne($this, $model, $foreignKey, $localKey, $joinType); + } + + /** + * BELONGS TO 鍏宠仈瀹氫箟 + * @access public + * @param string $model 妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + * @param array $alias 鍒悕瀹氫箟锛堝凡缁忓簾寮冿級 + * @param string $joinType JOIN绫诲瀷 + * @return BelongsTo + */ + public function belongsTo($model, $foreignKey = '', $localKey = '', $alias = [], $joinType = 'INNER') + { + // 璁板綍褰撳墠鍏宠仈淇℃伅 + $model = $this->parseModel($model); + $foreignKey = $foreignKey ?: $this->getForeignKey($model); + $localKey = $localKey ?: (new $model)->getPk(); + return new BelongsTo($this, $model, $foreignKey, $localKey, $joinType); + } + + /** + * HAS MANY 鍏宠仈瀹氫箟 + * @access public + * @param string $model 妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + * @return HasMany + */ + public function hasMany($model, $foreignKey = '', $localKey = '') + { + // 璁板綍褰撳墠鍏宠仈淇℃伅 + $model = $this->parseModel($model); + $localKey = $localKey ?: $this->getPk(); + $foreignKey = $foreignKey ?: $this->getForeignKey($this->name); + return new HasMany($this, $model, $foreignKey, $localKey); + } + + /** + * HAS MANY 杩滅▼鍏宠仈瀹氫箟 + * @access public + * @param string $model 妯″瀷鍚 + * @param string $through 涓棿妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $throughKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + * @return HasManyThrough + */ + public function hasManyThrough($model, $through, $foreignKey = '', $throughKey = '', $localKey = '') + { + // 璁板綍褰撳墠鍏宠仈淇℃伅 + $model = $this->parseModel($model); + $through = $this->parseModel($through); + $localKey = $localKey ?: $this->getPk(); + $foreignKey = $foreignKey ?: $this->getForeignKey($this->name); + $throughKey = $throughKey ?: $this->getForeignKey($through); + return new HasManyThrough($this, $model, $through, $foreignKey, $throughKey, $localKey); + } + + /** + * BELONGS TO MANY 鍏宠仈瀹氫箟 + * @access public + * @param string $model 妯″瀷鍚 + * @param string $table 涓棿琛ㄥ悕 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 褰撳墠妯″瀷鍏宠仈閿 + * @return BelongsToMany + */ + public function belongsToMany($model, $table = '', $foreignKey = '', $localKey = '') + { + // 璁板綍褰撳墠鍏宠仈淇℃伅 + $model = $this->parseModel($model); + $name = Loader::parseName(basename(str_replace('\\', '/', $model))); + $table = $table ?: $this->db(false)->getTable(Loader::parseName($this->name) . '_' . $name); + $foreignKey = $foreignKey ?: $name . '_id'; + $localKey = $localKey ?: $this->getForeignKey($this->name); + return new BelongsToMany($this, $model, $table, $foreignKey, $localKey); + } + + /** + * MORPH MANY 鍏宠仈瀹氫箟 + * @access public + * @param string $model 妯″瀷鍚 + * @param string|array $morph 澶氭佸瓧娈典俊鎭 + * @param string $type 澶氭佺被鍨 + * @return MorphMany + */ + public function morphMany($model, $morph = null, $type = '') + { + // 璁板綍褰撳墠鍏宠仈淇℃伅 + $model = $this->parseModel($model); + if (is_null($morph)) { + $trace = debug_backtrace(false, 2); + $morph = Loader::parseName($trace[1]['function']); + } + $type = $type ?: Loader::parseName($this->name); + if (is_array($morph)) { + list($morphType, $foreignKey) = $morph; + } else { + $morphType = $morph . '_type'; + $foreignKey = $morph . '_id'; + } + return new MorphMany($this, $model, $foreignKey, $morphType, $type); + } + + /** + * MORPH TO 鍏宠仈瀹氫箟 + * @access public + * @param string|array $morph 澶氭佸瓧娈典俊鎭 + * @param array $alias 澶氭佸埆鍚嶅畾涔 + * @return MorphTo + */ + public function morphTo($morph = null, $alias = []) + { + if (is_null($morph)) { + $trace = debug_backtrace(false, 2); + $morph = Loader::parseName($trace[1]['function']); + } + // 璁板綍褰撳墠鍏宠仈淇℃伅 + if (is_array($morph)) { + list($morphType, $foreignKey) = $morph; + } else { + $morphType = $morph . '_type'; + $foreignKey = $morph . '_id'; + } + return new MorphTo($this, $morphType, $foreignKey, $alias); + } + + public function __call($method, $args) + { + if (isset(static::$db)) { + $query = static::$db; + static::$db = null; + } else { + $query = $this->db(); + } + + if (method_exists($this, 'scope' . $method)) { + // 鍔ㄦ佽皟鐢ㄥ懡鍚嶈寖鍥 + $method = 'scope' . $method; + array_unshift($args, $query); + call_user_func_array([$this, $method], $args); + return $this; + } else { + return call_user_func_array([$query, $method], $args); + } + } + + public static function __callStatic($method, $params) + { + if (isset(static::$db)) { + $query = static::$db; + static::$db = null; + } else { + $query = (new static())->db(); + } + + return call_user_func_array([$query, $method], $params); + } + + /** + * 淇敼鍣 璁剧疆鏁版嵁瀵硅薄鐨勫 + * @access public + * @param string $name 鍚嶇О + * @param mixed $value 鍊 + * @return void + */ + public function __set($name, $value) + { + $this->setAttr($name, $value); + } + + /** + * 鑾峰彇鍣 鑾峰彇鏁版嵁瀵硅薄鐨勫 + * @access public + * @param string $name 鍚嶇О + * @return mixed + */ + public function __get($name) + { + return $this->getAttr($name); + } + + /** + * 妫娴嬫暟鎹璞$殑鍊 + * @access public + * @param string $name 鍚嶇О + * @return boolean + */ + public function __isset($name) + { + try { + if (array_key_exists($name, $this->data)) { + return true; + } else { + $this->getAttr($name); + return true; + } + } catch (InvalidArgumentException $e) { + return false; + } + + } + + /** + * 閿姣佹暟鎹璞$殑鍊 + * @access public + * @param string $name 鍚嶇О + * @return void + */ + public function __unset($name) + { + unset($this->data[$name]); + } + + public function __toString() + { + return $this->toJson(); + } + + // JsonSerializable + public function jsonSerialize() + { + return $this->toArray(); + } + + // ArrayAccess + public function offsetSet($name, $value) + { + $this->setAttr($name, $value); + } + + public function offsetExists($name) + { + return $this->__isset($name); + } + + public function offsetUnset($name) + { + $this->__unset($name); + } + + public function offsetGet($name) + { + return $this->getAttr($name); + } + + /** + * 瑙e簭鍒楀寲鍚庡鐞 + */ + public function __wakeup() + { + $this->initialize(); + } + + /** + * 妯″瀷浜嬩欢蹇嵎鏂规硶 + * @param $callback + * @param bool $override + */ + protected static function beforeInsert($callback, $override = false) + { + self::event('before_insert', $callback, $override); + } + + protected static function afterInsert($callback, $override = false) + { + self::event('after_insert', $callback, $override); + } + + protected static function beforeUpdate($callback, $override = false) + { + self::event('before_update', $callback, $override); + } + + protected static function afterUpdate($callback, $override = false) + { + self::event('after_update', $callback, $override); + } + + protected static function beforeWrite($callback, $override = false) + { + self::event('before_write', $callback, $override); + } + + protected static function afterWrite($callback, $override = false) + { + self::event('after_write', $callback, $override); + } + + protected static function beforeDelete($callback, $override = false) + { + self::event('before_delete', $callback, $override); + } + + protected static function afterDelete($callback, $override = false) + { + self::event('after_delete', $callback, $override); + } + +} diff --git a/thinkphp/library/think/Paginator.php b/thinkphp/library/think/Paginator.php new file mode 100644 index 000000000..0c1bea8a7 --- /dev/null +++ b/thinkphp/library/think/Paginator.php @@ -0,0 +1,369 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use ArrayAccess; +use ArrayIterator; +use Countable; +use IteratorAggregate; +use JsonSerializable; +use Traversable; + +abstract class Paginator implements ArrayAccess, Countable, IteratorAggregate, JsonSerializable +{ + /** @var bool 鏄惁涓虹畝娲佹ā寮 */ + protected $simple = false; + + /** @var Collection 鏁版嵁闆 */ + protected $items; + + /** @var integer 褰撳墠椤 */ + protected $currentPage; + + /** @var integer 鏈鍚庝竴椤 */ + protected $lastPage; + + /** @var integer|null 鏁版嵁鎬绘暟 */ + protected $total; + + /** @var integer 姣忛〉鐨勬暟閲 */ + protected $listRows; + + /** @var bool 鏄惁鏈変笅涓椤 */ + protected $hasMore; + + /** @var array 涓浜涢厤缃 */ + protected $options = [ + 'var_page' => 'page', + 'path' => '/', + 'query' => [], + 'fragment' => '', + ]; + + public function __construct($items, $listRows, $currentPage = null, $total = null, $simple = false, $options = []) + { + $this->options = array_merge($this->options, $options); + + $this->options['path'] = '/' != $this->options['path'] ? rtrim($this->options['path'], '/') : $this->options['path']; + + $this->simple = $simple; + $this->listRows = $listRows; + + if (!$items instanceof Collection) { + $items = Collection::make($items); + } + + if ($simple) { + $this->currentPage = $this->setCurrentPage($currentPage); + $this->hasMore = count($items) > ($this->listRows); + $items = $items->slice(0, $this->listRows); + } else { + $this->total = $total; + $this->lastPage = (int) ceil($total / $listRows); + $this->currentPage = $this->setCurrentPage($currentPage); + $this->hasMore = $this->currentPage < $this->lastPage; + } + $this->items = $items; + } + + /** + * @param $items + * @param $listRows + * @param null $currentPage + * @param bool $simple + * @param null $total + * @param array $options + * @return Paginator + */ + public static function make($items, $listRows, $currentPage = null, $total = null, $simple = false, $options = []) + { + return new static($items, $listRows, $currentPage, $total, $simple, $options); + } + + protected function setCurrentPage($currentPage) + { + if (!$this->simple && $currentPage > $this->lastPage) { + return $this->lastPage > 0 ? $this->lastPage : 1; + } + + return $currentPage; + } + + /** + * 鑾峰彇椤电爜瀵瑰簲鐨勯摼鎺 + * + * @param $page + * @return string + */ + protected function url($page) + { + if ($page <= 0) { + $page = 1; + } + + if (strpos($this->options['path'], '[PAGE]') === false) { + $parameters = [$this->options['var_page'] => $page]; + $path = $this->options['path']; + } else { + $parameters = []; + $path = str_replace('[PAGE]', $page, $this->options['path']); + } + if (count($this->options['query']) > 0) { + $parameters = array_merge($this->options['query'], $parameters); + } + $url = $path; + if (!empty($parameters)) { + $url .= '?' . urldecode(http_build_query($parameters, null, '&')); + } + return $url . $this->buildFragment(); + } + + /** + * 鑷姩鑾峰彇褰撳墠椤电爜 + * @param string $varPage + * @param int $default + * @return int + */ + public static function getCurrentPage($varPage = 'page', $default = 1) + { + $page = Request::instance()->request($varPage); + + if (filter_var($page, FILTER_VALIDATE_INT) !== false && (int) $page >= 1) { + return $page; + } + + return $default; + } + + /** + * 鑷姩鑾峰彇褰撳墠鐨刾ath + * @return string + */ + public static function getCurrentPath() + { + return Request::instance()->baseUrl(); + } + + public function total() + { + if ($this->simple) { + throw new \DomainException('not support total'); + } + return $this->total; + } + + public function listRows() + { + return $this->listRows; + } + + public function currentPage() + { + return $this->currentPage; + } + + public function lastPage() + { + if ($this->simple) { + throw new \DomainException('not support last'); + } + return $this->lastPage; + } + + /** + * 鏁版嵁鏄惁瓒冲鍒嗛〉 + * @return boolean + */ + public function hasPages() + { + return !(1 == $this->currentPage && !$this->hasMore); + } + + /** + * 鍒涘缓涓缁勫垎椤甸摼鎺 + * + * @param int $start + * @param int $end + * @return array + */ + public function getUrlRange($start, $end) + { + $urls = []; + + for ($page = $start; $page <= $end; $page++) { + $urls[$page] = $this->url($page); + } + + return $urls; + } + + /** + * 璁剧疆URL閿氱偣 + * + * @param string|null $fragment + * @return $this + */ + public function fragment($fragment) + { + $this->options['fragment'] = $fragment; + return $this; + } + + /** + * 娣诲姞URL鍙傛暟 + * + * @param array|string $key + * @param string|null $value + * @return $this + */ + public function appends($key, $value = null) + { + if (!is_array($key)) { + $queries = [$key => $value]; + } else { + $queries = $key; + } + + foreach ($queries as $k => $v) { + if ($k !== $this->options['var_page']) { + $this->options['query'][$k] = $v; + } + } + + return $this; + } + + /** + * 鏋勯犻敋鐐瑰瓧绗︿覆 + * + * @return string + */ + protected function buildFragment() + { + return $this->options['fragment'] ? '#' . $this->options['fragment'] : ''; + } + + /** + * 娓叉煋鍒嗛〉html + * @return mixed + */ + abstract public function render(); + + public function items() + { + return $this->items->all(); + } + + public function getCollection() + { + return $this->items; + } + + public function isEmpty() + { + return $this->items->isEmpty(); + } + + /** + * Retrieve an external iterator + * @return Traversable An instance of an object implementing Iterator or + * Traversable + */ + public function getIterator() + { + return new ArrayIterator($this->items->all()); + } + + /** + * Whether a offset exists + * @param mixed $offset + * @return bool + */ + public function offsetExists($offset) + { + return $this->items->offsetExists($offset); + } + + /** + * Offset to retrieve + * @param mixed $offset + * @return mixed + */ + public function offsetGet($offset) + { + return $this->items->offsetGet($offset); + } + + /** + * Offset to set + * @param mixed $offset + * @param mixed $value + */ + public function offsetSet($offset, $value) + { + $this->items->offsetSet($offset, $value); + } + + /** + * Offset to unset + * @param mixed $offset + * @return void + * @since 5.0.0 + */ + public function offsetUnset($offset) + { + $this->items->offsetUnset($offset); + } + + /** + * Count elements of an object + */ + public function count() + { + return $this->items->count(); + } + + public function __toString() + { + return (string) $this->render(); + } + + public function toArray() + { + try { + $total = $this->total(); + } catch (Exception $e) { + $total = null; + } + + return [ + 'total' => $total, + 'per_page' => $this->listRows(), + 'current_page' => $this->currentPage(), + 'data' => $this->items->toArray() + ]; + } + + /** + * Specify data which should be serialized to JSON + */ + public function jsonSerialize() + { + return $this->toArray(); + } + + public function __call($name, $arguments) + { + return call_user_func_array([$this->getCollection(), $name], $arguments); + } + +} diff --git a/thinkphp/library/think/Process.php b/thinkphp/library/think/Process.php new file mode 100644 index 000000000..6f3faa315 --- /dev/null +++ b/thinkphp/library/think/Process.php @@ -0,0 +1,1205 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\process\exception\Failed as ProcessFailedException; +use think\process\exception\Timeout as ProcessTimeoutException; +use think\process\pipes\Pipes; +use think\process\pipes\Unix as UnixPipes; +use think\process\pipes\Windows as WindowsPipes; +use think\process\Utils; + +class Process +{ + + const ERR = 'err'; + const OUT = 'out'; + + const STATUS_READY = 'ready'; + const STATUS_STARTED = 'started'; + const STATUS_TERMINATED = 'terminated'; + + const STDIN = 0; + const STDOUT = 1; + const STDERR = 2; + + const TIMEOUT_PRECISION = 0.2; + + private $callback; + private $commandline; + private $cwd; + private $env; + private $input; + private $starttime; + private $lastOutputTime; + private $timeout; + private $idleTimeout; + private $options; + private $exitcode; + private $fallbackExitcode; + private $processInformation; + private $outputDisabled = false; + private $stdout; + private $stderr; + private $enhanceWindowsCompatibility = true; + private $enhanceSigchildCompatibility; + private $process; + private $status = self::STATUS_READY; + private $incrementalOutputOffset = 0; + private $incrementalErrorOutputOffset = 0; + private $tty; + private $pty; + + private $useFileHandles = false; + + /** @var Pipes */ + private $processPipes; + + private $latestSignal; + + private static $sigchild; + + /** + * @var array + */ + public static $exitCodes = [ + 0 => 'OK', + 1 => 'General error', + 2 => 'Misuse of shell builtins', + 126 => 'Invoked command cannot execute', + 127 => 'Command not found', + 128 => 'Invalid exit argument', + // signals + 129 => 'Hangup', + 130 => 'Interrupt', + 131 => 'Quit and dump core', + 132 => 'Illegal instruction', + 133 => 'Trace/breakpoint trap', + 134 => 'Process aborted', + 135 => 'Bus error: "access to undefined portion of memory object"', + 136 => 'Floating point exception: "erroneous arithmetic operation"', + 137 => 'Kill (terminate immediately)', + 138 => 'User-defined 1', + 139 => 'Segmentation violation', + 140 => 'User-defined 2', + 141 => 'Write to pipe with no one reading', + 142 => 'Signal raised by alarm', + 143 => 'Termination (request to terminate)', + // 144 - not defined + 145 => 'Child process terminated, stopped (or continued*)', + 146 => 'Continue if stopped', + 147 => 'Stop executing temporarily', + 148 => 'Terminal stop signal', + 149 => 'Background process attempting to read from tty ("in")', + 150 => 'Background process attempting to write to tty ("out")', + 151 => 'Urgent data available on socket', + 152 => 'CPU time limit exceeded', + 153 => 'File size limit exceeded', + 154 => 'Signal raised by timer counting virtual time: "virtual timer expired"', + 155 => 'Profiling timer expired', + // 156 - not defined + 157 => 'Pollable event', + // 158 - not defined + 159 => 'Bad syscall', + ]; + + /** + * 鏋勯犳柟娉 + * @param string $commandline 鎸囦护 + * @param string|null $cwd 宸ヤ綔鐩綍 + * @param array|null $env 鐜鍙橀噺 + * @param string|null $input 杈撳叆 + * @param int|float|null $timeout 瓒呮椂鏃堕棿 + * @param array $options proc_open鐨勯夐」 + * @throws \RuntimeException + * @api + */ + public function __construct($commandline, $cwd = null, array $env = null, $input = null, $timeout = 60, array $options = []) + { + if (!function_exists('proc_open')) { + throw new \RuntimeException('The Process class relies on proc_open, which is not available on your PHP installation.'); + } + + $this->commandline = $commandline; + $this->cwd = $cwd; + + if (null === $this->cwd && (defined('ZEND_THREAD_SAFE') || '\\' === DS)) { + $this->cwd = getcwd(); + } + if (null !== $env) { + $this->setEnv($env); + } + + $this->input = $input; + $this->setTimeout($timeout); + $this->useFileHandles = '\\' === DS; + $this->pty = false; + $this->enhanceWindowsCompatibility = true; + $this->enhanceSigchildCompatibility = '\\' !== DS && $this->isSigchildEnabled(); + $this->options = array_replace([ + 'suppress_errors' => true, + 'binary_pipes' => true, + ], $options); + } + + public function __destruct() + { + $this->stop(); + } + + public function __clone() + { + $this->resetProcessData(); + } + + /** + * 杩愯鎸囦护 + * @param callback|null $callback + * @return int + */ + public function run($callback = null) + { + $this->start($callback); + + return $this->wait(); + } + + /** + * 杩愯鎸囦护 + * @param callable|null $callback + * @return self + * @throws \RuntimeException + * @throws ProcessFailedException + */ + public function mustRun($callback = null) + { + if ($this->isSigchildEnabled() && !$this->enhanceSigchildCompatibility) { + throw new \RuntimeException('This PHP has been compiled with --enable-sigchild. You must use setEnhanceSigchildCompatibility() to use this method.'); + } + + if (0 !== $this->run($callback)) { + throw new ProcessFailedException($this); + } + + return $this; + } + + /** + * 鍚姩杩涚▼骞跺啓鍒 STDIN 杈撳叆鍚庤繑鍥炪 + * @param callable|null $callback + * @throws \RuntimeException + * @throws \RuntimeException + * @throws \LogicException + */ + public function start($callback = null) + { + if ($this->isRunning()) { + throw new \RuntimeException('Process is already running'); + } + if ($this->outputDisabled && null !== $callback) { + throw new \LogicException('Output has been disabled, enable it to allow the use of a callback.'); + } + + $this->resetProcessData(); + $this->starttime = $this->lastOutputTime = microtime(true); + $this->callback = $this->buildCallback($callback); + $descriptors = $this->getDescriptors(); + + $commandline = $this->commandline; + + if ('\\' === DS && $this->enhanceWindowsCompatibility) { + $commandline = 'cmd /V:ON /E:ON /C "(' . $commandline . ')'; + foreach ($this->processPipes->getFiles() as $offset => $filename) { + $commandline .= ' ' . $offset . '>' . Utils::escapeArgument($filename); + } + $commandline .= '"'; + + if (!isset($this->options['bypass_shell'])) { + $this->options['bypass_shell'] = true; + } + } + + $this->process = proc_open($commandline, $descriptors, $this->processPipes->pipes, $this->cwd, $this->env, $this->options); + + if (!is_resource($this->process)) { + throw new \RuntimeException('Unable to launch a new process.'); + } + $this->status = self::STATUS_STARTED; + + if ($this->tty) { + return; + } + + $this->updateStatus(false); + $this->checkTimeout(); + } + + /** + * 閲嶅惎杩涚▼ + * @param callable|null $callback + * @return Process + * @throws \RuntimeException + * @throws \RuntimeException + */ + public function restart($callback = null) + { + if ($this->isRunning()) { + throw new \RuntimeException('Process is already running'); + } + + $process = clone $this; + $process->start($callback); + + return $process; + } + + /** + * 绛夊緟瑕佺粓姝㈢殑杩涚▼ + * @param callable|null $callback + * @return int + */ + public function wait($callback = null) + { + $this->requireProcessIsStarted(__FUNCTION__); + + $this->updateStatus(false); + if (null !== $callback) { + $this->callback = $this->buildCallback($callback); + } + + do { + $this->checkTimeout(); + $running = '\\' === DS ? $this->isRunning() : $this->processPipes->areOpen(); + $close = '\\' !== DS || !$running; + $this->readPipes(true, $close); + } while ($running); + + while ($this->isRunning()) { + usleep(1000); + } + + if ($this->processInformation['signaled'] && $this->processInformation['termsig'] !== $this->latestSignal) { + throw new \RuntimeException(sprintf('The process has been signaled with signal "%s".', $this->processInformation['termsig'])); + } + + return $this->exitcode; + } + + /** + * 鑾峰彇PID + * @return int|null + * @throws \RuntimeException + */ + public function getPid() + { + if ($this->isSigchildEnabled()) { + throw new \RuntimeException('This PHP has been compiled with --enable-sigchild. The process identifier can not be retrieved.'); + } + + $this->updateStatus(false); + + return $this->isRunning() ? $this->processInformation['pid'] : null; + } + + /** + * 灏嗕竴涓 POSIX 淇″彿鍙戦佸埌杩涚▼涓 + * @param int $signal + * @return Process + */ + public function signal($signal) + { + $this->doSignal($signal, true); + + return $this; + } + + /** + * 绂佺敤浠庡簳灞傝繃绋嬭幏鍙栬緭鍑哄拰閿欒杈撳嚭銆 + * @return Process + */ + public function disableOutput() + { + if ($this->isRunning()) { + throw new \RuntimeException('Disabling output while the process is running is not possible.'); + } + if (null !== $this->idleTimeout) { + throw new \LogicException('Output can not be disabled while an idle timeout is set.'); + } + + $this->outputDisabled = true; + + return $this; + } + + /** + * 寮鍚粠搴曞眰杩囩▼鑾峰彇杈撳嚭鍜岄敊璇緭鍑恒 + * @return Process + * @throws \RuntimeException + */ + public function enableOutput() + { + if ($this->isRunning()) { + throw new \RuntimeException('Enabling output while the process is running is not possible.'); + } + + $this->outputDisabled = false; + + return $this; + } + + /** + * 杈撳嚭鏄惁绂佺敤 + * @return bool + */ + public function isOutputDisabled() + { + return $this->outputDisabled; + } + + /** + * 鑾峰彇褰撳墠鐨勮緭鍑虹閬 + * @return string + * @throws \LogicException + * @throws \LogicException + * @api + */ + public function getOutput() + { + if ($this->outputDisabled) { + throw new \LogicException('Output has been disabled.'); + } + + $this->requireProcessIsStarted(__FUNCTION__); + + $this->readPipes(false, '\\' === DS ? !$this->processInformation['running'] : true); + + return $this->stdout; + } + + /** + * 浠ュ閲忔柟寮忚繑鍥炵殑杈撳嚭缁撴灉銆 + * @return string + */ + public function getIncrementalOutput() + { + $this->requireProcessIsStarted(__FUNCTION__); + + $data = $this->getOutput(); + + $latest = substr($data, $this->incrementalOutputOffset); + + if (false === $latest) { + return ''; + } + + $this->incrementalOutputOffset = strlen($data); + + return $latest; + } + + /** + * 娓呯┖杈撳嚭 + * @return Process + */ + public function clearOutput() + { + $this->stdout = ''; + $this->incrementalOutputOffset = 0; + + return $this; + } + + /** + * 杩斿洖褰撳墠鐨勯敊璇緭鍑虹殑杩囩▼ (STDERR)銆 + * @return string + */ + public function getErrorOutput() + { + if ($this->outputDisabled) { + throw new \LogicException('Output has been disabled.'); + } + + $this->requireProcessIsStarted(__FUNCTION__); + + $this->readPipes(false, '\\' === DS ? !$this->processInformation['running'] : true); + + return $this->stderr; + } + + /** + * 浠ュ閲忔柟寮忚繑鍥 errorOutput + * @return string + */ + public function getIncrementalErrorOutput() + { + $this->requireProcessIsStarted(__FUNCTION__); + + $data = $this->getErrorOutput(); + + $latest = substr($data, $this->incrementalErrorOutputOffset); + + if (false === $latest) { + return ''; + } + + $this->incrementalErrorOutputOffset = strlen($data); + + return $latest; + } + + /** + * 娓呯┖ errorOutput + * @return Process + */ + public function clearErrorOutput() + { + $this->stderr = ''; + $this->incrementalErrorOutputOffset = 0; + + return $this; + } + + /** + * 鑾峰彇閫鍑虹爜 + * @return null|int + */ + public function getExitCode() + { + if ($this->isSigchildEnabled() && !$this->enhanceSigchildCompatibility) { + throw new \RuntimeException('This PHP has been compiled with --enable-sigchild. You must use setEnhanceSigchildCompatibility() to use this method.'); + } + + $this->updateStatus(false); + + return $this->exitcode; + } + + /** + * 鑾峰彇閫鍑烘枃鏈 + * @return null|string + */ + public function getExitCodeText() + { + if (null === $exitcode = $this->getExitCode()) { + return; + } + + return isset(self::$exitCodes[$exitcode]) ? self::$exitCodes[$exitcode] : 'Unknown error'; + } + + /** + * 妫鏌ユ槸鍚︽垚鍔 + * @return bool + */ + public function isSuccessful() + { + return 0 === $this->getExitCode(); + } + + /** + * 鏄惁鏈崟鑾风殑淇″彿宸茶缁堟瀛愯繘绋 + * @return bool + */ + public function hasBeenSignaled() + { + $this->requireProcessIsTerminated(__FUNCTION__); + + if ($this->isSigchildEnabled()) { + throw new \RuntimeException('This PHP has been compiled with --enable-sigchild. Term signal can not be retrieved.'); + } + + $this->updateStatus(false); + + return $this->processInformation['signaled']; + } + + /** + * 杩斿洖瀵艰嚧瀛愯繘绋嬬粓姝㈠叾鎵ц鐨勬暟銆 + * @return int + */ + public function getTermSignal() + { + $this->requireProcessIsTerminated(__FUNCTION__); + + if ($this->isSigchildEnabled()) { + throw new \RuntimeException('This PHP has been compiled with --enable-sigchild. Term signal can not be retrieved.'); + } + + $this->updateStatus(false); + + return $this->processInformation['termsig']; + } + + /** + * 妫鏌ュ瓙杩涚▼淇″彿鏄惁宸插仠姝 + * @return bool + */ + public function hasBeenStopped() + { + $this->requireProcessIsTerminated(__FUNCTION__); + + $this->updateStatus(false); + + return $this->processInformation['stopped']; + } + + /** + * 杩斿洖瀵艰嚧瀛愯繘绋嬪仠姝㈠叾鎵ц鐨勬暟銆 + * @return int + */ + public function getStopSignal() + { + $this->requireProcessIsTerminated(__FUNCTION__); + + $this->updateStatus(false); + + return $this->processInformation['stopsig']; + } + + /** + * 妫鏌ユ槸鍚︽鍦ㄨ繍琛 + * @return bool + */ + public function isRunning() + { + if (self::STATUS_STARTED !== $this->status) { + return false; + } + + $this->updateStatus(false); + + return $this->processInformation['running']; + } + + /** + * 妫鏌ユ槸鍚﹀凡寮濮 + * @return bool + */ + public function isStarted() + { + return self::STATUS_READY != $this->status; + } + + /** + * 妫鏌ユ槸鍚﹀凡缁堟 + * @return bool + */ + public function isTerminated() + { + $this->updateStatus(false); + + return self::STATUS_TERMINATED == $this->status; + } + + /** + * 鑾峰彇褰撳墠鐨勭姸鎬 + * @return string + */ + public function getStatus() + { + $this->updateStatus(false); + + return $this->status; + } + + /** + * 缁堟杩涚▼ + */ + public function stop() + { + if ($this->isRunning()) { + if ('\\' === DS && !$this->isSigchildEnabled()) { + exec(sprintf('taskkill /F /T /PID %d 2>&1', $this->getPid()), $output, $exitCode); + if ($exitCode > 0) { + throw new \RuntimeException('Unable to kill the process'); + } + } else { + $pids = preg_split('/\s+/', `ps -o pid --no-heading --ppid {$this->getPid()}`); + foreach ($pids as $pid) { + if (is_numeric($pid)) { + posix_kill($pid, 9); + } + } + } + } + + $this->updateStatus(false); + if ($this->processInformation['running']) { + $this->close(); + } + + return $this->exitcode; + } + + /** + * 娣诲姞涓琛岃緭鍑 + * @param string $line + */ + public function addOutput($line) +{ + $this->lastOutputTime = microtime(true); + $this->stdout .= $line; + } + + /** + * 娣诲姞涓琛岄敊璇緭鍑 + * @param string $line + */ + public function addErrorOutput($line) +{ + $this->lastOutputTime = microtime(true); + $this->stderr .= $line; + } + + /** + * 鑾峰彇琚墽琛岀殑鎸囦护 + * @return string + */ + public function getCommandLine() +{ + return $this->commandline; + } + + /** + * 璁剧疆鎸囦护 + * @param string $commandline + * @return self + */ + public function setCommandLine($commandline) +{ + $this->commandline = $commandline; + + return $this; + } + + /** + * 鑾峰彇瓒呮椂鏃堕棿 + * @return float|null + */ + public function getTimeout() +{ + return $this->timeout; + } + + /** + * 鑾峰彇idle瓒呮椂鏃堕棿 + * @return float|null + */ + public function getIdleTimeout() +{ + return $this->idleTimeout; + } + + /** + * 璁剧疆瓒呮椂鏃堕棿 + * @param int|float|null $timeout + * @return self + */ + public function setTimeout($timeout) +{ + $this->timeout = $this->validateTimeout($timeout); + + return $this; + } + + /** + * 璁剧疆idle瓒呮椂鏃堕棿 + * @param int|float|null $timeout + * @return self + */ + public function setIdleTimeout($timeout) +{ + if (null !== $timeout && $this->outputDisabled) { + throw new \LogicException('Idle timeout can not be set while the output is disabled.'); + } + + $this->idleTimeout = $this->validateTimeout($timeout); + + return $this; + } + + /** + * 璁剧疆TTY + * @param bool $tty + * @return self + */ + public function setTty($tty) +{ + if ('\\' === DS && $tty) { + throw new \RuntimeException('TTY mode is not supported on Windows platform.'); + } + if ($tty && (!file_exists('/dev/tty') || !is_readable('/dev/tty'))) { + throw new \RuntimeException('TTY mode requires /dev/tty to be readable.'); + } + + $this->tty = (bool) $tty; + + return $this; + } + + /** + * 妫鏌ユ槸鍚︽槸tty妯″紡 + * @return bool + */ + public function isTty() +{ + return $this->tty; + } + + /** + * 璁剧疆pty妯″紡 + * @param bool $bool + * @return self + */ + public function setPty($bool) +{ + $this->pty = (bool) $bool; + + return $this; + } + + /** + * 鏄惁鏄痯ty妯″紡 + * @return bool + */ + public function isPty() +{ + return $this->pty; + } + + /** + * 鑾峰彇宸ヤ綔鐩綍 + * @return string|null + */ + public function getWorkingDirectory() +{ + if (null === $this->cwd) { + return getcwd() ?: null; + } + + return $this->cwd; + } + + /** + * 璁剧疆宸ヤ綔鐩綍 + * @param string $cwd + * @return self + */ + public function setWorkingDirectory($cwd) +{ + $this->cwd = $cwd; + + return $this; + } + + /** + * 鑾峰彇鐜鍙橀噺 + * @return array + */ + public function getEnv() +{ + return $this->env; + } + + /** + * 璁剧疆鐜鍙橀噺 + * @param array $env + * @return self + */ + public function setEnv(array $env) +{ + $env = array_filter($env, function ($value) { + return !is_array($value); + }); + + $this->env = []; + foreach ($env as $key => $value) { + $this->env[(binary) $key] = (binary) $value; + } + + return $this; + } + + /** + * 鑾峰彇杈撳叆 + * @return null|string + */ + public function getInput() +{ + return $this->input; + } + + /** + * 璁剧疆杈撳叆 + * @param mixed $input + * @return self + */ + public function setInput($input) +{ + if ($this->isRunning()) { + throw new \LogicException('Input can not be set while the process is running.'); + } + + $this->input = Utils::validateInput(sprintf('%s::%s', __CLASS__, __FUNCTION__), $input); + + return $this; + } + + /** + * 鑾峰彇proc_open鐨勯夐」 + * @return array + */ + public function getOptions() +{ + return $this->options; + } + + /** + * 璁剧疆proc_open鐨勯夐」 + * @param array $options + * @return self + */ + public function setOptions(array $options) +{ + $this->options = $options; + + return $this; + } + + /** + * 鏄惁鍏煎windows + * @return bool + */ + public function getEnhanceWindowsCompatibility() +{ + return $this->enhanceWindowsCompatibility; + } + + /** + * 璁剧疆鏄惁鍏煎windows + * @param bool $enhance + * @return self + */ + public function setEnhanceWindowsCompatibility($enhance) +{ + $this->enhanceWindowsCompatibility = (bool) $enhance; + + return $this; + } + + /** + * 杩斿洖鏄惁 sigchild 鍏煎妯″紡婵娲 + * @return bool + */ + public function getEnhanceSigchildCompatibility() +{ + return $this->enhanceSigchildCompatibility; + } + + /** + * 婵娲 sigchild 鍏煎鎬фā寮忋 + * @param bool $enhance + * @return self + */ + public function setEnhanceSigchildCompatibility($enhance) +{ + $this->enhanceSigchildCompatibility = (bool) $enhance; + + return $this; + } + + /** + * 鏄惁瓒呮椂 + */ + public function checkTimeout() +{ + if (self::STATUS_STARTED !== $this->status) { + return; + } + + if (null !== $this->timeout && $this->timeout < microtime(true) - $this->starttime) { + $this->stop(); + + throw new ProcessTimeoutException($this, ProcessTimeoutException::TYPE_GENERAL); + } + + if (null !== $this->idleTimeout && $this->idleTimeout < microtime(true) - $this->lastOutputTime) { + $this->stop(); + + throw new ProcessTimeoutException($this, ProcessTimeoutException::TYPE_IDLE); + } + } + + /** + * 鏄惁鏀寔pty + * @return bool + */ + public static function isPtySupported() +{ + static $result; + + if (null !== $result) { + return $result; + } + + if ('\\' === DS) { + return $result = false; + } + + $proc = @proc_open('echo 1', [['pty'], ['pty'], ['pty']], $pipes); + if (is_resource($proc)) { + proc_close($proc); + + return $result = true; + } + + return $result = false; + } + + /** + * 鍒涘缓鎵闇鐨 proc_open 鐨勬弿杩扮 + * @return array + */ + private function getDescriptors() +{ + if ('\\' === DS) { + $this->processPipes = WindowsPipes::create($this, $this->input); + } else { + $this->processPipes = UnixPipes::create($this, $this->input); + } + $descriptors = $this->processPipes->getDescriptors($this->outputDisabled); + + if (!$this->useFileHandles && $this->enhanceSigchildCompatibility && $this->isSigchildEnabled()) { + + $descriptors = array_merge($descriptors, [['pipe', 'w']]); + + $this->commandline = '(' . $this->commandline . ') 3>/dev/null; code=$?; echo $code >&3; exit $code'; + } + + return $descriptors; + } + + /** + * 寤虹珛 wait () 浣跨敤鐨勫洖璋冦 + * @param callable|null $callback + * @return callable + */ + protected function buildCallback($callback) +{ + $out = self::OUT; + $callback = function ($type, $data) use ($callback, $out) { + if ($out == $type) { + $this->addOutput($data); + } else { + $this->addErrorOutput($data); + } + + if (null !== $callback) { + call_user_func($callback, $type, $data); + } + }; + + return $callback; + } + + /** + * 鏇存柊鐘舵 + * @param bool $blocking + */ + protected function updateStatus($blocking) +{ + if (self::STATUS_STARTED !== $this->status) { + return; + } + + $this->processInformation = proc_get_status($this->process); + $this->captureExitCode(); + + $this->readPipes($blocking, '\\' === DS ? !$this->processInformation['running'] : true); + + if (!$this->processInformation['running']) { + $this->close(); + } + } + + /** + * 鏄惁寮鍚 '--enable-sigchild' + * @return bool + */ + protected function isSigchildEnabled() +{ + if (null !== self::$sigchild) { + return self::$sigchild; + } + + if (!function_exists('phpinfo')) { + return self::$sigchild = false; + } + + ob_start(); + phpinfo(INFO_GENERAL); + + return self::$sigchild = false !== strpos(ob_get_clean(), '--enable-sigchild'); + } + + /** + * 楠岃瘉鏄惁瓒呮椂 + * @param int|float|null $timeout + * @return float|null + */ + private function validateTimeout($timeout) +{ + $timeout = (float) $timeout; + + if (0.0 === $timeout) { + $timeout = null; + } elseif ($timeout < 0) { + throw new \InvalidArgumentException('The timeout value must be a valid positive integer or float number.'); + } + + return $timeout; + } + + /** + * 璇诲彇pipes + * @param bool $blocking + * @param bool $close + */ + private function readPipes($blocking, $close) +{ + $result = $this->processPipes->readAndWrite($blocking, $close); + + $callback = $this->callback; + foreach ($result as $type => $data) { + if (3 == $type) { + $this->fallbackExitcode = (int) $data; + } else { + $callback(self::STDOUT === $type ? self::OUT : self::ERR, $data); + } + } + } + + /** + * 鎹曡幏閫鍑虹爜 + */ + private function captureExitCode() +{ + if (isset($this->processInformation['exitcode']) && -1 != $this->processInformation['exitcode']) { + $this->exitcode = $this->processInformation['exitcode']; + } + } + + /** + * 鍏抽棴璧勬簮 + * @return int 閫鍑虹爜 + */ + private function close() +{ + $this->processPipes->close(); + if (is_resource($this->process)) { + $exitcode = proc_close($this->process); + } else { + $exitcode = -1; + } + + $this->exitcode = -1 !== $exitcode ? $exitcode : (null !== $this->exitcode ? $this->exitcode : -1); + $this->status = self::STATUS_TERMINATED; + + if (-1 === $this->exitcode && null !== $this->fallbackExitcode) { + $this->exitcode = $this->fallbackExitcode; + } elseif (-1 === $this->exitcode && $this->processInformation['signaled'] + && 0 < $this->processInformation['termsig'] + ) { + $this->exitcode = 128 + $this->processInformation['termsig']; + } + + return $this->exitcode; + } + + /** + * 閲嶇疆鏁版嵁 + */ + private function resetProcessData() +{ + $this->starttime = null; + $this->callback = null; + $this->exitcode = null; + $this->fallbackExitcode = null; + $this->processInformation = null; + $this->stdout = null; + $this->stderr = null; + $this->process = null; + $this->latestSignal = null; + $this->status = self::STATUS_READY; + $this->incrementalOutputOffset = 0; + $this->incrementalErrorOutputOffset = 0; + } + + /** + * 灏嗕竴涓 POSIX 淇″彿鍙戦佸埌杩涚▼涓 + * @param int $signal + * @param bool $throwException + * @return bool + */ + private function doSignal($signal, $throwException) +{ + if (!$this->isRunning()) { + if ($throwException) { + throw new \LogicException('Can not send signal on a non running process.'); + } + + return false; + } + + if ($this->isSigchildEnabled()) { + if ($throwException) { + throw new \RuntimeException('This PHP has been compiled with --enable-sigchild. The process can not be signaled.'); + } + + return false; + } + + if (true !== @proc_terminate($this->process, $signal)) { + if ($throwException) { + throw new \RuntimeException(sprintf('Error while sending signal `%s`.', $signal)); + } + + return false; + } + + $this->latestSignal = $signal; + + return true; + } + + /** + * 纭繚杩涚▼宸茬粡寮鍚 + * @param string $functionName + */ + private function requireProcessIsStarted($functionName) +{ + if (!$this->isStarted()) { + throw new \LogicException(sprintf('Process must be started before calling %s.', $functionName)); + } + } + + /** + * 纭繚杩涚▼宸茬粡缁堟 + * @param string $functionName + */ + private function requireProcessIsTerminated($functionName) +{ + if (!$this->isTerminated()) { + throw new \LogicException(sprintf('Process must be terminated before calling %s.', $functionName)); + } + } +} diff --git a/thinkphp/library/think/Request.php b/thinkphp/library/think/Request.php new file mode 100644 index 000000000..b72a5aaf8 --- /dev/null +++ b/thinkphp/library/think/Request.php @@ -0,0 +1,1602 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class Request +{ + /** + * @var object 瀵硅薄瀹炰緥 + */ + protected static $instance; + + protected $method; + /** + * @var string 鍩熷悕锛堝惈鍗忚鍜岀鍙o級 + */ + protected $domain; + + /** + * @var string URL鍦板潃 + */ + protected $url; + + /** + * @var string 鍩虹URL + */ + protected $baseUrl; + + /** + * @var string 褰撳墠鎵ц鐨勬枃浠 + */ + protected $baseFile; + + /** + * @var string 璁块棶鐨凴OOT鍦板潃 + */ + protected $root; + + /** + * @var string pathinfo + */ + protected $pathinfo; + + /** + * @var string pathinfo锛堜笉鍚悗缂锛 + */ + protected $path; + + /** + * @var array 褰撳墠璺敱淇℃伅 + */ + protected $routeInfo = []; + + /** + * @var array 褰撳墠璋冨害淇℃伅 + */ + protected $dispatch = []; + protected $module; + protected $controller; + protected $action; + // 褰撳墠璇█闆 + protected $langset; + + /** + * @var array 璇锋眰鍙傛暟 + */ + protected $param = []; + protected $get = []; + protected $post = []; + protected $request = []; + protected $route = []; + protected $put; + protected $session = []; + protected $file = []; + protected $cookie = []; + protected $server = []; + protected $header = []; + + /** + * @var array 璧勬簮绫诲瀷 + */ + protected $mimeType = [ + 'xml' => 'application/xml,text/xml,application/x-xml', + 'json' => 'application/json,text/x-json,application/jsonrequest,text/json', + 'js' => 'text/javascript,application/javascript,application/x-javascript', + 'css' => 'text/css', + 'rss' => 'application/rss+xml', + 'yaml' => 'application/x-yaml,text/yaml', + 'atom' => 'application/atom+xml', + 'pdf' => 'application/pdf', + 'text' => 'text/plain', + 'png' => 'image/png', + 'jpg' => 'image/jpg,image/jpeg,image/pjpeg', + 'gif' => 'image/gif', + 'csv' => 'text/csv', + 'html' => 'text/html,application/xhtml+xml,*/*', + ]; + + protected $content; + + // 鍏ㄥ眬杩囨护瑙勫垯 + protected $filter; + // Hook鎵╁睍鏂规硶 + protected static $hook = []; + // 缁戝畾鐨勫睘鎬 + protected $bind = []; + // php://input + protected $input; + // 璇锋眰缂撳瓨 + protected $cache; + // 缂撳瓨鏄惁妫鏌 + protected $isCheckCache; + + /** + * 鏋舵瀯鍑芥暟 + * @access protected + * @param array $options 鍙傛暟 + */ + protected function __construct($options = []) + { + foreach ($options as $name => $item) { + if (property_exists($this, $name)) { + $this->$name = $item; + } + } + if (is_null($this->filter)) { + $this->filter = Config::get('default_filter'); + } + // 淇濆瓨 php://input + $this->input = file_get_contents('php://input'); + } + + public function __call($method, $args) + { + if (array_key_exists($method, self::$hook)) { + array_unshift($args, $this); + return call_user_func_array(self::$hook[$method], $args); + } else { + throw new Exception('method not exists:' . __CLASS__ . '->' . $method); + } + } + + /** + * Hook 鏂规硶娉ㄥ叆 + * @access public + * @param string|array $method 鏂规硶鍚 + * @param mixed $callback callable + * @return void + */ + public static function hook($method, $callback = null) + { + if (is_array($method)) { + self::$hook = array_merge(self::$hook, $method); + } else { + self::$hook[$method] = $callback; + } + } + + /** + * 鍒濆鍖 + * @access public + * @param array $options 鍙傛暟 + * @return \think\Request + */ + public static function instance($options = []) + { + if (is_null(self::$instance)) { + self::$instance = new static($options); + } + return self::$instance; + } + + /** + * 鍒涘缓涓涓猆RL璇锋眰 + * @access public + * @param string $uri URL鍦板潃 + * @param string $method 璇锋眰绫诲瀷 + * @param array $params 璇锋眰鍙傛暟 + * @param array $cookie + * @param array $files + * @param array $server + * @param string $content + * @return \think\Request + */ + public static function create($uri, $method = 'GET', $params = [], $cookie = [], $files = [], $server = [], $content = null) + { + $server['PATH_INFO'] = ''; + $server['REQUEST_METHOD'] = strtoupper($method); + $info = parse_url($uri); + if (isset($info['host'])) { + $server['SERVER_NAME'] = $info['host']; + $server['HTTP_HOST'] = $info['host']; + } + if (isset($info['scheme'])) { + if ('https' === $info['scheme']) { + $server['HTTPS'] = 'on'; + $server['SERVER_PORT'] = 443; + } else { + unset($server['HTTPS']); + $server['SERVER_PORT'] = 80; + } + } + if (isset($info['port'])) { + $server['SERVER_PORT'] = $info['port']; + $server['HTTP_HOST'] = $server['HTTP_HOST'] . ':' . $info['port']; + } + if (isset($info['user'])) { + $server['PHP_AUTH_USER'] = $info['user']; + } + if (isset($info['pass'])) { + $server['PHP_AUTH_PW'] = $info['pass']; + } + if (!isset($info['path'])) { + $info['path'] = '/'; + } + $options = []; + $options[strtolower($method)] = $params; + $queryString = ''; + if (isset($info['query'])) { + parse_str(html_entity_decode($info['query']), $query); + if (!empty($params)) { + $params = array_replace($query, $params); + $queryString = http_build_query($query, '', '&'); + } else { + $params = $query; + $queryString = $info['query']; + } + } elseif (!empty($params)) { + $queryString = http_build_query($params, '', '&'); + } + if ($queryString) { + parse_str($queryString, $get); + $options['get'] = isset($options['get']) ? array_merge($get, $options['get']) : $get; + } + + $server['REQUEST_URI'] = $info['path'] . ('' !== $queryString ? '?' . $queryString : ''); + $server['QUERY_STRING'] = $queryString; + $options['cookie'] = $cookie; + $options['param'] = $params; + $options['file'] = $files; + $options['server'] = $server; + $options['url'] = $server['REQUEST_URI']; + $options['baseUrl'] = $info['path']; + $options['pathinfo'] = '/' == $info['path'] ? '/' : ltrim($info['path'], '/'); + $options['method'] = $server['REQUEST_METHOD']; + $options['domain'] = isset($info['scheme']) ? $info['scheme'] . '://' . $server['HTTP_HOST'] : ''; + $options['content'] = $content; + self::$instance = new self($options); + return self::$instance; + } + + /** + * 璁剧疆鎴栬幏鍙栧綋鍓嶅寘鍚崗璁殑鍩熷悕 + * @access public + * @param string $domain 鍩熷悕 + * @return string + */ + public function domain($domain = null) + { + if (!is_null($domain)) { + $this->domain = $domain; + return $this; + } elseif (!$this->domain) { + $this->domain = $this->scheme() . '://' . $this->host(); + } + return $this->domain; + } + + /** + * 璁剧疆鎴栬幏鍙栧綋鍓嶅畬鏁碪RL 鍖呮嫭QUERY_STRING + * @access public + * @param string|true $url URL鍦板潃 true 甯﹀煙鍚嶈幏鍙 + * @return string + */ + public function url($url = null) + { + if (!is_null($url) && true !== $url) { + $this->url = $url; + return $this; + } elseif (!$this->url) { + if (IS_CLI) { + $this->url = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : ''; + } elseif (isset($_SERVER['HTTP_X_REWRITE_URL'])) { + $this->url = $_SERVER['HTTP_X_REWRITE_URL']; + } elseif (isset($_SERVER['REQUEST_URI'])) { + $this->url = $_SERVER['REQUEST_URI']; + } elseif (isset($_SERVER['ORIG_PATH_INFO'])) { + $this->url = $_SERVER['ORIG_PATH_INFO'] . (!empty($_SERVER['QUERY_STRING']) ? '?' . $_SERVER['QUERY_STRING'] : ''); + } else { + $this->url = ''; + } + } + return true === $url ? $this->domain() . $this->url : $this->url; + } + + /** + * 璁剧疆鎴栬幏鍙栧綋鍓峌RL 涓嶅惈QUERY_STRING + * @access public + * @param string $url URL鍦板潃 + * @return string + */ + public function baseUrl($url = null) + { + if (!is_null($url) && true !== $url) { + $this->baseUrl = $url; + return $this; + } elseif (!$this->baseUrl) { + $str = $this->url(); + $this->baseUrl = strpos($str, '?') ? strstr($str, '?', true) : $str; + } + return true === $url ? $this->domain() . $this->baseUrl : $this->baseUrl; + } + + /** + * 璁剧疆鎴栬幏鍙栧綋鍓嶆墽琛岀殑鏂囦欢 SCRIPT_NAME + * @access public + * @param string $file 褰撳墠鎵ц鐨勬枃浠 + * @return string + */ + public function baseFile($file = null) + { + if (!is_null($file) && true !== $file) { + $this->baseFile = $file; + return $this; + } elseif (!$this->baseFile) { + $url = ''; + if (!IS_CLI) { + $script_name = basename($_SERVER['SCRIPT_FILENAME']); + if (basename($_SERVER['SCRIPT_NAME']) === $script_name) { + $url = $_SERVER['SCRIPT_NAME']; + } elseif (basename($_SERVER['PHP_SELF']) === $script_name) { + $url = $_SERVER['PHP_SELF']; + } elseif (isset($_SERVER['ORIG_SCRIPT_NAME']) && basename($_SERVER['ORIG_SCRIPT_NAME']) === $script_name) { + $url = $_SERVER['ORIG_SCRIPT_NAME']; + } elseif (($pos = strpos($_SERVER['PHP_SELF'], '/' . $script_name)) !== false) { + $url = substr($_SERVER['SCRIPT_NAME'], 0, $pos) . '/' . $script_name; + } elseif (isset($_SERVER['DOCUMENT_ROOT']) && strpos($_SERVER['SCRIPT_FILENAME'], $_SERVER['DOCUMENT_ROOT']) === 0) { + $url = str_replace('\\', '/', str_replace($_SERVER['DOCUMENT_ROOT'], '', $_SERVER['SCRIPT_FILENAME'])); + } + } + $this->baseFile = $url; + } + return true === $file ? $this->domain() . $this->baseFile : $this->baseFile; + } + + /** + * 璁剧疆鎴栬幏鍙朥RL璁块棶鏍瑰湴鍧 + * @access public + * @param string $url URL鍦板潃 + * @return string + */ + public function root($url = null) + { + if (!is_null($url) && true !== $url) { + $this->root = $url; + return $this; + } elseif (!$this->root) { + $file = $this->baseFile(); + if ($file && 0 !== strpos($this->url(), $file)) { + $file = str_replace('\\', '/', dirname($file)); + } + $this->root = rtrim($file, '/'); + } + return true === $url ? $this->domain() . $this->root : $this->root; + } + + /** + * 鑾峰彇褰撳墠璇锋眰URL鐨刾athinfo淇℃伅锛堝惈URL鍚庣紑锛 + * @access public + * @return string + */ + public function pathinfo() + { + if (is_null($this->pathinfo)) { + if (isset($_GET[Config::get('var_pathinfo')])) { + // 鍒ゆ柇URL閲岄潰鏄惁鏈夊吋瀹规ā寮忓弬鏁 + $_SERVER['PATH_INFO'] = $_GET[Config::get('var_pathinfo')]; + unset($_GET[Config::get('var_pathinfo')]); + } elseif (IS_CLI) { + // CLI妯″紡涓 index.php module/controller/action/params/... + $_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : ''; + } + + // 鍒嗘瀽PATHINFO淇℃伅 + if (!isset($_SERVER['PATH_INFO'])) { + foreach (Config::get('pathinfo_fetch') as $type) { + if (!empty($_SERVER[$type])) { + $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type], $_SERVER['SCRIPT_NAME'])) ? + substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type]; + break; + } + } + } + $this->pathinfo = empty($_SERVER['PATH_INFO']) ? '/' : ltrim($_SERVER['PATH_INFO'], '/'); + } + return $this->pathinfo; + } + + /** + * 鑾峰彇褰撳墠璇锋眰URL鐨刾athinfo淇℃伅(涓嶅惈URL鍚庣紑) + * @access public + * @return string + */ + public function path() + { + if (is_null($this->path)) { + $suffix = Config::get('url_html_suffix'); + $pathinfo = $this->pathinfo(); + if (false === $suffix) { + // 绂佹浼潤鎬佽闂 + $this->path = $pathinfo; + } elseif ($suffix) { + // 鍘婚櫎姝e父鐨刄RL鍚庣紑 + $this->path = preg_replace('/\.(' . ltrim($suffix, '.') . ')$/i', '', $pathinfo); + } else { + // 鍏佽浠讳綍鍚庣紑璁块棶 + $this->path = preg_replace('/\.' . $this->ext() . '$/i', '', $pathinfo); + } + } + return $this->path; + } + + /** + * 褰撳墠URL鐨勮闂悗缂 + * @access public + * @return string + */ + public function ext() + { + return pathinfo($this->pathinfo(), PATHINFO_EXTENSION); + } + + /** + * 鑾峰彇褰撳墠璇锋眰鐨勬椂闂 + * @access public + * @param bool $float 鏄惁浣跨敤娴偣绫诲瀷 + * @return integer|float + */ + public function time($float = false) + { + return $float ? $_SERVER['REQUEST_TIME_FLOAT'] : $_SERVER['REQUEST_TIME']; + } + + /** + * 褰撳墠璇锋眰鐨勮祫婧愮被鍨 + * @access public + * @return false|string + */ + public function type() + { + $accept = $this->server('HTTP_ACCEPT'); + if (empty($accept)) { + return false; + } + + foreach ($this->mimeType as $key => $val) { + $array = explode(',', $val); + foreach ($array as $k => $v) { + if (stristr($accept, $v)) { + return $key; + } + } + } + return false; + } + + /** + * 璁剧疆璧勬簮绫诲瀷 + * @access public + * @param string|array $type 璧勬簮绫诲瀷鍚 + * @param string $val 璧勬簮绫诲瀷 + * @return void + */ + public function mimeType($type, $val = '') + { + if (is_array($type)) { + $this->mimeType = array_merge($this->mimeType, $type); + } else { + $this->mimeType[$type] = $val; + } + } + + /** + * 褰撳墠鐨勮姹傜被鍨 + * @access public + * @param bool $method true 鑾峰彇鍘熷璇锋眰绫诲瀷 + * @return string + */ + public function method($method = false) + { + if (true === $method) { + // 鑾峰彇鍘熷璇锋眰绫诲瀷 + return IS_CLI ? 'GET' : (isset($this->server['REQUEST_METHOD']) ? $this->server['REQUEST_METHOD'] : $_SERVER['REQUEST_METHOD']); + } elseif (!$this->method) { + if (isset($_POST[Config::get('var_method')])) { + $this->method = strtoupper($_POST[Config::get('var_method')]); + $this->{$this->method}($_POST); + } elseif (isset($_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE'])) { + $this->method = strtoupper($_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE']); + } else { + $this->method = IS_CLI ? 'GET' : (isset($this->server['REQUEST_METHOD']) ? $this->server['REQUEST_METHOD'] : $_SERVER['REQUEST_METHOD']); + } + } + return $this->method; + } + + /** + * 鏄惁涓篏ET璇锋眰 + * @access public + * @return bool + */ + public function isGet() + { + return $this->method() == 'GET'; + } + + /** + * 鏄惁涓篜OST璇锋眰 + * @access public + * @return bool + */ + public function isPost() + { + return $this->method() == 'POST'; + } + + /** + * 鏄惁涓篜UT璇锋眰 + * @access public + * @return bool + */ + public function isPut() + { + return $this->method() == 'PUT'; + } + + /** + * 鏄惁涓篋ELTE璇锋眰 + * @access public + * @return bool + */ + public function isDelete() + { + return $this->method() == 'DELETE'; + } + + /** + * 鏄惁涓篐EAD璇锋眰 + * @access public + * @return bool + */ + public function isHead() + { + return $this->method() == 'HEAD'; + } + + /** + * 鏄惁涓篜ATCH璇锋眰 + * @access public + * @return bool + */ + public function isPatch() + { + return $this->method() == 'PATCH'; + } + + /** + * 鏄惁涓篛PTIONS璇锋眰 + * @access public + * @return bool + */ + public function isOptions() + { + return $this->method() == 'OPTIONS'; + } + + /** + * 鏄惁涓篶li + * @access public + * @return bool + */ + public function isCli() + { + return PHP_SAPI == 'cli'; + } + + /** + * 鏄惁涓篶gi + * @access public + * @return bool + */ + public function isCgi() + { + return strpos(PHP_SAPI, 'cgi') === 0; + } + + /** + * 鑾峰彇鑾峰彇褰撳墠璇锋眰鐨勫弬鏁 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function param($name = '', $default = null, $filter = '') + { + if (empty($this->param)) { + $method = $this->method(true); + // 鑷姩鑾峰彇璇锋眰鍙橀噺 + switch ($method) { + case 'POST': + $vars = $this->post(false); + break; + case 'PUT': + case 'DELETE': + case 'PATCH': + $vars = $this->put(false); + break; + default: + $vars = []; + } + // 褰撳墠璇锋眰鍙傛暟鍜孶RL鍦板潃涓殑鍙傛暟鍚堝苟 + $this->param = array_merge($this->get(false), $vars, $this->route(false)); + } + if (true === $name) { + // 鑾峰彇鍖呭惈鏂囦欢涓婁紶淇℃伅鐨勬暟缁 + $file = $this->file(); + $data = array_merge($this->param, $file); + return $this->input($data, '', $default, $filter); + } + return $this->input($this->param, $name, $default, $filter); + } + + /** + * 璁剧疆鑾峰彇鑾峰彇璺敱鍙傛暟 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function route($name = '', $default = null, $filter = '') + { + if (is_array($name)) { + $this->param = []; + return $this->route = array_merge($this->route, $name); + } + return $this->input($this->route, $name, $default, $filter); + } + + /** + * 璁剧疆鑾峰彇鑾峰彇GET鍙傛暟 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function get($name = '', $default = null, $filter = '') + { + if (empty($this->get)) { + $this->get = $_GET; + } + if (is_array($name)) { + $this->param = []; + return $this->get = array_merge($this->get, $name); + } + return $this->input($this->get, $name, $default, $filter); + } + + /** + * 璁剧疆鑾峰彇鑾峰彇POST鍙傛暟 + * @access public + * @param string $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function post($name = '', $default = null, $filter = '') + { + if (empty($this->post)) { + $content = $this->input; + if (empty($_POST) && 'application/json' == $this->contentType()) { + $this->post = (array) json_decode($content, true); + } else { + $this->post = $_POST; + } + } + if (is_array($name)) { + $this->param = []; + return $this->post = array_merge($this->post, $name); + } + return $this->input($this->post, $name, $default, $filter); + } + + /** + * 璁剧疆鑾峰彇鑾峰彇PUT鍙傛暟 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function put($name = '', $default = null, $filter = '') + { + if (is_null($this->put)) { + $content = $this->input; + if ('application/json' == $this->contentType()) { + $this->put = (array) json_decode($content, true); + } else { + parse_str($content, $this->put); + } + } + if (is_array($name)) { + $this->param = []; + return $this->put = is_null($this->put) ? $name : array_merge($this->put, $name); + } + + return $this->input($this->put, $name, $default, $filter); + } + + /** + * 璁剧疆鑾峰彇鑾峰彇DELETE鍙傛暟 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function delete($name = '', $default = null, $filter = '') + { + return $this->put($name, $default, $filter); + } + + /** + * 璁剧疆鑾峰彇鑾峰彇PATCH鍙傛暟 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function patch($name = '', $default = null, $filter = '') + { + return $this->put($name, $default, $filter); + } + + /** + * 鑾峰彇request鍙橀噺 + * @param string $name 鏁版嵁鍚嶇О + * @param string $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function request($name = '', $default = null, $filter = '') + { + if (empty($this->request)) { + $this->request = $_REQUEST; + } + if (is_array($name)) { + $this->param = []; + return $this->request = array_merge($this->request, $name); + } + return $this->input($this->request, $name, $default, $filter); + } + + /** + * 鑾峰彇session鏁版嵁 + * @access public + * @param string|array $name 鏁版嵁鍚嶇О + * @param string $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function session($name = '', $default = null, $filter = '') + { + if (empty($this->session)) { + $this->session = Session::get(); + } + if (is_array($name)) { + return $this->session = array_merge($this->session, $name); + } + return $this->input($this->session, $name, $default, $filter); + } + + /** + * 鑾峰彇cookie鍙傛暟 + * @access public + * @param string|array $name 鏁版嵁鍚嶇О + * @param string $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function cookie($name = '', $default = null, $filter = '') + { + if (empty($this->cookie)) { + $this->cookie = $_COOKIE; + } + if (is_array($name)) { + return $this->cookie = array_merge($this->cookie, $name); + } + return $this->input($this->cookie, $name, $default, $filter); + } + + /** + * 鑾峰彇server鍙傛暟 + * @access public + * @param string|array $name 鏁版嵁鍚嶇О + * @param string $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function server($name = '', $default = null, $filter = '') + { + if (empty($this->server)) { + $this->server = $_SERVER; + } + if (is_array($name)) { + return $this->server = array_merge($this->server, $name); + } + return $this->input($this->server, false === $name ? false : strtoupper($name), $default, $filter); + } + + /** + * 鑾峰彇涓婁紶鐨勬枃浠朵俊鎭 + * @access public + * @param string|array $name 鍚嶇О + * @return null|array|\think\File + */ + public function file($name = '') + { + if (empty($this->file)) { + $this->file = isset($_FILES) ? $_FILES : []; + } + if (is_array($name)) { + return $this->file = array_merge($this->file, $name); + } + $files = $this->file; + if (!empty($files)) { + // 澶勭悊涓婁紶鏂囦欢 + $array = []; + foreach ($files as $key => $file) { + if (is_array($file['name'])) { + $item = []; + $keys = array_keys($file); + $count = count($file['name']); + for ($i = 0; $i < $count; $i++) { + if (empty($file['tmp_name'][$i]) || !is_file($file['tmp_name'][$i])) { + continue; + } + $temp['key'] = $key; + foreach ($keys as $_key) { + $temp[$_key] = $file[$_key][$i]; + } + $item[] = (new File($temp['tmp_name']))->setUploadInfo($temp); + } + $array[$key] = $item; + } else { + if ($file instanceof File) { + $array[$key] = $file; + } else { + if (empty($file['tmp_name']) || !is_file($file['tmp_name'])) { + continue; + } + $array[$key] = (new File($file['tmp_name']))->setUploadInfo($file); + } + } + } + if (strpos($name, '.')) { + list($name, $sub) = explode('.', $name); + } + if ('' === $name) { + // 鑾峰彇鍏ㄩ儴鏂囦欢 + return $array; + } elseif (isset($sub) && isset($array[$name][$sub])) { + return $array[$name][$sub]; + } elseif (isset($array[$name])) { + return $array[$name]; + } + } + return; + } + + /** + * 鑾峰彇鐜鍙橀噺 + * @param string|array $name 鏁版嵁鍚嶇О + * @param string $default 榛樿鍊 + * @param string|array $filter 杩囨护鏂规硶 + * @return mixed + */ + public function env($name = '', $default = null, $filter = '') + { + if (empty($this->env)) { + $this->env = $_ENV; + } + if (is_array($name)) { + return $this->env = array_merge($this->env, $name); + } + return $this->input($this->env, false === $name ? false : strtoupper($name), $default, $filter); + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶇殑Header + * @access public + * @param string|array $name header鍚嶇О + * @param string $default 榛樿鍊 + * @return string + */ + public function header($name = '', $default = null) + { + if (empty($this->header)) { + $header = []; + if (function_exists('apache_request_headers') && $result = apache_request_headers()) { + $header = $result; + } else { + $server = $this->server ?: $_SERVER; + foreach ($server as $key => $val) { + if (0 === strpos($key, 'HTTP_')) { + $key = str_replace('_', '-', strtolower(substr($key, 5))); + $header[$key] = $val; + } + } + if (isset($server['CONTENT_TYPE'])) { + $header['content-type'] = $server['CONTENT_TYPE']; + } + if (isset($server['CONTENT_LENGTH'])) { + $header['content-length'] = $server['CONTENT_LENGTH']; + } + } + $this->header = array_change_key_case($header); + } + if (is_array($name)) { + return $this->header = array_merge($this->header, $name); + } + if ('' === $name) { + return $this->header; + } + $name = str_replace('_', '-', strtolower($name)); + return isset($this->header[$name]) ? $this->header[$name] : $default; + } + + /** + * 鑾峰彇鍙橀噺 鏀寔杩囨护鍜岄粯璁ゅ + * @param array $data 鏁版嵁婧 + * @param string|false $name 瀛楁鍚 + * @param mixed $default 榛樿鍊 + * @param string|array $filter 杩囨护鍑芥暟 + * @return mixed + */ + public function input($data = [], $name = '', $default = null, $filter = '') + { + if (false === $name) { + // 鑾峰彇鍘熷鏁版嵁 + return $data; + } + $name = (string) $name; + if ('' != $name) { + // 瑙f瀽name + if (strpos($name, '/')) { + list($name, $type) = explode('/', $name); + } else { + $type = 's'; + } + // 鎸.鎷嗗垎鎴愬缁存暟缁勮繘琛屽垽鏂 + foreach (explode('.', $name) as $val) { + if (isset($data[$val])) { + $data = $data[$val]; + } else { + // 鏃犺緭鍏ユ暟鎹紝杩斿洖榛樿鍊 + return $default; + } + } + if (is_object($data)) { + return $data; + } + } + + // 瑙f瀽杩囨护鍣 + if (is_null($filter)) { + $filter = []; + } else { + $filter = $filter ?: $this->filter; + if (is_string($filter)) { + $filter = explode(',', $filter); + } else { + $filter = (array) $filter; + } + } + + $filter[] = $default; + if (is_array($data)) { + array_walk_recursive($data, [$this, 'filterValue'], $filter); + reset($data); + } else { + $this->filterValue($data, $name, $filter); + } + + if (isset($type) && $data !== $default) { + // 寮哄埗绫诲瀷杞崲 + $this->typeCast($data, $type); + } + return $data; + } + + /** + * 璁剧疆鎴栬幏鍙栧綋鍓嶇殑杩囨护瑙勫垯 + * @param mixed $filter 杩囨护瑙勫垯 + * @return mixed + */ + public function filter($filter = null) + { + if (is_null($filter)) { + return $this->filter; + } else { + $this->filter = $filter; + } + } + + /** + * 閫掑綊杩囨护缁欏畾鐨勫 + * @param mixed $value 閿 + * @param mixed $key 閿悕 + * @param array $filters 杩囨护鏂规硶+榛樿鍊 + * @return mixed + */ + private function filterValue(&$value, $key, $filters) + { + $default = array_pop($filters); + foreach ($filters as $filter) { + if (is_callable($filter)) { + // 璋冪敤鍑芥暟鎴栬呮柟娉曡繃婊 + $value = call_user_func($filter, $value); + } elseif (is_scalar($value)) { + if (strpos($filter, '/')) { + // 姝e垯杩囨护 + if (!preg_match($filter, $value)) { + // 鍖归厤涓嶆垚鍔熻繑鍥為粯璁ゅ + $value = $default; + break; + } + } elseif (!empty($filter)) { + // filter鍑芥暟涓嶅瓨鍦ㄦ椂, 鍒欎娇鐢╢ilter_var杩涜杩囨护 + // filter涓洪潪鏁村舰鍊兼椂, 璋冪敤filter_id鍙栧緱杩囨护id + $value = filter_var($value, is_int($filter) ? $filter : filter_id($filter)); + if (false === $value) { + $value = $default; + break; + } + } + } + } + return $this->filterExp($value); + } + + /** + * 杩囨护琛ㄥ崟涓殑琛ㄨ揪寮 + * @param string $value + * @return void + */ + public function filterExp(&$value) + { + // 杩囨护鏌ヨ鐗规畩瀛楃 + if (is_string($value) && preg_match('/^(EXP|NEQ|GT|EGT|LT|ELT|OR|XOR|LIKE|NOTLIKE|NOT BETWEEN|NOTBETWEEN|BETWEEN|NOTIN|NOT IN|IN)$/i', $value)) { + $value .= ' '; + } + // TODO 鍏朵粬瀹夊叏杩囨护 + } + + /** + * 寮哄埗绫诲瀷杞崲 + * @param string $data + * @param string $type + * @return mixed + */ + private function typeCast(&$data, $type) + { + switch (strtolower($type)) { + // 鏁扮粍 + case 'a': + $data = (array) $data; + break; + // 鏁板瓧 + case 'd': + $data = (int) $data; + break; + // 娴偣 + case 'f': + $data = (float) $data; + break; + // 甯冨皵 + case 'b': + $data = (boolean) $data; + break; + // 瀛楃涓 + case 's': + default: + if (is_scalar($data)) { + $data = (string) $data; + } else { + throw new \InvalidArgumentException('variable type error锛' . gettype($data)); + } + } + } + + /** + * 鏄惁瀛樺湪鏌愪釜璇锋眰鍙傛暟 + * @access public + * @param string $name 鍙橀噺鍚 + * @param string $type 鍙橀噺绫诲瀷 + * @param bool $checkEmpty 鏄惁妫娴嬬┖鍊 + * @return mixed + */ + public function has($name, $type = 'param', $checkEmpty = false) + { + if (empty($this->$type)) { + $param = $this->$type(); + } else { + $param = $this->$type; + } + // 鎸.鎷嗗垎鎴愬缁存暟缁勮繘琛屽垽鏂 + foreach (explode('.', $name) as $val) { + if (isset($param[$val])) { + $param = $param[$val]; + } else { + return false; + } + } + return ($checkEmpty && '' === $param) ? false : true; + } + + /** + * 鑾峰彇鎸囧畾鐨勫弬鏁 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param string $type 鍙橀噺绫诲瀷 + * @return mixed + */ + public function only($name, $type = 'param') + { + $param = $this->$type(); + if (is_string($name)) { + $name = explode(',', $name); + } + $item = []; + foreach ($name as $key) { + if (isset($param[$key])) { + $item[$key] = $param[$key]; + } + } + return $item; + } + + /** + * 鎺掗櫎鎸囧畾鍙傛暟鑾峰彇 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param string $type 鍙橀噺绫诲瀷 + * @return mixed + */ + public function except($name, $type = 'param') + { + $param = $this->$type(); + if (is_string($name)) { + $name = explode(',', $name); + } + foreach ($name as $key) { + if (isset($param[$key])) { + unset($param[$key]); + } + } + return $param; + } + + /** + * 褰撳墠鏄惁ssl + * @access public + * @return bool + */ + public function isSsl() + { + $server = array_merge($_SERVER, $this->server); + if (isset($server['HTTPS']) && ('1' == $server['HTTPS'] || 'on' == strtolower($server['HTTPS']))) { + return true; + } elseif (isset($server['REQUEST_SCHEME']) && 'https' == $server['REQUEST_SCHEME']) { + return true; + } elseif (isset($server['SERVER_PORT']) && ('443' == $server['SERVER_PORT'])) { + return true; + } elseif (isset($server['HTTP_X_FORWARDED_PROTO']) && 'https' == $server['HTTP_X_FORWARDED_PROTO']) { + return true; + } + return false; + } + + /** + * 褰撳墠鏄惁Ajax璇锋眰 + * @access public + * @param bool $ajax true 鑾峰彇鍘熷ajax璇锋眰 + * @return bool + */ + public function isAjax($ajax = false) + { + $value = $this->server('HTTP_X_REQUESTED_WITH', '', 'strtolower'); + $result = ('xmlhttprequest' == $value) ? true : false; + if (true === $ajax) { + return $result; + } else { + return $this->param(Config::get('var_ajax')) ? true : $result; + } + } + + /** + * 褰撳墠鏄惁Pjax璇锋眰 + * @access public + * @param bool $pjax true 鑾峰彇鍘熷pjax璇锋眰 + * @return bool + */ + public function isPjax($pjax = false) + { + $result = !is_null($this->server('HTTP_X_PJAX')) ? true : false; + if (true === $pjax) { + return $result; + } else { + return $this->param(Config::get('var_pjax')) ? true : $result; + } + } + + /** + * 鑾峰彇瀹㈡埛绔疘P鍦板潃 + * @param integer $type 杩斿洖绫诲瀷 0 杩斿洖IP鍦板潃 1 杩斿洖IPV4鍦板潃鏁板瓧 + * @param boolean $adv 鏄惁杩涜楂樼骇妯″紡鑾峰彇锛堟湁鍙兘琚吉瑁咃級 + * @return mixed + */ + public function ip($type = 0, $adv = false) + { + $type = $type ? 1 : 0; + static $ip = null; + if (null !== $ip) { + return $ip[$type]; + } + + if ($adv) { + if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) { + $arr = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']); + $pos = array_search('unknown', $arr); + if (false !== $pos) { + unset($arr[$pos]); + } + $ip = trim(current($arr)); + } elseif (isset($_SERVER['HTTP_CLIENT_IP'])) { + $ip = $_SERVER['HTTP_CLIENT_IP']; + } elseif (isset($_SERVER['REMOTE_ADDR'])) { + $ip = $_SERVER['REMOTE_ADDR']; + } + } elseif (isset($_SERVER['REMOTE_ADDR'])) { + $ip = $_SERVER['REMOTE_ADDR']; + } + // IP鍦板潃鍚堟硶楠岃瘉 + $long = sprintf("%u", ip2long($ip)); + $ip = $long ? [$ip, $long] : ['0.0.0.0', 0]; + return $ip[$type]; + } + + /** + * 妫娴嬫槸鍚︿娇鐢ㄦ墜鏈鸿闂 + * @access public + * @return bool + */ + public function isMobile() + { + if (isset($_SERVER['HTTP_VIA']) && stristr($_SERVER['HTTP_VIA'], "wap")) { + return true; + } elseif (isset($_SERVER['HTTP_ACCEPT']) && strpos(strtoupper($_SERVER['HTTP_ACCEPT']), "VND.WAP.WML")) { + return true; + } elseif (isset($_SERVER['HTTP_X_WAP_PROFILE']) || isset($_SERVER['HTTP_PROFILE'])) { + return true; + } elseif (isset($_SERVER['HTTP_USER_AGENT']) && preg_match('/(blackberry|configuration\/cldc|hp |hp-|htc |htc_|htc-|iemobile|kindle|midp|mmp|motorola|mobile|nokia|opera mini|opera |Googlebot-Mobile|YahooSeeker\/M1A1-R2D2|android|iphone|ipod|mobi|palm|palmos|pocket|portalmmm|ppc;|smartphone|sonyericsson|sqh|spv|symbian|treo|up.browser|up.link|vodafone|windows ce|xda |xda_)/i', $_SERVER['HTTP_USER_AGENT'])) { + return true; + } else { + return false; + } + } + + /** + * 褰撳墠URL鍦板潃涓殑scheme鍙傛暟 + * @access public + * @return string + */ + public function scheme() + { + return $this->isSsl() ? 'https' : 'http'; + } + + /** + * 褰撳墠璇锋眰URL鍦板潃涓殑query鍙傛暟 + * @access public + * @return string + */ + public function query() + { + return $this->server('QUERY_STRING'); + } + + /** + * 褰撳墠璇锋眰鐨刪ost + * @access public + * @return string + */ + public function host() + { + return $this->server('HTTP_HOST'); + } + + /** + * 褰撳墠璇锋眰URL鍦板潃涓殑port鍙傛暟 + * @access public + * @return integer + */ + public function port() + { + return $this->server('SERVER_PORT'); + } + + /** + * 褰撳墠璇锋眰 SERVER_PROTOCOL + * @access public + * @return integer + */ + public function protocol() + { + return $this->server('SERVER_PROTOCOL'); + } + + /** + * 褰撳墠璇锋眰 REMOTE_PORT + * @access public + * @return integer + */ + public function remotePort() + { + return $this->server('REMOTE_PORT'); + } + + /** + * 褰撳墠璇锋眰 HTTP_CONTENT_TYPE + * @access public + * @return string + */ + public function contentType() + { + $contentType = $this->server('CONTENT_TYPE'); + if ($contentType) { + list($type) = explode(';', $contentType); + return trim($type); + } + return ''; + } + + /** + * 鑾峰彇褰撳墠璇锋眰鐨勮矾鐢变俊鎭 + * @access public + * @param array $route 璺敱鍚嶇О + * @return array + */ + public function routeInfo($route = []) + { + if (!empty($route)) { + $this->routeInfo = $route; + } else { + return $this->routeInfo; + } + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶈姹傜殑璋冨害淇℃伅 + * @access public + * @param array $dispatch 璋冨害淇℃伅 + * @return array + */ + public function dispatch($dispatch = null) + { + if (!is_null($dispatch)) { + $this->dispatch = $dispatch; + } + return $this->dispatch; + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶇殑妯″潡鍚 + * @access public + * @param string $module 妯″潡鍚 + * @return string|Request + */ + public function module($module = null) + { + if (!is_null($module)) { + $this->module = $module; + return $this; + } else { + return $this->module ?: ''; + } + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶇殑鎺у埗鍣ㄥ悕 + * @access public + * @param string $controller 鎺у埗鍣ㄥ悕 + * @return string|Request + */ + public function controller($controller = null) + { + if (!is_null($controller)) { + $this->controller = $controller; + return $this; + } else { + return $this->controller ?: ''; + } + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶇殑鎿嶄綔鍚 + * @access public + * @param string $action 鎿嶄綔鍚 + * @return string|Request + */ + public function action($action = null) + { + if (!is_null($action)) { + $this->action = $action; + return $this; + } else { + return $this->action ?: ''; + } + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶇殑璇█ + * @access public + * @param string $lang 璇█鍚 + * @return string|Request + */ + public function langset($lang = null) + { + if (!is_null($lang)) { + $this->langset = $lang; + return $this; + } else { + return $this->langset ?: ''; + } + } + + /** + * 璁剧疆鎴栬呰幏鍙栧綋鍓嶈姹傜殑content + * @access public + * @return string + */ + public function getContent() + { + if (is_null($this->content)) { + $this->content = $this->input; + } + return $this->content; + } + + /** + * 鑾峰彇褰撳墠璇锋眰鐨刾hp://input + * @access public + * @return string + */ + public function getInput() + { + return $this->input; + } + + /** + * 鐢熸垚璇锋眰浠ょ墝 + * @access public + * @param string $name 浠ょ墝鍚嶇О + * @param mixed $type 浠ょ墝鐢熸垚鏂规硶 + * @return string + */ + public function token($name = '__token__', $type = 'md5') + { + $type = is_callable($type) ? $type : 'md5'; + $token = call_user_func($type, $_SERVER['REQUEST_TIME_FLOAT']); + if ($this->isAjax()) { + header($name . ': ' . $token); + } + Session::set($name, $token); + return $token; + } + + /** + * 璁剧疆褰撳墠鍦板潃鐨勮姹傜紦瀛 + * @access public + * @param string $key 缂撳瓨鏍囪瘑锛屾敮鎸佸彉閲忚鍒 锛屼緥濡 item/:name/:id + * @param mixed $expire 缂撳瓨鏈夋晥鏈 + * @return void + */ + public function cache($key, $expire = null) + { + if (false !== $key && $this->isGet() && !$this->isCheckCache) { + // 鏍囪璇锋眰缂撳瓨妫鏌 + $this->isCheckCache = true; + if (false === $expire) { + // 鍏抽棴褰撳墠缂撳瓨 + return; + } + if ($key instanceof \Closure) { + $key = call_user_func_array($key, [$this]); + } elseif (true === $key) { + // 鑷姩缂撳瓨鍔熻兘 + $key = '__URL__'; + } elseif (strpos($key, '|')) { + list($key, $fun) = explode('|', $key); + } + // 鐗规畩瑙勫垯鏇挎崲 + if (false !== strpos($key, '__')) { + $key = str_replace(['__MODULE__', '__CONTROLLER__', '__ACTION__', '__URL__'], [$this->module, $this->controller, $this->action, md5($this->url())], $key); + } + + if (false !== strpos($key, ':')) { + $param = $this->param(); + foreach ($param as $item => $val) { + if (is_string($val) && false !== strpos($key, ':' . $item)) { + $key = str_replace(':' . $item, $val, $key); + } + } + } elseif (strpos($key, ']')) { + if ('[' . $this->ext() . ']' == $key) { + // 缂撳瓨鏌愪釜鍚庣紑鐨勮姹 + $key = md5($this->url()); + } else { + return; + } + } + if (isset($fun)) { + $key = $fun($key); + } + + if (strtotime($this->server('HTTP_IF_MODIFIED_SINCE')) + $expire > $_SERVER['REQUEST_TIME']) { + // 璇诲彇缂撳瓨 + $response = Response::create()->code(304); + throw new \think\exception\HttpResponseException($response); + } elseif (Cache::has($key)) { + list($content, $header) = Cache::get($key); + $response = Response::create($content)->header($header); + throw new \think\exception\HttpResponseException($response); + } else { + $this->cache = [$key, $expire]; + } + } + } + + /** + * 璇诲彇璇锋眰缂撳瓨璁剧疆 + * @access public + * @return array + */ + public function getCache() + { + return $this->cache; + } + + /** + * 璁剧疆褰撳墠璇锋眰缁戝畾鐨勫璞″疄渚 + * @access public + * @param string $name 缁戝畾鐨勫璞℃爣璇 + * @param mixed $obj 缁戝畾鐨勫璞″疄渚 + * @return mixed + */ + public function bind($name, $obj = null) + { + if (is_array($name)) { + $this->bind = array_merge($this->bind, $name); + } else { + $this->bind[$name] = $obj; + } + } + + public function __set($name, $value) + { + $this->bind[$name] = $value; + } + + public function __get($name) + { + return isset($this->bind[$name]) ? $this->bind[$name] : null; + } + + public function __isset($name) + { + return isset($this->bind[$name]); + } +} diff --git a/thinkphp/library/think/Response.php b/thinkphp/library/think/Response.php new file mode 100644 index 000000000..656198f03 --- /dev/null +++ b/thinkphp/library/think/Response.php @@ -0,0 +1,328 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\response\Json as JsonResponse; +use think\response\Jsonp as JsonpResponse; +use think\response\Redirect as RedirectResponse; +use think\response\View as ViewResponse; +use think\response\Xml as XmlResponse; + +class Response +{ + + // 鍘熷鏁版嵁 + protected $data; + + // 褰撳墠鐨刢ontentType + protected $contentType = 'text/html'; + + // 瀛楃闆 + protected $charset = 'utf-8'; + + //鐘舵 + protected $code = 200; + + // 杈撳嚭鍙傛暟 + protected $options = []; + // header鍙傛暟 + protected $header = []; + + protected $content = null; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param mixed $data 杈撳嚭鏁版嵁 + * @param int $code + * @param array $header + * @param array $options 杈撳嚭鍙傛暟 + */ + public function __construct($data = '', $code = 200, array $header = [], $options = []) + { + $this->data($data); + $this->header = $header; + $this->code = $code; + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + $this->contentType($this->contentType, $this->charset); + } + + /** + * 鍒涘缓Response瀵硅薄 + * @access public + * @param mixed $data 杈撳嚭鏁版嵁 + * @param string $type 杈撳嚭绫诲瀷 + * @param int $code + * @param array $header + * @param array $options 杈撳嚭鍙傛暟 + * @return Response|JsonResponse|ViewResponse|XmlResponse|RedirectResponse|JsonpResponse + */ + public static function create($data = '', $type = '', $code = 200, array $header = [], $options = []) + { + $type = empty($type) ? 'null' : strtolower($type); + + $class = false !== strpos($type, '\\') ? $type : '\\think\\response\\' . ucfirst($type); + if (class_exists($class)) { + $response = new $class($data, $code, $header, $options); + } else { + $response = new static($data, $code, $header, $options); + } + + return $response; + } + + /** + * 鍙戦佹暟鎹埌瀹㈡埛绔 + * @access public + * @return mixed + * @throws \InvalidArgumentException + */ + public function send() + { + // 澶勭悊杈撳嚭鏁版嵁 + $data = $this->getContent(); + + // Trace璋冭瘯娉ㄥ叆 + if (Env::get('app_trace', Config::get('app_trace'))) { + Debug::inject($this, $data); + } + + if (200 == $this->code) { + $cache = Request::instance()->getCache(); + if ($cache) { + $this->header['Cache-Control'] = 'max-age=' . $cache[1] . ',must-revalidate'; + $this->header['Last-Modified'] = gmdate('D, d M Y H:i:s') . ' GMT'; + $this->header['Expires'] = gmdate('D, d M Y H:i:s', $_SERVER['REQUEST_TIME'] + $cache[1]) . ' GMT'; + Cache::set($cache[0], [$data, $this->header], $cache[1]); + } + } + + if (!headers_sent() && !empty($this->header)) { + // 鍙戦佺姸鎬佺爜 + http_response_code($this->code); + // 鍙戦佸ご閮ㄤ俊鎭 + foreach ($this->header as $name => $val) { + header($name . ':' . $val); + } + } + + echo $data; + + if (function_exists('fastcgi_finish_request')) { + // 鎻愰珮椤甸潰鍝嶅簲 + fastcgi_finish_request(); + } + + // 鐩戝惉response_end + Hook::listen('response_end', $this); + + // 娓呯┖褰撴璇锋眰鏈夋晥鐨勬暟鎹 + if (!($this instanceof RedirectResponse)) { + Session::flush(); + } + } + + /** + * 澶勭悊鏁版嵁 + * @access protected + * @param mixed $data 瑕佸鐞嗙殑鏁版嵁 + * @return mixed + */ + protected function output($data) + { + return $data; + } + + /** + * 杈撳嚭鐨勫弬鏁 + * @access public + * @param mixed $options 杈撳嚭鍙傛暟 + * @return $this + */ + public function options($options = []) + { + $this->options = array_merge($this->options, $options); + return $this; + } + + /** + * 杈撳嚭鏁版嵁璁剧疆 + * @access public + * @param mixed $data 杈撳嚭鏁版嵁 + * @return $this + */ + public function data($data) + { + $this->data = $data; + return $this; + } + + /** + * 璁剧疆鍝嶅簲澶 + * @access public + * @param string|array $name 鍙傛暟鍚 + * @param string $value 鍙傛暟鍊 + * @return $this + */ + public function header($name, $value = null) + { + if (is_array($name)) { + $this->header = array_merge($this->header, $name); + } else { + $this->header[$name] = $value; + } + return $this; + } + + /** + * 璁剧疆椤甸潰杈撳嚭鍐呭 + * @param $content + * @return $this + */ + public function content($content) + { + if (null !== $content && !is_string($content) && !is_numeric($content) && !is_callable([ + $content, + '__toString', + ]) + ) { + throw new \InvalidArgumentException(sprintf('variable type error锛 %s', gettype($content))); + } + + $this->content = (string) $content; + + return $this; + } + + /** + * 鍙戦丠TTP鐘舵 + * @param integer $code 鐘舵佺爜 + * @return $this + */ + public function code($code) + { + $this->code = $code; + return $this; + } + + /** + * LastModified + * @param string $time + * @return $this + */ + public function lastModified($time) + { + $this->header['Last-Modified'] = $time; + return $this; + } + + /** + * Expires + * @param string $time + * @return $this + */ + public function expires($time) + { + $this->header['Expires'] = $time; + return $this; + } + + /** + * ETag + * @param string $eTag + * @return $this + */ + public function eTag($eTag) + { + $this->header['ETag'] = $eTag; + return $this; + } + + /** + * 椤甸潰缂撳瓨鎺у埗 + * @param string $cache 鐘舵佺爜 + * @return $this + */ + public function cacheControl($cache) + { + $this->header['Cache-control'] = $cache; + return $this; + } + + /** + * 椤甸潰杈撳嚭绫诲瀷 + * @param string $contentType 杈撳嚭绫诲瀷 + * @param string $charset 杈撳嚭缂栫爜 + * @return $this + */ + public function contentType($contentType, $charset = 'utf-8') + { + $this->header['Content-Type'] = $contentType . '; charset=' . $charset; + return $this; + } + + /** + * 鑾峰彇澶撮儴淇℃伅 + * @param string $name 澶撮儴鍚嶇О + * @return mixed + */ + public function getHeader($name = '') + { + if (!empty($name)) { + return isset($this->header[$name]) ? $this->header[$name] : null; + } else { + return $this->header; + } + } + + /** + * 鑾峰彇鍘熷鏁版嵁 + * @return mixed + */ + public function getData() + { + return $this->data; + } + + /** + * 鑾峰彇杈撳嚭鏁版嵁 + * @return mixed + */ + public function getContent() + { + if (null == $this->content) { + $content = $this->output($this->data); + + if (null !== $content && !is_string($content) && !is_numeric($content) && !is_callable([ + $content, + '__toString', + ]) + ) { + throw new \InvalidArgumentException(sprintf('variable type error锛 %s', gettype($content))); + } + + $this->content = (string) $content; + } + return $this->content; + } + + /** + * 鑾峰彇鐘舵佺爜 + * @return integer + */ + public function getCode() + { + return $this->code; + } +} diff --git a/thinkphp/library/think/Route.php b/thinkphp/library/think/Route.php new file mode 100644 index 000000000..1629d6364 --- /dev/null +++ b/thinkphp/library/think/Route.php @@ -0,0 +1,1590 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\exception\HttpException; + +class Route +{ + // 璺敱瑙勫垯 + private static $rules = [ + 'get' => [], + 'post' => [], + 'put' => [], + 'delete' => [], + 'patch' => [], + 'head' => [], + 'options' => [], + '*' => [], + 'alias' => [], + 'domain' => [], + 'pattern' => [], + 'name' => [], + ]; + + // REST璺敱鎿嶄綔鏂规硶瀹氫箟 + private static $rest = [ + 'index' => ['get', '', 'index'], + 'create' => ['get', '/create', 'create'], + 'edit' => ['get', '/:id/edit', 'edit'], + 'read' => ['get', '/:id', 'read'], + 'save' => ['post', '', 'save'], + 'update' => ['put', '/:id', 'update'], + 'delete' => ['delete', '/:id', 'delete'], + ]; + + // 涓嶅悓璇锋眰绫诲瀷鐨勬柟娉曞墠缂 + private static $methodPrefix = [ + 'get' => 'get', + 'post' => 'post', + 'put' => 'put', + 'delete' => 'delete', + 'patch' => 'patch', + ]; + + // 瀛愬煙鍚 + private static $subDomain = ''; + // 鍩熷悕缁戝畾 + private static $bind = []; + // 褰撳墠鍒嗙粍淇℃伅 + private static $group = []; + // 褰撳墠瀛愬煙鍚嶇粦瀹 + private static $domainBind; + private static $domainRule; + // 褰撳墠鍩熷悕 + private static $domain; + // 褰撳墠璺敱鎵ц杩囩▼涓殑鍙傛暟 + private static $option = []; + + /** + * 娉ㄥ唽鍙橀噺瑙勫垯 + * @access public + * @param string|array $name 鍙橀噺鍚 + * @param string $rule 鍙橀噺瑙勫垯 + * @return void + */ + public static function pattern($name = null, $rule = '') + { + if (is_array($name)) { + self::$rules['pattern'] = array_merge(self::$rules['pattern'], $name); + } else { + self::$rules['pattern'][$name] = $rule; + } + } + + /** + * 娉ㄥ唽瀛愬煙鍚嶉儴缃茶鍒 + * @access public + * @param string|array $domain 瀛愬煙鍚 + * @param mixed $rule 璺敱瑙勫垯 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function domain($domain, $rule = '', $option = [], $pattern = []) + { + if (is_array($domain)) { + foreach ($domain as $key => $item) { + self::domain($key, $item, $option, $pattern); + } + } elseif ($rule instanceof \Closure) { + // 鎵ц闂寘 + self::setDomain($domain); + call_user_func_array($rule, []); + self::setDomain(null); + } elseif (is_array($rule)) { + self::setDomain($domain); + self::group('', function () use ($rule) { + // 鍔ㄦ佹敞鍐屽煙鍚嶇殑璺敱瑙勫垯 + self::registerRules($rule); + }, $option, $pattern); + self::setDomain(null); + } else { + self::$rules['domain'][$domain]['[bind]'] = [$rule, $option, $pattern]; + } + } + + private static function setDomain($domain) + { + self::$domain = $domain; + } + + /** + * 璁剧疆璺敱缁戝畾 + * @access public + * @param mixed $bind 缁戝畾淇℃伅 + * @param string $type 缁戝畾绫诲瀷 榛樿涓簃odule 鏀寔 namespace class controller + * @return mixed + */ + public static function bind($bind, $type = 'module') + { + self::$bind = ['type' => $type, $type => $bind]; + } + + /** + * 璁剧疆鎴栬呰幏鍙栬矾鐢辨爣璇 + * @access public + * @param string|array $name 璺敱鍛藉悕鏍囪瘑 鏁扮粍琛ㄧず鎵归噺璁剧疆 + * @param array $value 璺敱鍦板潃鍙婂彉閲忎俊鎭 + * @return array + */ + public static function name($name = '', $value = null) + { + if (is_array($name)) { + return self::$rules['name'] = $name; + } elseif ('' === $name) { + return self::$rules['name']; + } elseif (!is_null($value)) { + self::$rules['name'][strtolower($name)][] = $value; + } else { + $name = strtolower($name); + return isset(self::$rules['name'][$name]) ? self::$rules['name'][$name] : null; + } + } + + /** + * 璇诲彇璺敱缁戝畾 + * @access public + * @param string $type 缁戝畾绫诲瀷 + * @return mixed + */ + public static function getBind($type) + { + return isset(self::$bind[$type]) ? self::$bind[$type] : null; + } + + /** + * 瀵煎叆閰嶇疆鏂囦欢鐨勮矾鐢辫鍒 + * @access public + * @param array $rule 璺敱瑙勫垯 + * @param string $type 璇锋眰绫诲瀷 + * @return void + */ + public static function import(array $rule, $type = '*') + { + // 妫鏌ュ煙鍚嶉儴缃 + if (isset($rule['__domain__'])) { + self::domain($rule['__domain__']); + unset($rule['__domain__']); + } + + // 妫鏌ュ彉閲忚鍒 + if (isset($rule['__pattern__'])) { + self::pattern($rule['__pattern__']); + unset($rule['__pattern__']); + } + + // 妫鏌ヨ矾鐢卞埆鍚 + if (isset($rule['__alias__'])) { + self::alias($rule['__alias__']); + unset($rule['__alias__']); + } + + // 妫鏌ヨ祫婧愯矾鐢 + if (isset($rule['__rest__'])) { + self::resource($rule['__rest__']); + unset($rule['__rest__']); + } + + self::registerRules($rule, strtolower($type)); + } + + // 鎵归噺娉ㄥ唽璺敱 + protected static function registerRules($rules, $type = '*') + { + foreach ($rules as $key => $val) { + if (is_numeric($key)) { + $key = array_shift($val); + } + if (empty($val)) { + continue; + } + if (is_string($key) && 0 === strpos($key, '[')) { + $key = substr($key, 1, -1); + self::group($key, $val); + } elseif (is_array($val)) { + self::setRule($key, $val[0], $type, $val[1], isset($val[2]) ? $val[2] : []); + } else { + self::setRule($key, $val, $type); + } + } + } + + /** + * 娉ㄥ唽璺敱瑙勫垯 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param string $type 璇锋眰绫诲瀷 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function rule($rule, $route = '', $type = '*', $option = [], $pattern = []) + { + $group = self::getGroup('name'); + if (!is_null($group)) { + // 璺敱鍒嗙粍 + $option = array_merge(self::getGroup('option'), $option); + $pattern = array_merge(self::getGroup('pattern'), $pattern); + } + + $type = strtolower($type); + + if (strpos($type, '|')) { + $option['method'] = $type; + $type = '*'; + } + if (is_array($rule) && empty($route)) { + foreach ($rule as $key => $val) { + if (is_numeric($key)) { + $key = array_shift($val); + } + if (is_array($val)) { + $route = $val[0]; + $option1 = array_merge($option, $val[1]); + $pattern1 = array_merge($pattern, isset($val[2]) ? $val[2] : []); + } else { + $route = $val; + } + self::setRule($key, $route, $type, isset($option1) ? $option1 : $option, isset($pattern1) ? $pattern1 : $pattern, $group); + } + } else { + self::setRule($rule, $route, $type, $option, $pattern, $group); + } + + } + + /** + * 璁剧疆璺敱瑙勫垯 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param string $type 璇锋眰绫诲瀷 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @param string $group 鎵灞炲垎缁 + * @return void + */ + protected static function setRule($rule, $route, $type = '*', $option = [], $pattern = [], $group = '') + { + if (is_array($rule)) { + $name = $rule[0]; + $rule = $rule[1]; + } elseif (is_string($route)) { + $name = $route; + } + if (!isset($option['complete_match'])) { + if (Config::get('route_complete_match')) { + $option['complete_match'] = true; + } elseif ('$' == substr($rule, -1, 1)) { + // 鏄惁瀹屾暣鍖归厤 + $option['complete_match'] = true; + } + } elseif (empty($option['complete_match']) && '$' == substr($rule, -1, 1)) { + // 鏄惁瀹屾暣鍖归厤 + $option['complete_match'] = true; + } + + if ('$' == substr($rule, -1, 1)) { + $rule = substr($rule, 0, -1); + } + + if ('/' != $rule || $group) { + $rule = trim($rule, '/'); + } + $vars = self::parseVar($rule); + if (isset($name)) { + $key = $group ? $group . ($rule ? '/' . $rule : '') : $rule; + self::name($name, [$key, $vars, self::$domain]); + } + if ($group) { + if ('*' != $type) { + $option['method'] = $type; + } + if (self::$domain) { + self::$rules['domain'][self::$domain]['*'][$group]['rule'][] = ['rule' => $rule, 'route' => $route, 'var' => $vars, 'option' => $option, 'pattern' => $pattern]; + } else { + self::$rules['*'][$group]['rule'][] = ['rule' => $rule, 'route' => $route, 'var' => $vars, 'option' => $option, 'pattern' => $pattern]; + } + } else { + if ('*' != $type && isset(self::$rules['*'][$rule])) { + unset(self::$rules['*'][$rule]); + } + if (self::$domain) { + self::$rules['domain'][self::$domain][$type][$rule] = ['rule' => $rule, 'route' => $route, 'var' => $vars, 'option' => $option, 'pattern' => $pattern]; + } else { + self::$rules[$type][$rule] = ['rule' => $rule, 'route' => $route, 'var' => $vars, 'option' => $option, 'pattern' => $pattern]; + } + if ('*' == $type) { + // 娉ㄥ唽璺敱蹇嵎鏂瑰紡 + foreach (['get', 'post', 'put', 'delete', 'patch', 'head', 'options'] as $method) { + if (self::$domain) { + self::$rules['domain'][self::$domain][$method][$rule] = true; + } else { + self::$rules[$method][$rule] = true; + } + } + } + } + } + + /** + * 璁剧疆褰撳墠鎵ц鐨勫弬鏁颁俊鎭 + * @access public + * @param array $options 鍙傛暟淇℃伅 + * @return mixed + */ + protected static function setOption($options = []) + { + self::$option[] = $options; + } + + /** + * 鑾峰彇褰撳墠鎵ц鐨勬墍鏈夊弬鏁颁俊鎭 + * @access public + * @return array + */ + public static function getOption() + { + return self::$option; + } + + /** + * 鑾峰彇褰撳墠鐨勫垎缁勪俊鎭 + * @access public + * @param string $type 鍒嗙粍淇℃伅鍚嶇О name option pattern + * @return mixed + */ + public static function getGroup($type) + { + if (isset(self::$group[$type])) { + return self::$group[$type]; + } else { + return 'name' == $type ? null : []; + } + } + + /** + * 璁剧疆褰撳墠鐨勮矾鐢卞垎缁 + * @access public + * @param string $name 鍒嗙粍鍚嶇О + * @param array $option 鍒嗙粍璺敱鍙傛暟 + * @param array $pattern 鍒嗙粍鍙橀噺瑙勫垯 + * @return void + */ + public static function setGroup($name, $option = [], $pattern = []) + { + self::$group['name'] = $name; + self::$group['option'] = $option ?: []; + self::$group['pattern'] = $pattern ?: []; + } + + /** + * 娉ㄥ唽璺敱鍒嗙粍 + * @access public + * @param string|array $name 鍒嗙粍鍚嶇О鎴栬呭弬鏁 + * @param array|\Closure $routes 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function group($name, $routes, $option = [], $pattern = []) + { + if (is_array($name)) { + $option = $name; + $name = isset($option['name']) ? $option['name'] : ''; + } + // 鍒嗙粍 + $currentGroup = self::getGroup('name'); + if ($currentGroup) { + $name = $currentGroup . ($name ? '/' . ltrim($name, '/') : ''); + } + if (!empty($name)) { + if ($routes instanceof \Closure) { + $currentOption = self::getGroup('option'); + $currentPattern = self::getGroup('pattern'); + self::setGroup($name, array_merge($currentOption, $option), array_merge($currentPattern, $pattern)); + call_user_func_array($routes, []); + self::setGroup($currentGroup, $currentOption, $currentPattern); + if ($currentGroup != $name) { + self::$rules['*'][$name]['route'] = ''; + self::$rules['*'][$name]['var'] = self::parseVar($name); + self::$rules['*'][$name]['option'] = $option; + self::$rules['*'][$name]['pattern'] = $pattern; + } + } else { + $item = []; + foreach ($routes as $key => $val) { + if (is_numeric($key)) { + $key = array_shift($val); + } + if (is_array($val)) { + $route = $val[0]; + $option1 = array_merge($option, isset($val[1]) ? $val[1] : []); + $pattern1 = array_merge($pattern, isset($val[2]) ? $val[2] : []); + } else { + $route = $val; + } + + $options = isset($option1) ? $option1 : $option; + $patterns = isset($pattern1) ? $pattern1 : $pattern; + if ('$' == substr($key, -1, 1)) { + // 鏄惁瀹屾暣鍖归厤 + $options['complete_match'] = true; + $key = substr($key, 0, -1); + } + $key = trim($key, '/'); + $vars = self::parseVar($key); + $item[] = ['rule' => $key, 'route' => $route, 'var' => $vars, 'option' => $options, 'pattern' => $patterns]; + // 璁剧疆璺敱鏍囪瘑 + self::name($route, [$name . ($key ? '/' . $key : ''), $vars, self::$domain]); + } + self::$rules['*'][$name] = ['rule' => $item, 'route' => '', 'var' => [], 'option' => $option, 'pattern' => $pattern]; + } + + foreach (['get', 'post', 'put', 'delete', 'patch', 'head', 'options'] as $method) { + if (!isset(self::$rules[$method][$name])) { + self::$rules[$method][$name] = true; + } elseif (is_array(self::$rules[$method][$name])) { + self::$rules[$method][$name] = array_merge(self::$rules['*'][$name], self::$rules[$method][$name]); + } + } + + } elseif ($routes instanceof \Closure) { + // 闂寘娉ㄥ唽 + $currentOption = self::getGroup('option'); + $currentPattern = self::getGroup('pattern'); + self::setGroup('', array_merge($currentOption, $option), array_merge($currentPattern, $pattern)); + call_user_func_array($routes, []); + self::setGroup($currentGroup, $currentOption, $currentPattern); + } else { + // 鎵归噺娉ㄥ唽璺敱 + self::rule($routes, '', '*', $option, $pattern); + } + } + + /** + * 娉ㄥ唽璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function any($rule, $route = '', $option = [], $pattern = []) + { + self::rule($rule, $route, '*', $option, $pattern); + } + + /** + * 娉ㄥ唽GET璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function get($rule, $route = '', $option = [], $pattern = []) + { + self::rule($rule, $route, 'GET', $option, $pattern); + } + + /** + * 娉ㄥ唽POST璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function post($rule, $route = '', $option = [], $pattern = []) + { + self::rule($rule, $route, 'POST', $option, $pattern); + } + + /** + * 娉ㄥ唽PUT璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function put($rule, $route = '', $option = [], $pattern = []) + { + self::rule($rule, $route, 'PUT', $option, $pattern); + } + + /** + * 娉ㄥ唽DELETE璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function delete($rule, $route = '', $option = [], $pattern = []) + { + self::rule($rule, $route, 'DELETE', $option, $pattern); + } + + /** + * 娉ㄥ唽PATCH璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function patch($rule, $route = '', $option = [], $pattern = []) + { + self::rule($rule, $route, 'PATCH', $option, $pattern); + } + + /** + * 娉ㄥ唽璧勬簮璺敱 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function resource($rule, $route = '', $option = [], $pattern = []) + { + if (is_array($rule)) { + foreach ($rule as $key => $val) { + if (is_array($val)) { + list($val, $option, $pattern) = array_pad($val, 3, []); + } + self::resource($key, $val, $option, $pattern); + } + } else { + if (strpos($rule, '.')) { + // 娉ㄥ唽宓屽璧勬簮璺敱 + $array = explode('.', $rule); + $last = array_pop($array); + $item = []; + foreach ($array as $val) { + $item[] = $val . '/:' . (isset($option['var'][$val]) ? $option['var'][$val] : $val . '_id'); + } + $rule = implode('/', $item) . '/' . $last; + } + // 娉ㄥ唽璧勬簮璺敱 + foreach (self::$rest as $key => $val) { + if ((isset($option['only']) && !in_array($key, $option['only'])) + || (isset($option['except']) && in_array($key, $option['except']))) { + continue; + } + if (isset($last) && strpos($val[1], ':id') && isset($option['var'][$last])) { + $val[1] = str_replace(':id', ':' . $option['var'][$last], $val[1]); + } elseif (strpos($val[1], ':id') && isset($option['var'][$rule])) { + $val[1] = str_replace(':id', ':' . $option['var'][$rule], $val[1]); + } + $item = ltrim($rule . $val[1], '/'); + $option['rest'] = $key; + self::rule($item . '$', $route . '/' . $val[2], $val[0], $option, $pattern); + } + } + } + + /** + * 娉ㄥ唽鎺у埗鍣ㄨ矾鐢 鎿嶄綔鏂规硶瀵瑰簲涓嶅悓鐨勮姹傚悗缂 + * @access public + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return void + */ + public static function controller($rule, $route = '', $option = [], $pattern = []) + { + foreach (self::$methodPrefix as $type => $val) { + self::$type($rule . '/:action', $route . '/' . $val . ':action', $option, $pattern); + } + } + + /** + * 娉ㄥ唽鍒悕璺敱 + * @access public + * @param string|array $rule 璺敱鍒悕 + * @param string $route 璺敱鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @return void + */ + public static function alias($rule = null, $route = '', $option = []) + { + if (is_array($rule)) { + self::$rules['alias'] = array_merge(self::$rules['alias'], $rule); + } else { + self::$rules['alias'][$rule] = $option ? [$route, $option] : $route; + } + } + + /** + * 璁剧疆涓嶅悓璇锋眰绫诲瀷涓嬮潰鐨勬柟娉曞墠缂 + * @access public + * @param string $method 璇锋眰绫诲瀷 + * @param string $prefix 绫诲瀷鍓嶇紑 + * @return void + */ + public static function setMethodPrefix($method, $prefix = '') + { + if (is_array($method)) { + self::$methodPrefix = array_merge(self::$methodPrefix, array_change_key_case($method)); + } else { + self::$methodPrefix[strtolower($method)] = $prefix; + } + } + + /** + * rest鏂规硶瀹氫箟鍜屼慨鏀 + * @access public + * @param string $name 鏂规硶鍚嶇О + * @param array|bool $resource 璧勬簮 + * @return void + */ + public static function rest($name, $resource = []) + { + if (is_array($name)) { + self::$rest = $resource ? $name : array_merge(self::$rest, $name); + } else { + self::$rest[$name] = $resource; + } + } + + /** + * 娉ㄥ唽鏈尮閰嶈矾鐢辫鍒欏悗鐨勫鐞 + * @access public + * @param string $route 璺敱鍦板潃 + * @param string $method 璇锋眰绫诲瀷 + * @param array $option 璺敱鍙傛暟 + * @return void + */ + public static function miss($route, $method = '*', $option = []) + { + self::rule('__miss__', $route, $method, $option, []); + } + + /** + * 娉ㄥ唽涓涓嚜鍔ㄨВ鏋愮殑URL璺敱 + * @access public + * @param string $route 璺敱鍦板潃 + * @return void + */ + public static function auto($route) + { + self::rule('__auto__', $route, '*', [], []); + } + + /** + * 鑾峰彇鎴栬呮壒閲忚缃矾鐢卞畾涔 + * @access public + * @param mixed $rules 璇锋眰绫诲瀷鎴栬呰矾鐢卞畾涔夋暟缁 + * @return array + */ + public static function rules($rules = '') + { + if (is_array($rules)) { + self::$rules = $rules; + } elseif ($rules) { + return true === $rules ? self::$rules : self::$rules[strtolower($rules)]; + } else { + $rules = self::$rules; + unset($rules['pattern'], $rules['alias'], $rules['domain'], $rules['name']); + return $rules; + } + } + + /** + * 妫娴嬪瓙鍩熷悕閮ㄧ讲 + * @access public + * @param Request $request Request璇锋眰瀵硅薄 + * @param array $currentRules 褰撳墠璺敱瑙勫垯 + * @param string $method 璇锋眰绫诲瀷 + * @return void + */ + public static function checkDomain($request, &$currentRules, $method = 'get') + { + // 鍩熷悕瑙勫垯 + $rules = self::$rules['domain']; + // 寮鍚瓙鍩熷悕閮ㄧ讲 鏀寔浜岀骇鍜屼笁绾у煙鍚 + if (!empty($rules)) { + $host = $request->host(); + if (isset($rules[$host])) { + // 瀹屾暣鍩熷悕鎴栬匢P閰嶇疆 + $item = $rules[$host]; + } else { + $rootDomain = Config::get('url_domain_root'); + if ($rootDomain) { + // 閰嶇疆鍩熷悕鏍 渚嬪 thinkphp.cn 163.com.cn 濡傛灉鏄浗瀹剁骇鍩熷悕 com.cn net.cn 涔嬬被鐨勫煙鍚嶉渶瑕侀厤缃 + $domain = explode('.', rtrim(stristr($host, $rootDomain, true), '.')); + } else { + $domain = explode('.', $host, -2); + } + // 瀛愬煙鍚嶉厤缃 + if (!empty($domain)) { + // 褰撳墠瀛愬煙鍚 + $subDomain = implode('.', $domain); + self::$subDomain = $subDomain; + $domain2 = array_pop($domain); + if ($domain) { + // 瀛樺湪涓夌骇鍩熷悕 + $domain3 = array_pop($domain); + } + if ($subDomain && isset($rules[$subDomain])) { + // 瀛愬煙鍚嶉厤缃 + $item = $rules[$subDomain]; + } elseif (isset($rules['*.' . $domain2]) && !empty($domain3)) { + // 娉涗笁绾у煙鍚 + $item = $rules['*.' . $domain2]; + $panDomain = $domain3; + } elseif (isset($rules['*']) && !empty($domain2)) { + // 娉涗簩绾у煙鍚 + if ('www' != $domain2) { + $item = $rules['*']; + $panDomain = $domain2; + } + } + } + } + if (!empty($item)) { + if (isset($panDomain)) { + // 淇濆瓨褰撳墠娉涘煙鍚 + $request->route(['__domain__' => $panDomain]); + } + if (isset($item['[bind]'])) { + // 瑙f瀽瀛愬煙鍚嶉儴缃茶鍒 + list($rule, $option, $pattern) = $item['[bind]']; + if (!empty($option['https']) && !$request->isSsl()) { + // https妫娴 + throw new HttpException(404, 'must use https request:' . $host); + } + + if (strpos($rule, '?')) { + // 浼犲叆鍏跺畠鍙傛暟 + $array = parse_url($rule); + $result = $array['path']; + parse_str($array['query'], $params); + if (isset($panDomain)) { + $pos = array_search('*', $params); + if (false !== $pos) { + // 娉涘煙鍚嶄綔涓哄弬鏁 + $params[$pos] = $panDomain; + } + } + $_GET = array_merge($_GET, $params); + } else { + $result = $rule; + } + + if (0 === strpos($result, '\\')) { + // 缁戝畾鍒板懡鍚嶇┖闂 渚嬪 \app\index\behavior + self::$bind = ['type' => 'namespace', 'namespace' => $result]; + } elseif (0 === strpos($result, '@')) { + // 缁戝畾鍒扮被 渚嬪 @app\index\controller\User + self::$bind = ['type' => 'class', 'class' => substr($result, 1)]; + } else { + // 缁戝畾鍒版ā鍧/鎺у埗鍣 渚嬪 index/user + self::$bind = ['type' => 'module', 'module' => $result]; + } + self::$domainBind = true; + } else { + self::$domainRule = $item; + $currentRules = isset($item[$method]) ? $item[$method] : $item['*']; + } + } + } + } + + /** + * 妫娴婾RL璺敱 + * @access public + * @param Request $request Request璇锋眰瀵硅薄 + * @param string $url URL鍦板潃 + * @param string $depr URL鍒嗛殧绗 + * @param bool $checkDomain 鏄惁妫娴嬪煙鍚嶈鍒 + * @return false|array + */ + public static function check($request, $url, $depr = '/', $checkDomain = false) + { + // 鍒嗛殧绗︽浛鎹 纭繚璺敱瀹氫箟浣跨敤缁熶竴鐨勫垎闅旂 + $url = str_replace($depr, '|', $url); + + if (strpos($url, '|') && isset(self::$rules['alias'][strstr($url, '|', true)])) { + // 妫娴嬭矾鐢卞埆鍚 + $result = self::checkRouteAlias($request, $url, $depr); + if (false !== $result) { + return $result; + } + } + $method = strtolower($request->method()); + // 鑾峰彇褰撳墠璇锋眰绫诲瀷鐨勮矾鐢辫鍒 + $rules = self::$rules[$method]; + // 妫娴嬪煙鍚嶉儴缃 + if ($checkDomain) { + self::checkDomain($request, $rules, $method); + } + // 妫娴婾RL缁戝畾 + $return = self::checkUrlBind($url, $rules, $depr); + if (false !== $return) { + return $return; + } + if ('|' != $url) { + $url = rtrim($url, '|'); + } + $item = str_replace('|', '/', $url); + if (isset($rules[$item])) { + // 闈欐佽矾鐢辫鍒欐娴 + $rule = $rules[$item]; + if (true === $rule) { + $rule = self::getRouteExpress($item); + } + if (!empty($rule['route']) && self::checkOption($rule['option'], $request)) { + self::setOption($rule['option']); + return self::parseRule($item, $rule['route'], $url, $rule['option']); + } + } + + // 璺敱瑙勫垯妫娴 + if (!empty($rules)) { + return self::checkRoute($request, $rules, $url, $depr); + } + return false; + } + + private static function getRouteExpress($key) + { + return self::$domainRule ? self::$domainRule['*'][$key] : self::$rules['*'][$key]; + } + + /** + * 妫娴嬭矾鐢辫鍒 + * @access private + * @param Request $request + * @param array $rules 璺敱瑙勫垯 + * @param string $url URL鍦板潃 + * @param string $depr URL鍒嗗壊绗 + * @param string $group 璺敱鍒嗙粍鍚 + * @param array $options 璺敱鍙傛暟锛堝垎缁勶級 + * @return mixed + */ + private static function checkRoute($request, $rules, $url, $depr = '/', $group = '', $options = []) + { + foreach ($rules as $key => $item) { + if (true === $item) { + $item = self::getRouteExpress($key); + } + if (!isset($item['rule'])) { + continue; + } + $rule = $item['rule']; + $route = $item['route']; + $vars = $item['var']; + $option = $item['option']; + $pattern = $item['pattern']; + + // 妫鏌ュ弬鏁版湁鏁堟 + if (!self::checkOption($option, $request)) { + continue; + } + + if (isset($option['ext'])) { + // 璺敱ext鍙傛暟 浼樺厛浜庣郴缁熼厤缃殑URL浼潤鎬佸悗缂鍙傛暟 + $url = preg_replace('/\.' . $request->ext() . '$/i', '', $url); + } + + if (is_array($rule)) { + // 鍒嗙粍璺敱 + $pos = strpos(str_replace('<', ':', $key), ':'); + if (false !== $pos) { + $str = substr($key, 0, $pos); + } else { + $str = $key; + } + if (is_string($str) && $str && 0 !== strpos(str_replace('|', '/', $url), $str)) { + continue; + } + self::setOption($option); + $result = self::checkRoute($request, $rule, $url, $depr, $key, $option); + if (false !== $result) { + return $result; + } + } elseif ($route) { + if ('__miss__' == $rule || '__auto__' == $rule) { + // 鎸囧畾鐗规畩璺敱 + $var = trim($rule, '__'); + ${$var} = $item; + continue; + } + if ($group) { + $rule = $group . ($rule ? '/' . ltrim($rule, '/') : ''); + } + + self::setOption($option); + if (isset($options['bind_model']) && isset($option['bind_model'])) { + $option['bind_model'] = array_merge($options['bind_model'], $option['bind_model']); + } + $result = self::checkRule($rule, $route, $url, $pattern, $option, $depr); + if (false !== $result) { + return $result; + } + } + } + if (isset($auto)) { + // 鑷姩瑙f瀽URL鍦板潃 + return self::parseUrl($auto['route'] . '/' . $url, $depr); + } elseif (isset($miss)) { + // 鏈尮閰嶆墍鏈夎矾鐢辩殑璺敱瑙勫垯澶勭悊 + return self::parseRule('', $miss['route'], $url, $miss['option']); + } + return false; + } + + /** + * 妫娴嬭矾鐢卞埆鍚 + * @access private + * @param Request $request + * @param string $url URL鍦板潃 + * @param string $depr URL鍒嗛殧绗 + * @return mixed + */ + private static function checkRouteAlias($request, $url, $depr) + { + $array = explode('|', $url); + $alias = array_shift($array); + $item = self::$rules['alias'][$alias]; + + if (is_array($item)) { + list($rule, $option) = $item; + $action = $array[0]; + if (isset($option['allow']) && !in_array($action, explode(',', $option['allow']))) { + // 鍏佽鎿嶄綔 + return false; + } elseif (isset($option['except']) && in_array($action, explode(',', $option['except']))) { + // 鎺掗櫎鎿嶄綔 + return false; + } + if (isset($option['method'][$action])) { + $option['method'] = $option['method'][$action]; + } + } else { + $rule = $item; + } + $bind = implode('|', $array); + // 鍙傛暟鏈夋晥鎬ф鏌 + if (isset($option) && !self::checkOption($option, $request)) { + // 璺敱涓嶅尮閰 + return false; + } elseif (0 === strpos($rule, '\\')) { + // 璺敱鍒扮被 + return self::bindToClass($bind, substr($rule, 1), $depr); + } elseif (0 === strpos($rule, '@')) { + // 璺敱鍒版帶鍒跺櫒绫 + return self::bindToController($bind, substr($rule, 1), $depr); + } else { + // 璺敱鍒版ā鍧/鎺у埗鍣 + return self::bindToModule($bind, $rule, $depr); + } + } + + /** + * 妫娴婾RL缁戝畾 + * @access private + * @param string $url URL鍦板潃 + * @param array $rules 璺敱瑙勫垯 + * @param string $depr URL鍒嗛殧绗 + * @return mixed + */ + private static function checkUrlBind(&$url, &$rules, $depr = '/') + { + if (!empty(self::$bind)) { + $type = self::$bind['type']; + $bind = self::$bind[$type]; + // 璁板綍缁戝畾淇℃伅 + App::$debug && Log::record('[ BIND ] ' . var_export($bind, true), 'info'); + // 濡傛灉鏈塙RL缁戝畾 鍒欒繘琛岀粦瀹氭娴 + switch ($type) { + case 'class': + // 缁戝畾鍒扮被 + return self::bindToClass($url, $bind, $depr); + case 'controller': + // 缁戝畾鍒版帶鍒跺櫒绫 + return self::bindToController($url, $bind, $depr); + case 'namespace': + // 缁戝畾鍒板懡鍚嶇┖闂 + return self::bindToNamespace($url, $bind, $depr); + } + } + return false; + } + + /** + * 缁戝畾鍒扮被 + * @access public + * @param string $url URL鍦板潃 + * @param string $class 绫诲悕锛堝甫鍛藉悕绌洪棿锛 + * @param string $depr URL鍒嗛殧绗 + * @return array + */ + public static function bindToClass($url, $class, $depr = '/') + { + $url = str_replace($depr, '|', $url); + $array = explode('|', $url, 2); + $action = !empty($array[0]) ? $array[0] : Config::get('default_action'); + if (!empty($array[1])) { + self::parseUrlParams($array[1]); + } + return ['type' => 'method', 'method' => [$class, $action]]; + } + + /** + * 缁戝畾鍒板懡鍚嶇┖闂 + * @access public + * @param string $url URL鍦板潃 + * @param string $namespace 鍛藉悕绌洪棿 + * @param string $depr URL鍒嗛殧绗 + * @return array + */ + public static function bindToNamespace($url, $namespace, $depr = '/') + { + $url = str_replace($depr, '|', $url); + $array = explode('|', $url, 3); + $class = !empty($array[0]) ? $array[0] : Config::get('default_controller'); + $method = !empty($array[1]) ? $array[1] : Config::get('default_action'); + if (!empty($array[2])) { + self::parseUrlParams($array[2]); + } + return ['type' => 'method', 'method' => [$namespace . '\\' . Loader::parseName($class, 1), $method]]; + } + + /** + * 缁戝畾鍒版帶鍒跺櫒绫 + * @access public + * @param string $url URL鍦板潃 + * @param string $controller 鎺у埗鍣ㄥ悕 锛堟敮鎸佸甫妯″潡鍚 index/user 锛 + * @param string $depr URL鍒嗛殧绗 + * @return array + */ + public static function bindToController($url, $controller, $depr = '/') + { + $url = str_replace($depr, '|', $url); + $array = explode('|', $url, 2); + $action = !empty($array[0]) ? $array[0] : Config::get('default_action'); + if (!empty($array[1])) { + self::parseUrlParams($array[1]); + } + return ['type' => 'controller', 'controller' => $controller . '/' . $action]; + } + + /** + * 缁戝畾鍒版ā鍧/鎺у埗鍣 + * @access public + * @param string $url URL鍦板潃 + * @param string $controller 鎺у埗鍣ㄧ被鍚嶏紙甯﹀懡鍚嶇┖闂达級 + * @param string $depr URL鍒嗛殧绗 + * @return array + */ + public static function bindToModule($url, $controller, $depr = '/') + { + $url = str_replace($depr, '|', $url); + $array = explode('|', $url, 2); + $action = !empty($array[0]) ? $array[0] : Config::get('default_action'); + if (!empty($array[1])) { + self::parseUrlParams($array[1]); + } + return ['type' => 'module', 'module' => $controller . '/' . $action]; + } + + /** + * 璺敱鍙傛暟鏈夋晥鎬ф鏌 + * @access private + * @param array $option 璺敱鍙傛暟 + * @param Request $request Request瀵硅薄 + * @return bool + */ + private static function checkOption($option, $request) + { + if ((isset($option['method']) && is_string($option['method']) && false === stripos($option['method'], $request->method())) + || (isset($option['ajax']) && $option['ajax'] && !$request->isAjax()) // Ajax妫娴 + || (isset($option['ajax']) && !$option['ajax'] && $request->isAjax()) // 闈濧jax妫娴 + || (isset($option['pjax']) && $option['pjax'] && !$request->isPjax()) // Pjax妫娴 + || (isset($option['pjax']) && !$option['pjax'] && $request->isPjax()) // 闈濸jax妫娴 + || (isset($option['ext']) && false === stripos('|' . $option['ext'] . '|', $request->ext() ? '|' . $request->ext() . '|' : '')) // 浼潤鎬佸悗缂妫娴 + || (isset($option['deny_ext']) && false !== stripos('|' . $option['deny_ext'] . '|', $request->ext() ? '|' . $request->ext() . '|' : '')) + || (isset($option['domain']) && !in_array($option['domain'], [$_SERVER['HTTP_HOST'], self::$subDomain])) // 鍩熷悕妫娴 + || (isset($option['https']) && $option['https'] && !$request->isSsl()) // https妫娴 + || (isset($option['https']) && !$option['https'] && $request->isSsl()) // https妫娴 + || (!empty($option['before_behavior']) && false === Hook::exec($option['before_behavior'])) // 琛屼负妫娴 + || (!empty($option['callback']) && is_callable($option['callback']) && false === call_user_func($option['callback'])) // 鑷畾涔夋娴 + ) { + return false; + } + return true; + } + + /** + * 妫娴嬭矾鐢辫鍒 + * @access private + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param string $url URL鍦板潃 + * @param array $pattern 鍙橀噺瑙勫垯 + * @param array $option 璺敱鍙傛暟 + * @param string $depr URL鍒嗛殧绗︼紙鍏ㄥ眬锛 + * @return array|false + */ + private static function checkRule($rule, $route, $url, $pattern, $option, $depr) + { + // 妫鏌ュ畬鏁磋鍒欏畾涔 + if (isset($pattern['__url__']) && !preg_match('/^' . $pattern['__url__'] . '/', str_replace('|', $depr, $url))) { + return false; + } + // 妫鏌ヨ矾鐢辩殑鍙傛暟鍒嗛殧绗 + if (isset($option['param_depr'])) { + $url = str_replace(['|', $option['param_depr']], [$depr, '|'], $url); + } + + $len1 = substr_count($url, '|'); + $len2 = substr_count($rule, '/'); + // 澶氫綑鍙傛暟鏄惁鍚堝苟 + $merge = !empty($option['merge_extra_vars']) ? true : false; + if ($merge && $len1 > $len2) { + $url = str_replace('|', $depr, $url); + $url = implode('|', explode($depr, $url, $len2 + 1)); + } + + if ($len1 >= $len2 || strpos($rule, '[')) { + if (!empty($option['complete_match'])) { + // 瀹屾暣鍖归厤 + if (!$merge && $len1 != $len2 && (false === strpos($rule, '[') || $len1 > $len2 || $len1 < $len2 - substr_count($rule, '['))) { + return false; + } + } + $pattern = array_merge(self::$rules['pattern'], $pattern); + if (false !== $match = self::match($url, $rule, $pattern, $merge)) { + // 鍖归厤鍒拌矾鐢辫鍒 + return self::parseRule($rule, $route, $url, $option, $match, $merge); + } + } + return false; + } + + /** + * 瑙f瀽妯″潡鐨刄RL鍦板潃 [妯″潡/鎺у埗鍣/鎿嶄綔?]鍙傛暟1=鍊1&鍙傛暟2=鍊2... + * @access public + * @param string $url URL鍦板潃 + * @param string $depr URL鍒嗛殧绗 + * @param bool $autoSearch 鏄惁鑷姩娣卞害鎼滅储鎺у埗鍣 + * @return array + */ + public static function parseUrl($url, $depr = '/', $autoSearch = false) + { + + if (isset(self::$bind['module'])) { + $bind = str_replace('/', $depr, self::$bind['module']); + // 濡傛灉鏈夋ā鍧/鎺у埗鍣ㄧ粦瀹 + $url = $bind . ('.' != substr($bind, -1) ? $depr : '') . ltrim($url, $depr); + } + $url = str_replace($depr, '|', $url); + list($path, $var) = self::parseUrlPath($url); + $route = [null, null, null]; + if (isset($path)) { + // 瑙f瀽妯″潡 + $module = Config::get('app_multi_module') ? array_shift($path) : null; + if ($autoSearch) { + // 鑷姩鎼滅储鎺у埗鍣 + $dir = APP_PATH . ($module ? $module . DS : '') . Config::get('url_controller_layer'); + $suffix = App::$suffix || Config::get('controller_suffix') ? ucfirst(Config::get('url_controller_layer')) : ''; + $item = []; + $find = false; + foreach ($path as $val) { + $item[] = $val; + $file = $dir . DS . str_replace('.', DS, $val) . $suffix . EXT; + $file = pathinfo($file, PATHINFO_DIRNAME) . DS . Loader::parseName(pathinfo($file, PATHINFO_FILENAME), 1) . EXT; + if (is_file($file)) { + $find = true; + break; + } else { + $dir .= DS . Loader::parseName($val); + } + } + if ($find) { + $controller = implode('.', $item); + $path = array_slice($path, count($item)); + } else { + $controller = array_shift($path); + } + } else { + // 瑙f瀽鎺у埗鍣 + $controller = !empty($path) ? array_shift($path) : null; + } + // 瑙f瀽鎿嶄綔 + $action = !empty($path) ? array_shift($path) : null; + // 瑙f瀽棰濆鍙傛暟 + self::parseUrlParams(empty($path) ? '' : implode('|', $path)); + // 灏佽璺敱 + $route = [$module, $controller, $action]; + // 妫鏌ュ湴鍧鏄惁琚畾涔夎繃璺敱 + $name = strtolower($module . '/' . Loader::parseName($controller, 1) . '/' . $action); + $name2 = ''; + if (empty($module) || isset($bind) && $module == $bind) { + $name2 = strtolower(Loader::parseName($controller, 1) . '/' . $action); + } + + if (isset(self::$rules['name'][$name]) || isset(self::$rules['name'][$name2])) { + throw new HttpException(404, 'invalid request:' . str_replace('|', $depr, $url)); + } + } + return ['type' => 'module', 'module' => $route]; + } + + /** + * 瑙f瀽URL鐨刾athinfo鍙傛暟鍜屽彉閲 + * @access private + * @param string $url URL鍦板潃 + * @return array + */ + private static function parseUrlPath($url) + { + // 鍒嗛殧绗︽浛鎹 纭繚璺敱瀹氫箟浣跨敤缁熶竴鐨勫垎闅旂 + $url = str_replace('|', '/', $url); + $url = trim($url, '/'); + $var = []; + if (false !== strpos($url, '?')) { + // [妯″潡/鎺у埗鍣/鎿嶄綔?]鍙傛暟1=鍊1&鍙傛暟2=鍊2... + $info = parse_url($url); + $path = explode('/', $info['path']); + parse_str($info['query'], $var); + } elseif (strpos($url, '/')) { + // [妯″潡/鎺у埗鍣/鎿嶄綔] + $path = explode('/', $url); + } elseif (false !== strpos($url, '=')) { + // 鍙傛暟1=鍊1&鍙傛暟2=鍊2... + parse_str($url, $var); + } else { + $path = [$url]; + } + return [$path, $var]; + } + + /** + * 妫娴婾RL鍜岃鍒欒矾鐢辨槸鍚﹀尮閰 + * @access private + * @param string $url URL鍦板潃 + * @param string $rule 璺敱瑙勫垯 + * @param array $pattern 鍙橀噺瑙勫垯 + * @return array|false + */ + private static function match($url, $rule, $pattern) + { + $m2 = explode('/', $rule); + $m1 = explode('|', $url); + + $var = []; + foreach ($m2 as $key => $val) { + // val涓畾涔変簡澶氫釜鍙橀噺 + if (false !== strpos($val, '<') && preg_match_all('/<(\w+(\??))>/', $val, $matches)) { + $value = []; + $replace = []; + foreach ($matches[1] as $name) { + if (strpos($name, '?')) { + $name = substr($name, 0, -1); + $replace[] = '(' . (isset($pattern[$name]) ? $pattern[$name] : '\w+') . ')?'; + } else { + $replace[] = '(' . (isset($pattern[$name]) ? $pattern[$name] : '\w+') . ')'; + } + $value[] = $name; + } + $val = str_replace($matches[0], $replace, $val); + if (preg_match('/^' . $val . '$/', isset($m1[$key]) ? $m1[$key] : '', $match)) { + array_shift($match); + foreach ($value as $k => $name) { + if (isset($match[$k])) { + $var[$name] = $match[$k]; + } + } + continue; + } else { + return false; + } + } + + if (0 === strpos($val, '[:')) { + // 鍙夊弬鏁 + $val = substr($val, 1, -1); + $optional = true; + } else { + $optional = false; + } + if (0 === strpos($val, ':')) { + // URL鍙橀噺 + $name = substr($val, 1); + if (!$optional && !isset($m1[$key])) { + return false; + } + if (isset($m1[$key]) && isset($pattern[$name])) { + // 妫鏌ュ彉閲忚鍒 + if ($pattern[$name] instanceof \Closure) { + $result = call_user_func_array($pattern[$name], [$m1[$key]]); + if (false === $result) { + return false; + } + } elseif (!preg_match('/^' . $pattern[$name] . '$/', $m1[$key])) { + return false; + } + } + $var[$name] = isset($m1[$key]) ? $m1[$key] : ''; + } elseif (!isset($m1[$key]) || 0 !== strcasecmp($val, $m1[$key])) { + return false; + } + } + // 鎴愬姛鍖归厤鍚庤繑鍥濽RL涓殑鍔ㄦ佸彉閲忔暟缁 + return $var; + } + + /** + * 瑙f瀽瑙勫垯璺敱 + * @access private + * @param string $rule 璺敱瑙勫垯 + * @param string $route 璺敱鍦板潃 + * @param string $pathinfo URL鍦板潃 + * @param array $option 璺敱鍙傛暟 + * @param array $matches 鍖归厤鐨勫彉閲 + * @return array + */ + private static function parseRule($rule, $route, $pathinfo, $option = [], $matches = []) + { + $request = Request::instance(); + // 瑙f瀽璺敱瑙勫垯 + if ($rule) { + $rule = explode('/', $rule); + // 鑾峰彇URL鍦板潃涓殑鍙傛暟 + $paths = explode('|', $pathinfo); + foreach ($rule as $item) { + $fun = ''; + if (0 === strpos($item, '[:')) { + $item = substr($item, 1, -1); + } + if (0 === strpos($item, ':')) { + $var = substr($item, 1); + $matches[$var] = array_shift($paths); + } else { + // 杩囨护URL涓殑闈欐佸彉閲 + array_shift($paths); + } + } + } else { + $paths = explode('|', $pathinfo); + } + + // 鑾峰彇璺敱鍦板潃瑙勫垯 + if (is_string($route) && isset($option['prefix'])) { + // 璺敱鍦板潃鍓嶇紑 + $route = $option['prefix'] . $route; + } + // 鏇挎崲璺敱鍦板潃涓殑鍙橀噺 + if (is_string($route) && !empty($matches)) { + foreach ($matches as $key => $val) { + if (false !== strpos($route, ':' . $key)) { + $route = str_replace(':' . $key, $val, $route); + } + } + } + + // 缁戝畾妯″瀷鏁版嵁 + if (isset($option['bind_model'])) { + $bind = []; + foreach ($option['bind_model'] as $key => $val) { + if ($val instanceof \Closure) { + $result = call_user_func_array($val, [$matches]); + } else { + if (is_array($val)) { + $fields = explode('&', $val[1]); + $model = $val[0]; + $exception = isset($val[2]) ? $val[2] : true; + } else { + $fields = ['id']; + $model = $val; + $exception = true; + } + $where = []; + $match = true; + foreach ($fields as $field) { + if (!isset($matches[$field])) { + $match = false; + break; + } else { + $where[$field] = $matches[$field]; + } + } + if ($match) { + $query = strpos($model, '\\') ? $model::where($where) : Loader::model($model)->where($where); + $result = $query->failException($exception)->find(); + } + } + if (!empty($result)) { + $bind[$key] = $result; + } + } + $request->bind($bind); + } + + // 瑙f瀽棰濆鍙傛暟 + self::parseUrlParams(empty($paths) ? '' : implode('|', $paths), $matches); + // 璁板綍鍖归厤鐨勮矾鐢变俊鎭 + $request->routeInfo(['rule' => $rule, 'route' => $route, 'option' => $option, 'var' => $matches]); + + // 妫娴嬭矾鐢盿fter琛屼负 + if (!empty($option['after_behavior'])) { + if ($option['after_behavior'] instanceof \Closure) { + $result = call_user_func_array($option['after_behavior'], []); + } else { + foreach ((array) $option['after_behavior'] as $behavior) { + $result = Hook::exec($behavior, ''); + if (!is_null($result)) { + break; + } + } + } + // 璺敱瑙勫垯閲嶅畾鍚 + if ($result instanceof Response) { + return ['type' => 'response', 'response' => $result]; + } elseif (is_array($result)) { + return $result; + } + } + + if ($route instanceof \Closure) { + // 鎵ц闂寘 + $result = ['type' => 'function', 'function' => $route]; + } elseif (0 === strpos($route, '/') || strpos($route, '://')) { + // 璺敱鍒伴噸瀹氬悜鍦板潃 + $result = ['type' => 'redirect', 'url' => $route, 'status' => isset($option['status']) ? $option['status'] : 301]; + } elseif (false !== strpos($route, '\\')) { + // 璺敱鍒版柟娉 + list($path, $var) = self::parseUrlPath($route); + $route = str_replace('/', '@', implode('/', $path)); + $method = strpos($route, '@') ? explode('@', $route) : $route; + $result = ['type' => 'method', 'method' => $method, 'var' => $var]; + } elseif (0 === strpos($route, '@')) { + // 璺敱鍒版帶鍒跺櫒 + $route = substr($route, 1); + list($route, $var) = self::parseUrlPath($route); + $result = ['type' => 'controller', 'controller' => implode('/', $route), 'var' => $var]; + $request->action(array_pop($route)); + $request->controller($route ? array_pop($route) : Config::get('default_controller')); + $request->module($route ? array_pop($route) : Config::get('default_module')); + App::$modulePath = APP_PATH . (Config::get('app_multi_module') ? $request->module() . DS : ''); + } else { + // 璺敱鍒版ā鍧/鎺у埗鍣/鎿嶄綔 + $result = self::parseModule($route); + } + // 寮鍚姹傜紦瀛 + if ($request->isGet() && isset($option['cache'])) { + $cache = $option['cache']; + if (is_array($cache)) { + list($key, $expire) = $cache; + } else { + $key = str_replace('|', '/', $pathinfo); + $expire = $cache; + } + $request->cache($key, $expire); + } + return $result; + } + + /** + * 瑙f瀽URL鍦板潃涓 妯″潡/鎺у埗鍣/鎿嶄綔 + * @access private + * @param string $url URL鍦板潃 + * @return array + */ + private static function parseModule($url) + { + list($path, $var) = self::parseUrlPath($url); + $action = array_pop($path); + $controller = !empty($path) ? array_pop($path) : null; + $module = Config::get('app_multi_module') && !empty($path) ? array_pop($path) : null; + $method = Request::instance()->method(); + if (Config::get('use_action_prefix') && !empty(self::$methodPrefix[$method])) { + // 鎿嶄綔鏂规硶鍓嶇紑鏀寔 + $action = 0 !== strpos($action, self::$methodPrefix[$method]) ? self::$methodPrefix[$method] . $action : $action; + } + // 璁剧疆褰撳墠璇锋眰鐨勮矾鐢卞彉閲 + Request::instance()->route($var); + // 璺敱鍒版ā鍧/鎺у埗鍣/鎿嶄綔 + return ['type' => 'module', 'module' => [$module, $controller, $action], 'convert' => false]; + } + + /** + * 瑙f瀽URL鍦板潃涓殑鍙傛暟Request瀵硅薄 + * @access private + * @param string $rule 璺敱瑙勫垯 + * @param array $var 鍙橀噺 + * @return void + */ + private static function parseUrlParams($url, &$var = []) + { + if ($url) { + if (Config::get('url_param_type')) { + $var += explode('|', $url); + } else { + preg_replace_callback('/(\w+)\|([^\|]+)/', function ($match) use (&$var) { + $var[$match[1]] = strip_tags($match[2]); + }, $url); + } + } + // 璁剧疆褰撳墠璇锋眰鐨勫弬鏁 + Request::instance()->route($var); + } + + // 鍒嗘瀽璺敱瑙勫垯涓殑鍙橀噺 + private static function parseVar($rule) + { + // 鎻愬彇璺敱瑙勫垯涓殑鍙橀噺 + $var = []; + foreach (explode('/', $rule) as $val) { + $optional = false; + if (false !== strpos($val, '<') && preg_match_all('/<(\w+(\??))>/', $val, $matches)) { + foreach ($matches[1] as $name) { + if (strpos($name, '?')) { + $name = substr($name, 0, -1); + $optional = true; + } else { + $optional = false; + } + $var[$name] = $optional ? 2 : 1; + } + } + + if (0 === strpos($val, '[:')) { + // 鍙夊弬鏁 + $optional = true; + $val = substr($val, 1, -1); + } + if (0 === strpos($val, ':')) { + // URL鍙橀噺 + $name = substr($val, 1); + $var[$name] = $optional ? 2 : 1; + } + } + return $var; + } +} diff --git a/thinkphp/library/think/Session.php b/thinkphp/library/think/Session.php new file mode 100644 index 000000000..35726220c --- /dev/null +++ b/thinkphp/library/think/Session.php @@ -0,0 +1,365 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\exception\ClassNotFoundException; + +class Session +{ + protected static $prefix = ''; + protected static $init = null; + + /** + * 璁剧疆鎴栬呰幏鍙杝ession浣滅敤鍩燂紙鍓嶇紑锛 + * @param string $prefix + * @return string|void + */ + public static function prefix($prefix = '') + { + if (empty($prefix) && null !== $prefix) { + return self::$prefix; + } else { + self::$prefix = $prefix; + } + } + + /** + * session鍒濆鍖 + * @param array $config + * @return void + * @throws \think\Exception + */ + public static function init(array $config = []) + { + if (empty($config)) { + $config = Config::get('session'); + } + // 璁板綍鍒濆鍖栦俊鎭 + App::$debug && Log::record('[ SESSION ] INIT ' . var_export($config, true), 'info'); + $isDoStart = false; + if (isset($config['use_trans_sid'])) { + ini_set('session.use_trans_sid', $config['use_trans_sid'] ? 1 : 0); + } + + // 鍚姩session + if (!empty($config['auto_start']) && PHP_SESSION_ACTIVE != session_status()) { + ini_set('session.auto_start', 0); + $isDoStart = true; + } + + if (isset($config['prefix'])) { + self::$prefix = $config['prefix']; + } + if (isset($config['var_session_id']) && isset($_REQUEST[$config['var_session_id']])) { + session_id($_REQUEST[$config['var_session_id']]); + } elseif (isset($config['id']) && !empty($config['id'])) { + session_id($config['id']); + } + if (isset($config['name'])) { + session_name($config['name']); + } + if (isset($config['path'])) { + session_save_path($config['path']); + } + if (isset($config['domain'])) { + ini_set('session.cookie_domain', $config['domain']); + } + if (isset($config['expire'])) { + ini_set('session.gc_maxlifetime', $config['expire']); + ini_set('session.cookie_lifetime', $config['expire']); + } + if (isset($config['secure'])) { + ini_set('session.cookie_secure', $config['secure']); + } + if (isset($config['httponly'])) { + ini_set('session.cookie_httponly', $config['httponly']); + } + if (isset($config['use_cookies'])) { + ini_set('session.use_cookies', $config['use_cookies'] ? 1 : 0); + } + if (isset($config['cache_limiter'])) { + session_cache_limiter($config['cache_limiter']); + } + if (isset($config['cache_expire'])) { + session_cache_expire($config['cache_expire']); + } + if (!empty($config['type'])) { + // 璇诲彇session椹卞姩 + $class = false !== strpos($config['type'], '\\') ? $config['type'] : '\\think\\session\\driver\\' . ucwords($config['type']); + + // 妫鏌ラ┍鍔ㄧ被 + if (!class_exists($class) || !session_set_save_handler(new $class($config))) { + throw new ClassNotFoundException('error session handler:' . $class, $class); + } + } + if ($isDoStart) { + session_start(); + self::$init = true; + } else { + self::$init = false; + } + } + + /** + * session鑷姩鍚姩鎴栬呭垵濮嬪寲 + * @return void + */ + public static function boot() + { + if (is_null(self::$init)) { + self::init(); + } elseif (false === self::$init) { + if (PHP_SESSION_ACTIVE != session_status()) { + session_start(); + } + self::$init = true; + } + } + + /** + * session璁剧疆 + * @param string $name session鍚嶇О + * @param mixed $value session鍊 + * @param string|null $prefix 浣滅敤鍩燂紙鍓嶇紑锛 + * @return void + */ + public static function set($name, $value = '', $prefix = null) + { + empty(self::$init) && self::boot(); + + $prefix = !is_null($prefix) ? $prefix : self::$prefix; + if (strpos($name, '.')) { + // 浜岀淮鏁扮粍璧嬪 + list($name1, $name2) = explode('.', $name); + if ($prefix) { + $_SESSION[$prefix][$name1][$name2] = $value; + } else { + $_SESSION[$name1][$name2] = $value; + } + } elseif ($prefix) { + $_SESSION[$prefix][$name] = $value; + } else { + $_SESSION[$name] = $value; + } + } + + /** + * session鑾峰彇 + * @param string $name session鍚嶇О + * @param string|null $prefix 浣滅敤鍩燂紙鍓嶇紑锛 + * @return mixed + */ + public static function get($name = '', $prefix = null) + { + empty(self::$init) && self::boot(); + $prefix = !is_null($prefix) ? $prefix : self::$prefix; + if ('' == $name) { + // 鑾峰彇鍏ㄩ儴鐨剆ession + $value = $prefix ? (!empty($_SESSION[$prefix]) ? $_SESSION[$prefix] : []) : $_SESSION; + } elseif ($prefix) { + // 鑾峰彇session + if (strpos($name, '.')) { + list($name1, $name2) = explode('.', $name); + $value = isset($_SESSION[$prefix][$name1][$name2]) ? $_SESSION[$prefix][$name1][$name2] : null; + } else { + $value = isset($_SESSION[$prefix][$name]) ? $_SESSION[$prefix][$name] : null; + } + } else { + if (strpos($name, '.')) { + list($name1, $name2) = explode('.', $name); + $value = isset($_SESSION[$name1][$name2]) ? $_SESSION[$name1][$name2] : null; + } else { + $value = isset($_SESSION[$name]) ? $_SESSION[$name] : null; + } + } + return $value; + } + + /** + * session鑾峰彇骞跺垹闄 + * @param string $name session鍚嶇О + * @param string|null $prefix 浣滅敤鍩燂紙鍓嶇紑锛 + * @return mixed + */ + public static function pull($name, $prefix = null) + { + $result = self::get($name, $prefix); + if ($result) { + self::delete($name, $prefix); + return $result; + } else { + return; + } + } + + /** + * session璁剧疆 涓嬩竴娆¤姹傛湁鏁 + * @param string $name session鍚嶇О + * @param mixed $value session鍊 + * @param string|null $prefix 浣滅敤鍩燂紙鍓嶇紑锛 + * @return void + */ + public static function flash($name, $value) + { + self::set($name, $value); + if (!self::has('__flash__.__time__')) { + self::set('__flash__.__time__', $_SERVER['REQUEST_TIME_FLOAT']); + } + self::push('__flash__', $name); + } + + /** + * 娓呯┖褰撳墠璇锋眰鐨剆ession鏁版嵁 + * @return void + */ + public static function flush() + { + if (self::$init) { + $item = self::get('__flash__'); + + if (!empty($item)) { + $time = $item['__time__']; + if ($_SERVER['REQUEST_TIME_FLOAT'] > $time) { + unset($item['__time__']); + self::delete($item); + self::set('__flash__', []); + } + } + } + } + + /** + * 鍒犻櫎session鏁版嵁 + * @param string|array $name session鍚嶇О + * @param string|null $prefix 浣滅敤鍩燂紙鍓嶇紑锛 + * @return void + */ + public static function delete($name, $prefix = null) + { + empty(self::$init) && self::boot(); + $prefix = !is_null($prefix) ? $prefix : self::$prefix; + if (is_array($name)) { + foreach ($name as $key) { + self::delete($key, $prefix); + } + } elseif (strpos($name, '.')) { + list($name1, $name2) = explode('.', $name); + if ($prefix) { + unset($_SESSION[$prefix][$name1][$name2]); + } else { + unset($_SESSION[$name1][$name2]); + } + } else { + if ($prefix) { + unset($_SESSION[$prefix][$name]); + } else { + unset($_SESSION[$name]); + } + } + } + + /** + * 娓呯┖session鏁版嵁 + * @param string|null $prefix 浣滅敤鍩燂紙鍓嶇紑锛 + * @return void + */ + public static function clear($prefix = null) + { + empty(self::$init) && self::boot(); + $prefix = !is_null($prefix) ? $prefix : self::$prefix; + if ($prefix) { + unset($_SESSION[$prefix]); + } else { + $_SESSION = []; + } + } + + /** + * 鍒ゆ柇session鏁版嵁 + * @param string $name session鍚嶇О + * @param string|null $prefix + * @return bool + */ + public static function has($name, $prefix = null) + { + empty(self::$init) && self::boot(); + $prefix = !is_null($prefix) ? $prefix : self::$prefix; + if (strpos($name, '.')) { + // 鏀寔鏁扮粍 + list($name1, $name2) = explode('.', $name); + return $prefix ? isset($_SESSION[$prefix][$name1][$name2]) : isset($_SESSION[$name1][$name2]); + } else { + return $prefix ? isset($_SESSION[$prefix][$name]) : isset($_SESSION[$name]); + } + } + + /** + * 娣诲姞鏁版嵁鍒颁竴涓猻ession鏁扮粍 + * @param string $key + * @param mixed $value + * @return void + */ + public static function push($key, $value) + { + $array = self::get($key); + if (is_null($array)) { + $array = []; + } + $array[] = $value; + self::set($key, $array); + } + + /** + * 鍚姩session + * @return void + */ + public static function start() + { + session_start(); + self::$init = true; + } + + /** + * 閿姣乻ession + * @return void + */ + public static function destroy() + { + if (!empty($_SESSION)) { + $_SESSION = []; + } + session_unset(); + session_destroy(); + self::$init = null; + } + + /** + * 閲嶆柊鐢熸垚session_id + * @param bool $delete 鏄惁鍒犻櫎鍏宠仈浼氳瘽鏂囦欢 + * @return void + */ + private static function regenerate($delete = false) + { + session_regenerate_id($delete); + } + + /** + * 鏆傚仠session + * @return void + */ + public static function pause() + { + // 鏆傚仠session + session_write_close(); + self::$init = false; + } +} diff --git a/thinkphp/library/think/Template.php b/thinkphp/library/think/Template.php new file mode 100644 index 000000000..66ba9e187 --- /dev/null +++ b/thinkphp/library/think/Template.php @@ -0,0 +1,1151 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\exception\TemplateNotFoundException; + +/** + * ThinkPHP鍒嗙鍑烘潵鐨勬ā鏉垮紩鎿 + * 鏀寔XML鏍囩鍜屾櫘閫氭爣绛剧殑妯℃澘瑙f瀽 + * 缂栬瘧鍨嬫ā鏉垮紩鎿 鏀寔鍔ㄦ佺紦瀛 + */ +class Template +{ + // 妯℃澘鍙橀噺 + protected $data = []; + // 寮曟搸閰嶇疆 + protected $config = [ + 'view_path' => '', // 妯℃澘璺緞 + 'view_base' => '', + 'view_suffix' => 'html', // 榛樿妯℃澘鏂囦欢鍚庣紑 + 'view_depr' => DS, + 'cache_suffix' => 'php', // 榛樿妯℃澘缂撳瓨鍚庣紑 + 'tpl_deny_func_list' => 'echo,exit', // 妯℃澘寮曟搸绂佺敤鍑芥暟 + 'tpl_deny_php' => false, // 榛樿妯℃澘寮曟搸鏄惁绂佺敤PHP鍘熺敓浠g爜 + 'tpl_begin' => '{', // 妯℃澘寮曟搸鏅氭爣绛惧紑濮嬫爣璁 + 'tpl_end' => '}', // 妯℃澘寮曟搸鏅氭爣绛剧粨鏉熸爣璁 + 'strip_space' => false, // 鏄惁鍘婚櫎妯℃澘鏂囦欢閲岄潰鐨刪tml绌烘牸涓庢崲琛 + 'tpl_cache' => true, // 鏄惁寮鍚ā鏉跨紪璇戠紦瀛,璁句负false鍒欐瘡娆¢兘浼氶噸鏂扮紪璇 + 'compile_type' => 'file', // 妯℃澘缂栬瘧绫诲瀷 + 'cache_prefix' => '', // 妯℃澘缂撳瓨鍓嶇紑鏍囪瘑锛屽彲浠ュ姩鎬佹敼鍙 + 'cache_time' => 0, // 妯℃澘缂撳瓨鏈夋晥鏈 0 涓烘案涔咃紝(浠ユ暟瀛椾负鍊硷紝鍗曚綅:绉) + 'layout_on' => false, // 甯冨眬妯℃澘寮鍏 + 'layout_name' => 'layout', // 甯冨眬妯℃澘鍏ュ彛鏂囦欢 + 'layout_item' => '{__CONTENT__}', // 甯冨眬妯℃澘鐨勫唴瀹规浛鎹㈡爣璇 + 'taglib_begin' => '{', // 鏍囩搴撴爣绛惧紑濮嬫爣璁 + 'taglib_end' => '}', // 鏍囩搴撴爣绛剧粨鏉熸爣璁 + 'taglib_load' => true, // 鏄惁浣跨敤鍐呯疆鏍囩搴撲箣澶栫殑鍏跺畠鏍囩搴擄紝榛樿鑷姩妫娴 + 'taglib_build_in' => 'cx', // 鍐呯疆鏍囩搴撳悕绉(鏍囩浣跨敤涓嶅繀鎸囧畾鏍囩搴撳悕绉),浠ラ楀彿鍒嗛殧 娉ㄦ剰瑙f瀽椤哄簭 + 'taglib_pre_load' => '', // 闇瑕侀澶栧姞杞界殑鏍囩搴(椤绘寚瀹氭爣绛惧簱鍚嶇О)锛屽涓互閫楀彿鍒嗛殧 + 'display_cache' => false, // 妯℃澘娓叉煋缂撳瓨 + 'cache_id' => '', // 妯℃澘缂撳瓨ID + 'tpl_replace_string' => [], + 'tpl_var_identify' => 'array', // .璇硶鍙橀噺璇嗗埆锛宎rray|object|'', 涓虹┖鏃惰嚜鍔ㄨ瘑鍒 + ]; + + private $literal = []; + private $includeFile = []; // 璁板綍鎵鏈夋ā鏉垮寘鍚殑鏂囦欢璺緞鍙婃洿鏂版椂闂 + protected $storage; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + */ + public function __construct(array $config = []) + { + $this->config['cache_path'] = TEMP_PATH; + $this->config = array_merge($this->config, $config); + $this->config['taglib_begin'] = $this->stripPreg($this->config['taglib_begin']); + $this->config['taglib_end'] = $this->stripPreg($this->config['taglib_end']); + $this->config['tpl_begin'] = $this->stripPreg($this->config['tpl_begin']); + $this->config['tpl_end'] = $this->stripPreg($this->config['tpl_end']); + + // 鍒濆鍖栨ā鏉跨紪璇戝瓨鍌ㄥ櫒 + $type = $this->config['compile_type'] ? $this->config['compile_type'] : 'File'; + $class = false !== strpos($type, '\\') ? $type : '\\think\\template\\driver\\' . ucwords($type); + $this->storage = new $class(); + } + + /** + * 瀛楃涓叉浛鎹 閬垮厤姝e垯娣锋穯 + * @access private + * @param string $str + * @return string + */ + private function stripPreg($str) + { + return str_replace( + ['{', '}', '(', ')', '|', '[', ']', '-', '+', '*', '.', '^', '?'], + ['\{', '\}', '\(', '\)', '\|', '\[', '\]', '\-', '\+', '\*', '\.', '\^', '\?'], + $str); + } + + /** + * 妯℃澘鍙橀噺璧嬪 + * @access public + * @param mixed $name + * @param mixed $value + * @return void + */ + public function assign($name, $value = '') + { + if (is_array($name)) { + $this->data = array_merge($this->data, $name); + } else { + $this->data[$name] = $value; + } + } + + /** + * 妯℃澘寮曟搸鍙傛暟璧嬪 + * @access public + * @param mixed $name + * @param mixed $value + */ + public function __set($name, $value) + { + $this->config[$name] = $value; + } + + /** + * 妯℃澘寮曟搸閰嶇疆椤 + * @access public + * @param array|string $config + * @return void|array + */ + public function config($config) + { + if (is_array($config)) { + $this->config = array_merge($this->config, $config); + } elseif (isset($this->config[$config])) { + return $this->config[$config]; + } else { + return; + } + } + + /** + * 妯℃澘鍙橀噺鑾峰彇 + * @access public + * @param string $name 鍙橀噺鍚 + * @return mixed + */ + public function get($name = '') + { + if ('' == $name) { + return $this->data; + } else { + $data = $this->data; + foreach (explode('.', $name) as $key => $val) { + if (isset($data[$val])) { + $data = $data[$val]; + } else { + $data = null; + break; + } + } + return $data; + } + } + + /** + * 娓叉煋妯℃澘鏂囦欢 + * @access public + * @param string $template 妯℃澘鏂囦欢 + * @param array $vars 妯℃澘鍙橀噺 + * @param array $config 妯℃澘鍙傛暟 + * @return void + */ + public function fetch($template, $vars = [], $config = []) + { + if ($vars) { + $this->data = $vars; + } + if ($config) { + $this->config($config); + } + if (!empty($this->config['cache_id']) && $this->config['display_cache']) { + // 璇诲彇娓叉煋缂撳瓨 + $cacheContent = Cache::get($this->config['cache_id']); + if (false !== $cacheContent) { + echo $cacheContent; + return; + } + } + $template = $this->parseTemplateFile($template); + if ($template) { + $cacheFile = $this->config['cache_path'] . $this->config['cache_prefix'] . md5($template) . '.' . ltrim($this->config['cache_suffix'], '.'); + if (!$this->checkCache($cacheFile)) { + // 缂撳瓨鏃犳晥 閲嶆柊妯℃澘缂栬瘧 + $content = file_get_contents($template); + $this->compiler($content, $cacheFile); + } + // 椤甸潰缂撳瓨 + ob_start(); + ob_implicit_flush(0); + // 璇诲彇缂栬瘧瀛樺偍 + $this->storage->read($cacheFile, $this->data); + // 鑾峰彇骞舵竻绌虹紦瀛 + $content = ob_get_clean(); + if (!empty($this->config['cache_id']) && $this->config['display_cache']) { + // 缂撳瓨椤甸潰杈撳嚭 + Cache::set($this->config['cache_id'], $content, $this->config['cache_time']); + } + echo $content; + } + } + + /** + * 娓叉煋妯℃澘鍐呭 + * @access public + * @param string $content 妯℃澘鍐呭 + * @param array $vars 妯℃澘鍙橀噺 + * @param array $config 妯℃澘鍙傛暟 + * @return void + */ + public function display($content, $vars = [], $config = []) + { + if ($vars) { + $this->data = $vars; + } + if ($config) { + $this->config($config); + } + $cacheFile = $this->config['cache_path'] . $this->config['cache_prefix'] . md5($content) . '.' . ltrim($this->config['cache_suffix'], '.'); + if (!$this->checkCache($cacheFile)) { + // 缂撳瓨鏃犳晥 妯℃澘缂栬瘧 + $this->compiler($content, $cacheFile); + } + // 璇诲彇缂栬瘧瀛樺偍 + $this->storage->read($cacheFile, $this->data); + } + + /** + * 璁剧疆甯冨眬 + * @access public + * @param mixed $name 甯冨眬妯℃澘鍚嶇О false 鍒欏叧闂竷灞 + * @param string $replace 甯冨眬妯℃澘鍐呭鏇挎崲鏍囪瘑 + * @return object + */ + public function layout($name, $replace = '') + { + if (false === $name) { + // 鍏抽棴甯冨眬 + $this->config['layout_on'] = false; + } else { + // 寮鍚竷灞 + $this->config['layout_on'] = true; + // 鍚嶇О蹇呴』涓哄瓧绗︿覆 + if (is_string($name)) { + $this->config['layout_name'] = $name; + } + if (!empty($replace)) { + $this->config['layout_item'] = $replace; + } + } + return $this; + } + + /** + * 妫鏌ョ紪璇戠紦瀛樻槸鍚︽湁鏁 + * 濡傛灉鏃犳晥鍒欓渶瑕侀噸鏂扮紪璇 + * @access private + * @param string $cacheFile 缂撳瓨鏂囦欢鍚 + * @return boolean + */ + private function checkCache($cacheFile) + { + // 鏈紑鍚紦瀛樺姛鑳 + if (!$this->config['tpl_cache']) { + return false; + } + // 缂撳瓨鏂囦欢涓嶅瓨鍦 + if (!is_file($cacheFile)) { + return false; + } + // 璇诲彇缂撳瓨鏂囦欢澶辫触 + if (!$handle = @fopen($cacheFile, "r")) { + return false; + } + // 璇诲彇绗竴琛 + preg_match('/\/\*(.+?)\*\//', fgets($handle), $matches); + if (!isset($matches[1])) { + return false; + } + $includeFile = unserialize($matches[1]); + if (!is_array($includeFile)) { + return false; + } + // 妫鏌ユā鏉挎枃浠舵槸鍚︽湁鏇存柊 + foreach ($includeFile as $path => $time) { + if (is_file($path) && filemtime($path) > $time) { + // 妯℃澘鏂囦欢濡傛灉鏈夋洿鏂板垯缂撳瓨闇瑕佹洿鏂 + return false; + } + } + // 妫鏌ョ紪璇戝瓨鍌ㄦ槸鍚︽湁鏁 + return $this->storage->check($cacheFile, $this->config['cache_time']); + } + + /** + * 妫鏌ョ紪璇戠紦瀛樻槸鍚﹀瓨鍦 + * @access public + * @param string $cacheId 缂撳瓨鐨刬d + * @return boolean + */ + public function isCache($cacheId) + { + if ($cacheId && $this->config['display_cache']) { + // 缂撳瓨椤甸潰杈撳嚭 + return Cache::has($cacheId); + } + return false; + } + + /** + * 缂栬瘧妯℃澘鏂囦欢鍐呭 + * @access private + * @param string $content 妯℃澘鍐呭 + * @param string $cacheFile 缂撳瓨鏂囦欢鍚 + * @return void + */ + private function compiler(&$content, $cacheFile) + { + // 鍒ゆ柇鏄惁鍚敤甯冨眬 + if ($this->config['layout_on']) { + if (false !== strpos($content, '{__NOLAYOUT__}')) { + // 鍙互鍗曠嫭瀹氫箟涓嶄娇鐢ㄥ竷灞 + $content = str_replace('{__NOLAYOUT__}', '', $content); + } else { + // 璇诲彇甯冨眬妯℃澘 + $layoutFile = $this->parseTemplateFile($this->config['layout_name']); + if ($layoutFile) { + // 鏇挎崲甯冨眬鐨勪富浣撳唴瀹 + $content = str_replace($this->config['layout_item'], $content, file_get_contents($layoutFile)); + } + } + } else { + $content = str_replace('{__NOLAYOUT__}', '', $content); + } + + // 妯℃澘瑙f瀽 + $this->parse($content); + if ($this->config['strip_space']) { + /* 鍘婚櫎html绌烘牸涓庢崲琛 */ + $find = ['~>\s+<~', '~>(\s+\n|\r)~']; + $replace = ['><', '>']; + $content = preg_replace($find, $replace, $content); + } + // 浼樺寲鐢熸垚鐨刾hp浠g爜 + $content = preg_replace('/\?>\s*<\?php\s(?!echo\b)/s', '', $content); + // 妯℃澘杩囨护杈撳嚭 + $replace = $this->config['tpl_replace_string']; + $content = str_replace(array_keys($replace), array_values($replace), $content); + // 娣诲姞瀹夊叏浠g爜鍙婃ā鏉垮紩鐢ㄨ褰 + $content = 'includeFile) . '*/ ?>' . "\n" . $content; + // 缂栬瘧瀛樺偍 + $this->storage->write($cacheFile, $content); + $this->includeFile = []; + return; + } + + /** + * 妯℃澘瑙f瀽鍏ュ彛 + * 鏀寔鏅氭爣绛惧拰TagLib瑙f瀽 鏀寔鑷畾涔夋爣绛惧簱 + * @access public + * @param string $content 瑕佽В鏋愮殑妯℃澘鍐呭 + * @return void + */ + public function parse(&$content) + { + // 鍐呭涓虹┖涓嶈В鏋 + if (empty($content)) { + return; + } + // 鏇挎崲literal鏍囩鍐呭 + $this->parseLiteral($content); + // 瑙f瀽缁ф壙 + $this->parseExtend($content); + // 瑙f瀽甯冨眬 + $this->parseLayout($content); + // 妫鏌nclude璇硶 + $this->parseInclude($content); + // 鏇挎崲鍖呭惈鏂囦欢涓璴iteral鏍囩鍐呭 + $this->parseLiteral($content); + // 妫鏌HP璇硶 + $this->parsePhp($content); + + // 鑾峰彇闇瑕佸紩鍏ョ殑鏍囩搴撳垪琛 + // 鏍囩搴撳彧闇瑕佸畾涔変竴娆★紝鍏佽寮曞叆澶氫釜涓娆 + // 涓鑸斁鍦ㄦ枃浠剁殑鏈鍓嶉潰 + // 鏍煎紡锛 + // 褰揟AGLIB_LOAD閰嶇疆涓簍rue鏃舵墠浼氳繘琛屾娴 + if ($this->config['taglib_load']) { + $tagLibs = $this->getIncludeTagLib($content); + if (!empty($tagLibs)) { + // 瀵瑰鍏ョ殑TagLib杩涜瑙f瀽 + foreach ($tagLibs as $tagLibName) { + $this->parseTagLib($tagLibName, $content); + } + } + } + // 棰勫厛鍔犺浇鐨勬爣绛惧簱 鏃犻渶鍦ㄦ瘡涓ā鏉夸腑浣跨敤taglib鏍囩鍔犺浇 浣嗗繀椤讳娇鐢ㄦ爣绛惧簱XML鍓嶇紑 + if ($this->config['taglib_pre_load']) { + $tagLibs = explode(',', $this->config['taglib_pre_load']); + foreach ($tagLibs as $tag) { + $this->parseTagLib($tag, $content); + } + } + // 鍐呯疆鏍囩搴 鏃犻渶浣跨敤taglib鏍囩瀵煎叆灏卞彲浠ヤ娇鐢 骞朵笖涓嶉渶浣跨敤鏍囩搴揦ML鍓嶇紑 + $tagLibs = explode(',', $this->config['taglib_build_in']); + foreach ($tagLibs as $tag) { + $this->parseTagLib($tag, $content, true); + } + // 瑙f瀽鏅氭ā鏉挎爣绛 {$tagName} + $this->parseTag($content); + + // 杩樺師琚浛鎹㈢殑Literal鏍囩 + $this->parseLiteral($content, true); + return; + } + + /** + * 妫鏌HP璇硶 + * @access private + * @param string $content 瑕佽В鏋愮殑妯℃澘鍐呭 + * @return void + * @throws \think\Exception + */ + private function parsePhp(&$content) + { + // 鐭爣绛剧殑鎯呭喌瑕佸皢' . "\n", $content); + // PHP璇硶妫鏌 + if ($this->config['tpl_deny_php'] && false !== strpos($content, 'getRegex('layout'), $content, $matches)) { + // 鏇挎崲Layout鏍囩 + $content = str_replace($matches[0], '', $content); + // 瑙f瀽Layout鏍囩 + $array = $this->parseAttr($matches[0]); + if (!$this->config['layout_on'] || $this->config['layout_name'] != $array['name']) { + // 璇诲彇甯冨眬妯℃澘 + $layoutFile = $this->parseTemplateFile($array['name']); + if ($layoutFile) { + $replace = isset($array['replace']) ? $array['replace'] : $this->config['layout_item']; + // 鏇挎崲甯冨眬鐨勪富浣撳唴瀹 + $content = str_replace($replace, $content, file_get_contents($layoutFile)); + } + } + } else { + $content = str_replace('{__NOLAYOUT__}', '', $content); + } + return; + } + + /** + * 瑙f瀽妯℃澘涓殑include鏍囩 + * @access private + * @param string $content 瑕佽В鏋愮殑妯℃澘鍐呭 + * @return void + */ + private function parseInclude(&$content) + { + $regex = $this->getRegex('include'); + $func = function ($template) use (&$func, &$regex, &$content) { + if (preg_match_all($regex, $template, $matches, PREG_SET_ORDER)) { + foreach ($matches as $match) { + $array = $this->parseAttr($match[0]); + $file = $array['file']; + unset($array['file']); + // 鍒嗘瀽妯℃澘鏂囦欢鍚嶅苟璇诲彇鍐呭 + $parseStr = $this->parseTemplateName($file); + foreach ($array as $k => $v) { + // 浠$寮澶村瓧绗︿覆杞崲鎴愭ā鏉垮彉閲 + if (0 === strpos($v, '$')) { + $v = $this->get(substr($v, 1)); + } + $parseStr = str_replace('[' . $k . ']', $v, $parseStr); + } + $content = str_replace($match[0], $parseStr, $content); + // 鍐嶆瀵瑰寘鍚枃浠惰繘琛屾ā鏉垮垎鏋 + $func($parseStr); + } + unset($matches); + } + }; + // 鏇挎崲妯℃澘涓殑include鏍囩 + $func($content); + return; + } + + /** + * 瑙f瀽妯℃澘涓殑extend鏍囩 + * @access private + * @param string $content 瑕佽В鏋愮殑妯℃澘鍐呭 + * @return void + */ + private function parseExtend(&$content) + { + $regex = $this->getRegex('extend'); + $array = $blocks = $baseBlocks = []; + $extend = ''; + $func = function ($template) use (&$func, &$regex, &$array, &$extend, &$blocks, &$baseBlocks) { + if (preg_match($regex, $template, $matches)) { + if (!isset($array[$matches['name']])) { + $array[$matches['name']] = 1; + // 璇诲彇缁ф壙妯℃澘 + $extend = $this->parseTemplateName($matches['name']); + // 閫掑綊妫鏌ョ户鎵 + $func($extend); + // 鍙栧緱block鏍囩鍐呭 + $blocks = array_merge($blocks, $this->parseBlock($template)); + return; + } + } else { + // 鍙栧緱椤跺眰妯℃澘block鏍囩鍐呭 + $baseBlocks = $this->parseBlock($template, true); + if (empty($extend)) { + // 鏃爀xtend鏍囩浣嗘湁block鏍囩鐨勬儏鍐 + $extend = $template; + } + } + }; + + $func($content); + if (!empty($extend)) { + if ($baseBlocks) { + $children = []; + foreach ($baseBlocks as $name => $val) { + $replace = $val['content']; + if (!empty($children[$name])) { + // 濡傛灉鍖呭惈鏈夊瓙block鏍囩 + foreach ($children[$name] as $key) { + $replace = str_replace($baseBlocks[$key]['begin'] . $baseBlocks[$key]['content'] . $baseBlocks[$key]['end'], $blocks[$key]['content'], $replace); + } + } + if (isset($blocks[$name])) { + // 甯︽湁{__block__}琛ㄧず涓庢墍缁ф壙妯℃澘鐨勭浉搴旀爣绛惧悎骞讹紝鑰屼笉鏄鐩 + $replace = str_replace(['{__BLOCK__}', '{__block__}'], $replace, $blocks[$name]['content']); + if (!empty($val['parent'])) { + // 濡傛灉涓嶆槸鏈椤跺眰鐨刡lock鏍囩 + $parent = $val['parent']; + if (isset($blocks[$parent])) { + $blocks[$parent]['content'] = str_replace($blocks[$name]['begin'] . $blocks[$name]['content'] . $blocks[$name]['end'], $replace, $blocks[$parent]['content']); + } + $blocks[$name]['content'] = $replace; + $children[$parent][] = $name; + continue; + } + } elseif (!empty($val['parent'])) { + // 濡傛灉瀛愭爣绛炬病鏈夎缁ф壙鍒欑敤鍘熷 + $children[$val['parent']][] = $name; + $blocks[$name] = $val; + } + if (!$val['parent']) { + // 鏇挎崲妯℃澘涓殑椤剁骇block鏍囩 + $extend = str_replace($val['begin'] . $val['content'] . $val['end'], $replace, $extend); + } + } + } + $content = $extend; + unset($blocks, $baseBlocks); + } + return; + } + + /** + * 鏇挎崲椤甸潰涓殑literal鏍囩 + * @access private + * @param string $content 妯℃澘鍐呭 + * @param boolean $restore 鏄惁涓鸿繕鍘 + * @return void + */ + private function parseLiteral(&$content, $restore = false) + { + $regex = $this->getRegex($restore ? 'restoreliteral' : 'literal'); + if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER)) { + if (!$restore) { + $count = count($this->literal); + // 鏇挎崲literal鏍囩 + foreach ($matches as $match) { + $this->literal[] = substr($match[0], strlen($match[1]), -strlen($match[2])); + $content = str_replace($match[0], "", $content); + $count++; + } + } else { + // 杩樺師literal鏍囩 + foreach ($matches as $match) { + $content = str_replace($match[0], $this->literal[$match[1]], $content); + } + // 娓呯┖literal璁板綍 + $this->literal = []; + } + unset($matches); + } + return; + } + + /** + * 鑾峰彇妯℃澘涓殑block鏍囩 + * @access private + * @param string $content 妯℃澘鍐呭 + * @param boolean $sort 鏄惁鎺掑簭 + * @return array + */ + private function parseBlock(&$content, $sort = false) + { + $regex = $this->getRegex('block'); + $result = []; + if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER | PREG_OFFSET_CAPTURE)) { + $right = $keys = []; + foreach ($matches as $match) { + if (empty($match['name'][0])) { + if (count($right) > 0) { + $tag = array_pop($right); + $start = $tag['offset'] + strlen($tag['tag']); + $length = $match[0][1] - $start; + $result[$tag['name']] = [ + 'begin' => $tag['tag'], + 'content' => substr($content, $start, $length), + 'end' => $match[0][0], + 'parent' => count($right) ? end($right)['name'] : '', + ]; + $keys[$tag['name']] = $match[0][1]; + } + } else { + // 鏍囩澶村帇鍏ユ爤 + $right[] = [ + 'name' => $match[2][0], + 'offset' => $match[0][1], + 'tag' => $match[0][0], + ]; + } + } + unset($right, $matches); + if ($sort) { + // 鎸塨lock鏍囩缁撴潫绗﹀湪妯℃澘涓殑浣嶇疆鎺掑簭 + array_multisort($keys, $result); + } + } + return $result; + } + + /** + * 鎼滅储妯℃澘椤甸潰涓寘鍚殑TagLib搴 + * 骞惰繑鍥炲垪琛 + * @access private + * @param string $content 妯℃澘鍐呭 + * @return array|null + */ + private function getIncludeTagLib(&$content) + { + // 鎼滅储鏄惁鏈塗agLib鏍囩 + if (preg_match($this->getRegex('taglib'), $content, $matches)) { + // 鏇挎崲TagLib鏍囩 + $content = str_replace($matches[0], '', $content); + return explode(',', $matches['name']); + } + return; + } + + /** + * TagLib搴撹В鏋 + * @access public + * @param string $tagLib 瑕佽В鏋愮殑鏍囩搴 + * @param string $content 瑕佽В鏋愮殑妯℃澘鍐呭 + * @param boolean $hide 鏄惁闅愯棌鏍囩搴撳墠缂 + * @return void + */ + public function parseTagLib($tagLib, &$content, $hide = false) + { + if (false !== strpos($tagLib, '\\')) { + // 鏀寔鎸囧畾鏍囩搴撶殑鍛藉悕绌洪棿 + $className = $tagLib; + $tagLib = substr($tagLib, strrpos($tagLib, '\\') + 1); + } else { + $className = '\\think\\template\\taglib\\' . ucwords($tagLib); + } + $tLib = new $className($this); + $tLib->parseTag($content, $hide ? '' : $tagLib); + return; + } + + /** + * 鍒嗘瀽鏍囩灞炴 + * @access public + * @param string $str 灞炴у瓧绗︿覆 + * @param string $name 涓嶄负绌烘椂杩斿洖鎸囧畾鐨勫睘鎬у悕 + * @return array + */ + public function parseAttr($str, $name = null) + { + $regex = '/\s+(?>(?P[\w-]+)\s*)=(?>\s*)([\"\'])(?P(?:(?!\\2).)*)\\2/is'; + $array = []; + if (preg_match_all($regex, $str, $matches, PREG_SET_ORDER)) { + foreach ($matches as $match) { + $array[$match['name']] = $match['value']; + } + unset($matches); + } + if (!empty($name) && isset($array[$name])) { + return $array[$name]; + } else { + return $array; + } + } + + /** + * 妯℃澘鏍囩瑙f瀽 + * 鏍煎紡锛 {TagName:args [|content] } + * @access private + * @param string $content 瑕佽В鏋愮殑妯℃澘鍐呭 + * @return void + */ + private function parseTag(&$content) + { + $regex = $this->getRegex('tag'); + if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER)) { + foreach ($matches as $match) { + $str = stripslashes($match[1]); + $flag = substr($str, 0, 1); + switch ($flag) { + case '$': + // 瑙f瀽妯℃澘鍙橀噺 鏍煎紡 {$varName} + // 鏄惁甯︽湁?鍙 + if (false !== $pos = strpos($str, '?')) { + $array = preg_split('/([!=]={1,2}|(?<]={0,1})/', substr($str, 0, $pos), 2, PREG_SPLIT_DELIM_CAPTURE); + $name = $array[0]; + $this->parseVar($name); + $this->parseVarFunction($name); + + $str = trim(substr($str, $pos + 1)); + $this->parseVar($str); + $first = substr($str, 0, 1); + if (strpos($name, ')')) { + // $name涓哄璞℃垨鏄嚜鍔ㄨ瘑鍒紝鎴栬呭惈鏈夊嚱鏁 + if (isset($array[1])) { + $this->parseVar($array[2]); + $name .= $array[1] . $array[2]; + } + switch ($first) { + case '?': + $str = ''; + break; + case '=': + $str = ''; + break; + default: + $str = ''; + } + } else { + if (isset($array[1])) { + $this->parseVar($array[2]); + $_name = ' && ' . $name . $array[1] . $array[2]; + } else { + $_name = ''; + } + // $name涓烘暟缁 + switch ($first) { + case '?': + // {$varname??'xxx'} $varname鏈夊畾涔夊垯杈撳嚭$varname,鍚﹀垯杈撳嚭xxx + $str = ''; + break; + case '=': + // {$varname?='xxx'} $varname涓虹湡鏃舵墠杈撳嚭xxx + $str = ''; + break; + case ':': + // {$varname?:'xxx'} $varname涓虹湡鏃惰緭鍑$varname,鍚﹀垯杈撳嚭xxx + $str = ''; + break; + default: + if (strpos($str, ':')) { + // {$varname ? 'a' : 'b'} $varname涓虹湡鏃惰緭鍑篴,鍚﹀垯杈撳嚭b + $str = ''; + } else { + $str = ''; + } + } + } + } else { + $this->parseVar($str); + $this->parseVarFunction($str); + $str = ''; + } + break; + case ':': + // 杈撳嚭鏌愪釜鍑芥暟鐨勭粨鏋 + $str = substr($str, 1); + $this->parseVar($str); + $str = ''; + break; + case '~': + // 鎵ц鏌愪釜鍑芥暟 + $str = substr($str, 1); + $this->parseVar($str); + $str = ''; + break; + case '-': + case '+': + // 杈撳嚭璁$畻 + $this->parseVar($str); + $str = ''; + break; + case '/': + // 娉ㄩ噴鏍囩 + $flag2 = substr($str, 1, 1); + if ('/' == $flag2 || ('*' == $flag2 && substr(rtrim($str), -2) == '*/')) { + $str = ''; + } + break; + default: + // 鏈瘑鍒殑鏍囩鐩存帴杩斿洖 + $str = $this->config['tpl_begin'] . $str . $this->config['tpl_end']; + break; + } + $content = str_replace($match[0], $str, $content); + } + unset($matches); + } + return; + } + + /** + * 妯℃澘鍙橀噺瑙f瀽,鏀寔浣跨敤鍑芥暟 + * 鏍煎紡锛 {$varname|function1|function2=arg1,arg2} + * @access public + * @param string $varStr 鍙橀噺鏁版嵁 + * @return void + */ + public function parseVar(&$varStr) + { + $varStr = trim($varStr); + if (preg_match_all('/\$[a-zA-Z_](?>\w*)(?:[:\.][0-9a-zA-Z_](?>\w*))+/', $varStr, $matches, PREG_OFFSET_CAPTURE)) { + static $_varParseList = []; + while ($matches[0]) { + $match = array_pop($matches[0]); + //濡傛灉宸茬粡瑙f瀽杩囪鍙橀噺瀛椾覆锛屽垯鐩存帴杩斿洖鍙橀噺鍊 + if (isset($_varParseList[$match[0]])) { + $parseStr = $_varParseList[$match[0]]; + } else { + if (strpos($match[0], '.')) { + $vars = explode('.', $match[0]); + $first = array_shift($vars); + if ('$Think' == $first) { + // 鎵鏈変互Think.鎵撳ご鐨勪互鐗规畩鍙橀噺瀵瑰緟 鏃犻渶妯℃澘璧嬪煎氨鍙互杈撳嚭 + $parseStr = $this->parseThinkVar($vars); + } elseif ('$Request' == $first) { + // 鑾峰彇Request璇锋眰瀵硅薄鍙傛暟 + $method = array_shift($vars); + if (!empty($vars)) { + $params = implode('.', $vars); + if ('true' != $params) { + $params = '\'' . $params . '\''; + } + } else { + $params = ''; + } + $parseStr = '\think\Request::instance()->' . $method . '(' . $params . ')'; + } else { + switch ($this->config['tpl_var_identify']) { + case 'array': // 璇嗗埆涓烘暟缁 + $parseStr = $first . '[\'' . implode('\'][\'', $vars) . '\']'; + break; + case 'obj': // 璇嗗埆涓哄璞 + $parseStr = $first . '->' . implode('->', $vars); + break; + default: // 鑷姩鍒ゆ柇鏁扮粍鎴栧璞 + $parseStr = '(is_array(' . $first . ')?' . $first . '[\'' . implode('\'][\'', $vars) . '\']:' . $first . '->' . implode('->', $vars) . ')'; + } + } + } else { + $parseStr = str_replace(':', '->', $match[0]); + } + $_varParseList[$match[0]] = $parseStr; + } + $varStr = substr_replace($varStr, $parseStr, $match[1], strlen($match[0])); + } + unset($matches); + } + return; + } + + /** + * 瀵规ā鏉夸腑浣跨敤浜嗗嚱鏁扮殑鍙橀噺杩涜瑙f瀽 + * 鏍煎紡 {$varname|function1|function2=arg1,arg2} + * @access public + * @param string $varStr 鍙橀噺瀛楃涓 + * @return void + */ + public function parseVarFunction(&$varStr) + { + if (false == strpos($varStr, '|')) { + return; + } + static $_varFunctionList = []; + $_key = md5($varStr); + //濡傛灉宸茬粡瑙f瀽杩囪鍙橀噺瀛椾覆锛屽垯鐩存帴杩斿洖鍙橀噺鍊 + if (isset($_varFunctionList[$_key])) { + $varStr = $_varFunctionList[$_key]; + } else { + $varArray = explode('|', $varStr); + // 鍙栧緱鍙橀噺鍚嶇О + $name = array_shift($varArray); + // 瀵瑰彉閲忎娇鐢ㄥ嚱鏁 + $length = count($varArray); + // 鍙栧緱妯℃澘绂佹浣跨敤鍑芥暟鍒楄〃 + $template_deny_funs = explode(',', $this->config['tpl_deny_func_list']); + for ($i = 0; $i < $length; $i++) { + $args = explode('=', $varArray[$i], 2); + // 妯℃澘鍑芥暟杩囨护 + $fun = trim($args[0]); + switch ($fun) { + case 'default': // 鐗规畩妯℃澘鍑芥暟 + if (false === strpos($name, '(')) { + $name = '(isset(' . $name . ') && (' . $name . ' !== \'\')?' . $name . ':' . $args[1] . ')'; + } else { + $name = '(' . $name . ' !== \'\'?' . $name . ':' . $args[1] . ')'; + } + break; + default: // 閫氱敤妯℃澘鍑芥暟 + if (!in_array($fun, $template_deny_funs)) { + if (isset($args[1])) { + if (strstr($args[1], '###')) { + $args[1] = str_replace('###', $name, $args[1]); + $name = "$fun($args[1])"; + } else { + $name = "$fun($name,$args[1])"; + } + } else { + if (!empty($args[0])) { + $name = "$fun($name)"; + } + } + } + } + } + $_varFunctionList[$_key] = $name; + $varStr = $name; + } + return; + } + + /** + * 鐗规畩妯℃澘鍙橀噺瑙f瀽 + * 鏍煎紡 浠 $Think. 鎵撳ご鐨勫彉閲忓睘浜庣壒娈婃ā鏉垮彉閲 + * @access public + * @param array $vars 鍙橀噺鏁扮粍 + * @return string + */ + public function parseThinkVar($vars) + { + $type = strtoupper(trim(array_shift($vars))); + $param = implode('.', $vars); + if ($vars) { + switch ($type) { + case 'SERVER': + $parseStr = '\\think\\Request::instance()->server(\'' . $param . '\')'; + break; + case 'GET': + $parseStr = '\\think\\Request::instance()->get(\'' . $param . '\')'; + break; + case 'POST': + $parseStr = '\\think\\Request::instance()->post(\'' . $param . '\')'; + break; + case 'COOKIE': + $parseStr = '\\think\\Cookie::get(\'' . $param . '\')'; + break; + case 'SESSION': + $parseStr = '\\think\\Session::get(\'' . $param . '\')'; + break; + case 'ENV': + $parseStr = '\\think\\Request::instance()->env(\'' . $param . '\')'; + break; + case 'REQUEST': + $parseStr = '\\think\\Request::instance()->request(\'' . $param . '\')'; + break; + case 'CONST': + $parseStr = strtoupper($param); + break; + case 'LANG': + $parseStr = '\\think\\Lang::get(\'' . $param . '\')'; + break; + case 'CONFIG': + $parseStr = '\\think\\Config::get(\'' . $param . '\')'; + break; + default: + $parseStr = '\'\''; + break; + } + } else { + switch ($type) { + case 'NOW': + $parseStr = "date('Y-m-d g:i a',time())"; + break; + case 'VERSION': + $parseStr = 'THINK_VERSION'; + break; + case 'LDELIM': + $parseStr = '\'' . ltrim($this->config['tpl_begin'], '\\') . '\''; + break; + case 'RDELIM': + $parseStr = '\'' . ltrim($this->config['tpl_end'], '\\') . '\''; + break; + default: + if (defined($type)) { + $parseStr = $type; + } else { + $parseStr = ''; + } + } + } + return $parseStr; + } + + /** + * 鍒嗘瀽鍔犺浇鐨勬ā鏉挎枃浠跺苟璇诲彇鍐呭 鏀寔澶氫釜妯℃澘鏂囦欢璇诲彇 + * @access private + * @param string $templateName 妯℃澘鏂囦欢鍚 + * @return string + */ + private function parseTemplateName($templateName) + { + $array = explode(',', $templateName); + $parseStr = ''; + foreach ($array as $templateName) { + if (empty($templateName)) { + continue; + } + if (0 === strpos($templateName, '$')) { + //鏀寔鍔犺浇鍙橀噺鏂囦欢鍚 + $templateName = $this->get(substr($templateName, 1)); + } + $template = $this->parseTemplateFile($templateName); + if ($template) { + // 鑾峰彇妯℃澘鏂囦欢鍐呭 + $parseStr .= file_get_contents($template); + } + } + return $parseStr; + } + + /** + * 瑙f瀽妯℃澘鏂囦欢鍚 + * @access private + * @param string $template 鏂囦欢鍚 + * @return string|false + */ + private function parseTemplateFile($template) + { + if ('' == pathinfo($template, PATHINFO_EXTENSION)) { + if (strpos($template, '@')) { + list($module, $template) = explode('@', $template); + } + if (0 !== strpos($template, '/')) { + $template = str_replace(['/', ':'], $this->config['view_depr'], $template); + } else { + $template = str_replace(['/', ':'], $this->config['view_depr'], substr($template, 1)); + } + if ($this->config['view_base']) { + $module = isset($module) ? $module : Request::instance()->module(); + $path = $this->config['view_base'] . ($module ? $module . DS : ''); + } else { + $path = isset($module) ? APP_PATH . $module . DS . basename($this->config['view_path']) . DS : $this->config['view_path']; + } + $template = $path . $template . '.' . ltrim($this->config['view_suffix'], '.'); + } + + if (is_file($template)) { + // 璁板綍妯℃澘鏂囦欢鐨勬洿鏂版椂闂 + $this->includeFile[$template] = filemtime($template); + return $template; + } else { + throw new TemplateNotFoundException('template not exists:' . $template, $template); + } + } + + /** + * 鎸夋爣绛剧敓鎴愭鍒 + * @access private + * @param string $tagName 鏍囩鍚 + * @return string + */ + private function getRegex($tagName) + { + $regex = ''; + if ('tag' == $tagName) { + $begin = $this->config['tpl_begin']; + $end = $this->config['tpl_end']; + if (strlen(ltrim($begin, '\\')) == 1 && strlen(ltrim($end, '\\')) == 1) { + $regex = $begin . '((?:[\$]{1,2}[a-wA-w_]|[\:\~][\$a-wA-w_]|[+]{2}[\$][a-wA-w_]|[-]{2}[\$][a-wA-w_]|\/[\*\/])(?>[^' . $end . ']*))' . $end; + } else { + $regex = $begin . '((?:[\$]{1,2}[a-wA-w_]|[\:\~][\$a-wA-w_]|[+]{2}[\$][a-wA-w_]|[-]{2}[\$][a-wA-w_]|\/[\*\/])(?>(?:(?!' . $end . ').)*))' . $end; + } + } else { + $begin = $this->config['taglib_begin']; + $end = $this->config['taglib_end']; + $single = strlen(ltrim($begin, '\\')) == 1 && strlen(ltrim($end, '\\')) == 1 ? true : false; + switch ($tagName) { + case 'block': + if ($single) { + $regex = $begin . '(?:' . $tagName . '\b(?>(?:(?!name=).)*)\bname=([\'\"])(?P[\$\w\-\/\.]+)\\1(?>[^' . $end . ']*)|\/' . $tagName . ')' . $end; + } else { + $regex = $begin . '(?:' . $tagName . '\b(?>(?:(?!name=).)*)\bname=([\'\"])(?P[\$\w\-\/\.]+)\\1(?>(?:(?!' . $end . ').)*)|\/' . $tagName . ')' . $end; + } + break; + case 'literal': + if ($single) { + $regex = '(' . $begin . $tagName . '\b(?>[^' . $end . ']*)' . $end . ')'; + $regex .= '(?:(?>[^' . $begin . ']*)(?>(?!' . $begin . '(?>' . $tagName . '\b[^' . $end . ']*|\/' . $tagName . ')' . $end . ')' . $begin . '[^' . $begin . ']*)*)'; + $regex .= '(' . $begin . '\/' . $tagName . $end . ')'; + } else { + $regex = '(' . $begin . $tagName . '\b(?>(?:(?!' . $end . ').)*)' . $end . ')'; + $regex .= '(?:(?>(?:(?!' . $begin . ').)*)(?>(?!' . $begin . '(?>' . $tagName . '\b(?>(?:(?!' . $end . ').)*)|\/' . $tagName . ')' . $end . ')' . $begin . '(?>(?:(?!' . $begin . ').)*))*)'; + $regex .= '(' . $begin . '\/' . $tagName . $end . ')'; + } + break; + case 'restoreliteral': + $regex = ''; + break; + case 'include': + $name = 'file'; + case 'taglib': + case 'layout': + case 'extend': + if (empty($name)) { + $name = 'name'; + } + if ($single) { + $regex = $begin . $tagName . '\b(?>(?:(?!' . $name . '=).)*)\b' . $name . '=([\'\"])(?P[\$\w\-\/\.\:@,\\\\]+)\\1(?>[^' . $end . ']*)' . $end; + } else { + $regex = $begin . $tagName . '\b(?>(?:(?!' . $name . '=).)*)\b' . $name . '=([\'\"])(?P[\$\w\-\/\.\:@,\\\\]+)\\1(?>(?:(?!' . $end . ').)*)' . $end; + } + break; + } + } + return '/' . $regex . '/is'; + } +} diff --git a/thinkphp/library/think/Url.php b/thinkphp/library/think/Url.php new file mode 100644 index 000000000..51c846d73 --- /dev/null +++ b/thinkphp/library/think/Url.php @@ -0,0 +1,319 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class Url +{ + // 鐢熸垚URL鍦板潃鐨剅oot + protected static $root; + protected static $bindCheck; + + /** + * URL鐢熸垚 鏀寔璺敱鍙嶅皠 + * @param string $url 璺敱鍦板潃 + * @param string|array $vars 鍙傛暟锛堟敮鎸佹暟缁勫拰瀛楃涓诧級a=val&b=val2... ['a'=>'val1', 'b'=>'val2'] + * @param string|bool $suffix 浼潤鎬佸悗缂锛岄粯璁や负true琛ㄧず鑾峰彇閰嶇疆鍊 + * @param boolean|string $domain 鏄惁鏄剧ず鍩熷悕 鎴栬呯洿鎺ヤ紶鍏ュ煙鍚 + * @return string + */ + public static function build($url = '', $vars = '', $suffix = true, $domain = false) + { + if (false === $domain && Route::rules('domain')) { + $domain = true; + } + // 瑙f瀽URL + if (0 === strpos($url, '[') && $pos = strpos($url, ']')) { + // [name] 琛ㄧず浣跨敤璺敱鍛藉悕鏍囪瘑鐢熸垚URL + $name = substr($url, 1, $pos - 1); + $url = 'name' . substr($url, $pos + 1); + } + if (false === strpos($url, '://') && 0 !== strpos($url, '/')) { + $info = parse_url($url); + $url = !empty($info['path']) ? $info['path'] : ''; + if (isset($info['fragment'])) { + // 瑙f瀽閿氱偣 + $anchor = $info['fragment']; + if (false !== strpos($anchor, '?')) { + // 瑙f瀽鍙傛暟 + list($anchor, $info['query']) = explode('?', $anchor, 2); + } + if (false !== strpos($anchor, '@')) { + // 瑙f瀽鍩熷悕 + list($anchor, $domain) = explode('@', $anchor, 2); + } + } elseif (strpos($url, '@') && false === strpos($url, '\\')) { + // 瑙f瀽鍩熷悕 + list($url, $domain) = explode('@', $url, 2); + } + } + + // 瑙f瀽鍙傛暟 + if (is_string($vars)) { + // aaa=1&bbb=2 杞崲鎴愭暟缁 + parse_str($vars, $vars); + } + + if ($url) { + $rule = Route::name(isset($name) ? $name : $url . (isset($info['query']) ? '?' . $info['query'] : '')); + if (is_null($rule) && isset($info['query'])) { + $rule = Route::name($url); + // 瑙f瀽鍦板潃閲岄潰鍙傛暟 鍚堝苟鍒皏ars + parse_str($info['query'], $params); + $vars = array_merge($params, $vars); + unset($info['query']); + } + } + if (!empty($rule) && $match = self::getRuleUrl($rule, $vars)) { + // 鍖归厤璺敱鍛藉悕鏍囪瘑 + $url = $match[0]; + // 鏇挎崲鍙夊垎闅旂 + $url = preg_replace(['/(\W)\?$/', '/(\W)\?/'], ['', '\1'], $url); + if (!empty($match[1])) { + $domain = $match[1]; + } + } elseif (!empty($rule) && isset($name)) { + throw new \InvalidArgumentException('route name not exists:' . $name); + } else { + // 妫鏌ュ埆鍚嶈矾鐢 + $alias = Route::rules('alias'); + $matchAlias = false; + if ($alias) { + // 鍒悕璺敱瑙f瀽 + foreach ($alias as $key => $val) { + if (is_array($val)) { + $val = $val[0]; + } + if (0 === strpos($url, $val)) { + $url = $key . substr($url, strlen($val)); + $matchAlias = true; + break; + } + } + } + if (!$matchAlias) { + // 璺敱鏍囪瘑涓嶅瓨鍦 鐩存帴瑙f瀽 + $url = self::parseUrl($url, $domain); + } + if (isset($info['query'])) { + // 瑙f瀽鍦板潃閲岄潰鍙傛暟 鍚堝苟鍒皏ars + parse_str($info['query'], $params); + $vars = array_merge($params, $vars); + } + } + + // 妫娴婾RL缁戝畾 + if (!self::$bindCheck) { + $type = Route::getBind('type'); + if ($type) { + $bind = Route::getBind($type); + if (0 === strpos($url, $bind)) { + $url = substr($url, strlen($bind) + 1); + } + } + } + // 杩樺師URL鍒嗛殧绗 + $depr = Config::get('pathinfo_depr'); + $url = str_replace('/', $depr, $url); + + // URL鍚庣紑 + $suffix = in_array($url, ['/', '']) ? '' : self::parseSuffix($suffix); + // 閿氱偣 + $anchor = !empty($anchor) ? '#' . $anchor : ''; + // 鍙傛暟缁勮 + if (!empty($vars)) { + // 娣诲姞鍙傛暟 + if (Config::get('url_common_param')) { + $vars = urldecode(http_build_query($vars)); + $url .= $suffix . '?' . $vars . $anchor; + } else { + $paramType = Config::get('url_param_type'); + foreach ($vars as $var => $val) { + if ('' !== trim($val)) { + if ($paramType) { + $url .= $depr . urlencode($val); + } else { + $url .= $depr . $var . $depr . urlencode($val); + } + } + } + $url .= $suffix . $anchor; + } + } else { + $url .= $suffix . $anchor; + } + // 妫娴嬪煙鍚 + $domain = self::parseDomain($url, $domain); + // URL缁勮 + $url = $domain . (self::$root ?: Request::instance()->root()) . '/' . ltrim($url, '/'); + self::$bindCheck = false; + return $url; + } + + // 鐩存帴瑙f瀽URL鍦板潃 + protected static function parseUrl($url, &$domain) + { + $request = Request::instance(); + if (0 === strpos($url, '/')) { + // 鐩存帴浣滀负璺敱鍦板潃瑙f瀽 + $url = substr($url, 1); + } elseif (false !== strpos($url, '\\')) { + // 瑙f瀽鍒扮被 + $url = ltrim(str_replace('\\', '/', $url), '/'); + } elseif (0 === strpos($url, '@')) { + // 瑙f瀽鍒版帶鍒跺櫒 + $url = substr($url, 1); + } else { + // 瑙f瀽鍒 妯″潡/鎺у埗鍣/鎿嶄綔 + $module = $request->module(); + $domains = Route::rules('domain'); + if (true === $domain && 2 == substr_count($url, '/')) { + $current = $request->host(); + $match = []; + $pos = []; + foreach ($domains as $key => $item) { + if (isset($item['[bind]']) && 0 === strpos($url, $item['[bind]'][0])) { + $pos[$key] = strlen($item['[bind]'][0]) + 1; + $match[] = $key; + $module = ''; + } + } + if ($match) { + $domain = current($match); + foreach ($match as $item) { + if (0 === strpos($current, $item)) { + $domain = $item; + } + } + self::$bindCheck = true; + $url = substr($url, $pos[$domain]); + } + } elseif ($domain) { + if (isset($domains[$domain]['[bind]'][0])) { + $bindModule = $domains[$domain]['[bind]'][0]; + if ($bindModule && !in_array($bindModule[0], ['\\', '@'])) { + $module = ''; + } + } + } + $module = $module ? $module . '/' : ''; + + $controller = Loader::parseName($request->controller()); + if ('' == $url) { + // 绌哄瓧绗︿覆杈撳嚭褰撳墠鐨 妯″潡/鎺у埗鍣/鎿嶄綔 + $url = $module . $controller . '/' . $request->action(); + } else { + $path = explode('/', $url); + $action = Config::get('url_convert') ? strtolower(array_pop($path)) : array_pop($path); + $controller = empty($path) ? $controller : (Config::get('url_convert') ? Loader::parseName(array_pop($path)) : array_pop($path)); + $module = empty($path) ? $module : array_pop($path) . '/'; + $url = $module . $controller . '/' . $action; + } + } + return $url; + } + + // 妫娴嬪煙鍚 + protected static function parseDomain(&$url, $domain) + { + if (!$domain) { + return ''; + } + $request = Request::instance(); + $rootDomain = Config::get('url_domain_root'); + if (true === $domain) { + // 鑷姩鍒ゆ柇鍩熷悕 + $domain = $request->host(); + + $domains = Route::rules('domain'); + if ($domains) { + $route_domain = array_keys($domains); + foreach ($route_domain as $domain_prefix) { + if (0 === strpos($domain_prefix, '*.') && strpos($domain, ltrim($domain_prefix, '*.')) !== false) { + foreach ($domains as $key => $rule) { + $rule = is_array($rule) ? $rule[0] : $rule; + if (is_string($rule) && false === strpos($key, '*') && 0 === strpos($url, $rule)) { + $url = ltrim($url, $rule); + $domain = $key; + // 鐢熸垚瀵瑰簲瀛愬煙鍚 + if (!empty($rootDomain)) { + $domain .= $rootDomain; + } + break; + } elseif (false !== strpos($key, '*')) { + if (!empty($rootDomain)) { + $domain .= $rootDomain; + } + break; + } + } + } + } + } + + } else { + if (empty($rootDomain)) { + $host = $request->host(); + $rootDomain = substr_count($host, '.') > 1 ? substr(strstr($host, '.'), 1) : $host; + } + if (!strpos($domain, $rootDomain)) { + $domain .= '.' . $rootDomain; + } + } + return ($request->isSsl() ? 'https://' : 'http://') . $domain; + } + + // 瑙f瀽URL鍚庣紑 + protected static function parseSuffix($suffix) + { + if ($suffix) { + $suffix = true === $suffix ? Config::get('url_html_suffix') : $suffix; + if ($pos = strpos($suffix, '|')) { + $suffix = substr($suffix, 0, $pos); + } + } + return (empty($suffix) || 0 === strpos($suffix, '.')) ? $suffix : '.' . $suffix; + } + + // 鍖归厤璺敱鍦板潃 + public static function getRuleUrl($rule, &$vars = []) + { + foreach ($rule as $item) { + list($url, $pattern, $domain) = $item; + if (empty($pattern)) { + return [$url, $domain]; + } + foreach ($pattern as $key => $val) { + if (isset($vars[$key])) { + $url = str_replace(['[:' . $key . ']', '<' . $key . '?>', ':' . $key . '', '<' . $key . '>'], $vars[$key], $url); + unset($vars[$key]); + $result = [$url, $domain]; + } elseif (2 == $val) { + $url = str_replace(['/[:' . $key . ']', '[:' . $key . ']', '<' . $key . '?>'], '', $url); + $result = [$url, $domain]; + } else { + break; + } + } + if (isset($result)) { + return $result; + } + } + return false; + } + + // 鎸囧畾褰撳墠鐢熸垚URL鍦板潃鐨剅oot + public static function root($root) + { + self::$root = $root; + Request::instance()->root($root); + } +} diff --git a/thinkphp/library/think/Validate.php b/thinkphp/library/think/Validate.php new file mode 100644 index 000000000..ac9e6d55f --- /dev/null +++ b/thinkphp/library/think/Validate.php @@ -0,0 +1,1274 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\exception\ClassNotFoundException; + +class Validate +{ + // 瀹炰緥 + protected static $instance; + + // 鑷畾涔夌殑楠岃瘉绫诲瀷 + protected static $type = []; + + // 楠岃瘉绫诲瀷鍒悕 + protected $alias = [ + '>' => 'gt', '>=' => 'egt', '<' => 'lt', '<=' => 'elt', '=' => 'eq', 'same' => 'eq', + ]; + + // 褰撳墠楠岃瘉鐨勮鍒 + protected $rule = []; + + // 楠岃瘉鎻愮ず淇℃伅 + protected $message = []; + // 楠岃瘉瀛楁鎻忚堪 + protected $field = []; + + // 楠岃瘉瑙勫垯榛樿鎻愮ず淇℃伅 + protected static $typeMsg = [ + 'require' => ':attribute涓嶈兘涓虹┖', + 'number' => ':attribute蹇呴』鏄暟瀛', + 'float' => ':attribute蹇呴』鏄诞鐐规暟', + 'boolean' => ':attribute蹇呴』鏄竷灏斿', + 'email' => ':attribute鏍煎紡涓嶇', + 'array' => ':attribute蹇呴』鏄暟缁', + 'accepted' => ':attribute蹇呴』鏄痽es銆乷n鎴栬1', + 'date' => ':attribute鏍煎紡涓嶇鍚', + 'file' => ':attribute涓嶆槸鏈夋晥鐨勪笂浼犳枃浠', + 'image' => ':attribute涓嶆槸鏈夋晥鐨勫浘鍍忔枃浠', + 'alpha' => ':attribute鍙兘鏄瓧姣', + 'alphaNum' => ':attribute鍙兘鏄瓧姣嶅拰鏁板瓧', + 'alphaDash' => ':attribute鍙兘鏄瓧姣嶃佹暟瀛楀拰涓嬪垝绾縚鍙婄牬鎶樺彿-', + 'activeUrl' => ':attribute涓嶆槸鏈夋晥鐨勫煙鍚嶆垨鑰匢P', + 'chs' => ':attribute鍙兘鏄眽瀛', + 'chsAlpha' => ':attribute鍙兘鏄眽瀛椼佸瓧姣', + 'chsAlphaNum' => ':attribute鍙兘鏄眽瀛椼佸瓧姣嶅拰鏁板瓧', + 'chsDash' => ':attribute鍙兘鏄眽瀛椼佸瓧姣嶃佹暟瀛楀拰涓嬪垝绾縚鍙婄牬鎶樺彿-', + 'url' => ':attribute涓嶆槸鏈夋晥鐨刄RL鍦板潃', + 'ip' => ':attribute涓嶆槸鏈夋晥鐨処P鍦板潃', + 'dateFormat' => ':attribute蹇呴』浣跨敤鏃ユ湡鏍煎紡 :rule', + 'in' => ':attribute蹇呴』鍦 :rule 鑼冨洿鍐', + 'notIn' => ':attribute涓嶈兘鍦 :rule 鑼冨洿鍐', + 'between' => ':attribute鍙兘鍦 :1 - :2 涔嬮棿', + 'notBetween' => ':attribute涓嶈兘鍦 :1 - :2 涔嬮棿', + 'length' => ':attribute闀垮害涓嶇鍚堣姹 :rule', + 'max' => ':attribute闀垮害涓嶈兘瓒呰繃 :rule', + 'min' => ':attribute闀垮害涓嶈兘灏忎簬 :rule', + 'after' => ':attribute鏃ユ湡涓嶈兘灏忎簬 :rule', + 'before' => ':attribute鏃ユ湡涓嶈兘瓒呰繃 :rule', + 'expire' => '涓嶅湪鏈夋晥鏈熷唴 :rule', + 'allowIp' => '涓嶅厑璁哥殑IP璁块棶', + 'denyIp' => '绂佹鐨処P璁块棶', + 'confirm' => ':attribute鍜岀‘璁ゅ瓧娈:2涓嶄竴鑷', + 'different' => ':attribute鍜屾瘮杈冨瓧娈:2涓嶈兘鐩稿悓', + 'egt' => ':attribute蹇呴』澶т簬绛変簬 :rule', + 'gt' => ':attribute蹇呴』澶т簬 :rule', + 'elt' => ':attribute蹇呴』灏忎簬绛変簬 :rule', + 'lt' => ':attribute蹇呴』灏忎簬 :rule', + 'eq' => ':attribute蹇呴』绛変簬 :rule', + 'unique' => ':attribute宸插瓨鍦', + 'regex' => ':attribute涓嶇鍚堟寚瀹氳鍒', + 'method' => '鏃犳晥鐨勮姹傜被鍨', + 'token' => '浠ょ墝鏁版嵁鏃犳晥', + 'fileSize' => '涓婁紶鏂囦欢澶у皬涓嶇', + 'fileExt' => '涓婁紶鏂囦欢鍚庣紑涓嶇', + 'fileMime' => '涓婁紶鏂囦欢绫诲瀷涓嶇', + + ]; + + // 褰撳墠楠岃瘉鍦烘櫙 + protected $currentScene = null; + + // 姝e垯琛ㄨ揪寮 regex = ['zip'=>'\d{6}',...] + protected $regex = []; + + // 楠岃瘉鍦烘櫙 scene = ['edit'=>'name1,name2,...'] + protected $scene = []; + + // 楠岃瘉澶辫触閿欒淇℃伅 + protected $error = []; + + // 鎵归噺楠岃瘉 + protected $batch = false; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param array $rules 楠岃瘉瑙勫垯 + * @param array $message 楠岃瘉鎻愮ず淇℃伅 + * @param array $field 楠岃瘉瀛楁鎻忚堪淇℃伅 + */ + public function __construct(array $rules = [], $message = [], $field = []) + { + $this->rule = array_merge($this->rule, $rules); + $this->message = array_merge($this->message, $message); + $this->field = array_merge($this->field, $field); + } + + /** + * 瀹炰緥鍖栭獙璇 + * @access public + * @param array $rules 楠岃瘉瑙勫垯 + * @param array $message 楠岃瘉鎻愮ず淇℃伅 + * @param array $field 楠岃瘉瀛楁鎻忚堪淇℃伅 + * @return Validate + */ + public static function make($rules = [], $message = [], $field = []) + { + if (is_null(self::$instance)) { + self::$instance = new self($rules, $message, $field); + } + return self::$instance; + } + + /** + * 娣诲姞瀛楁楠岃瘉瑙勫垯 + * @access protected + * @param string|array $name 瀛楁鍚嶇О鎴栬呰鍒欐暟缁 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return Validate + */ + public function rule($name, $rule = '') + { + if (is_array($name)) { + $this->rule = array_merge($this->rule, $name); + } else { + $this->rule[$name] = $rule; + } + return $this; + } + + /** + * 娉ㄥ唽楠岃瘉锛堢被鍨嬶級瑙勫垯 + * @access public + * @param string $type 楠岃瘉瑙勫垯绫诲瀷 + * @param mixed $callback callback鏂规硶(鎴栭棴鍖) + * @return void + */ + public static function extend($type, $callback = null) + { + if (is_array($type)) { + self::$type = array_merge(self::$type, $type); + } else { + self::$type[$type] = $callback; + } + } + + /** + * 鑾峰彇楠岃瘉瑙勫垯鐨勯粯璁ゆ彁绀轰俊鎭 + * @access protected + * @param string|array $type 楠岃瘉瑙勫垯绫诲瀷鍚嶇О鎴栬呮暟缁 + * @param string $msg 楠岃瘉鎻愮ず淇℃伅 + * @return void + */ + public static function setTypeMsg($type, $msg = null) + { + if (is_array($type)) { + self::$typeMsg = array_merge(self::$typeMsg, $type); + } else { + self::$typeMsg[$type] = $msg; + } + } + + /** + * 璁剧疆鎻愮ず淇℃伅 + * @access public + * @param string|array $name 瀛楁鍚嶇О + * @param string $message 鎻愮ず淇℃伅 + * @return Validate + */ + public function message($name, $message = '') + { + if (is_array($name)) { + $this->message = array_merge($this->message, $name); + } else { + $this->message[$name] = $message; + } + return $this; + } + + /** + * 璁剧疆楠岃瘉鍦烘櫙 + * @access public + * @param string|array $name 鍦烘櫙鍚嶆垨鑰呭満鏅缃暟缁 + * @param mixed $fields 瑕侀獙璇佺殑瀛楁 + * @return Validate + */ + public function scene($name, $fields = null) + { + if (is_array($name)) { + $this->scene = array_merge($this->scene, $name); + }if (is_null($fields)) { + // 璁剧疆褰撳墠鍦烘櫙 + $this->currentScene = $name; + } else { + // 璁剧疆楠岃瘉鍦烘櫙 + $this->scene[$name] = $fields; + } + return $this; + } + + /** + * 鍒ゆ柇鏄惁瀛樺湪鏌愪釜楠岃瘉鍦烘櫙 + * @access public + * @param string $name 鍦烘櫙鍚 + * @return bool + */ + public function hasScene($name) + { + return isset($this->scene[$name]); + } + + /** + * 璁剧疆鎵归噺楠岃瘉 + * @access public + * @param bool $batch 鏄惁鎵归噺楠岃瘉 + * @return Validate + */ + public function batch($batch = true) + { + $this->batch = $batch; + return $this; + } + + /** + * 鏁版嵁鑷姩楠岃瘉 + * @access public + * @param array $data 鏁版嵁 + * @param mixed $rules 楠岃瘉瑙勫垯 + * @param string $scene 楠岃瘉鍦烘櫙 + * @return bool + */ + public function check($data, $rules = [], $scene = '') + { + $this->error = []; + + if (empty($rules)) { + // 璇诲彇楠岃瘉瑙勫垯 + $rules = $this->rule; + } + + // 鍒嗘瀽楠岃瘉瑙勫垯 + $scene = $this->getScene($scene); + if (is_array($scene)) { + // 澶勭悊鍦烘櫙楠岃瘉瀛楁 + $change = []; + $array = []; + foreach ($scene as $k => $val) { + if (is_numeric($k)) { + $array[] = $val; + } else { + $array[] = $k; + $change[$k] = $val; + } + } + } + + foreach ($rules as $key => $item) { + // field => rule1|rule2... field=>['rule1','rule2',...] + if (is_numeric($key)) { + // [field,rule1|rule2,msg1|msg2] + $key = $item[0]; + $rule = $item[1]; + if (isset($item[2])) { + $msg = is_string($item[2]) ? explode('|', $item[2]) : $item[2]; + } else { + $msg = []; + } + } else { + $rule = $item; + $msg = []; + } + if (strpos($key, '|')) { + // 瀛楁|鎻忚堪 鐢ㄤ簬鎸囧畾灞炴у悕绉 + list($key, $title) = explode('|', $key); + } else { + $title = isset($this->field[$key]) ? $this->field[$key] : $key; + } + + // 鍦烘櫙妫娴 + if (!empty($scene)) { + if ($scene instanceof \Closure && !call_user_func_array($scene, [$key, $data])) { + continue; + } elseif (is_array($scene)) { + if (!in_array($key, $array)) { + continue; + } elseif (isset($change[$key])) { + // 閲嶈浇鏌愪釜楠岃瘉瑙勫垯 + $rule = $change[$key]; + } + } + } + + // 鑾峰彇鏁版嵁 鏀寔浜岀淮鏁扮粍 + $value = $this->getDataValue($data, $key); + + // 瀛楁楠岃瘉 + if ($rule instanceof \Closure) { + // 鍖垮悕鍑芥暟楠岃瘉 鏀寔浼犲叆褰撳墠瀛楁鍜屾墍鏈夊瓧娈典袱涓暟鎹 + $result = call_user_func_array($rule, [$value, $data]); + } else { + $result = $this->checkItem($key, $value, $rule, $data, $title, $msg); + } + + if (true !== $result) { + // 娌℃湁杩斿洖true 鍒欒〃绀洪獙璇佸け璐 + if (!empty($this->batch)) { + // 鎵归噺楠岃瘉 + if (is_array($result)) { + $this->error = array_merge($this->error, $result); + } else { + $this->error[$key] = $result; + } + } else { + $this->error = $result; + return false; + } + } + } + return !empty($this->error) ? false : true; + } + + /** + * 楠岃瘉鍗曚釜瀛楁瑙勫垯 + * @access protected + * @param string $field 瀛楁鍚 + * @param mixed $value 瀛楁鍊 + * @param mixed $rules 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @param string $title 瀛楁鎻忚堪 + * @param array $msg 鎻愮ず淇℃伅 + * @return mixed + */ + protected function checkItem($field, $value, $rules, $data, $title = '', $msg = []) + { + // 鏀寔澶氳鍒欓獙璇 require|in:a,b,c|... 鎴栬 ['require','in'=>'a,b,c',...] + if (is_string($rules)) { + $rules = explode('|', $rules); + } + $i = 0; + foreach ($rules as $key => $rule) { + if ($rule instanceof \Closure) { + $result = call_user_func_array($rule, [$value, $data]); + $info = is_numeric($key) ? '' : $key; + } else { + // 鍒ゆ柇楠岃瘉绫诲瀷 + if (is_numeric($key)) { + if (strpos($rule, ':')) { + list($type, $rule) = explode(':', $rule, 2); + if (isset($this->alias[$type])) { + // 鍒ゆ柇鍒悕 + $type = $this->alias[$type]; + } + $info = $type; + } elseif (method_exists($this, $rule)) { + $type = $rule; + $info = $rule; + $rule = ''; + } else { + $type = 'is'; + $info = $rule; + } + } else { + $info = $type = $key; + } + + // 濡傛灉涓嶆槸require 鏈夋暟鎹墠浼氳楠岃瘉 + if (0 === strpos($info, 'require') || (!is_null($value) && '' !== $value)) { + // 楠岃瘉绫诲瀷 + $callback = isset(self::$type[$type]) ? self::$type[$type] : [$this, $type]; + // 楠岃瘉鏁版嵁 + $result = call_user_func_array($callback, [$value, $rule, $data, $field, $title]); + } else { + $result = true; + } + } + + if (false === $result) { + // 楠岃瘉澶辫触 杩斿洖閿欒淇℃伅 + if (isset($msg[$i])) { + $message = $msg[$i]; + if (is_string($message) && strpos($message, '{%') === 0) { + $message = Lang::get(substr($message, 2, -1)); + } + } else { + $message = $this->getRuleMsg($field, $title, $info, $rule); + } + return $message; + } elseif (true !== $result) { + // 杩斿洖鑷畾涔夐敊璇俊鎭 + if (is_string($result) && false !== strpos($result, ':')) { + $result = str_replace([':attribute', ':rule'], [$title, (string) $rule], $result); + } + return $result; + } + $i++; + } + return $result; + } + + /** + * 楠岃瘉鏄惁鍜屾煇涓瓧娈电殑鍊间竴鑷 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @param string $field 瀛楁鍚 + * @return bool + */ + protected function confirm($value, $rule, $data, $field = '') + { + if ('' == $rule) { + if (strpos($field, '_confirm')) { + $rule = strstr($field, '_confirm', true); + } else { + $rule = $field . '_confirm'; + } + } + return $this->getDataValue($data, $rule) === $value; + } + + /** + * 楠岃瘉鏄惁鍜屾煇涓瓧娈电殑鍊兼槸鍚︿笉鍚 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @return bool + */ + protected function different($value, $rule, $data) + { + return $this->getDataValue($data, $rule) != $value; + } + + /** + * 楠岃瘉鏄惁澶т簬绛変簬鏌愪釜鍊 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function egt($value, $rule) + { + return $value >= $rule; + } + + /** + * 楠岃瘉鏄惁澶т簬鏌愪釜鍊 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function gt($value, $rule) + { + return $value > $rule; + } + + /** + * 楠岃瘉鏄惁灏忎簬绛変簬鏌愪釜鍊 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function elt($value, $rule) + { + return $value <= $rule; + } + + /** + * 楠岃瘉鏄惁灏忎簬鏌愪釜鍊 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function lt($value, $rule) + { + return $value < $rule; + } + + /** + * 楠岃瘉鏄惁绛変簬鏌愪釜鍊 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function eq($value, $rule) + { + return $value == $rule; + } + + /** + * 楠岃瘉瀛楁鍊兼槸鍚︿负鏈夋晥鏍煎紡 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param string $rule 楠岃瘉瑙勫垯 + * @param array $data 楠岃瘉鏁版嵁 + * @return bool + */ + protected function is($value, $rule, $data = []) + { + switch ($rule) { + case 'require': + // 蹇呴』 + $result = !empty($value) || '0' == $value; + break; + case 'accepted': + // 鎺ュ彈 + $result = in_array($value, ['1', 'on', 'yes']); + break; + case 'date': + // 鏄惁鏄竴涓湁鏁堟棩鏈 + $result = false !== strtotime($value); + break; + case 'alpha': + // 鍙厑璁稿瓧姣 + $result = $this->regex($value, '/^[A-Za-z]+$/'); + break; + case 'alphaNum': + // 鍙厑璁稿瓧姣嶅拰鏁板瓧 + $result = $this->regex($value, '/^[A-Za-z0-9]+$/'); + break; + case 'alphaDash': + // 鍙厑璁稿瓧姣嶃佹暟瀛楀拰涓嬪垝绾 鐮存姌鍙 + $result = $this->regex($value, '/^[A-Za-z0-9\-\_]+$/'); + break; + case 'chs': + // 鍙厑璁告眽瀛 + $result = $this->regex($value, '/^[\x{4e00}-\x{9fa5}]+$/u'); + break; + case 'chsAlpha': + // 鍙厑璁告眽瀛椼佸瓧姣 + $result = $this->regex($value, '/^[\x{4e00}-\x{9fa5}a-zA-Z]+$/u'); + break; + case 'chsAlphaNum': + // 鍙厑璁告眽瀛椼佸瓧姣嶅拰鏁板瓧 + $result = $this->regex($value, '/^[\x{4e00}-\x{9fa5}a-zA-Z0-9]+$/u'); + break; + case 'chsDash': + // 鍙厑璁告眽瀛椼佸瓧姣嶃佹暟瀛楀拰涓嬪垝绾縚鍙婄牬鎶樺彿- + $result = $this->regex($value, '/^[\x{4e00}-\x{9fa5}a-zA-Z0-9\_\-]+$/u'); + break; + case 'activeUrl': + // 鏄惁涓烘湁鏁堢殑缃戝潃 + $result = checkdnsrr($value); + break; + case 'ip': + // 鏄惁涓篒P鍦板潃 + $result = $this->filter($value, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4 | FILTER_FLAG_IPV6); + break; + case 'url': + // 鏄惁涓轰竴涓猆RL鍦板潃 + $result = $this->filter($value, FILTER_VALIDATE_URL); + break; + case 'float': + // 鏄惁涓篺loat + $result = $this->filter($value, FILTER_VALIDATE_FLOAT); + break; + case 'number': + $result = is_numeric($value); + break; + case 'integer': + // 鏄惁涓烘暣鍨 + $result = $this->filter($value, FILTER_VALIDATE_INT); + break; + case 'email': + // 鏄惁涓洪偖绠卞湴鍧 + $result = $this->filter($value, FILTER_VALIDATE_EMAIL); + break; + case 'boolean': + // 鏄惁涓哄竷灏斿 + $result = in_array($value, [true, false, 0, 1, '0', '1'], true); + break; + case 'array': + // 鏄惁涓烘暟缁 + $result = is_array($value); + break; + case 'file': + $result = $value instanceof File; + break; + case 'image': + $result = $value instanceof File && in_array($this->getImageType($value->getRealPath()), [1, 2, 3, 6]); + break; + case 'token': + $result = $this->token($value, '__token__', $data); + break; + default: + if (isset(self::$type[$rule])) { + // 娉ㄥ唽鐨勯獙璇佽鍒 + $result = call_user_func_array(self::$type[$rule], [$value]); + } else { + // 姝e垯楠岃瘉 + $result = $this->regex($value, $rule); + } + } + return $result; + } + + // 鍒ゆ柇鍥惧儚绫诲瀷 + protected function getImageType($image) + { + if (function_exists('exif_imagetype')) { + return exif_imagetype($image); + } else { + $info = getimagesize($image); + return $info[2]; + } + } + + /** + * 楠岃瘉鏄惁涓哄悎鏍肩殑鍩熷悕鎴栬匢P 鏀寔A锛孧X锛孨S锛孲OA锛孭TR锛孋NAME锛孉AAA锛孉6锛 SRV锛孨APTR锛孴XT 鎴栬 ANY绫诲瀷 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function activeUrl($value, $rule) + { + if (!in_array($rule, ['A', 'MX', 'NS', 'SOA', 'PTR', 'CNAME', 'AAAA', 'A6', 'SRV', 'NAPTR', 'TXT', 'ANY'])) { + $rule = 'MX'; + } + return checkdnsrr($value, $rule); + } + + /** + * 楠岃瘉鏄惁鏈夋晥IP + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 ipv4 ipv6 + * @return bool + */ + protected function ip($value, $rule) + { + if (!in_array($rule, ['ipv4', 'ipv6'])) { + $rule = 'ipv4'; + } + return $this->filter($value, FILTER_VALIDATE_IP, 'ipv6' == $rule ? FILTER_FLAG_IPV6 : FILTER_FLAG_IPV4); + } + + /** + * 楠岃瘉涓婁紶鏂囦欢鍚庣紑 + * @access protected + * @param mixed $file 涓婁紶鏂囦欢 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function fileExt($file, $rule) + { + if (!($file instanceof File)) { + return false; + } + if (is_string($rule)) { + $rule = explode(',', $rule); + } + if (is_array($file)) { + foreach ($file as $item) { + if (!$item->checkExt($rule)) { + return false; + } + } + return true; + } else { + return $file->checkExt($rule); + } + } + + /** + * 楠岃瘉涓婁紶鏂囦欢绫诲瀷 + * @access protected + * @param mixed $file 涓婁紶鏂囦欢 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function fileMime($file, $rule) + { + if (!($file instanceof File)) { + return false; + } + if (is_string($rule)) { + $rule = explode(',', $rule); + } + if (is_array($file)) { + foreach ($file as $item) { + if (!$item->checkMime($rule)) { + return false; + } + } + return true; + } else { + return $file->checkMime($rule); + } + } + + /** + * 楠岃瘉涓婁紶鏂囦欢澶у皬 + * @access protected + * @param mixed $file 涓婁紶鏂囦欢 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function fileSize($file, $rule) + { + if (!($file instanceof File)) { + return false; + } + if (is_array($file)) { + foreach ($file as $item) { + if (!$item->checkSize($rule)) { + return false; + } + } + return true; + } else { + return $file->checkSize($rule); + } + } + + /** + * 楠岃瘉鍥剧墖鐨勫楂樺強绫诲瀷 + * @access protected + * @param mixed $file 涓婁紶鏂囦欢 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function image($file, $rule) + { + if (!($file instanceof File)) { + return false; + } + if ($rule) { + $rule = explode(',', $rule); + list($width, $height, $type) = getimagesize($file->getRealPath()); + if (isset($rule[2])) { + $imageType = strtolower($rule[2]); + if ('jpeg' == $imageType) { + $imageType = 'jpg'; + } + if (image_type_to_extension($type, false) != $imageType) { + return false; + } + } + + list($w, $h) = $rule; + return $w == $width && $h == $height; + } else { + return in_array($this->getImageType($file->getRealPath()), [1, 2, 3, 6]); + } + } + + /** + * 楠岃瘉璇锋眰绫诲瀷 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function method($value, $rule) + { + $method = Request::instance()->method(); + return strtoupper($rule) == $method; + } + + /** + * 楠岃瘉鏃堕棿鍜屾棩鏈熸槸鍚︾鍚堟寚瀹氭牸寮 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function dateFormat($value, $rule) + { + $info = date_parse_from_format($rule, $value); + return 0 == $info['warning_count'] && 0 == $info['error_count']; + } + + /** + * 楠岃瘉鏄惁鍞竴 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 鏍煎紡锛氭暟鎹〃,瀛楁鍚,鎺掗櫎ID,涓婚敭鍚 + * @param array $data 鏁版嵁 + * @param string $field 楠岃瘉瀛楁鍚 + * @return bool + */ + protected function unique($value, $rule, $data, $field) + { + if (is_string($rule)) { + $rule = explode(',', $rule); + } + if (false !== strpos($rule[0], '\\')) { + // 鎸囧畾妯″瀷绫 + $db = new $rule[0]; + } else { + try { + $db = Loader::model($rule[0]); + } catch (ClassNotFoundException $e) { + $db = Db::name($rule[0]); + } + } + $key = isset($rule[1]) ? $rule[1] : $field; + + if (strpos($key, '^')) { + // 鏀寔澶氫釜瀛楁楠岃瘉 + $fields = explode('^', $key); + foreach ($fields as $key) { + $map[$key] = $data[$key]; + } + } elseif (strpos($key, '=')) { + parse_str($key, $map); + } else { + $map[$key] = $data[$field]; + } + + $pk = strval(isset($rule[3]) ? $rule[3] : $db->getPk()); + if (isset($rule[2])) { + $map[$pk] = ['neq', $rule[2]]; + } elseif (isset($data[$pk])) { + $map[$pk] = ['neq', $data[$pk]]; + } + + if ($db->where($map)->field($pk)->find()) { + return false; + } + return true; + } + + /** + * 浣跨敤琛屼负绫婚獙璇 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @return mixed + */ + protected function behavior($value, $rule, $data) + { + return Hook::exec($rule, '', $data); + } + + /** + * 浣跨敤filter_var鏂瑰紡楠岃瘉 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function filter($value, $rule) + { + if (is_string($rule) && strpos($rule, ',')) { + list($rule, $param) = explode(',', $rule); + } elseif (is_array($rule)) { + $param = isset($rule[1]) ? $rule[1] : null; + } else { + $param = null; + } + return false !== filter_var($value, is_int($rule) ? $rule : filter_id($rule), $param); + } + + /** + * 楠岃瘉鏌愪釜瀛楁绛変簬鏌愪釜鍊肩殑鏃跺欏繀椤 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @return bool + */ + protected function requireIf($value, $rule, $data) + { + list($field, $val) = explode(',', $rule); + if ($this->getDataValue($data, $field) == $val) { + return !empty($value); + } else { + return true; + } + } + + /** + * 閫氳繃鍥炶皟鏂规硶楠岃瘉鏌愪釜瀛楁鏄惁蹇呴』 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @return bool + */ + protected function requireCallback($value, $rule, $data) + { + $result = call_user_func_array($rule, [$value, $data]); + if ($result) { + return !empty($value); + } else { + return true; + } + } + + /** + * 楠岃瘉鏌愪釜瀛楁鏈夊肩殑鎯呭喌涓嬪繀椤 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @return bool + */ + protected function requireWith($value, $rule, $data) + { + $val = $this->getDataValue($data, $rule); + if (!empty($val)) { + return !empty($value); + } else { + return true; + } + } + + /** + * 楠岃瘉鏄惁鍦ㄨ寖鍥村唴 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function in($value, $rule) + { + return in_array($value, is_array($rule) ? $rule : explode(',', $rule)); + } + + /** + * 楠岃瘉鏄惁涓嶅湪鏌愪釜鑼冨洿 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function notIn($value, $rule) + { + return !in_array($value, is_array($rule) ? $rule : explode(',', $rule)); + } + + /** + * between楠岃瘉鏁版嵁 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function between($value, $rule) + { + if (is_string($rule)) { + $rule = explode(',', $rule); + } + list($min, $max) = $rule; + return $value >= $min && $value <= $max; + } + + /** + * 浣跨敤notbetween楠岃瘉鏁版嵁 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function notBetween($value, $rule) + { + if (is_string($rule)) { + $rule = explode(',', $rule); + } + list($min, $max) = $rule; + return $value < $min || $value > $max; + } + + /** + * 楠岃瘉鏁版嵁闀垮害 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function length($value, $rule) + { + if (is_array($value)) { + $length = count($value); + } elseif ($value instanceof File) { + $length = $value->getSize(); + } else { + $length = mb_strlen((string) $value); + } + + if (strpos($rule, ',')) { + // 闀垮害鍖洪棿 + list($min, $max) = explode(',', $rule); + return $length >= $min && $length <= $max; + } else { + // 鎸囧畾闀垮害 + return $length == $rule; + } + } + + /** + * 楠岃瘉鏁版嵁鏈澶ч暱搴 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function max($value, $rule) + { + if (is_array($value)) { + $length = count($value); + } elseif ($value instanceof File) { + $length = $value->getSize(); + } else { + $length = mb_strlen((string) $value); + } + return $length <= $rule; + } + + /** + * 楠岃瘉鏁版嵁鏈灏忛暱搴 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function min($value, $rule) + { + if (is_array($value)) { + $length = count($value); + } elseif ($value instanceof File) { + $length = $value->getSize(); + } else { + $length = mb_strlen((string) $value); + } + return $length >= $rule; + } + + /** + * 楠岃瘉鏃ユ湡 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function after($value, $rule) + { + return strtotime($value) >= strtotime($rule); + } + + /** + * 楠岃瘉鏃ユ湡 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function before($value, $rule) + { + return strtotime($value) <= strtotime($rule); + } + + /** + * 楠岃瘉鏈夋晥鏈 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return bool + */ + protected function expire($value, $rule) + { + if (is_string($rule)) { + $rule = explode(',', $rule); + } + list($start, $end) = $rule; + if (!is_numeric($start)) { + $start = strtotime($start); + } + + if (!is_numeric($end)) { + $end = strtotime($end); + } + return $_SERVER['REQUEST_TIME'] >= $start && $_SERVER['REQUEST_TIME'] <= $end; + } + + /** + * 楠岃瘉IP璁稿彲 + * @access protected + * @param string $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return mixed + */ + protected function allowIp($value, $rule) + { + return in_array($_SERVER['REMOTE_ADDR'], is_array($rule) ? $rule : explode(',', $rule)); + } + + /** + * 楠岃瘉IP绂佺敤 + * @access protected + * @param string $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @return mixed + */ + protected function denyIp($value, $rule) + { + return !in_array($_SERVER['REMOTE_ADDR'], is_array($rule) ? $rule : explode(',', $rule)); + } + + /** + * 浣跨敤姝e垯楠岃瘉鏁版嵁 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 姝e垯瑙勫垯鎴栬呴瀹氫箟姝e垯鍚 + * @return mixed + */ + protected function regex($value, $rule) + { + if (isset($this->regex[$rule])) { + $rule = $this->regex[$rule]; + } + if (0 !== strpos($rule, '/') && !preg_match('/\/[imsU]{0,4}$/', $rule)) { + // 涓嶆槸姝e垯琛ㄨ揪寮忓垯涓ょ琛ヤ笂/ + $rule = '/^' . $rule . '$/'; + } + return 1 === preg_match($rule, (string) $value); + } + + /** + * 楠岃瘉琛ㄥ崟浠ょ墝 + * @access protected + * @param mixed $value 瀛楁鍊 + * @param mixed $rule 楠岃瘉瑙勫垯 + * @param array $data 鏁版嵁 + * @return bool + */ + protected function token($value, $rule, $data) + { + $rule = !empty($rule) ? $rule : '__token__'; + if (!isset($data[$rule]) || !Session::has($rule)) { + // 浠ょ墝鏁版嵁鏃犳晥 + return false; + } + + // 浠ょ墝楠岃瘉 + if (isset($data[$rule]) && Session::get($rule) === $data[$rule]) { + // 闃叉閲嶅鎻愪氦 + Session::delete($rule); // 楠岃瘉瀹屾垚閿姣乻ession + return true; + } + // 寮鍚疶OKEN閲嶇疆 + Session::delete($rule); + return false; + } + + // 鑾峰彇閿欒淇℃伅 + public function getError() + { + return $this->error; + } + + /** + * 鑾峰彇鏁版嵁鍊 + * @access protected + * @param array $data 鏁版嵁 + * @param string $key 鏁版嵁鏍囪瘑 鏀寔浜岀淮 + * @return mixed + */ + protected function getDataValue($data, $key) + { + if (strpos($key, '.')) { + // 鏀寔浜岀淮鏁扮粍楠岃瘉 + list($name1, $name2) = explode('.', $key); + $value = isset($data[$name1][$name2]) ? $data[$name1][$name2] : null; + } else { + $value = isset($data[$key]) ? $data[$key] : null; + } + return $value; + } + + /** + * 鑾峰彇楠岃瘉瑙勫垯鐨勯敊璇彁绀轰俊鎭 + * @access protected + * @param string $attribute 瀛楁鑻辨枃鍚 + * @param string $title 瀛楁鎻忚堪鍚 + * @param string $type 楠岃瘉瑙勫垯鍚嶇О + * @param mixed $rule 楠岃瘉瑙勫垯鏁版嵁 + * @return string + */ + protected function getRuleMsg($attribute, $title, $type, $rule) + { + if (isset($this->message[$attribute . '.' . $type])) { + $msg = $this->message[$attribute . '.' . $type]; + } elseif (isset($this->message[$attribute][$type])) { + $msg = $this->message[$attribute][$type]; + } elseif (isset($this->message[$attribute])) { + $msg = $this->message[$attribute]; + } elseif (isset(self::$typeMsg[$type])) { + $msg = self::$typeMsg[$type]; + } else { + $msg = $title . '瑙勫垯閿欒'; + } + + if (is_string($msg) && 0 === strpos($msg, '{%')) { + $msg = Lang::get(substr($msg, 2, -1)); + } + + if (is_string($msg) && is_scalar($rule) && false !== strpos($msg, ':')) { + // 鍙橀噺鏇挎崲 + if (is_string($rule) && strpos($rule, ',')) { + $array = array_pad(explode(',', $rule), 3, ''); + } else { + $array = array_pad([], 3, ''); + } + $msg = str_replace( + [':attribute', ':rule', ':1', ':2', ':3'], + [$title, (string) $rule, $array[0], $array[1], $array[2]], + $msg); + } + return $msg; + } + + /** + * 鑾峰彇鏁版嵁楠岃瘉鐨勫満鏅 + * @access protected + * @param string $scene 楠岃瘉鍦烘櫙 + * @return array + */ + protected function getScene($scene = '') + { + if (empty($scene)) { + // 璇诲彇鎸囧畾鍦烘櫙 + $scene = $this->currentScene; + } + + if (!empty($scene) && isset($this->scene[$scene])) { + // 濡傛灉璁剧疆浜嗛獙璇侀傜敤鍦烘櫙 + $scene = $this->scene[$scene]; + if (is_string($scene)) { + $scene = explode(',', $scene); + } + } else { + $scene = []; + } + return $scene; + } + + public static function __callStatic($method, $params) + { + $class = self::make(); + if (method_exists($class, $method)) { + return call_user_func_array([$class, $method], $params); + } else { + throw new \BadMethodCallException('method not exists:' . __CLASS__ . '->' . $method); + } + } +} diff --git a/thinkphp/library/think/View.php b/thinkphp/library/think/View.php new file mode 100644 index 000000000..020e5789e --- /dev/null +++ b/thinkphp/library/think/View.php @@ -0,0 +1,236 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +class View +{ + // 瑙嗗浘瀹炰緥 + protected static $instance; + // 妯℃澘寮曟搸瀹炰緥 + public $engine; + // 妯℃澘鍙橀噺 + protected $data = []; + // 鐢ㄤ簬闈欐佽祴鍊肩殑妯℃澘鍙橀噺 + protected static $var = []; + // 瑙嗗浘杈撳嚭鏇挎崲 + protected $replace = []; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param array $engine 妯℃澘寮曟搸鍙傛暟 + * @param array $replace 瀛楃涓叉浛鎹㈠弬鏁 + */ + public function __construct($engine = [], $replace = []) + { + // 鍒濆鍖栨ā鏉垮紩鎿 + $this->engine((array) $engine); + // 鍩虹鏇挎崲瀛楃涓 + $request = Request::instance(); + $base = $request->root(); + $root = strpos($base, '.') ? ltrim(dirname($base), DS) : $base; + if ('' != $root) { + $root = '/' . ltrim($root, '/'); + } + $baseReplace = [ + '__ROOT__' => $root, + '__URL__' => $base . '/' . $request->module() . '/' . Loader::parseName($request->controller()), + '__STATIC__' => $root . '/static', + '__CSS__' => $root . '/static/css', + '__JS__' => $root . '/static/js', + ]; + $this->replace = array_merge($baseReplace, (array) $replace); + } + + /** + * 鍒濆鍖栬鍥 + * @access public + * @param array $engine 妯℃澘寮曟搸鍙傛暟 + * @param array $replace 瀛楃涓叉浛鎹㈠弬鏁 + * @return object + */ + public static function instance($engine = [], $replace = []) + { + if (is_null(self::$instance)) { + self::$instance = new self($engine, $replace); + } + return self::$instance; + } + + /** + * 妯℃澘鍙橀噺闈欐佽祴鍊 + * @access public + * @param mixed $name 鍙橀噺鍚 + * @param mixed $value 鍙橀噺鍊 + * @return void + */ + public static function share($name, $value = '') + { + if (is_array($name)) { + self::$var = array_merge(self::$var, $name); + } else { + self::$var[$name] = $value; + } + } + + /** + * 妯℃澘鍙橀噺璧嬪 + * @access public + * @param mixed $name 鍙橀噺鍚 + * @param mixed $value 鍙橀噺鍊 + * @return $this + */ + public function assign($name, $value = '') + { + if (is_array($name)) { + $this->data = array_merge($this->data, $name); + } else { + $this->data[$name] = $value; + } + return $this; + } + + /** + * 璁剧疆褰撳墠妯℃澘瑙f瀽鐨勫紩鎿 + * @access public + * @param array|string $options 寮曟搸鍙傛暟 + * @return $this + */ + public function engine($options = []) + { + if (is_string($options)) { + $type = $options; + $options = []; + } else { + $type = !empty($options['type']) ? $options['type'] : 'Think'; + } + + $class = false !== strpos($type, '\\') ? $type : '\\think\\view\\driver\\' . ucfirst($type); + if (isset($options['type'])) { + unset($options['type']); + } + $this->engine = new $class($options); + return $this; + } + + /** + * 閰嶇疆妯℃澘寮曟搸 + * @access private + * @param string|array $name 鍙傛暟鍚 + * @param mixed $value 鍙傛暟鍊 + * @return void + */ + public function config($name, $value = null) + { + $this->engine->config($name, $value); + return $this; + } + + /** + * 瑙f瀽鍜岃幏鍙栨ā鏉垮唴瀹 鐢ㄤ簬杈撳嚭 + * @param string $template 妯℃澘鏂囦欢鍚嶆垨鑰呭唴瀹 + * @param array $vars 妯℃澘杈撳嚭鍙橀噺 + * @param array $replace 鏇挎崲鍐呭 + * @param array $config 妯℃澘鍙傛暟 + * @param bool $renderContent 鏄惁娓叉煋鍐呭 + * @return string + * @throws Exception + */ + public function fetch($template = '', $vars = [], $replace = [], $config = [], $renderContent = false) + { + // 妯℃澘鍙橀噺 + $vars = array_merge(self::$var, $this->data, $vars); + + // 椤甸潰缂撳瓨 + ob_start(); + ob_implicit_flush(0); + + // 娓叉煋杈撳嚭 + $method = $renderContent ? 'display' : 'fetch'; + $this->engine->$method($template, $vars, $config); + + // 鑾峰彇骞舵竻绌虹紦瀛 + $content = ob_get_clean(); + // 鍐呭杩囨护鏍囩 + Hook::listen('view_filter', $content); + // 鍏佽鐢ㄦ埛鑷畾涔夋ā鏉跨殑瀛楃涓叉浛鎹 + $replace = array_merge($this->replace, $replace); + if (!empty($replace)) { + $content = strtr($content, $replace); + } + return $content; + } + + /** + * 瑙嗗浘鍐呭鏇挎崲 + * @access public + * @param string|array $content 琚浛鎹㈠唴瀹癸紙鏀寔鎵归噺鏇挎崲锛 + * @param string $replace 鏇挎崲鍐呭 + * @return $this + */ + public function replace($content, $replace = '') + { + if (is_array($content)) { + $this->replace = array_merge($this->replace, $content); + } else { + $this->replace[$content] = $replace; + } + return $this; + } + + /** + * 娓叉煋鍐呭杈撳嚭 + * @access public + * @param string $content 鍐呭 + * @param array $vars 妯℃澘杈撳嚭鍙橀噺 + * @param array $replace 鏇挎崲鍐呭 + * @param array $config 妯℃澘鍙傛暟 + * @return mixed + */ + public function display($content, $vars = [], $replace = [], $config = []) + { + return $this->fetch($content, $vars, $replace, $config, true); + } + + /** + * 妯℃澘鍙橀噺璧嬪 + * @access public + * @param string $name 鍙橀噺鍚 + * @param mixed $value 鍙橀噺鍊 + */ + public function __set($name, $value) + { + $this->data[$name] = $value; + } + + /** + * 鍙栧緱妯℃澘鏄剧ず鍙橀噺鐨勫 + * @access protected + * @param string $name 妯℃澘鍙橀噺 + * @return mixed + */ + public function __get($name) + { + return $this->data[$name]; + } + + /** + * 妫娴嬫ā鏉垮彉閲忔槸鍚﹁缃 + * @access public + * @param string $name 妯℃澘鍙橀噺鍚 + * @return bool + */ + public function __isset($name) + { + return isset($this->data[$name]); + } +} diff --git a/thinkphp/library/think/cache/Driver.php b/thinkphp/library/think/cache/Driver.php new file mode 100644 index 000000000..688507a81 --- /dev/null +++ b/thinkphp/library/think/cache/Driver.php @@ -0,0 +1,209 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache; + +/** + * 缂撳瓨鍩虹绫 + */ +abstract class Driver +{ + protected $handler = null; + protected $options = []; + protected $tag; + + /** + * 鍒ゆ柇缂撳瓨鏄惁瀛樺湪 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + abstract public function has($name); + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + abstract public function get($name, $default = false); + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param int $expire 鏈夋晥鏃堕棿 0涓烘案涔 + * @return boolean + */ + abstract public function set($name, $value, $expire = null); + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + abstract public function inc($name, $step = 1); + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + abstract public function dec($name, $step = 1); + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + abstract public function rm($name); + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return boolean + */ + abstract public function clear($tag = null); + + /** + * 鑾峰彇瀹為檯鐨勭紦瀛樻爣璇 + * @access public + * @param string $name 缂撳瓨鍚 + * @return string + */ + protected function getCacheKey($name) + { + return $this->options['prefix'] . $name; + } + + /** + * 璇诲彇缂撳瓨骞跺垹闄 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return mixed + */ + public function pull($name) + { + $result = $this->get($name, false); + if ($result) { + $this->rm($name); + return $result; + } else { + return; + } + } + + /** + * 濡傛灉涓嶅瓨鍦ㄥ垯鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param int $expire 鏈夋晥鏃堕棿 0涓烘案涔 + * @return mixed + */ + public function remember($name, $value, $expire = null) + { + if (!$this->has($name)) { + if ($value instanceof \Closure) { + $value = call_user_func($value); + } + $this->set($name, $value, $expire); + } else { + $value = $this->get($name); + } + return $value; + } + + /** + * 缂撳瓨鏍囩 + * @access public + * @param string $name 鏍囩鍚 + * @param string|array $keys 缂撳瓨鏍囪瘑 + * @param bool $overlay 鏄惁瑕嗙洊 + * @return $this + */ + public function tag($name, $keys = null, $overlay = false) + { + if (is_null($keys)) { + $this->tag = $name; + } else { + $key = 'tag_' . md5($name); + if (is_string($keys)) { + $keys = explode(',', $keys); + } + $keys = array_map([$this, 'getCacheKey'], $keys); + if ($overlay) { + $value = $keys; + } else { + $value = array_unique(array_merge($this->getTagItem($name), $keys)); + } + $this->set($key, implode(',', $value)); + } + return $this; + } + + /** + * 鏇存柊鏍囩 + * @access public + * @param string $name 缂撳瓨鏍囪瘑 + * @return void + */ + protected function setTagItem($name) + { + if ($this->tag) { + $key = 'tag_' . md5($this->tag); + $this->tag = null; + if ($this->has($key)) { + $value = $this->get($key); + $value .= ',' . $name; + } else { + $value = $name; + } + $this->set($key, $value); + } + } + + /** + * 鑾峰彇鏍囩鍖呭惈鐨勭紦瀛樻爣璇 + * @access public + * @param string $tag 缂撳瓨鏍囩 + * @return array + */ + protected function getTagItem($tag) + { + $key = 'tag_' . md5($tag); + $value = $this->get($key); + if ($value) { + return explode(',', $value); + } else { + return []; + } + } + + /** + * 杩斿洖鍙ユ焺瀵硅薄锛屽彲鎵ц鍏跺畠楂樼骇鏂规硶 + * + * @access public + * @return object + */ + public function handler() + { + return $this->handler; + } +} diff --git a/thinkphp/library/think/cache/driver/File.php b/thinkphp/library/think/cache/driver/File.php new file mode 100644 index 000000000..dba02c3e2 --- /dev/null +++ b/thinkphp/library/think/cache/driver/File.php @@ -0,0 +1,247 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +/** + * 鏂囦欢绫诲瀷缂撳瓨绫 + * @author liu21st + */ +class File extends Driver +{ + protected $options = [ + 'expire' => 0, + 'cache_subdir' => true, + 'prefix' => '', + 'path' => CACHE_PATH, + 'data_compress' => false, + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options + */ + public function __construct($options = []) + { + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + if (substr($this->options['path'], -1) != DS) { + $this->options['path'] .= DS; + } + $this->init(); + } + + /** + * 鍒濆鍖栨鏌 + * @access private + * @return boolean + */ + private function init() + { + // 鍒涘缓椤圭洰缂撳瓨鐩綍 + if (!is_dir($this->options['path'])) { + if (mkdir($this->options['path'], 0755, true)) { + return true; + } + } + return false; + } + + /** + * 鍙栧緱鍙橀噺鐨勫瓨鍌ㄦ枃浠跺悕 + * @access protected + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return string + */ + protected function getCacheKey($name) + { + $name = md5($name); + if ($this->options['cache_subdir']) { + // 浣跨敤瀛愮洰褰 + $name = substr($name, 0, 2) . DS . substr($name, 2); + } + if ($this->options['prefix']) { + $name = $this->options['prefix'] . DS . $name; + } + $filename = $this->options['path'] . $name . '.php'; + $dir = dirname($filename); + if (!is_dir($dir)) { + mkdir($dir, 0755, true); + } + return $filename; + } + + /** + * 鍒ゆ柇缂撳瓨鏄惁瀛樺湪 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + return $this->get($name) ? true : false; + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $filename = $this->getCacheKey($name); + if (!is_file($filename)) { + return $default; + } + $content = file_get_contents($filename); + if (false !== $content) { + $expire = (int) substr($content, 8, 12); + if (0 != $expire && $_SERVER['REQUEST_TIME'] > filemtime($filename) + $expire) { + //缂撳瓨杩囨湡鍒犻櫎缂撳瓨鏂囦欢 + $this->unlink($filename); + return $default; + } + $content = substr($content, 20, -3); + if ($this->options['data_compress'] && function_exists('gzcompress')) { + //鍚敤鏁版嵁鍘嬬缉 + $content = gzuncompress($content); + } + $content = unserialize($content); + return $content; + } else { + return $default; + } + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param int $expire 鏈夋晥鏃堕棿 0涓烘案涔 + * @return boolean + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + $filename = $this->getCacheKey($name); + if ($this->tag && !is_file($filename)) { + $first = true; + } + $data = serialize($value); + if ($this->options['data_compress'] && function_exists('gzcompress')) { + //鏁版嵁鍘嬬缉 + $data = gzcompress($data, 3); + } + $data = ""; + $result = file_put_contents($filename, $data); + if ($result) { + isset($first) && $this->setTagItem($filename); + clearstatcache(); + return true; + } else { + return false; + } + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + if ($this->has($name)) { + $value = $this->get($name) + $step; + } else { + $value = $step; + } + return $this->set($name, $value, 0) ? $value : false; + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + if ($this->has($name)) { + $value = $this->get($name) - $step; + } else { + $value = $step; + } + return $this->set($name, $value, 0) ? $value : false; + } + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + public function rm($name) + { + return $this->unlink($this->getCacheKey($name)); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return boolean + */ + public function clear($tag = null) + { + if ($tag) { + // 鎸囧畾鏍囩娓呴櫎 + $keys = $this->getTagItem($tag); + foreach ($keys as $key) { + $this->unlink($key); + } + $this->rm('tag_' . md5($tag)); + return true; + } + $files = (array) glob($this->options['path'] . ($this->options['prefix'] ? $this->options['prefix'] . DS : '') . '*'); + foreach ($files as $path) { + if (is_dir($path)) { + array_map('unlink', glob($path . '/*.php')); + } else { + unlink($path); + } + } + return true; + } + + /** + * 鍒ゆ柇鏂囦欢鏄惁瀛樺湪鍚庯紝鍒犻櫎 + * @param $path + * @return bool + * @author byron sampson + * @return boolean + */ + private function unlink($path) + { + return is_file($path) && unlink($path); + } + +} diff --git a/thinkphp/library/think/cache/driver/Lite.php b/thinkphp/library/think/cache/driver/Lite.php new file mode 100644 index 000000000..57727651b --- /dev/null +++ b/thinkphp/library/think/cache/driver/Lite.php @@ -0,0 +1,185 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +/** + * 鏂囦欢绫诲瀷缂撳瓨绫 + * @author liu21st + */ +class Lite extends Driver +{ + protected $options = [ + 'prefix' => '', + 'path' => '', + 'expire' => 0, // 绛変簬 10*365*24*3600锛10骞达級 + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * + * @param array $options + */ + public function __construct($options = []) + { + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + if (substr($this->options['path'], -1) != DS) { + $this->options['path'] .= DS; + } + + } + + /** + * 鍙栧緱鍙橀噺鐨勫瓨鍌ㄦ枃浠跺悕 + * @access protected + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return string + */ + protected function getCacheKey($name) + { + return $this->options['path'] . $this->options['prefix'] . md5($name) . '.php'; + } + + /** + * 鍒ゆ柇缂撳瓨鏄惁瀛樺湪 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return mixed + */ + public function has($name) + { + return $this->get($name) ? true : false; + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $filename = $this->getCacheKey($name); + if (is_file($filename)) { + // 鍒ゆ柇鏄惁杩囨湡 + $mtime = filemtime($filename); + if ($mtime < $_SERVER['REQUEST_TIME']) { + // 娓呴櫎宸茬粡杩囨湡鐨勬枃浠 + unlink($filename); + return $default; + } + return include $filename; + } else { + return $default; + } + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param int $expire 鏈夋晥鏃堕棿 0涓烘案涔 + * @return bool + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + // 妯℃嫙姘镐箙 + if (0 === $expire) { + $expire = 10 * 365 * 24 * 3600; + } + $filename = $this->getCacheKey($name); + if ($this->tag && !is_file($filename)) { + $first = true; + } + $ret = file_put_contents($filename, ("setTagItem($filename); + touch($filename, $_SERVER['REQUEST_TIME'] + $expire); + } + return $ret; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + if ($this->has($name)) { + $value = $this->get($name) + $step; + } else { + $value = $step; + } + return $this->set($name, $value, 0) ? $value : false; + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + if ($this->has($name)) { + $value = $this->get($name) - $step; + } else { + $value = $step; + } + return $this->set($name, $value, 0) ? $value : false; + } + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + public function rm($name) + { + return unlink($this->getCacheKey($name)); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return bool + */ + public function clear($tag = null) + { + if ($tag) { + // 鎸囧畾鏍囩娓呴櫎 + $keys = $this->getTagItem($tag); + foreach ($keys as $key) { + unlink($key); + } + $this->rm('tag_' . md5($tag)); + return true; + } + array_map("unlink", glob($this->options['path'] . ($this->options['prefix'] ? $this->options['prefix'] . DS : '') . '*.php')); + } +} diff --git a/thinkphp/library/think/cache/driver/Memcache.php b/thinkphp/library/think/cache/driver/Memcache.php new file mode 100644 index 000000000..d41939d82 --- /dev/null +++ b/thinkphp/library/think/cache/driver/Memcache.php @@ -0,0 +1,171 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +class Memcache extends Driver +{ + protected $options = [ + 'host' => '127.0.0.1', + 'port' => 11211, + 'expire' => 0, + 'timeout' => 0, // 瓒呮椂鏃堕棿锛堝崟浣嶏細姣锛 + 'persistent' => true, + 'prefix' => '', + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options 缂撳瓨鍙傛暟 + * @access public + * @throws \BadFunctionCallException + */ + public function __construct($options = []) + { + if (!extension_loaded('memcache')) { + throw new \BadFunctionCallException('not support: memcache'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + $this->handler = new \Memcache; + // 鏀寔闆嗙兢 + $hosts = explode(',', $this->options['host']); + $ports = explode(',', $this->options['port']); + if (empty($ports[0])) { + $ports[0] = 11211; + } + // 寤虹珛杩炴帴 + foreach ((array) $hosts as $i => $host) { + $port = isset($ports[$i]) ? $ports[$i] : $ports[0]; + $this->options['timeout'] > 0 ? + $this->handler->addServer($host, $port, $this->options['persistent'], 1, $this->options['timeout']) : + $this->handler->addServer($host, $port, $this->options['persistent'], 1); + } + } + + /** + * 鍒ゆ柇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + $key = $this->getCacheKey($name); + return $this->handler->get($key) ? true : false; + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $result = $this->handler->get($this->getCacheKey($name)); + return false !== $result ? $result : $default; + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param integer $expire 鏈夋晥鏃堕棿锛堢锛 + * @return bool + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + if ($this->tag && !$this->has($name)) { + $first = true; + } + $key = $this->getCacheKey($name); + if ($this->handler->set($key, $value, 0, $expire)) { + isset($first) && $this->setTagItem($key); + return true; + } + return false; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + $key = $this->getCacheKey($name); + return $this->handler->increment($key, $step); + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + $key = $this->getCacheKey($name); + $value = $this->handler->get($key) - $step; + $res = $this->handler->set($key, $value); + if (!$res) { + return false; + } else { + return $value; + } + } + + /** + * 鍒犻櫎缂撳瓨 + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param bool|false $ttl + * @return bool + */ + public function rm($name, $ttl = false) + { + $key = $this->getCacheKey($name); + return false === $ttl ? + $this->handler->delete($key) : + $this->handler->delete($key, $ttl); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return bool + */ + public function clear($tag = null) + { + if ($tag) { + // 鎸囧畾鏍囩娓呴櫎 + $keys = $this->getTagItem($tag); + foreach ($keys as $key) { + $this->handler->delete($key); + } + $this->rm('tag_' . md5($tag)); + return true; + } + return $this->handler->flush(); + } +} diff --git a/thinkphp/library/think/cache/driver/Memcached.php b/thinkphp/library/think/cache/driver/Memcached.php new file mode 100644 index 000000000..35fafd07e --- /dev/null +++ b/thinkphp/library/think/cache/driver/Memcached.php @@ -0,0 +1,181 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +class Memcached extends Driver +{ + protected $options = [ + 'host' => '127.0.0.1', + 'port' => 11211, + 'expire' => 0, + 'timeout' => 0, // 瓒呮椂鏃堕棿锛堝崟浣嶏細姣锛 + 'prefix' => '', + 'username' => '', //璐﹀彿 + 'password' => '', //瀵嗙爜 + 'option' => [], + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options 缂撳瓨鍙傛暟 + * @access public + */ + public function __construct($options = []) + { + if (!extension_loaded('memcached')) { + throw new \BadFunctionCallException('not support: memcached'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + $this->handler = new \Memcached; + if (!empty($this->options['option'])) { + $this->handler->setOptions($this->options['option']); + } + // 璁剧疆杩炴帴瓒呮椂鏃堕棿锛堝崟浣嶏細姣锛 + if ($this->options['timeout'] > 0) { + $this->handler->setOption(\Memcached::OPT_CONNECT_TIMEOUT, $this->options['timeout']); + } + // 鏀寔闆嗙兢 + $hosts = explode(',', $this->options['host']); + $ports = explode(',', $this->options['port']); + if (empty($ports[0])) { + $ports[0] = 11211; + } + // 寤虹珛杩炴帴 + $servers = []; + foreach ((array) $hosts as $i => $host) { + $servers[] = [$host, (isset($ports[$i]) ? $ports[$i] : $ports[0]), 1]; + } + $this->handler->addServers($servers); + if ('' != $this->options['username']) { + $this->handler->setOption(\Memcached::OPT_BINARY_PROTOCOL, true); + $this->handler->setSaslAuthData($this->options['username'], $this->options['password']); + } + } + + /** + * 鍒ゆ柇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + $key = $this->getCacheKey($name); + return $this->handler->get($key) ? true : false; + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $result = $this->handler->get($this->getCacheKey($name)); + return false !== $result ? $result : $default; + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param integer $expire 鏈夋晥鏃堕棿锛堢锛 + * @return bool + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + if ($this->tag && !$this->has($name)) { + $first = true; + } + $key = $this->getCacheKey($name); + $expire = 0 == $expire ? 0 : $_SERVER['REQUEST_TIME'] + $expire; + if ($this->handler->set($key, $value, $expire)) { + isset($first) && $this->setTagItem($key); + return true; + } + return false; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + $key = $this->getCacheKey($name); + return $this->handler->increment($key, $step); + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + $key = $this->getCacheKey($name); + $value = $this->handler->get($key) - $step; + $res = $this->handler->set($key, $value); + if (!$res) { + return false; + } else { + return $value; + } + } + + /** + * 鍒犻櫎缂撳瓨 + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param bool|false $ttl + * @return bool + */ + public function rm($name, $ttl = false) + { + $key = $this->getCacheKey($name); + return false === $ttl ? + $this->handler->delete($key) : + $this->handler->delete($key, $ttl); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return bool + */ + public function clear($tag = null) + { + if ($tag) { + // 鎸囧畾鏍囩娓呴櫎 + $keys = $this->getTagItem($tag); + $this->handler->deleteMulti($keys); + $this->rm('tag_' . md5($tag)); + return true; + } + return $this->handler->flush(); + } +} diff --git a/thinkphp/library/think/cache/driver/Redis.php b/thinkphp/library/think/cache/driver/Redis.php new file mode 100644 index 000000000..360f515a1 --- /dev/null +++ b/thinkphp/library/think/cache/driver/Redis.php @@ -0,0 +1,176 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +/** + * Redis缂撳瓨椹卞姩锛岄傚悎鍗曟満閮ㄧ讲銆佹湁鍓嶇浠g悊瀹炵幇楂樺彲鐢ㄧ殑鍦烘櫙锛屾ц兘鏈濂 + * 鏈夐渶瑕佸湪涓氬姟灞傚疄鐜拌鍐欏垎绂汇佹垨鑰呬娇鐢≧edisCluster鐨勯渶姹傦紝璇蜂娇鐢≧edisd椹卞姩 + * + * 瑕佹眰瀹夎phpredis鎵╁睍锛歨ttps://github.com/nicolasff/phpredis + * @author 灏樼紭 <130775@qq.com> + */ +class Redis extends Driver +{ + protected $options = [ + 'host' => '127.0.0.1', + 'port' => 6379, + 'password' => '', + 'select' => 0, + 'timeout' => 0, + 'expire' => 0, + 'persistent' => false, + 'prefix' => '', + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options 缂撳瓨鍙傛暟 + * @access public + */ + public function __construct($options = []) + { + if (!extension_loaded('redis')) { + throw new \BadFunctionCallException('not support: redis'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + $func = $this->options['persistent'] ? 'pconnect' : 'connect'; + $this->handler = new \Redis; + $this->handler->$func($this->options['host'], $this->options['port'], $this->options['timeout']); + + if ('' != $this->options['password']) { + $this->handler->auth($this->options['password']); + } + + if (0 != $this->options['select']) { + $this->handler->select($this->options['select']); + } + } + + /** + * 鍒ゆ柇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + return $this->handler->get($this->getCacheKey($name)) ? true : false; + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $value = $this->handler->get($this->getCacheKey($name)); + if (is_null($value)) { + return $default; + } + $jsonData = json_decode($value, true); + // 妫娴嬫槸鍚︿负JSON鏁版嵁 true 杩斿洖JSON瑙f瀽鏁扮粍, false杩斿洖婧愭暟鎹 byron sampson + return (null === $jsonData) ? $value : $jsonData; + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param integer $expire 鏈夋晥鏃堕棿锛堢锛 + * @return boolean + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + if ($this->tag && !$this->has($name)) { + $first = true; + } + $key = $this->getCacheKey($name); + //瀵规暟缁/瀵硅薄鏁版嵁杩涜缂撳瓨澶勭悊锛屼繚璇佹暟鎹畬鏁存 byron sampson + $value = (is_object($value) || is_array($value)) ? json_encode($value) : $value; + if (is_int($expire) && $expire) { + $result = $this->handler->setex($key, $expire, $value); + } else { + $result = $this->handler->set($key, $value); + } + isset($first) && $this->setTagItem($key); + return $result; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + $key = $this->getCacheKey($name); + return $this->handler->incrby($key, $step); + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + $key = $this->getCacheKey($name); + return $this->handler->decrby($key, $step); + } + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + public function rm($name) + { + return $this->handler->delete($this->getCacheKey($name)); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return boolean + */ + public function clear($tag = null) + { + if ($tag) { + // 鎸囧畾鏍囩娓呴櫎 + $keys = $this->getTagItem($tag); + foreach ($keys as $key) { + $this->handler->delete($key); + } + $this->rm('tag_' . md5($tag)); + return true; + } + return $this->handler->flushDB(); + } + +} diff --git a/thinkphp/library/think/cache/driver/Sqlite.php b/thinkphp/library/think/cache/driver/Sqlite.php new file mode 100644 index 000000000..76c592d80 --- /dev/null +++ b/thinkphp/library/think/cache/driver/Sqlite.php @@ -0,0 +1,195 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +/** + * Sqlite缂撳瓨椹卞姩 + * @author liu21st + */ +class Sqlite extends Driver +{ + protected $options = [ + 'db' => ':memory:', + 'table' => 'sharedmemory', + 'prefix' => '', + 'expire' => 0, + 'persistent' => false, + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options 缂撳瓨鍙傛暟 + * @throws \BadFunctionCallException + * @access public + */ + public function __construct($options = []) + { + if (!extension_loaded('sqlite')) { + throw new \BadFunctionCallException('not support: sqlite'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + $func = $this->options['persistent'] ? 'sqlite_popen' : 'sqlite_open'; + $this->handler = $func($this->options['db']); + } + + /** + * 鑾峰彇瀹為檯鐨勭紦瀛樻爣璇 + * @access public + * @param string $name 缂撳瓨鍚 + * @return string + */ + protected function getCacheKey($name) + { + return $this->options['prefix'] . sqlite_escape_string($name); + } + + /** + * 鍒ゆ柇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + $name = $this->getCacheKey($name); + $sql = 'SELECT value FROM ' . $this->options['table'] . ' WHERE var=\'' . $name . '\' AND (expire=0 OR expire >' . $_SERVER['REQUEST_TIME'] . ') LIMIT 1'; + $result = sqlite_query($this->handler, $sql); + return sqlite_num_rows($result); + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $name = $this->getCacheKey($name); + $sql = 'SELECT value FROM ' . $this->options['table'] . ' WHERE var=\'' . $name . '\' AND (expire=0 OR expire >' . $_SERVER['REQUEST_TIME'] . ') LIMIT 1'; + $result = sqlite_query($this->handler, $sql); + if (sqlite_num_rows($result)) { + $content = sqlite_fetch_single($result); + if (function_exists('gzcompress')) { + //鍚敤鏁版嵁鍘嬬缉 + $content = gzuncompress($content); + } + return unserialize($content); + } + return $default; + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param integer $expire 鏈夋晥鏃堕棿锛堢锛 + * @return boolean + */ + public function set($name, $value, $expire = null) + { + $name = $this->getCacheKey($name); + $value = sqlite_escape_string(serialize($value)); + if (is_null($expire)) { + $expire = $this->options['expire']; + } + $expire = (0 == $expire) ? 0 : ($_SERVER['REQUEST_TIME'] + $expire); //缂撳瓨鏈夋晥鏈熶负0琛ㄧず姘镐箙缂撳瓨 + if (function_exists('gzcompress')) { + //鏁版嵁鍘嬬缉 + $value = gzcompress($value, 3); + } + if ($this->tag) { + $tag = $this->tag; + $this->tag = null; + } else { + $tag = ''; + } + $sql = 'REPLACE INTO ' . $this->options['table'] . ' (var, value, expire, tag) VALUES (\'' . $name . '\', \'' . $value . '\', \'' . $expire . '\', \'' . $tag . '\')'; + if (sqlite_query($this->handler, $sql)) { + return true; + } + return false; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + if ($this->has($name)) { + $value = $this->get($name) + $step; + } else { + $value = $step; + } + return $this->set($name, $value, 0) ? $value : false; + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + if ($this->has($name)) { + $value = $this->get($name) - $step; + } else { + $value = $step; + } + return $this->set($name, $value, 0) ? $value : false; + } + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + public function rm($name) + { + $name = $this->getCacheKey($name); + $sql = 'DELETE FROM ' . $this->options['table'] . ' WHERE var=\'' . $name . '\''; + sqlite_query($this->handler, $sql); + return true; + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return boolean + */ + public function clear($tag = null) + { + if ($tag) { + $name = sqlite_escape_string($tag); + $sql = 'DELETE FROM ' . $this->options['table'] . ' WHERE tag=\'' . $name . '\''; + sqlite_query($this->handler, $sql); + return true; + } + $sql = 'DELETE FROM ' . $this->options['table']; + sqlite_query($this->handler, $sql); + return true; + } +} diff --git a/thinkphp/library/think/cache/driver/Wincache.php b/thinkphp/library/think/cache/driver/Wincache.php new file mode 100644 index 000000000..5be8d0df2 --- /dev/null +++ b/thinkphp/library/think/cache/driver/Wincache.php @@ -0,0 +1,149 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +/** + * Wincache缂撳瓨椹卞姩 + * @author liu21st + */ +class Wincache extends Driver +{ + protected $options = [ + 'prefix' => '', + 'expire' => 0, + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options 缂撳瓨鍙傛暟 + * @throws \BadFunctionCallException + * @access public + */ + public function __construct($options = []) + { + if (!function_exists('wincache_ucache_info')) { + throw new \BadFunctionCallException('not support: WinCache'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + } + + /** + * 鍒ゆ柇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + $key = $this->getCacheKey($name); + return wincache_ucache_exists($key); + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $key = $this->getCacheKey($name); + return wincache_ucache_exists($key) ? wincache_ucache_get($key) : $default; + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param integer $expire 鏈夋晥鏃堕棿锛堢锛 + * @return boolean + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + $key = $this->getCacheKey($name); + if ($this->tag && !$this->has($name)) { + $first = true; + } + if (wincache_ucache_set($key, $value, $expire)) { + isset($first) && $this->setTagItem($key); + return true; + } + return false; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + $key = $this->getCacheKey($name); + return wincache_ucache_inc($key, $step); + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + $key = $this->getCacheKey($name); + return wincache_ucache_dec($key, $step); + } + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + public function rm($name) + { + return wincache_ucache_delete($this->getCacheKey($name)); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return boolean + */ + public function clear($tag = null) + { + if ($tag) { + $keys = $this->getTagItem($tag); + foreach ($keys as $key) { + wincache_ucache_delete($key); + } + $this->rm('tag_' . md5($tag)); + return true; + } else { + return wincache_ucache_clear(); + } + } + +} diff --git a/thinkphp/library/think/cache/driver/Xcache.php b/thinkphp/library/think/cache/driver/Xcache.php new file mode 100644 index 000000000..317a4ee30 --- /dev/null +++ b/thinkphp/library/think/cache/driver/Xcache.php @@ -0,0 +1,152 @@ + +// +---------------------------------------------------------------------- + +namespace think\cache\driver; + +use think\cache\Driver; + +/** + * Xcache缂撳瓨椹卞姩 + * @author liu21st + */ +class Xcache extends Driver +{ + protected $options = [ + 'prefix' => '', + 'expire' => 0, + ]; + + /** + * 鏋舵瀯鍑芥暟 + * @param array $options 缂撳瓨鍙傛暟 + * @access public + * @throws \BadFunctionCallException + */ + public function __construct($options = []) + { + if (!function_exists('xcache_info')) { + throw new \BadFunctionCallException('not support: Xcache'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + } + + /** + * 鍒ゆ柇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return bool + */ + public function has($name) + { + $key = $this->getCacheKey($name); + return xcache_isset($key); + } + + /** + * 璇诲彇缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function get($name, $default = false) + { + $key = $this->getCacheKey($name); + return xcache_isset($key) ? xcache_get($key) : $default; + } + + /** + * 鍐欏叆缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param mixed $value 瀛樺偍鏁版嵁 + * @param integer $expire 鏈夋晥鏃堕棿锛堢锛 + * @return boolean + */ + public function set($name, $value, $expire = null) + { + if (is_null($expire)) { + $expire = $this->options['expire']; + } + if ($this->tag && !$this->has($name)) { + $first = true; + } + $key = $this->getCacheKey($name); + if (xcache_set($key, $value, $expire)) { + isset($first) && $this->setTagItem($key); + return true; + } + return false; + } + + /** + * 鑷缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function inc($name, $step = 1) + { + $key = $this->getCacheKey($name); + return xcache_inc($key, $step); + } + + /** + * 鑷噺缂撳瓨锛堥拡瀵规暟鍊肩紦瀛橈級 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @param int $step 姝ラ暱 + * @return false|int + */ + public function dec($name, $step = 1) + { + $key = $this->getCacheKey($name); + return xcache_dec($key, $step); + } + + /** + * 鍒犻櫎缂撳瓨 + * @access public + * @param string $name 缂撳瓨鍙橀噺鍚 + * @return boolean + */ + public function rm($name) + { + return xcache_unset($this->getCacheKey($name)); + } + + /** + * 娓呴櫎缂撳瓨 + * @access public + * @param string $tag 鏍囩鍚 + * @return boolean + */ + public function clear($tag = null) + { + if ($tag) { + // 鎸囧畾鏍囩娓呴櫎 + $keys = $this->getTagItem($tag); + foreach ($keys as $key) { + xcache_unset($key); + } + $this->rm('tag_' . md5($tag)); + return true; + } + if (function_exists('xcache_unset_by_prefix')) { + return xcache_unset_by_prefix($this->options['prefix']); + } else { + return false; + } + } +} diff --git a/thinkphp/library/think/config/driver/Ini.php b/thinkphp/library/think/config/driver/Ini.php new file mode 100644 index 000000000..a223a5785 --- /dev/null +++ b/thinkphp/library/think/config/driver/Ini.php @@ -0,0 +1,24 @@ + +// +---------------------------------------------------------------------- + +namespace think\config\driver; + +class Ini +{ + public function parse($config) + { + if (is_file($config)) { + return parse_ini_file($config, true); + } else { + return parse_ini_string($config, true); + } + } +} diff --git a/thinkphp/library/think/config/driver/Json.php b/thinkphp/library/think/config/driver/Json.php new file mode 100644 index 000000000..557f75fe0 --- /dev/null +++ b/thinkphp/library/think/config/driver/Json.php @@ -0,0 +1,24 @@ + +// +---------------------------------------------------------------------- + +namespace think\config\driver; + +class Json +{ + public function parse($config) + { + if (is_file($config)) { + $config = file_get_contents($config); + } + $result = json_decode($config, true); + return $result; + } +} diff --git a/thinkphp/library/think/config/driver/Xml.php b/thinkphp/library/think/config/driver/Xml.php new file mode 100644 index 000000000..b573a5627 --- /dev/null +++ b/thinkphp/library/think/config/driver/Xml.php @@ -0,0 +1,31 @@ + +// +---------------------------------------------------------------------- + +namespace think\config\driver; + +class Xml +{ + public function parse($config) + { + if (is_file($config)) { + $content = simplexml_load_file($config); + } else { + $content = simplexml_load_string($config); + } + $result = (array) $content; + foreach ($result as $key => $val) { + if (is_object($val)) { + $result[$key] = (array) $val; + } + } + return $result; + } +} diff --git a/thinkphp/library/think/console/Command.php b/thinkphp/library/think/console/Command.php new file mode 100644 index 000000000..d0caad2f6 --- /dev/null +++ b/thinkphp/library/think/console/Command.php @@ -0,0 +1,470 @@ + +// +---------------------------------------------------------------------- + +namespace think\console; + +use think\Console; +use think\console\input\Argument; +use think\console\input\Definition; +use think\console\input\Option; + +class Command +{ + + /** @var Console */ + private $console; + private $name; + private $aliases = []; + private $definition; + private $help; + private $description; + private $ignoreValidationErrors = false; + private $consoleDefinitionMerged = false; + private $consoleDefinitionMergedWithArgs = false; + private $code; + private $synopsis = []; + private $usages = []; + + /** @var Input */ + protected $input; + + /** @var Output */ + protected $output; + + /** + * 鏋勯犳柟娉 + * @param string|null $name 鍛戒护鍚嶇О,濡傛灉娌℃湁璁剧疆鍒欐瘮濡傚湪 configure() 閲岃缃 + * @throws \LogicException + * @api + */ + public function __construct($name = null) + { + $this->definition = new Definition(); + + if (null !== $name) { + $this->setName($name); + } + + $this->configure(); + + if (!$this->name) { + throw new \LogicException(sprintf('The command defined in "%s" cannot have an empty name.', get_class($this))); + } + } + + /** + * 蹇界暐楠岃瘉閿欒 + */ + public function ignoreValidationErrors() + { + $this->ignoreValidationErrors = true; + } + + /** + * 璁剧疆鎺у埗鍙 + * @param Console $console + */ + public function setConsole(Console $console = null) + { + $this->console = $console; + } + + /** + * 鑾峰彇鎺у埗鍙 + * @return Console + * @api + */ + public function getConsole() + { + return $this->console; + } + + /** + * 鏄惁鏈夋晥 + * @return bool + */ + public function isEnabled() + { + return true; + } + + /** + * 閰嶇疆鎸囦护 + */ + protected function configure() + { + } + + /** + * 鎵ц鎸囦护 + * @param Input $input + * @param Output $output + * @return null|int + * @throws \LogicException + * @see setCode() + */ + protected function execute(Input $input, Output $output) + { + throw new \LogicException('You must override the execute() method in the concrete command class.'); + } + + /** + * 鐢ㄦ埛楠岃瘉 + * @param Input $input + * @param Output $output + */ + protected function interact(Input $input, Output $output) + { + } + + /** + * 鍒濆鍖 + * @param Input $input An InputInterface instance + * @param Output $output An OutputInterface instance + */ + protected function initialize(Input $input, Output $output) + { + } + + /** + * 鎵ц + * @param Input $input + * @param Output $output + * @return int + * @throws \Exception + * @see setCode() + * @see execute() + */ + public function run(Input $input, Output $output) + { + $this->input = $input; + $this->output = $output; + + $this->getSynopsis(true); + $this->getSynopsis(false); + + $this->mergeConsoleDefinition(); + + try { + $input->bind($this->definition); + } catch (\Exception $e) { + if (!$this->ignoreValidationErrors) { + throw $e; + } + } + + $this->initialize($input, $output); + + if ($input->isInteractive()) { + $this->interact($input, $output); + } + + $input->validate(); + + if ($this->code) { + $statusCode = call_user_func($this->code, $input, $output); + } else { + $statusCode = $this->execute($input, $output); + } + + return is_numeric($statusCode) ? (int) $statusCode : 0; + } + + /** + * 璁剧疆鎵ц浠g爜 + * @param callable $code callable(InputInterface $input, OutputInterface $output) + * @return Command + * @throws \InvalidArgumentException + * @see execute() + */ + public function setCode(callable $code) + { + if (!is_callable($code)) { + throw new \InvalidArgumentException('Invalid callable provided to Command::setCode.'); + } + + if (PHP_VERSION_ID >= 50400 && $code instanceof \Closure) { + $r = new \ReflectionFunction($code); + if (null === $r->getClosureThis()) { + $code = \Closure::bind($code, $this); + } + } + + $this->code = $code; + + return $this; + } + + /** + * 鍚堝苟鍙傛暟瀹氫箟 + * @param bool $mergeArgs + */ + public function mergeConsoleDefinition($mergeArgs = true) + { + if (null === $this->console + || (true === $this->consoleDefinitionMerged + && ($this->consoleDefinitionMergedWithArgs || !$mergeArgs)) + ) { + return; + } + + if ($mergeArgs) { + $currentArguments = $this->definition->getArguments(); + $this->definition->setArguments($this->console->getDefinition()->getArguments()); + $this->definition->addArguments($currentArguments); + } + + $this->definition->addOptions($this->console->getDefinition()->getOptions()); + + $this->consoleDefinitionMerged = true; + if ($mergeArgs) { + $this->consoleDefinitionMergedWithArgs = true; + } + } + + /** + * 璁剧疆鍙傛暟瀹氫箟 + * @param array|Definition $definition + * @return Command + * @api + */ + public function setDefinition($definition) + { + if ($definition instanceof Definition) { + $this->definition = $definition; + } else { + $this->definition->setDefinition($definition); + } + + $this->consoleDefinitionMerged = false; + + return $this; + } + + /** + * 鑾峰彇鍙傛暟瀹氫箟 + * @return Definition + * @api + */ + public function getDefinition() + { + return $this->definition; + } + + /** + * 鑾峰彇褰撳墠鎸囦护鐨勫弬鏁板畾涔 + * @return Definition + */ + public function getNativeDefinition() + { + return $this->getDefinition(); + } + + /** + * 娣诲姞鍙傛暟 + * @param string $name 鍚嶇О + * @param int $mode 绫诲瀷 + * @param string $description 鎻忚堪 + * @param mixed $default 榛樿鍊 + * @return Command + */ + public function addArgument($name, $mode = null, $description = '', $default = null) + { + $this->definition->addArgument(new Argument($name, $mode, $description, $default)); + + return $this; + } + + /** + * 娣诲姞閫夐」 + * @param string $name 閫夐」鍚嶇О + * @param string $shortcut 鍒悕 + * @param int $mode 绫诲瀷 + * @param string $description 鎻忚堪 + * @param mixed $default 榛樿鍊 + * @return Command + */ + public function addOption($name, $shortcut = null, $mode = null, $description = '', $default = null) + { + $this->definition->addOption(new Option($name, $shortcut, $mode, $description, $default)); + + return $this; + } + + /** + * 璁剧疆鎸囦护鍚嶇О + * @param string $name + * @return Command + * @throws \InvalidArgumentException + */ + public function setName($name) + { + $this->validateName($name); + + $this->name = $name; + + return $this; + } + + /** + * 鑾峰彇鎸囦护鍚嶇О + * @return string + */ + public function getName() + { + return $this->name; + } + + /** + * 璁剧疆鎻忚堪 + * @param string $description + * @return Command + */ + public function setDescription($description) + { + $this->description = $description; + + return $this; + } + + /** + * 鑾峰彇鎻忚堪 + * @return string + */ + public function getDescription() + { + return $this->description; + } + + /** + * 璁剧疆甯姪淇℃伅 + * @param string $help + * @return Command + */ + public function setHelp($help) + { + $this->help = $help; + + return $this; + } + + /** + * 鑾峰彇甯姪淇℃伅 + * @return string + */ + public function getHelp() + { + return $this->help; + } + + /** + * 鎻忚堪淇℃伅 + * @return string + */ + public function getProcessedHelp() + { + $name = $this->name; + + $placeholders = [ + '%command.name%', + '%command.full_name%', + ]; + $replacements = [ + $name, + $_SERVER['PHP_SELF'] . ' ' . $name, + ]; + + return str_replace($placeholders, $replacements, $this->getHelp()); + } + + /** + * 璁剧疆鍒悕 + * @param string[] $aliases + * @return Command + * @throws \InvalidArgumentException + */ + public function setAliases($aliases) + { + if (!is_array($aliases) && !$aliases instanceof \Traversable) { + throw new \InvalidArgumentException('$aliases must be an array or an instance of \Traversable'); + } + + foreach ($aliases as $alias) { + $this->validateName($alias); + } + + $this->aliases = $aliases; + + return $this; + } + + /** + * 鑾峰彇鍒悕 + * @return array + */ + public function getAliases() + { + return $this->aliases; + } + + /** + * 鑾峰彇绠浠 + * @param bool $short 鏄惁绠鍗曠殑 + * @return string + */ + public function getSynopsis($short = false) + { + $key = $short ? 'short' : 'long'; + + if (!isset($this->synopsis[$key])) { + $this->synopsis[$key] = trim(sprintf('%s %s', $this->name, $this->definition->getSynopsis($short))); + } + + return $this->synopsis[$key]; + } + + /** + * 娣诲姞鐢ㄦ硶浠嬬粛 + * @param string $usage + * @return $this + */ + public function addUsage($usage) + { + if (0 !== strpos($usage, $this->name)) { + $usage = sprintf('%s %s', $this->name, $usage); + } + + $this->usages[] = $usage; + + return $this; + } + + /** + * 鑾峰彇鐢ㄦ硶浠嬬粛 + * @return array + */ + public function getUsages() + { + return $this->usages; + } + + /** + * 楠岃瘉鎸囦护鍚嶇О + * @param string $name + * @throws \InvalidArgumentException + */ + private function validateName($name) + { + if (!preg_match('/^[^\:]++(\:[^\:]++)*$/', $name)) { + throw new \InvalidArgumentException(sprintf('Command name "%s" is invalid.', $name)); + } + } +} diff --git a/thinkphp/library/think/console/Input.php b/thinkphp/library/think/console/Input.php new file mode 100644 index 000000000..2482dfdc0 --- /dev/null +++ b/thinkphp/library/think/console/Input.php @@ -0,0 +1,464 @@ + +// +---------------------------------------------------------------------- + +namespace think\console; + +use think\console\input\Argument; +use think\console\input\Definition; +use think\console\input\Option; + +class Input +{ + + /** + * @var Definition + */ + protected $definition; + + /** + * @var Option[] + */ + protected $options = []; + + /** + * @var Argument[] + */ + protected $arguments = []; + + protected $interactive = true; + + private $tokens; + private $parsed; + + public function __construct($argv = null) + { + if (null === $argv) { + $argv = $_SERVER['argv']; + // 鍘婚櫎鍛戒护鍚 + array_shift($argv); + } + + $this->tokens = $argv; + + $this->definition = new Definition(); + } + + protected function setTokens(array $tokens) + { + $this->tokens = $tokens; + } + + /** + * 缁戝畾瀹炰緥 + * @param Definition $definition A InputDefinition instance + */ + public function bind(Definition $definition) + { + $this->arguments = []; + $this->options = []; + $this->definition = $definition; + + $this->parse(); + } + + /** + * 瑙f瀽鍙傛暟 + */ + protected function parse() + { + $parseOptions = true; + $this->parsed = $this->tokens; + while (null !== $token = array_shift($this->parsed)) { + if ($parseOptions && '' == $token) { + $this->parseArgument($token); + } elseif ($parseOptions && '--' == $token) { + $parseOptions = false; + } elseif ($parseOptions && 0 === strpos($token, '--')) { + $this->parseLongOption($token); + } elseif ($parseOptions && '-' === $token[0] && '-' !== $token) { + $this->parseShortOption($token); + } else { + $this->parseArgument($token); + } + } + } + + /** + * 瑙f瀽鐭夐」 + * @param string $token 褰撳墠鐨勬寚浠. + */ + private function parseShortOption($token) + { + $name = substr($token, 1); + + if (strlen($name) > 1) { + if ($this->definition->hasShortcut($name[0]) + && $this->definition->getOptionForShortcut($name[0])->acceptValue() + ) { + $this->addShortOption($name[0], substr($name, 1)); + } else { + $this->parseShortOptionSet($name); + } + } else { + $this->addShortOption($name, null); + } + } + + /** + * 瑙f瀽鐭夐」 + * @param string $name 褰撳墠鎸囦护 + * @throws \RuntimeException + */ + private function parseShortOptionSet($name) + { + $len = strlen($name); + for ($i = 0; $i < $len; ++$i) { + if (!$this->definition->hasShortcut($name[$i])) { + throw new \RuntimeException(sprintf('The "-%s" option does not exist.', $name[$i])); + } + + $option = $this->definition->getOptionForShortcut($name[$i]); + if ($option->acceptValue()) { + $this->addLongOption($option->getName(), $i === $len - 1 ? null : substr($name, $i + 1)); + + break; + } else { + $this->addLongOption($option->getName(), null); + } + } + } + + /** + * 瑙f瀽瀹屾暣閫夐」 + * @param string $token 褰撳墠鎸囦护 + */ + private function parseLongOption($token) + { + $name = substr($token, 2); + + if (false !== $pos = strpos($name, '=')) { + $this->addLongOption(substr($name, 0, $pos), substr($name, $pos + 1)); + } else { + $this->addLongOption($name, null); + } + } + + /** + * 瑙f瀽鍙傛暟 + * @param string $token 褰撳墠鎸囦护 + * @throws \RuntimeException + */ + private function parseArgument($token) + { + $c = count($this->arguments); + + if ($this->definition->hasArgument($c)) { + $arg = $this->definition->getArgument($c); + + $this->arguments[$arg->getName()] = $arg->isArray() ? [$token] : $token; + + } elseif ($this->definition->hasArgument($c - 1) && $this->definition->getArgument($c - 1)->isArray()) { + $arg = $this->definition->getArgument($c - 1); + + $this->arguments[$arg->getName()][] = $token; + } else { + throw new \RuntimeException('Too many arguments.'); + } + } + + /** + * 娣诲姞涓涓煭閫夐」鐨勫 + * @param string $shortcut 鐭悕绉 + * @param mixed $value 鍊 + * @throws \RuntimeException + */ + private function addShortOption($shortcut, $value) + { + if (!$this->definition->hasShortcut($shortcut)) { + throw new \RuntimeException(sprintf('The "-%s" option does not exist.', $shortcut)); + } + + $this->addLongOption($this->definition->getOptionForShortcut($shortcut)->getName(), $value); + } + + /** + * 娣诲姞涓涓畬鏁撮夐」鐨勫 + * @param string $name 閫夐」鍚 + * @param mixed $value 鍊 + * @throws \RuntimeException + */ + private function addLongOption($name, $value) + { + if (!$this->definition->hasOption($name)) { + throw new \RuntimeException(sprintf('The "--%s" option does not exist.', $name)); + } + + $option = $this->definition->getOption($name); + + if (false === $value) { + $value = null; + } + + if (null !== $value && !$option->acceptValue()) { + throw new \RuntimeException(sprintf('The "--%s" option does not accept a value.', $name, $value)); + } + + if (null === $value && $option->acceptValue() && count($this->parsed)) { + $next = array_shift($this->parsed); + if (isset($next[0]) && '-' !== $next[0]) { + $value = $next; + } elseif (empty($next)) { + $value = ''; + } else { + array_unshift($this->parsed, $next); + } + } + + if (null === $value) { + if ($option->isValueRequired()) { + throw new \RuntimeException(sprintf('The "--%s" option requires a value.', $name)); + } + + if (!$option->isArray()) { + $value = $option->isValueOptional() ? $option->getDefault() : true; + } + } + + if ($option->isArray()) { + $this->options[$name][] = $value; + } else { + $this->options[$name] = $value; + } + } + + /** + * 鑾峰彇绗竴涓弬鏁 + * @return string|null + */ + public function getFirstArgument() + { + foreach ($this->tokens as $token) { + if ($token && '-' === $token[0]) { + continue; + } + + return $token; + } + return; + } + + /** + * 妫鏌ュ師濮嬪弬鏁版槸鍚﹀寘鍚煇涓 + * @param string|array $values 闇瑕佹鏌ョ殑鍊 + * @return bool + */ + public function hasParameterOption($values) + { + $values = (array) $values; + + foreach ($this->tokens as $token) { + foreach ($values as $value) { + if ($token === $value || 0 === strpos($token, $value . '=')) { + return true; + } + } + } + + return false; + } + + /** + * 鑾峰彇鍘熷閫夐」鐨勫 + * @param string|array $values 闇瑕佹鏌ョ殑鍊 + * @param mixed $default 榛樿鍊 + * @return mixed The option value + */ + public function getParameterOption($values, $default = false) + { + $values = (array) $values; + $tokens = $this->tokens; + + while (0 < count($tokens)) { + $token = array_shift($tokens); + + foreach ($values as $value) { + if ($token === $value || 0 === strpos($token, $value . '=')) { + if (false !== $pos = strpos($token, '=')) { + return substr($token, $pos + 1); + } + + return array_shift($tokens); + } + } + } + + return $default; + } + + /** + * 楠岃瘉杈撳叆 + * @throws \RuntimeException + */ + public function validate() + { + if (count($this->arguments) < $this->definition->getArgumentRequiredCount()) { + throw new \RuntimeException('Not enough arguments.'); + } + } + + /** + * 妫鏌ヨ緭鍏ユ槸鍚︽槸浜や簰鐨 + * @return bool + */ + public function isInteractive() + { + return $this->interactive; + } + + /** + * 璁剧疆杈撳叆鐨勪氦浜 + * @param bool + */ + public function setInteractive($interactive) + { + $this->interactive = (bool) $interactive; + } + + /** + * 鑾峰彇鎵鏈夌殑鍙傛暟 + * @return Argument[] + */ + public function getArguments() + { + return array_merge($this->definition->getArgumentDefaults(), $this->arguments); + } + + /** + * 鏍规嵁鍚嶇О鑾峰彇鍙傛暟 + * @param string $name 鍙傛暟鍚 + * @return mixed + * @throws \InvalidArgumentException + */ + public function getArgument($name) + { + if (!$this->definition->hasArgument($name)) { + throw new \InvalidArgumentException(sprintf('The "%s" argument does not exist.', $name)); + } + + return isset($this->arguments[$name]) ? $this->arguments[$name] : $this->definition->getArgument($name) + ->getDefault(); + } + + /** + * 璁剧疆鍙傛暟鐨勫 + * @param string $name 鍙傛暟鍚 + * @param string $value 鍊 + * @throws \InvalidArgumentException + */ + public function setArgument($name, $value) + { + if (!$this->definition->hasArgument($name)) { + throw new \InvalidArgumentException(sprintf('The "%s" argument does not exist.', $name)); + } + + $this->arguments[$name] = $value; + } + + /** + * 妫鏌ユ槸鍚﹀瓨鍦ㄦ煇涓弬鏁 + * @param string|int $name 鍙傛暟鍚嶆垨浣嶇疆 + * @return bool + */ + public function hasArgument($name) + { + return $this->definition->hasArgument($name); + } + + /** + * 鑾峰彇鎵鏈夌殑閫夐」 + * @return Option[] + */ + public function getOptions() + { + return array_merge($this->definition->getOptionDefaults(), $this->options); + } + + /** + * 鑾峰彇閫夐」鍊 + * @param string $name 閫夐」鍚嶇О + * @return mixed + * @throws \InvalidArgumentException + */ + public function getOption($name) + { + if (!$this->definition->hasOption($name)) { + throw new \InvalidArgumentException(sprintf('The "%s" option does not exist.', $name)); + } + + return isset($this->options[$name]) ? $this->options[$name] : $this->definition->getOption($name)->getDefault(); + } + + /** + * 璁剧疆閫夐」鍊 + * @param string $name 閫夐」鍚 + * @param string|bool $value 鍊 + * @throws \InvalidArgumentException + */ + public function setOption($name, $value) + { + if (!$this->definition->hasOption($name)) { + throw new \InvalidArgumentException(sprintf('The "%s" option does not exist.', $name)); + } + + $this->options[$name] = $value; + } + + /** + * 鏄惁鏈夋煇涓夐」 + * @param string $name 閫夐」鍚 + * @return bool + */ + public function hasOption($name) + { + return $this->definition->hasOption($name) && isset($this->options[$name]); + } + + /** + * 杞箟鎸囦护 + * @param string $token + * @return string + */ + public function escapeToken($token) + { + return preg_match('{^[\w-]+$}', $token) ? $token : escapeshellarg($token); + } + + /** + * 杩斿洖浼犻掔粰鍛戒护鐨勫弬鏁扮殑瀛楃涓 + * @return string + */ + public function __toString() + { + $tokens = array_map(function ($token) { + if (preg_match('{^(-[^=]+=)(.+)}', $token, $match)) { + return $match[1] . $this->escapeToken($match[2]); + } + + if ($token && '-' !== $token[0]) { + return $this->escapeToken($token); + } + + return $token; + }, $this->tokens); + + return implode(' ', $tokens); + } +} diff --git a/thinkphp/library/think/console/LICENSE b/thinkphp/library/think/console/LICENSE new file mode 100644 index 000000000..0abe056e1 --- /dev/null +++ b/thinkphp/library/think/console/LICENSE @@ -0,0 +1,19 @@ +Copyright (c) 2004-2016 Fabien Potencier + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is furnished +to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. \ No newline at end of file diff --git a/thinkphp/library/think/console/Output.php b/thinkphp/library/think/console/Output.php new file mode 100644 index 000000000..65dc9fb81 --- /dev/null +++ b/thinkphp/library/think/console/Output.php @@ -0,0 +1,222 @@ + +// +---------------------------------------------------------------------- + +namespace think\console; + +use Exception; +use think\console\output\Ask; +use think\console\output\Descriptor; +use think\console\output\driver\Buffer; +use think\console\output\driver\Console; +use think\console\output\driver\Nothing; +use think\console\output\Question; +use think\console\output\question\Choice; +use think\console\output\question\Confirmation; + +/** + * Class Output + * @package think\console + * + * @see \think\console\output\driver\Console::setDecorated + * @method void setDecorated($decorated) + * + * @see \think\console\output\driver\Buffer::fetch + * @method string fetch() + * + * @method void info($message) + * @method void error($message) + * @method void comment($message) + * @method void warning($message) + * @method void highlight($message) + * @method void question($message) + */ +class Output +{ + const VERBOSITY_QUIET = 0; + const VERBOSITY_NORMAL = 1; + const VERBOSITY_VERBOSE = 2; + const VERBOSITY_VERY_VERBOSE = 3; + const VERBOSITY_DEBUG = 4; + + const OUTPUT_NORMAL = 0; + const OUTPUT_RAW = 1; + const OUTPUT_PLAIN = 2; + + private $verbosity = self::VERBOSITY_NORMAL; + + /** @var Buffer|Console|Nothing */ + private $handle = null; + + protected $styles = [ + 'info', + 'error', + 'comment', + 'question', + 'highlight', + 'warning' + ]; + + public function __construct($driver = 'console') + { + $class = '\\think\\console\\output\\driver\\' . ucwords($driver); + + $this->handle = new $class($this); + } + + public function ask(Input $input, $question, $default = null, $validator = null) + { + $question = new Question($question, $default); + $question->setValidator($validator); + + return $this->askQuestion($input, $question); + } + + public function askHidden(Input $input, $question, $validator = null) + { + $question = new Question($question); + + $question->setHidden(true); + $question->setValidator($validator); + + return $this->askQuestion($input, $question); + } + + public function confirm(Input $input, $question, $default = true) + { + return $this->askQuestion($input, new Confirmation($question, $default)); + } + + /** + * {@inheritdoc} + */ + public function choice(Input $input, $question, array $choices, $default = null) + { + if (null !== $default) { + $values = array_flip($choices); + $default = $values[$default]; + } + + return $this->askQuestion($input, new Choice($question, $choices, $default)); + } + + protected function askQuestion(Input $input, Question $question) + { + $ask = new Ask($input, $this, $question); + $answer = $ask->run(); + + if ($input->isInteractive()) { + $this->newLine(); + } + + return $answer; + } + + protected function block($style, $message) + { + $this->writeln("<{$style}>{$message}"); + } + + /** + * 杈撳嚭绌鸿 + * @param int $count + */ + public function newLine($count = 1) + { + $this->write(str_repeat(PHP_EOL, $count)); + } + + /** + * 杈撳嚭淇℃伅骞舵崲琛 + * @param string $messages + * @param int $type + */ + public function writeln($messages, $type = self::OUTPUT_NORMAL) + { + $this->write($messages, true, $type); + } + + /** + * 杈撳嚭淇℃伅 + * @param string $messages + * @param bool $newline + * @param int $type + */ + public function write($messages, $newline = false, $type = self::OUTPUT_NORMAL) + { + $this->handle->write($messages, $newline, $type); + } + + public function renderException(\Exception $e) + { + $this->handle->renderException($e); + } + + /** + * {@inheritdoc} + */ + public function setVerbosity($level) + { + $this->verbosity = (int) $level; + } + + /** + * {@inheritdoc} + */ + public function getVerbosity() + { + return $this->verbosity; + } + + public function isQuiet() + { + return self::VERBOSITY_QUIET === $this->verbosity; + } + + public function isVerbose() + { + return self::VERBOSITY_VERBOSE <= $this->verbosity; + } + + public function isVeryVerbose() + { + return self::VERBOSITY_VERY_VERBOSE <= $this->verbosity; + } + + public function isDebug() + { + return self::VERBOSITY_DEBUG <= $this->verbosity; + } + + public function describe($object, array $options = []) + { + $descriptor = new Descriptor(); + $options = array_merge([ + 'raw_text' => false, + ], $options); + + $descriptor->describe($this, $object, $options); + } + + public function __call($method, $args) + { + if (in_array($method, $this->styles)) { + array_unshift($args, $method); + return call_user_func_array([$this, 'block'], $args); + } + + if ($this->handle && method_exists($this->handle, $method)) { + return call_user_func_array([$this->handle, $method], $args); + } else { + throw new Exception('method not exists:' . __CLASS__ . '->' . $method); + } + } + +} diff --git a/thinkphp/library/think/console/bin/README.md b/thinkphp/library/think/console/bin/README.md new file mode 100644 index 000000000..9acc52fba --- /dev/null +++ b/thinkphp/library/think/console/bin/README.md @@ -0,0 +1 @@ +console 宸ュ叿浣跨敤 hiddeninput.exe 鍦 windows 涓婇殣钘忓瘑鐮佽緭鍏ワ紝璇ヤ簩杩涘埗鏂囦欢鐢辩涓夋柟鎻愪緵锛岀浉鍏虫簮鐮佸拰鍏朵粬缁嗚妭鍙互鍦 [Hidden Input](https://github.com/Seldaek/hidden-input) 鎵惧埌銆 diff --git a/thinkphp/library/think/console/bin/hiddeninput.exe b/thinkphp/library/think/console/bin/hiddeninput.exe new file mode 100644 index 000000000..c8cf65e8d Binary files /dev/null and b/thinkphp/library/think/console/bin/hiddeninput.exe differ diff --git a/thinkphp/library/think/console/command/Build.php b/thinkphp/library/think/console/command/Build.php new file mode 100644 index 000000000..39806c3f4 --- /dev/null +++ b/thinkphp/library/think/console/command/Build.php @@ -0,0 +1,56 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\command; + +use think\console\Command; +use think\console\Input; +use think\console\input\Option; +use think\console\Output; + +class Build extends Command +{ + + /** + * {@inheritdoc} + */ + protected function configure() + { + $this->setName('build') + ->setDefinition([ + new Option('config', null, Option::VALUE_OPTIONAL, "build.php path"), + new Option('module', null, Option::VALUE_OPTIONAL, "module name"), + ]) + ->setDescription('Build Application Dirs'); + } + + protected function execute(Input $input, Output $output) + { + if ($input->hasOption('module')) { + \think\Build::module($input->getOption('module')); + $output->writeln("Successed"); + return; + } + + if ($input->hasOption('config')) { + $build = include $input->getOption('config'); + } else { + $build = include APP_PATH . 'build.php'; + } + if (empty($build)) { + $output->writeln("Build Config Is Empty"); + return; + } + \think\Build::run($build); + $output->writeln("Successed"); + + } +} diff --git a/thinkphp/library/think/console/command/Clear.php b/thinkphp/library/think/console/command/Clear.php new file mode 100644 index 000000000..020febd3a --- /dev/null +++ b/thinkphp/library/think/console/command/Clear.php @@ -0,0 +1,44 @@ + +// +---------------------------------------------------------------------- +namespace think\console\command; + +use think\console\Command; +use think\console\Input; +use think\console\input\Option; +use think\console\Output; + +class Clear extends Command +{ + protected function configure() + { + // 鎸囦护閰嶇疆 + $this + ->setName('clear') + ->addOption('path', 'd', Option::VALUE_OPTIONAL, 'path to clear', null) + ->setDescription('Clear runtime file'); + } + + protected function execute(Input $input, Output $output) + { + $path = $input->getOption('path') ?: RUNTIME_PATH; + $files = scandir($path); + if ($files) { + foreach ($files as $file) { + if ('.' != $file && '..' != $file && is_dir($path . $file)) { + array_map('unlink', glob($path . $file . '/*.*')); + } elseif (is_file($path . $file)) { + unlink($path . $file); + } + } + } + $output->writeln("Clear Successed"); + } +} diff --git a/thinkphp/library/think/console/command/Help.php b/thinkphp/library/think/console/command/Help.php new file mode 100644 index 000000000..bae2c6533 --- /dev/null +++ b/thinkphp/library/think/console/command/Help.php @@ -0,0 +1,69 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\command; + +use think\console\Command; +use think\console\Input; +use think\console\input\Argument as InputArgument; +use think\console\input\Option as InputOption; +use think\console\Output; + +class Help extends Command +{ + + private $command; + + /** + * {@inheritdoc} + */ + protected function configure() + { + $this->ignoreValidationErrors(); + + $this->setName('help')->setDefinition([ + new InputArgument('command_name', InputArgument::OPTIONAL, 'The command name', 'help'), + new InputOption('raw', null, InputOption::VALUE_NONE, 'To output raw command help'), + ])->setDescription('Displays help for a command')->setHelp(<<%command.name% command displays help for a given command: + + php %command.full_name% list + +To display the list of available commands, please use the list command. +EOF + ); + } + + /** + * Sets the command. + * @param Command $command The command to set + */ + public function setCommand(Command $command) + { + $this->command = $command; + } + + /** + * {@inheritdoc} + */ + protected function execute(Input $input, Output $output) + { + if (null === $this->command) { + $this->command = $this->getConsole()->find($input->getArgument('command_name')); + } + + $output->describe($this->command, [ + 'raw_text' => $input->getOption('raw'), + ]); + + $this->command = null; + } +} diff --git a/thinkphp/library/think/console/command/Lists.php b/thinkphp/library/think/console/command/Lists.php new file mode 100644 index 000000000..084ddaa23 --- /dev/null +++ b/thinkphp/library/think/console/command/Lists.php @@ -0,0 +1,74 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\command; + +use think\console\Command; +use think\console\Input; +use think\console\Output; +use think\console\input\Argument as InputArgument; +use think\console\input\Option as InputOption; +use think\console\input\Definition as InputDefinition; + +class Lists extends Command +{ + + /** + * {@inheritdoc} + */ + protected function configure() + { + $this->setName('list')->setDefinition($this->createDefinition())->setDescription('Lists commands')->setHelp(<<%command.name% command lists all commands: + + php %command.full_name% + +You can also display the commands for a specific namespace: + + php %command.full_name% test + +It's also possible to get raw list of commands (useful for embedding command runner): + + php %command.full_name% --raw +EOF + ); + } + + /** + * {@inheritdoc} + */ + public function getNativeDefinition() + { + return $this->createDefinition(); + } + + /** + * {@inheritdoc} + */ + protected function execute(Input $input, Output $output) + { + $output->describe($this->getConsole(), [ + 'raw_text' => $input->getOption('raw'), + 'namespace' => $input->getArgument('namespace'), + ]); + } + + /** + * {@inheritdoc} + */ + private function createDefinition() + { + return new InputDefinition([ + new InputArgument('namespace', InputArgument::OPTIONAL, 'The namespace name'), + new InputOption('raw', null, InputOption::VALUE_NONE, 'To output raw command list') + ]); + } +} diff --git a/thinkphp/library/think/console/command/Make.php b/thinkphp/library/think/console/command/Make.php new file mode 100644 index 000000000..b508f8d0b --- /dev/null +++ b/thinkphp/library/think/console/command/Make.php @@ -0,0 +1,109 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\command; + +use think\Config; +use think\console\Command; +use think\console\Input; +use think\console\input\Argument; +use think\console\Output; + +abstract class Make extends Command +{ + + protected $type; + + abstract protected function getStub(); + + protected function configure() + { + $this->addArgument('name', Argument::REQUIRED, "The name of the class"); + } + + protected function execute(Input $input, Output $output) + { + + $name = trim($input->getArgument('name')); + + $classname = $this->getClassName($name); + + $pathname = $this->getPathName($classname); + + if (is_file($pathname)) { + $output->writeln('' . $this->type . ' already exists!'); + return false; + } + + if (!is_dir(dirname($pathname))) { + mkdir(strtolower(dirname($pathname)), 0755, true); + } + + file_put_contents($pathname, $this->buildClass($classname)); + + $output->writeln('' . $this->type . ' created successfully.'); + + } + + protected function buildClass($name) + { + $stub = file_get_contents($this->getStub()); + + $namespace = trim(implode('\\', array_slice(explode('\\', $name), 0, -1)), '\\'); + + $class = str_replace($namespace . '\\', '', $name); + + return str_replace(['{%className%}', '{%namespace%}', '{%app_namespace%}'], [ + $class, + $namespace, + Config::get('app_namespace') + ], $stub); + + } + + protected function getPathName($name) + { + $name = str_replace(Config::get('app_namespace') . '\\', '', $name); + + return APP_PATH . str_replace('\\', '/', $name) . '.php'; + } + + protected function getClassName($name) + { + $appNamespace = Config::get('app_namespace'); + + if (strpos($name, $appNamespace . '\\') === 0) { + return $name; + } + + if (Config::get('app_multi_module')) { + if (strpos($name, '/')) { + list($module, $name) = explode('/', $name, 2); + } else { + $module = 'common'; + } + } else { + $module = null; + } + + if (strpos($name, '/') !== false) { + $name = str_replace('/', '\\', $name); + } + + return $this->getNamespace($appNamespace, $module) . '\\' . $name; + } + + protected function getNamespace($appNamespace, $module) + { + return $module ? ($appNamespace . '\\' . $module) : $appNamespace; + } + +} diff --git a/thinkphp/library/think/console/command/make/Controller.php b/thinkphp/library/think/console/command/make/Controller.php new file mode 100644 index 000000000..afa7be905 --- /dev/null +++ b/thinkphp/library/think/console/command/make/Controller.php @@ -0,0 +1,50 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\command\make; + +use think\Config; +use think\console\command\Make; +use think\console\input\Option; + +class Controller extends Make +{ + + protected $type = "Controller"; + + protected function configure() + { + parent::configure(); + $this->setName('make:controller') + ->addOption('plain', null, Option::VALUE_NONE, 'Generate an empty controller class.') + ->setDescription('Create a new resource controller class'); + } + + protected function getStub() + { + if ($this->input->getOption('plain')) { + return __DIR__ . '/stubs/controller.plain.stub'; + } + + return __DIR__ . '/stubs/controller.stub'; + } + + protected function getClassName($name) + { + return parent::getClassName($name) . (Config::get('controller_suffix') ? ucfirst(Config::get('url_controller_layer')) : ''); + } + + protected function getNamespace($appNamespace, $module) + { + return parent::getNamespace($appNamespace, $module) . '\controller'; + } + +} diff --git a/thinkphp/library/think/console/command/make/Model.php b/thinkphp/library/think/console/command/make/Model.php new file mode 100644 index 000000000..d4e9b5dd0 --- /dev/null +++ b/thinkphp/library/think/console/command/make/Model.php @@ -0,0 +1,36 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\command\make; + +use think\console\command\Make; + +class Model extends Make +{ + protected $type = "Model"; + + protected function configure() + { + parent::configure(); + $this->setName('make:model') + ->setDescription('Create a new model class'); + } + + protected function getStub() + { + return __DIR__ . '/stubs/model.stub'; + } + + protected function getNamespace($appNamespace, $module) + { + return parent::getNamespace($appNamespace, $module) . '\model'; + } +} diff --git a/thinkphp/library/think/console/command/make/stubs/controller.plain.stub b/thinkphp/library/think/console/command/make/stubs/controller.plain.stub new file mode 100644 index 000000000..b7539dcf6 --- /dev/null +++ b/thinkphp/library/think/console/command/make/stubs/controller.plain.stub @@ -0,0 +1,10 @@ + +// +---------------------------------------------------------------------- +namespace think\console\command\optimize; + +use think\App; +use think\console\Command; +use think\console\Input; +use think\console\Output; + +class Autoload extends Command +{ + + protected function configure() + { + $this->setName('optimize:autoload') + ->setDescription('Optimizes PSR0 and PSR4 packages to be loaded with classmaps too, good for production.'); + } + + protected function execute(Input $input, Output $output) + { + + $classmapFile = << realpath(rtrim(APP_PATH)), + 'think\\' => LIB_PATH . 'think', + 'behavior\\' => LIB_PATH . 'behavior', + 'traits\\' => LIB_PATH . 'traits', + '' => realpath(rtrim(EXTEND_PATH)) + ]; + + krsort($namespacesToScan); + $classMap = []; + foreach ($namespacesToScan as $namespace => $dir) { + + if (!is_dir($dir)) { + continue; + } + + $namespaceFilter = $namespace === '' ? null : $namespace; + $classMap = $this->addClassMapCode($dir, $namespaceFilter, $classMap); + } + + ksort($classMap); + foreach ($classMap as $class => $code) { + $classmapFile .= ' ' . var_export($class, true) . ' => ' . $code; + } + $classmapFile .= "];\n"; + + if (!is_dir(RUNTIME_PATH)) { + @mkdir(RUNTIME_PATH, 0755, true); + } + + file_put_contents(RUNTIME_PATH . 'classmap' . EXT, $classmapFile); + + $output->writeln('Succeed!'); + } + + protected function addClassMapCode($dir, $namespace, $classMap) + { + foreach ($this->createMap($dir, $namespace) as $class => $path) { + + $pathCode = $this->getPathCode($path) . ",\n"; + + if (!isset($classMap[$class])) { + $classMap[$class] = $pathCode; + } elseif ($classMap[$class] !== $pathCode && !preg_match('{/(test|fixture|example|stub)s?/}i', strtr($classMap[$class] . ' ' . $path, '\\', '/'))) { + $this->output->writeln( + 'Warning: Ambiguous class resolution, "' . $class . '"' . + ' was found in both "' . str_replace(["',\n"], [ + '' + ], $classMap[$class]) . '" and "' . $path . '", the first will be used.' + ); + } + } + return $classMap; + } + + protected function getPathCode($path) + { + + $baseDir = ''; + $appPath = $this->normalizePath(realpath(APP_PATH)); + $libPath = $this->normalizePath(realpath(LIB_PATH)); + $extendPath = $this->normalizePath(realpath(EXTEND_PATH)); + $path = $this->normalizePath($path); + + if (strpos($path, $libPath . '/') === 0) { + $path = substr($path, strlen(LIB_PATH)); + $baseDir = 'LIB_PATH'; + } elseif (strpos($path, $appPath . '/') === 0) { + $path = substr($path, strlen($appPath) + 1); + $baseDir = 'APP_PATH'; + } elseif (strpos($path, $extendPath . '/') === 0) { + $path = substr($path, strlen($extendPath) + 1); + $baseDir = 'EXTEND_PATH'; + } + + if ($path !== false) { + $baseDir .= " . "; + } + + return $baseDir . (($path !== false) ? var_export($path, true) : ""); + } + + protected function normalizePath($path) + { + $parts = []; + $path = strtr($path, '\\', '/'); + $prefix = ''; + $absolute = false; + + if (preg_match('{^([0-9a-z]+:(?://(?:[a-z]:)?)?)}i', $path, $match)) { + $prefix = $match[1]; + $path = substr($path, strlen($prefix)); + } + + if (substr($path, 0, 1) === '/') { + $absolute = true; + $path = substr($path, 1); + } + + $up = false; + foreach (explode('/', $path) as $chunk) { + if ('..' === $chunk && ($absolute || $up)) { + array_pop($parts); + $up = !(empty($parts) || '..' === end($parts)); + } elseif ('.' !== $chunk && '' !== $chunk) { + $parts[] = $chunk; + $up = '..' !== $chunk; + } + } + + return $prefix . ($absolute ? '/' : '') . implode('/', $parts); + } + + protected function createMap($path, $namespace = null) + { + if (is_string($path)) { + if (is_file($path)) { + $path = [new \SplFileInfo($path)]; + } elseif (is_dir($path)) { + + $objects = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($path), \RecursiveIteratorIterator::SELF_FIRST); + + $path = []; + + /** @var \SplFileInfo $object */ + foreach ($objects as $object) { + if ($object->isFile() && $object->getExtension() == 'php') { + $path[] = $object; + } + } + } else { + throw new \RuntimeException( + 'Could not scan for classes inside "' . $path . + '" which does not appear to be a file nor a folder' + ); + } + } + + $map = []; + + /** @var \SplFileInfo $file */ + foreach ($path as $file) { + $filePath = $file->getRealPath(); + + if (pathinfo($filePath, PATHINFO_EXTENSION) != 'php') { + continue; + } + + $classes = $this->findClasses($filePath); + + foreach ($classes as $class) { + if (null !== $namespace && 0 !== strpos($class, $namespace)) { + continue; + } + + if (!isset($map[$class])) { + $map[$class] = $filePath; + } elseif ($map[$class] !== $filePath && !preg_match('{/(test|fixture|example|stub)s?/}i', strtr($map[$class] . ' ' . $filePath, '\\', '/'))) { + $this->output->writeln( + 'Warning: Ambiguous class resolution, "' . $class . '"' . + ' was found in both "' . $map[$class] . '" and "' . $filePath . '", the first will be used.' + ); + } + } + } + + return $map; + } + + protected function findClasses($path) + { + $extraTypes = '|trait'; + + $contents = @php_strip_whitespace($path); + if (!$contents) { + if (!file_exists($path)) { + $message = 'File at "%s" does not exist, check your classmap definitions'; + } elseif (!is_readable($path)) { + $message = 'File at "%s" is not readable, check its permissions'; + } elseif ('' === trim(file_get_contents($path))) { + return []; + } else { + $message = 'File at "%s" could not be parsed as PHP, it may be binary or corrupted'; + } + $error = error_get_last(); + if (isset($error['message'])) { + $message .= PHP_EOL . 'The following message may be helpful:' . PHP_EOL . $error['message']; + } + throw new \RuntimeException(sprintf($message, $path)); + } + + if (!preg_match('{\b(?:class|interface' . $extraTypes . ')\s}i', $contents)) { + return []; + } + + // strip heredocs/nowdocs + $contents = preg_replace('{<<<\s*(\'?)(\w+)\\1(?:\r\n|\n|\r)(?:.*?)(?:\r\n|\n|\r)\\2(?=\r\n|\n|\r|;)}s', 'null', $contents); + // strip strings + $contents = preg_replace('{"[^"\\\\]*+(\\\\.[^"\\\\]*+)*+"|\'[^\'\\\\]*+(\\\\.[^\'\\\\]*+)*+\'}s', 'null', $contents); + // strip leading non-php code if needed + if (substr($contents, 0, 2) !== '.+<\?}s', '?>'); + if (false !== $pos && false === strpos(substr($contents, $pos), '])(?Pclass|interface' . $extraTypes . ') \s++ (?P[a-zA-Z_\x7f-\xff:][a-zA-Z0-9_\x7f-\xff:\-]*+) + | \b(?])(?Pnamespace) (?P\s++[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+(?:\s*+\\\\\s*+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*+)*+)? \s*+ [\{;] + ) + }ix', $contents, $matches); + + $classes = []; + $namespace = ''; + + for ($i = 0, $len = count($matches['type']); $i < $len; $i++) { + if (!empty($matches['ns'][$i])) { + $namespace = str_replace([' ', "\t", "\r", "\n"], '', $matches['nsname'][$i]) . '\\'; + } else { + $name = $matches['name'][$i]; + if ($name[0] === ':') { + $name = 'xhp' . substr(str_replace(['-', ':'], ['_', '__'], $name), 1); + } elseif ($matches['type'][$i] === 'enum') { + $name = rtrim($name, ':'); + } + $classes[] = ltrim($namespace . $name, '\\'); + } + } + + return $classes; + } + +} diff --git a/thinkphp/library/think/console/command/optimize/Config.php b/thinkphp/library/think/console/command/optimize/Config.php new file mode 100644 index 000000000..cadfe5ee9 --- /dev/null +++ b/thinkphp/library/think/console/command/optimize/Config.php @@ -0,0 +1,93 @@ + +// +---------------------------------------------------------------------- +namespace think\console\command\optimize; + +use think\Config as ThinkConfig; +use think\console\Command; +use think\console\Input; +use think\console\input\Argument; +use think\console\Output; + +class Config extends Command +{ + /** @var Output */ + protected $output; + + protected function configure() + { + $this->setName('optimize:config') + ->addArgument('module', Argument::OPTIONAL, 'Build module config cache .') + ->setDescription('Build config and common file cache.'); + } + + protected function execute(Input $input, Output $output) + { + if ($input->hasArgument('module')) { + $module = $input->getArgument('module') . DS; + } else { + $module = ''; + } + + $content = 'buildCacheContent($module); + + if (!is_dir(RUNTIME_PATH . $module)) { + @mkdir(RUNTIME_PATH . $module, 0755, true); + } + + file_put_contents(RUNTIME_PATH . $module . 'init' . EXT, $content); + + $output->writeln('Succeed!'); + } + + protected function buildCacheContent($module) + { + $content = ''; + $path = realpath(APP_PATH . $module) . DS; + + if ($module) { + // 鍔犺浇妯″潡閰嶇疆 + $config = ThinkConfig::load(CONF_PATH . $module . 'config' . CONF_EXT); + + // 璇诲彇鏁版嵁搴撻厤缃枃浠 + $filename = CONF_PATH . $module . 'database' . CONF_EXT; + ThinkConfig::load($filename, 'database'); + + // 鍔犺浇搴旂敤鐘舵侀厤缃 + if ($config['app_status']) { + $config = ThinkConfig::load(CONF_PATH . $module . $config['app_status'] . CONF_EXT); + } + // 璇诲彇鎵╁睍閰嶇疆鏂囦欢 + 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; + ThinkConfig::load($filename, pathinfo($file, PATHINFO_FILENAME)); + } + } + } + } + + // 鍔犺浇琛屼负鎵╁睍鏂囦欢 + if (is_file(CONF_PATH . $module . 'tags' . EXT)) { + $content .= '\think\Hook::import(' . (var_export(include CONF_PATH . $module . 'tags' . EXT, true)) . ');' . PHP_EOL; + } + + // 鍔犺浇鍏叡鏂囦欢 + if (is_file($path . 'common' . EXT)) { + $content .= substr(php_strip_whitespace($path . 'common' . EXT), 5) . PHP_EOL; + } + + $content .= '\think\Config::set(' . var_export(ThinkConfig::get(), true) . ');'; + return $content; + } +} diff --git a/thinkphp/library/think/console/command/optimize/Route.php b/thinkphp/library/think/console/command/optimize/Route.php new file mode 100644 index 000000000..911e4c147 --- /dev/null +++ b/thinkphp/library/think/console/command/optimize/Route.php @@ -0,0 +1,70 @@ + +// +---------------------------------------------------------------------- +namespace think\console\command\optimize; + +use think\console\Command; +use think\console\Input; +use think\console\Output; + +class Route extends Command +{ + /** @var Output */ + protected $output; + + protected function configure() + { + $this->setName('optimize:route') + ->setDescription('Build route cache.'); + } + + protected function execute(Input $input, Output $output) + { + file_put_contents(RUNTIME_PATH . 'route.php', $this->buildRouteCache()); + $output->writeln('Succeed!'); + } + + protected function buildRouteCache() + { + $files = \think\Config::get('route_config_file'); + foreach ($files as $file) { + if (is_file(CONF_PATH . $file . CONF_EXT)) { + $config = include CONF_PATH . $file . CONF_EXT; + if (is_array($config)) { + \think\Route::import($config); + } + } + } + $rules = \think\Route::rules(true); + array_walk_recursive($rules, [$this, 'buildClosure']); + $content = 'getStartLine(); + $endLine = $reflection->getEndLine(); + $file = $reflection->getFileName(); + $item = file($file); + $content = ''; + for ($i = $startLine - 1; $i <= $endLine - 1; $i++) { + $content .= $item[$i]; + } + $start = strpos($content, 'function'); + $end = strrpos($content, '}'); + $value = '[__start__' . substr($content, $start, $end - $start + 1) . '__end__]'; + } + } +} diff --git a/thinkphp/library/think/console/command/optimize/Schema.php b/thinkphp/library/think/console/command/optimize/Schema.php new file mode 100644 index 000000000..6ac38a36a --- /dev/null +++ b/thinkphp/library/think/console/command/optimize/Schema.php @@ -0,0 +1,111 @@ + +// +---------------------------------------------------------------------- +namespace think\console\command\optimize; + +use think\App; +use think\console\Command; +use think\console\Input; +use think\console\input\Option; +use think\console\Output; +use think\Db; + +class Schema extends Command +{ + /** @var Output */ + protected $output; + + protected function configure() + { + $this->setName('optimize:schema') + ->addOption('db', null, Option::VALUE_REQUIRED, 'db name .') + ->addOption('table', null, Option::VALUE_REQUIRED, 'table name .') + ->addOption('module', null, Option::VALUE_REQUIRED, 'module name .') + ->setDescription('Build database schema cache.'); + } + + protected function execute(Input $input, Output $output) + { + if (!is_dir(RUNTIME_PATH . 'schema')) { + @mkdir(RUNTIME_PATH . 'schema', 0755, true); + } + if ($input->hasOption('module')) { + $module = $input->getOption('module'); + // 璇诲彇妯″瀷 + $list = scandir(APP_PATH . $module . DS . 'model'); + $app = App::$namespace; + foreach ($list as $file) { + if ('.' == $file || '..' == $file) { + continue; + } + $class = '\\' . $app . '\\' . $module . '\\model\\' . pathinfo($file, PATHINFO_FILENAME); + $this->buildModelSchema($class); + } + $output->writeln('Succeed!'); + return; + } elseif ($input->hasOption('table')) { + $table = $input->getOption('table'); + if (!strpos($table, '.')) { + $dbName = Db::getConfig('database'); + } + $tables[] = $table; + } elseif ($input->hasOption('db')) { + $dbName = $input->getOption('db'); + $tables = Db::getTables($dbName); + } elseif (!\think\Config::get('app_multi_module')) { + $app = App::$namespace; + $list = scandir(APP_PATH . 'model'); + foreach ($list as $file) { + if ('.' == $file || '..' == $file) { + continue; + } + $class = '\\' . $app . '\\model\\' . pathinfo($file, PATHINFO_FILENAME); + $this->buildModelSchema($class); + } + $output->writeln('Succeed!'); + return; + } else { + $tables = Db::getTables(); + } + + $db = isset($dbName) ? $dbName . '.' : ''; + $this->buildDataBaseSchema($tables, $db); + + $output->writeln('Succeed!'); + } + + protected function buildModelSchema($class) + { + $reflect = new \ReflectionClass($class); + if (!$reflect->isAbstract() && $reflect->isSubclassOf('\think\Model')) { + $table = $class::getTable(); + $dbName = $class::getConfig('database'); + $content = 'getFields($table); + $content .= var_export($info, true) . ';'; + file_put_contents(RUNTIME_PATH . 'schema' . DS . $dbName . '.' . $table . EXT, $content); + } + } + + protected function buildDataBaseSchema($tables, $db) + { + if ('' == $db) { + $dbName = Db::getConfig('database') . '.'; + } else { + $dbName = $db; + } + foreach ($tables as $table) { + $content = ' +// +---------------------------------------------------------------------- + +namespace think\console\input; + +class Argument +{ + + const REQUIRED = 1; + const OPTIONAL = 2; + const IS_ARRAY = 4; + + private $name; + private $mode; + private $default; + private $description; + + /** + * 鏋勯犳柟娉 + * @param string $name 鍙傛暟鍚 + * @param int $mode 鍙傛暟绫诲瀷: self::REQUIRED 鎴栬 self::OPTIONAL + * @param string $description 鎻忚堪 + * @param mixed $default 榛樿鍊 (浠 self::OPTIONAL 绫诲瀷鏈夋晥) + * @throws \InvalidArgumentException + */ + public function __construct($name, $mode = null, $description = '', $default = null) + { + if (null === $mode) { + $mode = self::OPTIONAL; + } elseif (!is_int($mode) || $mode > 7 || $mode < 1) { + throw new \InvalidArgumentException(sprintf('Argument mode "%s" is not valid.', $mode)); + } + + $this->name = $name; + $this->mode = $mode; + $this->description = $description; + + $this->setDefault($default); + } + + /** + * 鑾峰彇鍙傛暟鍚 + * @return string + */ + public function getName() + { + return $this->name; + } + + /** + * 鏄惁蹇呴』 + * @return bool + */ + public function isRequired() + { + return self::REQUIRED === (self::REQUIRED & $this->mode); + } + + /** + * 璇ュ弬鏁版槸鍚︽帴鍙楁暟缁 + * @return bool + */ + public function isArray() + { + return self::IS_ARRAY === (self::IS_ARRAY & $this->mode); + } + + /** + * 璁剧疆榛樿鍊 + * @param mixed $default 榛樿鍊 + * @throws \LogicException + */ + public function setDefault($default = null) + { + if (self::REQUIRED === $this->mode && null !== $default) { + throw new \LogicException('Cannot set a default value except for InputArgument::OPTIONAL mode.'); + } + + if ($this->isArray()) { + if (null === $default) { + $default = []; + } elseif (!is_array($default)) { + throw new \LogicException('A default value for an array argument must be an array.'); + } + } + + $this->default = $default; + } + + /** + * 鑾峰彇榛樿鍊 + * @return mixed + */ + public function getDefault() + { + return $this->default; + } + + /** + * 鑾峰彇鎻忚堪 + * @return string + */ + public function getDescription() + { + return $this->description; + } +} diff --git a/thinkphp/library/think/console/input/Definition.php b/thinkphp/library/think/console/input/Definition.php new file mode 100644 index 000000000..c71977ec3 --- /dev/null +++ b/thinkphp/library/think/console/input/Definition.php @@ -0,0 +1,375 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\input; + +class Definition +{ + + /** + * @var Argument[] + */ + private $arguments; + + private $requiredCount; + private $hasAnArrayArgument = false; + private $hasOptional; + + /** + * @var Option[] + */ + private $options; + private $shortcuts; + + /** + * 鏋勯犳柟娉 + * @param array $definition + * @api + */ + public function __construct(array $definition = []) + { + $this->setDefinition($definition); + } + + /** + * 璁剧疆鎸囦护鐨勫畾涔 + * @param array $definition 瀹氫箟鐨勬暟缁 + */ + public function setDefinition(array $definition) + { + $arguments = []; + $options = []; + foreach ($definition as $item) { + if ($item instanceof Option) { + $options[] = $item; + } else { + $arguments[] = $item; + } + } + + $this->setArguments($arguments); + $this->setOptions($options); + } + + /** + * 璁剧疆鍙傛暟 + * @param Argument[] $arguments 鍙傛暟鏁扮粍 + */ + public function setArguments($arguments = []) + { + $this->arguments = []; + $this->requiredCount = 0; + $this->hasOptional = false; + $this->hasAnArrayArgument = false; + $this->addArguments($arguments); + } + + /** + * 娣诲姞鍙傛暟 + * @param Argument[] $arguments 鍙傛暟鏁扮粍 + * @api + */ + public function addArguments($arguments = []) + { + if (null !== $arguments) { + foreach ($arguments as $argument) { + $this->addArgument($argument); + } + } + } + + /** + * 娣诲姞涓涓弬鏁 + * @param Argument $argument 鍙傛暟 + * @throws \LogicException + */ + public function addArgument(Argument $argument) + { + if (isset($this->arguments[$argument->getName()])) { + throw new \LogicException(sprintf('An argument with name "%s" already exists.', $argument->getName())); + } + + if ($this->hasAnArrayArgument) { + throw new \LogicException('Cannot add an argument after an array argument.'); + } + + if ($argument->isRequired() && $this->hasOptional) { + throw new \LogicException('Cannot add a required argument after an optional one.'); + } + + if ($argument->isArray()) { + $this->hasAnArrayArgument = true; + } + + if ($argument->isRequired()) { + ++$this->requiredCount; + } else { + $this->hasOptional = true; + } + + $this->arguments[$argument->getName()] = $argument; + } + + /** + * 鏍规嵁鍚嶇О鎴栬呬綅缃幏鍙栧弬鏁 + * @param string|int $name 鍙傛暟鍚嶆垨鑰呬綅缃 + * @return Argument 鍙傛暟 + * @throws \InvalidArgumentException + */ + public function getArgument($name) + { + if (!$this->hasArgument($name)) { + throw new \InvalidArgumentException(sprintf('The "%s" argument does not exist.', $name)); + } + + $arguments = is_int($name) ? array_values($this->arguments) : $this->arguments; + + return $arguments[$name]; + } + + /** + * 鏍规嵁鍚嶇О鎴栦綅缃鏌ユ槸鍚﹀叿鏈夋煇涓弬鏁 + * @param string|int $name 鍙傛暟鍚嶆垨鑰呬綅缃 + * @return bool + * @api + */ + public function hasArgument($name) + { + $arguments = is_int($name) ? array_values($this->arguments) : $this->arguments; + + return isset($arguments[$name]); + } + + /** + * 鑾峰彇鎵鏈夌殑鍙傛暟 + * @return Argument[] 鍙傛暟鏁扮粍 + */ + public function getArguments() + { + return $this->arguments; + } + + /** + * 鑾峰彇鍙傛暟鏁伴噺 + * @return int + */ + public function getArgumentCount() + { + return $this->hasAnArrayArgument ? PHP_INT_MAX : count($this->arguments); + } + + /** + * 鑾峰彇蹇呭~鐨勫弬鏁扮殑鏁伴噺 + * @return int + */ + public function getArgumentRequiredCount() + { + return $this->requiredCount; + } + + /** + * 鑾峰彇鍙傛暟榛樿鍊 + * @return array + */ + public function getArgumentDefaults() + { + $values = []; + foreach ($this->arguments as $argument) { + $values[$argument->getName()] = $argument->getDefault(); + } + + return $values; + } + + /** + * 璁剧疆閫夐」 + * @param Option[] $options 閫夐」鏁扮粍 + */ + public function setOptions($options = []) + { + $this->options = []; + $this->shortcuts = []; + $this->addOptions($options); + } + + /** + * 娣诲姞閫夐」 + * @param Option[] $options 閫夐」鏁扮粍 + * @api + */ + public function addOptions($options = []) + { + foreach ($options as $option) { + $this->addOption($option); + } + } + + /** + * 娣诲姞涓涓夐」 + * @param Option $option 閫夐」 + * @throws \LogicException + * @api + */ + public function addOption(Option $option) + { + if (isset($this->options[$option->getName()]) && !$option->equals($this->options[$option->getName()])) { + throw new \LogicException(sprintf('An option named "%s" already exists.', $option->getName())); + } + + if ($option->getShortcut()) { + foreach (explode('|', $option->getShortcut()) as $shortcut) { + if (isset($this->shortcuts[$shortcut]) + && !$option->equals($this->options[$this->shortcuts[$shortcut]]) + ) { + throw new \LogicException(sprintf('An option with shortcut "%s" already exists.', $shortcut)); + } + } + } + + $this->options[$option->getName()] = $option; + if ($option->getShortcut()) { + foreach (explode('|', $option->getShortcut()) as $shortcut) { + $this->shortcuts[$shortcut] = $option->getName(); + } + } + } + + /** + * 鏍规嵁鍚嶇О鑾峰彇閫夐」 + * @param string $name 閫夐」鍚 + * @return Option + * @throws \InvalidArgumentException + * @api + */ + public function getOption($name) + { + if (!$this->hasOption($name)) { + throw new \InvalidArgumentException(sprintf('The "--%s" option does not exist.', $name)); + } + + return $this->options[$name]; + } + + /** + * 鏍规嵁鍚嶇О妫鏌ユ槸鍚︽湁杩欎釜閫夐」 + * @param string $name 閫夐」鍚 + * @return bool + * @api + */ + public function hasOption($name) + { + return isset($this->options[$name]); + } + + /** + * 鑾峰彇鎵鏈夐夐」 + * @return Option[] + * @api + */ + public function getOptions() + { + return $this->options; + } + + /** + * 鏍规嵁鍚嶇О妫鏌ユ煇涓夐」鏄惁鏈夌煭鍚嶇О + * @param string $name 鐭悕绉 + * @return bool + */ + public function hasShortcut($name) + { + return isset($this->shortcuts[$name]); + } + + /** + * 鏍规嵁鐭悕绉拌幏鍙栭夐」 + * @param string $shortcut 鐭悕绉 + * @return Option + */ + public function getOptionForShortcut($shortcut) + { + return $this->getOption($this->shortcutToName($shortcut)); + } + + /** + * 鑾峰彇鎵鏈夐夐」鐨勯粯璁ゅ + * @return array + */ + public function getOptionDefaults() + { + $values = []; + foreach ($this->options as $option) { + $values[$option->getName()] = $option->getDefault(); + } + + return $values; + } + + /** + * 鏍规嵁鐭悕绉拌幏鍙栭夐」鍚 + * @param string $shortcut 鐭悕绉 + * @return string + * @throws \InvalidArgumentException + */ + private function shortcutToName($shortcut) + { + if (!isset($this->shortcuts[$shortcut])) { + throw new \InvalidArgumentException(sprintf('The "-%s" option does not exist.', $shortcut)); + } + + return $this->shortcuts[$shortcut]; + } + + /** + * 鑾峰彇璇ユ寚浠ょ殑浠嬬粛 + * @param bool $short 鏄惁绠娲佷粙缁 + * @return string + */ + public function getSynopsis($short = false) + { + $elements = []; + + if ($short && $this->getOptions()) { + $elements[] = '[options]'; + } elseif (!$short) { + foreach ($this->getOptions() as $option) { + $value = ''; + if ($option->acceptValue()) { + $value = sprintf(' %s%s%s', $option->isValueOptional() ? '[' : '', strtoupper($option->getName()), $option->isValueOptional() ? ']' : ''); + } + + $shortcut = $option->getShortcut() ? sprintf('-%s|', $option->getShortcut()) : ''; + $elements[] = sprintf('[%s--%s%s]', $shortcut, $option->getName(), $value); + } + } + + if (count($elements) && $this->getArguments()) { + $elements[] = '[--]'; + } + + foreach ($this->getArguments() as $argument) { + $element = '<' . $argument->getName() . '>'; + if (!$argument->isRequired()) { + $element = '[' . $element . ']'; + } elseif ($argument->isArray()) { + $element .= ' (' . $element . ')'; + } + + if ($argument->isArray()) { + $element .= '...'; + } + + $elements[] = $element; + } + + return implode(' ', $elements); + } +} diff --git a/thinkphp/library/think/console/input/Option.php b/thinkphp/library/think/console/input/Option.php new file mode 100644 index 000000000..e5707c9ae --- /dev/null +++ b/thinkphp/library/think/console/input/Option.php @@ -0,0 +1,190 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\input; + +class Option +{ + + const VALUE_NONE = 1; + const VALUE_REQUIRED = 2; + const VALUE_OPTIONAL = 4; + const VALUE_IS_ARRAY = 8; + + private $name; + private $shortcut; + private $mode; + private $default; + private $description; + + /** + * 鏋勯犳柟娉 + * @param string $name 閫夐」鍚 + * @param string|array $shortcut 鐭悕绉,澶氫釜鐢▅闅斿紑鎴栬呬娇鐢ㄦ暟缁 + * @param int $mode 閫夐」绫诲瀷(鍙夌被鍨嬩负 self::VALUE_*) + * @param string $description 鎻忚堪 + * @param mixed $default 榛樿鍊 (绫诲瀷涓 self::VALUE_REQUIRED 鎴栬 self::VALUE_NONE 鐨勬椂鍊欏繀椤讳负null) + * @throws \InvalidArgumentException + */ + public function __construct($name, $shortcut = null, $mode = null, $description = '', $default = null) + { + if (0 === strpos($name, '--')) { + $name = substr($name, 2); + } + + if (empty($name)) { + throw new \InvalidArgumentException('An option name cannot be empty.'); + } + + if (empty($shortcut)) { + $shortcut = null; + } + + if (null !== $shortcut) { + if (is_array($shortcut)) { + $shortcut = implode('|', $shortcut); + } + $shortcuts = preg_split('{(\|)-?}', ltrim($shortcut, '-')); + $shortcuts = array_filter($shortcuts); + $shortcut = implode('|', $shortcuts); + + if (empty($shortcut)) { + throw new \InvalidArgumentException('An option shortcut cannot be empty.'); + } + } + + if (null === $mode) { + $mode = self::VALUE_NONE; + } elseif (!is_int($mode) || $mode > 15 || $mode < 1) { + throw new \InvalidArgumentException(sprintf('Option mode "%s" is not valid.', $mode)); + } + + $this->name = $name; + $this->shortcut = $shortcut; + $this->mode = $mode; + $this->description = $description; + + if ($this->isArray() && !$this->acceptValue()) { + throw new \InvalidArgumentException('Impossible to have an option mode VALUE_IS_ARRAY if the option does not accept a value.'); + } + + $this->setDefault($default); + } + + /** + * 鑾峰彇鐭悕绉 + * @return string + */ + public function getShortcut() + { + return $this->shortcut; + } + + /** + * 鑾峰彇閫夐」鍚 + * @return string + */ + public function getName() + { + return $this->name; + } + + /** + * 鏄惁鍙互璁剧疆鍊 + * @return bool 绫诲瀷涓嶆槸 self::VALUE_NONE 鐨勬椂鍊欒繑鍥瀟rue,鍏朵粬鍧囪繑鍥瀎alse + */ + public function acceptValue() + { + return $this->isValueRequired() || $this->isValueOptional(); + } + + /** + * 鏄惁蹇呴』 + * @return bool 绫诲瀷鏄 self::VALUE_REQUIRED 鐨勬椂鍊欒繑鍥瀟rue,鍏朵粬鍧囪繑鍥瀎alse + */ + public function isValueRequired() + { + return self::VALUE_REQUIRED === (self::VALUE_REQUIRED & $this->mode); + } + + /** + * 鏄惁鍙 + * @return bool 绫诲瀷鏄 self::VALUE_OPTIONAL 鐨勬椂鍊欒繑鍥瀟rue,鍏朵粬鍧囪繑鍥瀎alse + */ + public function isValueOptional() + { + return self::VALUE_OPTIONAL === (self::VALUE_OPTIONAL & $this->mode); + } + + /** + * 閫夐」鍊兼槸鍚︽帴鍙楁暟缁 + * @return bool 绫诲瀷鏄 self::VALUE_IS_ARRAY 鐨勬椂鍊欒繑鍥瀟rue,鍏朵粬鍧囪繑鍥瀎alse + */ + public function isArray() + { + return self::VALUE_IS_ARRAY === (self::VALUE_IS_ARRAY & $this->mode); + } + + /** + * 璁剧疆榛樿鍊 + * @param mixed $default 榛樿鍊 + * @throws \LogicException + */ + public function setDefault($default = null) + { + if (self::VALUE_NONE === (self::VALUE_NONE & $this->mode) && null !== $default) { + throw new \LogicException('Cannot set a default value when using InputOption::VALUE_NONE mode.'); + } + + if ($this->isArray()) { + if (null === $default) { + $default = []; + } elseif (!is_array($default)) { + throw new \LogicException('A default value for an array option must be an array.'); + } + } + + $this->default = $this->acceptValue() ? $default : false; + } + + /** + * 鑾峰彇榛樿鍊 + * @return mixed + */ + public function getDefault() + { + return $this->default; + } + + /** + * 鑾峰彇鎻忚堪鏂囧瓧 + * @return string + */ + public function getDescription() + { + return $this->description; + } + + /** + * 妫鏌ユ墍缁欓夐」鏄惁鏄綋鍓嶈繖涓 + * @param Option $option + * @return bool + */ + public function equals(Option $option) + { + return $option->getName() === $this->getName() + && $option->getShortcut() === $this->getShortcut() + && $option->getDefault() === $this->getDefault() + && $option->isArray() === $this->isArray() + && $option->isValueRequired() === $this->isValueRequired() + && $option->isValueOptional() === $this->isValueOptional(); + } +} diff --git a/thinkphp/library/think/console/output/Ask.php b/thinkphp/library/think/console/output/Ask.php new file mode 100644 index 000000000..3933eb296 --- /dev/null +++ b/thinkphp/library/think/console/output/Ask.php @@ -0,0 +1,340 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output; + +use think\console\Input; +use think\console\Output; +use think\console\output\question\Choice; +use think\console\output\question\Confirmation; + +class Ask +{ + private static $stty; + + private static $shell; + + /** @var Input */ + protected $input; + + /** @var Output */ + protected $output; + + /** @var Question */ + protected $question; + + public function __construct(Input $input, Output $output, Question $question) + { + $this->input = $input; + $this->output = $output; + $this->question = $question; + } + + public function run() + { + if (!$this->input->isInteractive()) { + return $this->question->getDefault(); + } + + if (!$this->question->getValidator()) { + return $this->doAsk(); + } + + $that = $this; + + $interviewer = function () use ($that) { + return $that->doAsk(); + }; + + return $this->validateAttempts($interviewer); + } + + protected function doAsk() + { + $this->writePrompt(); + + $inputStream = STDIN; + $autocomplete = $this->question->getAutocompleterValues(); + + if (null === $autocomplete || !$this->hasSttyAvailable()) { + $ret = false; + if ($this->question->isHidden()) { + try { + $ret = trim($this->getHiddenResponse($inputStream)); + } catch (\RuntimeException $e) { + if (!$this->question->isHiddenFallback()) { + throw $e; + } + } + } + + if (false === $ret) { + $ret = fgets($inputStream, 4096); + if (false === $ret) { + throw new \RuntimeException('Aborted'); + } + $ret = trim($ret); + } + } else { + $ret = trim($this->autocomplete($inputStream)); + } + + $ret = strlen($ret) > 0 ? $ret : $this->question->getDefault(); + + if ($normalizer = $this->question->getNormalizer()) { + return $normalizer($ret); + } + + return $ret; + } + + private function autocomplete($inputStream) + { + $autocomplete = $this->question->getAutocompleterValues(); + $ret = ''; + + $i = 0; + $ofs = -1; + $matches = $autocomplete; + $numMatches = count($matches); + + $sttyMode = shell_exec('stty -g'); + + shell_exec('stty -icanon -echo'); + + while (!feof($inputStream)) { + $c = fread($inputStream, 1); + + if ("\177" === $c) { + if (0 === $numMatches && 0 !== $i) { + --$i; + $this->output->write("\033[1D"); + } + + if ($i === 0) { + $ofs = -1; + $matches = $autocomplete; + $numMatches = count($matches); + } else { + $numMatches = 0; + } + + $ret = substr($ret, 0, $i); + } elseif ("\033" === $c) { + $c .= fread($inputStream, 2); + + if (isset($c[2]) && ('A' === $c[2] || 'B' === $c[2])) { + if ('A' === $c[2] && -1 === $ofs) { + $ofs = 0; + } + + if (0 === $numMatches) { + continue; + } + + $ofs += ('A' === $c[2]) ? -1 : 1; + $ofs = ($numMatches + $ofs) % $numMatches; + } + } elseif (ord($c) < 32) { + if ("\t" === $c || "\n" === $c) { + if ($numMatches > 0 && -1 !== $ofs) { + $ret = $matches[$ofs]; + $this->output->write(substr($ret, $i)); + $i = strlen($ret); + } + + if ("\n" === $c) { + $this->output->write($c); + break; + } + + $numMatches = 0; + } + + continue; + } else { + $this->output->write($c); + $ret .= $c; + ++$i; + + $numMatches = 0; + $ofs = 0; + + foreach ($autocomplete as $value) { + if (0 === strpos($value, $ret) && $i !== strlen($value)) { + $matches[$numMatches++] = $value; + } + } + } + + $this->output->write("\033[K"); + + if ($numMatches > 0 && -1 !== $ofs) { + $this->output->write("\0337"); + $this->output->highlight(substr($matches[$ofs], $i)); + $this->output->write("\0338"); + } + } + + shell_exec(sprintf('stty %s', $sttyMode)); + + return $ret; + } + + protected function getHiddenResponse($inputStream) + { + if ('\\' === DIRECTORY_SEPARATOR) { + $exe = __DIR__ . '/../bin/hiddeninput.exe'; + + $value = rtrim(shell_exec($exe)); + $this->output->writeln(''); + + if (isset($tmpExe)) { + unlink($tmpExe); + } + + return $value; + } + + if ($this->hasSttyAvailable()) { + $sttyMode = shell_exec('stty -g'); + + shell_exec('stty -echo'); + $value = fgets($inputStream, 4096); + shell_exec(sprintf('stty %s', $sttyMode)); + + if (false === $value) { + throw new \RuntimeException('Aborted'); + } + + $value = trim($value); + $this->output->writeln(''); + + return $value; + } + + if (false !== $shell = $this->getShell()) { + $readCmd = $shell === 'csh' ? 'set mypassword = $<' : 'read -r mypassword'; + $command = sprintf("/usr/bin/env %s -c 'stty -echo; %s; stty echo; echo \$mypassword'", $shell, $readCmd); + $value = rtrim(shell_exec($command)); + $this->output->writeln(''); + + return $value; + } + + throw new \RuntimeException('Unable to hide the response.'); + } + + protected function validateAttempts($interviewer) + { + /** @var \Exception $error */ + $error = null; + $attempts = $this->question->getMaxAttempts(); + while (null === $attempts || $attempts--) { + if (null !== $error) { + $this->output->error($error->getMessage()); + } + + try { + return call_user_func($this->question->getValidator(), $interviewer()); + } catch (\Exception $error) { + } + } + + throw $error; + } + + /** + * 鏄剧ず闂鐨勬彁绀轰俊鎭 + */ + protected function writePrompt() + { + $text = $this->question->getQuestion(); + $default = $this->question->getDefault(); + + switch (true) { + case null === $default: + $text = sprintf(' %s:', $text); + + break; + + case $this->question instanceof Confirmation: + $text = sprintf(' %s (yes/no) [%s]:', $text, $default ? 'yes' : 'no'); + + break; + + case $this->question instanceof Choice && $this->question->isMultiselect(): + $choices = $this->question->getChoices(); + $default = explode(',', $default); + + foreach ($default as $key => $value) { + $default[$key] = $choices[trim($value)]; + } + + $text = sprintf(' %s [%s]:', $text, implode(', ', $default)); + + break; + + case $this->question instanceof Choice: + $choices = $this->question->getChoices(); + $text = sprintf(' %s [%s]:', $text, $choices[$default]); + + break; + + default: + $text = sprintf(' %s [%s]:', $text, $default); + } + + $this->output->writeln($text); + + if ($this->question instanceof Choice) { + $width = max(array_map('strlen', array_keys($this->question->getChoices()))); + + foreach ($this->question->getChoices() as $key => $value) { + $this->output->writeln(sprintf(" [%-${width}s] %s", $key, $value)); + } + } + + $this->output->write(' > '); + } + + private function getShell() + { + if (null !== self::$shell) { + return self::$shell; + } + + self::$shell = false; + + if (file_exists('/usr/bin/env')) { + $test = "/usr/bin/env %s -c 'echo OK' 2> /dev/null"; + foreach (['bash', 'zsh', 'ksh', 'csh'] as $sh) { + if ('OK' === rtrim(shell_exec(sprintf($test, $sh)))) { + self::$shell = $sh; + break; + } + } + } + + return self::$shell; + } + + private function hasSttyAvailable() + { + if (null !== self::$stty) { + return self::$stty; + } + + exec('stty 2>&1', $output, $exitcode); + + return self::$stty = $exitcode === 0; + } +} diff --git a/thinkphp/library/think/console/output/Descriptor.php b/thinkphp/library/think/console/output/Descriptor.php new file mode 100644 index 000000000..6d98d53c7 --- /dev/null +++ b/thinkphp/library/think/console/output/Descriptor.php @@ -0,0 +1,319 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output; + +use think\Console; +use think\console\Command; +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\descriptor\Console as ConsoleDescription; + +class Descriptor +{ + + /** + * @var Output + */ + protected $output; + + /** + * {@inheritdoc} + */ + public function describe(Output $output, $object, array $options = []) + { + $this->output = $output; + + switch (true) { + case $object instanceof InputArgument: + $this->describeInputArgument($object, $options); + break; + case $object instanceof InputOption: + $this->describeInputOption($object, $options); + break; + case $object instanceof InputDefinition: + $this->describeInputDefinition($object, $options); + break; + case $object instanceof Command: + $this->describeCommand($object, $options); + break; + case $object instanceof Console: + $this->describeConsole($object, $options); + break; + default: + throw new \InvalidArgumentException(sprintf('Object of type "%s" is not describable.', get_class($object))); + } + } + + /** + * 杈撳嚭鍐呭 + * @param string $content + * @param bool $decorated + */ + protected function write($content, $decorated = false) + { + $this->output->write($content, false, $decorated ? Output::OUTPUT_NORMAL : Output::OUTPUT_RAW); + } + + /** + * 鎻忚堪鍙傛暟 + * @param InputArgument $argument + * @param array $options + * @return string|mixed + */ + protected function describeInputArgument(InputArgument $argument, array $options = []) + { + if (null !== $argument->getDefault() + && (!is_array($argument->getDefault()) + || count($argument->getDefault())) + ) { + $default = sprintf(' [default: %s]', $this->formatDefaultValue($argument->getDefault())); + } else { + $default = ''; + } + + $totalWidth = isset($options['total_width']) ? $options['total_width'] : strlen($argument->getName()); + $spacingWidth = $totalWidth - strlen($argument->getName()) + 2; + + $this->writeText(sprintf(" %s%s%s%s", $argument->getName(), str_repeat(' ', $spacingWidth), // + 17 = 2 spaces + + + 2 spaces + preg_replace('/\s*\R\s*/', PHP_EOL . str_repeat(' ', $totalWidth + 17), $argument->getDescription()), $default), $options); + } + + /** + * 鎻忚堪閫夐」 + * @param InputOption $option + * @param array $options + * @return string|mixed + */ + protected function describeInputOption(InputOption $option, array $options = []) + { + if ($option->acceptValue() && null !== $option->getDefault() + && (!is_array($option->getDefault()) + || count($option->getDefault())) + ) { + $default = sprintf(' [default: %s]', $this->formatDefaultValue($option->getDefault())); + } else { + $default = ''; + } + + $value = ''; + if ($option->acceptValue()) { + $value = '=' . strtoupper($option->getName()); + + if ($option->isValueOptional()) { + $value = '[' . $value . ']'; + } + } + + $totalWidth = isset($options['total_width']) ? $options['total_width'] : $this->calculateTotalWidthForOptions([$option]); + $synopsis = sprintf('%s%s', $option->getShortcut() ? sprintf('-%s, ', $option->getShortcut()) : ' ', sprintf('--%s%s', $option->getName(), $value)); + + $spacingWidth = $totalWidth - strlen($synopsis) + 2; + + $this->writeText(sprintf(" %s%s%s%s%s", $synopsis, str_repeat(' ', $spacingWidth), // + 17 = 2 spaces + + + 2 spaces + preg_replace('/\s*\R\s*/', "\n" . str_repeat(' ', $totalWidth + 17), $option->getDescription()), $default, $option->isArray() ? ' (multiple values allowed)' : ''), $options); + } + + /** + * 鎻忚堪杈撳叆 + * @param InputDefinition $definition + * @param array $options + * @return string|mixed + */ + protected function describeInputDefinition(InputDefinition $definition, array $options = []) + { + $totalWidth = $this->calculateTotalWidthForOptions($definition->getOptions()); + foreach ($definition->getArguments() as $argument) { + $totalWidth = max($totalWidth, strlen($argument->getName())); + } + + if ($definition->getArguments()) { + $this->writeText('Arguments:', $options); + $this->writeText("\n"); + foreach ($definition->getArguments() as $argument) { + $this->describeInputArgument($argument, array_merge($options, ['total_width' => $totalWidth])); + $this->writeText("\n"); + } + } + + if ($definition->getArguments() && $definition->getOptions()) { + $this->writeText("\n"); + } + + if ($definition->getOptions()) { + $laterOptions = []; + + $this->writeText('Options:', $options); + foreach ($definition->getOptions() as $option) { + if (strlen($option->getShortcut()) > 1) { + $laterOptions[] = $option; + continue; + } + $this->writeText("\n"); + $this->describeInputOption($option, array_merge($options, ['total_width' => $totalWidth])); + } + foreach ($laterOptions as $option) { + $this->writeText("\n"); + $this->describeInputOption($option, array_merge($options, ['total_width' => $totalWidth])); + } + } + } + + /** + * 鎻忚堪鎸囦护 + * @param Command $command + * @param array $options + * @return string|mixed + */ + protected function describeCommand(Command $command, array $options = []) + { + $command->getSynopsis(true); + $command->getSynopsis(false); + $command->mergeConsoleDefinition(false); + + $this->writeText('Usage:', $options); + foreach (array_merge([$command->getSynopsis(true)], $command->getAliases(), $command->getUsages()) as $usage) { + $this->writeText("\n"); + $this->writeText(' ' . $usage, $options); + } + $this->writeText("\n"); + + $definition = $command->getNativeDefinition(); + if ($definition->getOptions() || $definition->getArguments()) { + $this->writeText("\n"); + $this->describeInputDefinition($definition, $options); + $this->writeText("\n"); + } + + if ($help = $command->getProcessedHelp()) { + $this->writeText("\n"); + $this->writeText('Help:', $options); + $this->writeText("\n"); + $this->writeText(' ' . str_replace("\n", "\n ", $help), $options); + $this->writeText("\n"); + } + } + + /** + * 鎻忚堪鎺у埗鍙 + * @param Console $console + * @param array $options + * @return string|mixed + */ + protected function describeConsole(Console $console, array $options = []) + { + $describedNamespace = isset($options['namespace']) ? $options['namespace'] : null; + $description = new ConsoleDescription($console, $describedNamespace); + + if (isset($options['raw_text']) && $options['raw_text']) { + $width = $this->getColumnWidth($description->getCommands()); + + foreach ($description->getCommands() as $command) { + $this->writeText(sprintf("%-${width}s %s", $command->getName(), $command->getDescription()), $options); + $this->writeText("\n"); + } + } else { + if ('' != $help = $console->getHelp()) { + $this->writeText("$help\n\n", $options); + } + + $this->writeText("Usage:\n", $options); + $this->writeText(" command [options] [arguments]\n\n", $options); + + $this->describeInputDefinition(new InputDefinition($console->getDefinition()->getOptions()), $options); + + $this->writeText("\n"); + $this->writeText("\n"); + + $width = $this->getColumnWidth($description->getCommands()); + + if ($describedNamespace) { + $this->writeText(sprintf('Available commands for the "%s" namespace:', $describedNamespace), $options); + } else { + $this->writeText('Available commands:', $options); + } + + // add commands by namespace + foreach ($description->getNamespaces() as $namespace) { + if (!$describedNamespace && ConsoleDescription::GLOBAL_NAMESPACE !== $namespace['id']) { + $this->writeText("\n"); + $this->writeText(' ' . $namespace['id'] . '', $options); + } + + foreach ($namespace['commands'] as $name) { + $this->writeText("\n"); + $spacingWidth = $width - strlen($name); + $this->writeText(sprintf(" %s%s%s", $name, str_repeat(' ', $spacingWidth), $description->getCommand($name) + ->getDescription()), $options); + } + } + + $this->writeText("\n"); + } + } + + /** + * {@inheritdoc} + */ + private function writeText($content, array $options = []) + { + $this->write(isset($options['raw_text']) + && $options['raw_text'] ? strip_tags($content) : $content, isset($options['raw_output']) ? !$options['raw_output'] : true); + } + + /** + * 鏍煎紡鍖 + * @param mixed $default + * @return string + */ + private function formatDefaultValue($default) + { + return json_encode($default, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE); + } + + /** + * @param Command[] $commands + * @return int + */ + private function getColumnWidth(array $commands) + { + $width = 0; + foreach ($commands as $command) { + $width = strlen($command->getName()) > $width ? strlen($command->getName()) : $width; + } + + return $width + 2; + } + + /** + * @param InputOption[] $options + * @return int + */ + private function calculateTotalWidthForOptions($options) + { + $totalWidth = 0; + foreach ($options as $option) { + $nameLength = 4 + strlen($option->getName()) + 2; // - + shortcut + , + whitespace + name + -- + + if ($option->acceptValue()) { + $valueLength = 1 + strlen($option->getName()); // = + value + $valueLength += $option->isValueOptional() ? 2 : 0; // [ + ] + + $nameLength += $valueLength; + } + $totalWidth = max($totalWidth, $nameLength); + } + + return $totalWidth; + } +} diff --git a/thinkphp/library/think/console/output/Formatter.php b/thinkphp/library/think/console/output/Formatter.php new file mode 100644 index 000000000..f8bee5527 --- /dev/null +++ b/thinkphp/library/think/console/output/Formatter.php @@ -0,0 +1,198 @@ + +// +---------------------------------------------------------------------- +namespace think\console\output; + +use think\console\output\formatter\Stack as StyleStack; +use think\console\output\formatter\Style; + +class Formatter +{ + + private $decorated = false; + private $styles = []; + private $styleStack; + + /** + * 杞箟 + * @param string $text + * @return string + */ + public static function escape($text) + { + return preg_replace('/([^\\\\]?)setStyle('error', new Style('white', 'red')); + $this->setStyle('info', new Style('green')); + $this->setStyle('comment', new Style('yellow')); + $this->setStyle('question', new Style('black', 'cyan')); + $this->setStyle('highlight', new Style('red')); + $this->setStyle('warning', new Style('black', 'yellow')); + + $this->styleStack = new StyleStack(); + } + + /** + * 璁剧疆澶栬鏍囪瘑 + * @param bool $decorated 鏄惁缇庡寲鏂囧瓧 + */ + public function setDecorated($decorated) + { + $this->decorated = (bool) $decorated; + } + + /** + * 鑾峰彇澶栬鏍囪瘑 + * @return bool + */ + public function isDecorated() + { + return $this->decorated; + } + + /** + * 娣诲姞涓涓柊鏍峰紡 + * @param string $name 鏍峰紡鍚 + * @param Style $style 鏍峰紡瀹炰緥 + */ + public function setStyle($name, Style $style) + { + $this->styles[strtolower($name)] = $style; + } + + /** + * 鏄惁鏈夎繖涓牱寮 + * @param string $name + * @return bool + */ + public function hasStyle($name) + { + return isset($this->styles[strtolower($name)]); + } + + /** + * 鑾峰彇鏍峰紡 + * @param string $name + * @return Style + * @throws \InvalidArgumentException + */ + public function getStyle($name) + { + if (!$this->hasStyle($name)) { + throw new \InvalidArgumentException(sprintf('Undefined style: %s', $name)); + } + + return $this->styles[strtolower($name)]; + } + + /** + * 浣跨敤鎵缁欑殑鏍峰紡鏍煎紡鍖栨枃瀛 + * @param string $message 鏂囧瓧 + * @return string + */ + public function format($message) + { + $offset = 0; + $output = ''; + $tagRegex = '[a-z][a-z0-9_=;-]*'; + preg_match_all("#<(($tagRegex) | /($tagRegex)?)>#isx", $message, $matches, PREG_OFFSET_CAPTURE); + foreach ($matches[0] as $i => $match) { + $pos = $match[1]; + $text = $match[0]; + + if (0 != $pos && '\\' == $message[$pos - 1]) { + continue; + } + + $output .= $this->applyCurrentStyle(substr($message, $offset, $pos - $offset)); + $offset = $pos + strlen($text); + + if ($open = '/' != $text[1]) { + $tag = $matches[1][$i][0]; + } else { + $tag = isset($matches[3][$i][0]) ? $matches[3][$i][0] : ''; + } + + if (!$open && !$tag) { + // + $this->styleStack->pop(); + } elseif (false === $style = $this->createStyleFromString(strtolower($tag))) { + $output .= $this->applyCurrentStyle($text); + } elseif ($open) { + $this->styleStack->push($style); + } else { + $this->styleStack->pop($style); + } + } + + $output .= $this->applyCurrentStyle(substr($message, $offset)); + + return str_replace('\\<', '<', $output); + } + + /** + * @return StyleStack + */ + public function getStyleStack() + { + return $this->styleStack; + } + + /** + * 鏍规嵁瀛楃涓插垱寤烘柊鐨勬牱寮忓疄渚 + * @param string $string + * @return Style|bool + */ + private function createStyleFromString($string) + { + if (isset($this->styles[$string])) { + return $this->styles[$string]; + } + + if (!preg_match_all('/([^=]+)=([^;]+)(;|$)/', strtolower($string), $matches, PREG_SET_ORDER)) { + return false; + } + + $style = new Style(); + foreach ($matches as $match) { + array_shift($match); + + if ('fg' == $match[0]) { + $style->setForeground($match[1]); + } elseif ('bg' == $match[0]) { + $style->setBackground($match[1]); + } else { + try { + $style->setOption($match[1]); + } catch (\InvalidArgumentException $e) { + return false; + } + } + } + + return $style; + } + + /** + * 浠庡爢鏍堝簲鐢ㄦ牱寮忓埌鏂囧瓧 + * @param string $text 鏂囧瓧 + * @return string + */ + private function applyCurrentStyle($text) + { + return $this->isDecorated() && strlen($text) > 0 ? $this->styleStack->getCurrent()->apply($text) : $text; + } +} diff --git a/thinkphp/library/think/console/output/Question.php b/thinkphp/library/think/console/output/Question.php new file mode 100644 index 000000000..03975f274 --- /dev/null +++ b/thinkphp/library/think/console/output/Question.php @@ -0,0 +1,211 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output; + +class Question +{ + + private $question; + private $attempts; + private $hidden = false; + private $hiddenFallback = true; + private $autocompleterValues; + private $validator; + private $default; + private $normalizer; + + /** + * 鏋勯犳柟娉 + * @param string $question 闂 + * @param mixed $default 榛樿绛旀 + */ + public function __construct($question, $default = null) + { + $this->question = $question; + $this->default = $default; + } + + /** + * 鑾峰彇闂 + * @return string + */ + public function getQuestion() + { + return $this->question; + } + + /** + * 鑾峰彇榛樿绛旀 + * @return mixed + */ + public function getDefault() + { + return $this->default; + } + + /** + * 鏄惁闅愯棌绛旀 + * @return bool + */ + public function isHidden() + { + return $this->hidden; + } + + /** + * 闅愯棌绛旀 + * @param bool $hidden + * @return Question + */ + public function setHidden($hidden) + { + if ($this->autocompleterValues) { + throw new \LogicException('A hidden question cannot use the autocompleter.'); + } + + $this->hidden = (bool) $hidden; + + return $this; + } + + /** + * 涓嶈兘琚殣钘忔槸鍚︽挙閿 + * @return bool + */ + public function isHiddenFallback() + { + return $this->hiddenFallback; + } + + /** + * 璁剧疆涓嶈兘琚殣钘忕殑鏃跺欑殑鎿嶄綔 + * @param bool $fallback + * @return Question + */ + public function setHiddenFallback($fallback) + { + $this->hiddenFallback = (bool) $fallback; + + return $this; + } + + /** + * 鑾峰彇鑷姩瀹屾垚 + * @return null|array|\Traversable + */ + public function getAutocompleterValues() + { + return $this->autocompleterValues; + } + + /** + * 璁剧疆鑷姩瀹屾垚鐨勫 + * @param null|array|\Traversable $values + * @return Question + * @throws \InvalidArgumentException + * @throws \LogicException + */ + public function setAutocompleterValues($values) + { + if (is_array($values) && $this->isAssoc($values)) { + $values = array_merge(array_keys($values), array_values($values)); + } + + if (null !== $values && !is_array($values)) { + if (!$values instanceof \Traversable || $values instanceof \Countable) { + throw new \InvalidArgumentException('Autocompleter values can be either an array, `null` or an object implementing both `Countable` and `Traversable` interfaces.'); + } + } + + if ($this->hidden) { + throw new \LogicException('A hidden question cannot use the autocompleter.'); + } + + $this->autocompleterValues = $values; + + return $this; + } + + /** + * 璁剧疆绛旀鐨勯獙璇佸櫒 + * @param null|callable $validator + * @return Question The current instance + */ + public function setValidator($validator) + { + $this->validator = $validator; + + return $this; + } + + /** + * 鑾峰彇楠岃瘉鍣 + * @return null|callable + */ + public function getValidator() + { + return $this->validator; + } + + /** + * 璁剧疆鏈澶ч噸璇曟鏁 + * @param null|int $attempts + * @return Question + * @throws \InvalidArgumentException + */ + public function setMaxAttempts($attempts) + { + if (null !== $attempts && $attempts < 1) { + throw new \InvalidArgumentException('Maximum number of attempts must be a positive value.'); + } + + $this->attempts = $attempts; + + return $this; + } + + /** + * 鑾峰彇鏈澶ч噸璇曟鏁 + * @return null|int + */ + public function getMaxAttempts() + { + return $this->attempts; + } + + /** + * 璁剧疆鍝嶅簲鐨勫洖璋 + * @param string|\Closure $normalizer + * @return Question + */ + public function setNormalizer($normalizer) + { + $this->normalizer = $normalizer; + + return $this; + } + + /** + * 鑾峰彇鍝嶅簲鍥炶皟 + * The normalizer can ba a callable (a string), a closure or a class implementing __invoke. + * @return string|\Closure + */ + public function getNormalizer() + { + return $this->normalizer; + } + + protected function isAssoc($array) + { + return (bool) count(array_filter(array_keys($array), 'is_string')); + } +} diff --git a/thinkphp/library/think/console/output/descriptor/Console.php b/thinkphp/library/think/console/output/descriptor/Console.php new file mode 100644 index 000000000..4648b68e6 --- /dev/null +++ b/thinkphp/library/think/console/output/descriptor/Console.php @@ -0,0 +1,149 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\descriptor; + +use think\Console as ThinkConsole; +use think\console\Command; + +class Console +{ + + const GLOBAL_NAMESPACE = '_global'; + + /** + * @var ThinkConsole + */ + private $console; + + /** + * @var null|string + */ + private $namespace; + + /** + * @var array + */ + private $namespaces; + + /** + * @var Command[] + */ + private $commands; + + /** + * @var Command[] + */ + private $aliases; + + /** + * 鏋勯犳柟娉 + * @param ThinkConsole $console + * @param string|null $namespace + */ + public function __construct(ThinkConsole $console, $namespace = null) + { + $this->console = $console; + $this->namespace = $namespace; + } + + /** + * @return array + */ + public function getNamespaces() + { + if (null === $this->namespaces) { + $this->inspectConsole(); + } + + return $this->namespaces; + } + + /** + * @return Command[] + */ + public function getCommands() + { + if (null === $this->commands) { + $this->inspectConsole(); + } + + return $this->commands; + } + + /** + * @param string $name + * @return Command + * @throws \InvalidArgumentException + */ + public function getCommand($name) + { + if (!isset($this->commands[$name]) && !isset($this->aliases[$name])) { + throw new \InvalidArgumentException(sprintf('Command %s does not exist.', $name)); + } + + return isset($this->commands[$name]) ? $this->commands[$name] : $this->aliases[$name]; + } + + private function inspectConsole() + { + $this->commands = []; + $this->namespaces = []; + + $all = $this->console->all($this->namespace ? $this->console->findNamespace($this->namespace) : null); + foreach ($this->sortCommands($all) as $namespace => $commands) { + $names = []; + + /** @var Command $command */ + foreach ($commands as $name => $command) { + if (!$command->getName()) { + continue; + } + + if ($command->getName() === $name) { + $this->commands[$name] = $command; + } else { + $this->aliases[$name] = $command; + } + + $names[] = $name; + } + + $this->namespaces[$namespace] = ['id' => $namespace, 'commands' => $names]; + } + } + + /** + * @param array $commands + * @return array + */ + private function sortCommands(array $commands) + { + $namespacedCommands = []; + foreach ($commands as $name => $command) { + $key = $this->console->extractNamespace($name, 1); + if (!$key) { + $key = self::GLOBAL_NAMESPACE; + } + + $namespacedCommands[$key][$name] = $command; + } + ksort($namespacedCommands); + + foreach ($namespacedCommands as &$commandsSet) { + ksort($commandsSet); + } + // unset reference to keep scope clear + unset($commandsSet); + + return $namespacedCommands; + } +} diff --git a/thinkphp/library/think/console/output/driver/Buffer.php b/thinkphp/library/think/console/output/driver/Buffer.php new file mode 100644 index 000000000..c77a2ec4b --- /dev/null +++ b/thinkphp/library/think/console/output/driver/Buffer.php @@ -0,0 +1,52 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\driver; + +use think\console\Output; + +class Buffer +{ + /** + * @var string + */ + private $buffer = ''; + + public function __construct(Output $output) + { + // do nothing + } + + public function fetch() + { + $content = $this->buffer; + $this->buffer = ''; + return $content; + } + + public function write($messages, $newline = false, $options = Output::OUTPUT_NORMAL) + { + $messages = (array) $messages; + + foreach ($messages as $message) { + $this->buffer .= $message; + } + if ($newline) { + $this->buffer .= "\n"; + } + } + + public function renderException(\Exception $e) + { + // do nothing + } + +} diff --git a/thinkphp/library/think/console/output/driver/Console.php b/thinkphp/library/think/console/output/driver/Console.php new file mode 100644 index 000000000..3fbe224f9 --- /dev/null +++ b/thinkphp/library/think/console/output/driver/Console.php @@ -0,0 +1,368 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\driver; + +use think\console\Output; +use think\console\output\Formatter; + +class Console +{ + + /** @var Resource */ + private $stdout; + + /** @var Formatter */ + private $formatter; + + private $terminalDimensions; + + /** @var Output */ + private $output; + + public function __construct(Output $output) + { + $this->output = $output; + $this->formatter = new Formatter(); + $this->stdout = $this->openOutputStream(); + $decorated = $this->hasColorSupport($this->stdout); + $this->formatter->setDecorated($decorated); + } + + public function setDecorated($decorated) + { + $this->formatter->setDecorated($decorated); + } + + public function write($messages, $newline = false, $type = Output::OUTPUT_NORMAL, $stream = null) + { + if (Output::VERBOSITY_QUIET === $this->output->getVerbosity()) { + return; + } + + $messages = (array) $messages; + + foreach ($messages as $message) { + switch ($type) { + case Output::OUTPUT_NORMAL: + $message = $this->formatter->format($message); + break; + case Output::OUTPUT_RAW: + break; + case Output::OUTPUT_PLAIN: + $message = strip_tags($this->formatter->format($message)); + break; + default: + throw new \InvalidArgumentException(sprintf('Unknown output type given (%s)', $type)); + } + + $this->doWrite($message, $newline, $stream); + } + } + + public function renderException(\Exception $e) + { + $stderr = $this->openErrorStream(); + $decorated = $this->hasColorSupport($stderr); + $this->formatter->setDecorated($decorated); + + do { + $title = sprintf(' [%s] ', get_class($e)); + + $len = $this->stringWidth($title); + + $width = $this->getTerminalWidth() ? $this->getTerminalWidth() - 1 : PHP_INT_MAX; + + if (defined('HHVM_VERSION') && $width > 1 << 31) { + $width = 1 << 31; + } + $lines = []; + foreach (preg_split('/\r?\n/', $e->getMessage()) as $line) { + foreach ($this->splitStringByWidth($line, $width - 4) as $line) { + + $lineLength = $this->stringWidth(preg_replace('/\[[^m]*m/', '', $line)) + 4; + $lines[] = [$line, $lineLength]; + + $len = max($lineLength, $len); + } + } + + $messages = ['', '']; + $messages[] = $emptyLine = sprintf('%s', str_repeat(' ', $len)); + $messages[] = sprintf('%s%s', $title, str_repeat(' ', max(0, $len - $this->stringWidth($title)))); + foreach ($lines as $line) { + $messages[] = sprintf(' %s %s', $line[0], str_repeat(' ', $len - $line[1])); + } + $messages[] = $emptyLine; + $messages[] = ''; + $messages[] = ''; + + $this->write($messages, true, Output::OUTPUT_NORMAL, $stderr); + + if (Output::VERBOSITY_VERBOSE <= $this->output->getVerbosity()) { + $this->write('Exception trace:', true, Output::OUTPUT_NORMAL, $stderr); + + // exception related properties + $trace = $e->getTrace(); + array_unshift($trace, [ + 'function' => '', + 'file' => $e->getFile() !== null ? $e->getFile() : 'n/a', + 'line' => $e->getLine() !== null ? $e->getLine() : 'n/a', + 'args' => [], + ]); + + for ($i = 0, $count = count($trace); $i < $count; ++$i) { + $class = isset($trace[$i]['class']) ? $trace[$i]['class'] : ''; + $type = isset($trace[$i]['type']) ? $trace[$i]['type'] : ''; + $function = $trace[$i]['function']; + $file = isset($trace[$i]['file']) ? $trace[$i]['file'] : 'n/a'; + $line = isset($trace[$i]['line']) ? $trace[$i]['line'] : 'n/a'; + + $this->write(sprintf(' %s%s%s() at %s:%s', $class, $type, $function, $file, $line), true, Output::OUTPUT_NORMAL, $stderr); + } + + $this->write('', true, Output::OUTPUT_NORMAL, $stderr); + $this->write('', true, Output::OUTPUT_NORMAL, $stderr); + } + } while ($e = $e->getPrevious()); + + } + + /** + * 鑾峰彇缁堢瀹藉害 + * @return int|null + */ + protected function getTerminalWidth() + { + $dimensions = $this->getTerminalDimensions(); + + return $dimensions[0]; + } + + /** + * 鑾峰彇缁堢楂樺害 + * @return int|null + */ + protected function getTerminalHeight() + { + $dimensions = $this->getTerminalDimensions(); + + return $dimensions[1]; + } + + /** + * 鑾峰彇褰撳墠缁堢鐨勫昂瀵 + * @return array + */ + public function getTerminalDimensions() + { + if ($this->terminalDimensions) { + return $this->terminalDimensions; + } + + if ('\\' === DS) { + if (preg_match('/^(\d+)x\d+ \(\d+x(\d+)\)$/', trim(getenv('ANSICON')), $matches)) { + return [(int) $matches[1], (int) $matches[2]]; + } + if (preg_match('/^(\d+)x(\d+)$/', $this->getMode(), $matches)) { + return [(int) $matches[1], (int) $matches[2]]; + } + } + + if ($sttyString = $this->getSttyColumns()) { + if (preg_match('/rows.(\d+);.columns.(\d+);/i', $sttyString, $matches)) { + return [(int) $matches[2], (int) $matches[1]]; + } + if (preg_match('/;.(\d+).rows;.(\d+).columns/i', $sttyString, $matches)) { + return [(int) $matches[2], (int) $matches[1]]; + } + } + + return [null, null]; + } + + /** + * 鑾峰彇stty鍒楁暟 + * @return string + */ + private function getSttyColumns() + { + if (!function_exists('proc_open')) { + return; + } + + $descriptorspec = [1 => ['pipe', 'w'], 2 => ['pipe', 'w']]; + $process = proc_open('stty -a | grep columns', $descriptorspec, $pipes, null, null, ['suppress_errors' => true]); + if (is_resource($process)) { + $info = stream_get_contents($pipes[1]); + fclose($pipes[1]); + fclose($pipes[2]); + proc_close($process); + + return $info; + } + return; + } + + /** + * 鑾峰彇缁堢妯″紡 + * @return string x 鎴 null + */ + private function getMode() + { + if (!function_exists('proc_open')) { + return; + } + + $descriptorspec = [1 => ['pipe', 'w'], 2 => ['pipe', 'w']]; + $process = proc_open('mode CON', $descriptorspec, $pipes, null, null, ['suppress_errors' => true]); + if (is_resource($process)) { + $info = stream_get_contents($pipes[1]); + fclose($pipes[1]); + fclose($pipes[2]); + proc_close($process); + + if (preg_match('/--------+\r?\n.+?(\d+)\r?\n.+?(\d+)\r?\n/', $info, $matches)) { + return $matches[2] . 'x' . $matches[1]; + } + } + return; + } + + private function stringWidth($string) + { + if (!function_exists('mb_strwidth')) { + return strlen($string); + } + + if (false === $encoding = mb_detect_encoding($string)) { + return strlen($string); + } + + return mb_strwidth($string, $encoding); + } + + private function splitStringByWidth($string, $width) + { + if (!function_exists('mb_strwidth')) { + return str_split($string, $width); + } + + if (false === $encoding = mb_detect_encoding($string)) { + return str_split($string, $width); + } + + $utf8String = mb_convert_encoding($string, 'utf8', $encoding); + $lines = []; + $line = ''; + foreach (preg_split('//u', $utf8String) as $char) { + if (mb_strwidth($line . $char, 'utf8') <= $width) { + $line .= $char; + continue; + } + $lines[] = str_pad($line, $width); + $line = $char; + } + if (strlen($line)) { + $lines[] = count($lines) ? str_pad($line, $width) : $line; + } + + mb_convert_variables($encoding, 'utf8', $lines); + + return $lines; + } + + private function isRunningOS400() + { + $checks = [ + function_exists('php_uname') ? php_uname('s') : '', + getenv('OSTYPE'), + PHP_OS, + ]; + return false !== stripos(implode(';', $checks), 'OS400'); + } + + /** + * 褰撳墠鐜鏄惁鏀寔鍐欏叆鎺у埗鍙拌緭鍑哄埌stdout. + * + * @return bool + */ + protected function hasStdoutSupport() + { + return false === $this->isRunningOS400(); + } + + /** + * 褰撳墠鐜鏄惁鏀寔鍐欏叆鎺у埗鍙拌緭鍑哄埌stderr. + * + * @return bool + */ + protected function hasStderrSupport() + { + return false === $this->isRunningOS400(); + } + + /** + * @return resource + */ + private function openOutputStream() + { + if (!$this->hasStdoutSupport()) { + return fopen('php://output', 'w'); + } + return @fopen('php://stdout', 'w') ?: fopen('php://output', 'w'); + } + + /** + * @return resource + */ + private function openErrorStream() + { + return fopen($this->hasStderrSupport() ? 'php://stderr' : 'php://output', 'w'); + } + + /** + * 灏嗘秷鎭啓鍏ュ埌杈撳嚭銆 + * @param string $message 娑堟伅 + * @param bool $newline 鏄惁鍙﹁捣涓琛 + * @param null $stream + */ + protected function doWrite($message, $newline, $stream = null) + { + if (null === $stream) { + $stream = $this->stdout; + } + if (false === @fwrite($stream, $message . ($newline ? PHP_EOL : ''))) { + throw new \RuntimeException('Unable to write output.'); + } + + fflush($stream); + } + + /** + * 鏄惁鏀寔鐫鑹 + * @param $stream + * @return bool + */ + protected function hasColorSupport($stream) + { + if (DIRECTORY_SEPARATOR === '\\') { + return + '10.0.10586' === PHP_WINDOWS_VERSION_MAJOR . '.' . PHP_WINDOWS_VERSION_MINOR . '.' . PHP_WINDOWS_VERSION_BUILD + || false !== getenv('ANSICON') + || 'ON' === getenv('ConEmuANSI') + || 'xterm' === getenv('TERM'); + } + + return function_exists('posix_isatty') && @posix_isatty($stream); + } + +} diff --git a/thinkphp/library/think/console/output/driver/Nothing.php b/thinkphp/library/think/console/output/driver/Nothing.php new file mode 100644 index 000000000..9a55f7779 --- /dev/null +++ b/thinkphp/library/think/console/output/driver/Nothing.php @@ -0,0 +1,33 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\driver; + +use think\console\Output; + +class Nothing +{ + + public function __construct(Output $output) + { + // do nothing + } + + public function write($messages, $newline = false, $options = Output::OUTPUT_NORMAL) + { + // do nothing + } + + public function renderException(\Exception $e) + { + // do nothing + } +} diff --git a/thinkphp/library/think/console/output/formatter/Stack.php b/thinkphp/library/think/console/output/formatter/Stack.php new file mode 100644 index 000000000..4864a3f29 --- /dev/null +++ b/thinkphp/library/think/console/output/formatter/Stack.php @@ -0,0 +1,116 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\formatter; + +class Stack +{ + + /** + * @var Style[] + */ + private $styles; + + /** + * @var Style + */ + private $emptyStyle; + + /** + * 鏋勯犳柟娉 + * @param Style|null $emptyStyle + */ + public function __construct(Style $emptyStyle = null) + { + $this->emptyStyle = $emptyStyle ?: new Style(); + $this->reset(); + } + + /** + * 閲嶇疆鍫嗘爤 + */ + public function reset() + { + $this->styles = []; + } + + /** + * 鎺ㄤ竴涓牱寮忚繘鍏ュ爢鏍 + * @param Style $style + */ + public function push(Style $style) + { + $this->styles[] = $style; + } + + /** + * 浠庡爢鏍堜腑寮瑰嚭涓涓牱寮 + * @param Style|null $style + * @return Style + * @throws \InvalidArgumentException + */ + public function pop(Style $style = null) + { + if (empty($this->styles)) { + return $this->emptyStyle; + } + + if (null === $style) { + return array_pop($this->styles); + } + + /** + * @var int $index + * @var Style $stackedStyle + */ + foreach (array_reverse($this->styles, true) as $index => $stackedStyle) { + if ($style->apply('') === $stackedStyle->apply('')) { + $this->styles = array_slice($this->styles, 0, $index); + + return $stackedStyle; + } + } + + throw new \InvalidArgumentException('Incorrectly nested style tag found.'); + } + + /** + * 璁$畻鍫嗘爤鐨勫綋鍓嶆牱寮忋 + * @return Style + */ + public function getCurrent() + { + if (empty($this->styles)) { + return $this->emptyStyle; + } + + return $this->styles[count($this->styles) - 1]; + } + + /** + * @param Style $emptyStyle + * @return Stack + */ + public function setEmptyStyle(Style $emptyStyle) + { + $this->emptyStyle = $emptyStyle; + + return $this; + } + + /** + * @return Style + */ + public function getEmptyStyle() + { + return $this->emptyStyle; + } +} diff --git a/thinkphp/library/think/console/output/formatter/Style.php b/thinkphp/library/think/console/output/formatter/Style.php new file mode 100644 index 000000000..d9b099987 --- /dev/null +++ b/thinkphp/library/think/console/output/formatter/Style.php @@ -0,0 +1,189 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\formatter; + +class Style +{ + + private static $availableForegroundColors = [ + 'black' => ['set' => 30, 'unset' => 39], + 'red' => ['set' => 31, 'unset' => 39], + 'green' => ['set' => 32, 'unset' => 39], + 'yellow' => ['set' => 33, 'unset' => 39], + 'blue' => ['set' => 34, 'unset' => 39], + 'magenta' => ['set' => 35, 'unset' => 39], + 'cyan' => ['set' => 36, 'unset' => 39], + 'white' => ['set' => 37, 'unset' => 39], + ]; + private static $availableBackgroundColors = [ + 'black' => ['set' => 40, 'unset' => 49], + 'red' => ['set' => 41, 'unset' => 49], + 'green' => ['set' => 42, 'unset' => 49], + 'yellow' => ['set' => 43, 'unset' => 49], + 'blue' => ['set' => 44, 'unset' => 49], + 'magenta' => ['set' => 45, 'unset' => 49], + 'cyan' => ['set' => 46, 'unset' => 49], + 'white' => ['set' => 47, 'unset' => 49], + ]; + private static $availableOptions = [ + 'bold' => ['set' => 1, 'unset' => 22], + 'underscore' => ['set' => 4, 'unset' => 24], + 'blink' => ['set' => 5, 'unset' => 25], + 'reverse' => ['set' => 7, 'unset' => 27], + 'conceal' => ['set' => 8, 'unset' => 28], + ]; + + private $foreground; + private $background; + private $options = []; + + /** + * 鍒濆鍖栬緭鍑虹殑鏍峰紡 + * @param string|null $foreground 瀛椾綋棰滆壊 + * @param string|null $background 鑳屾櫙鑹 + * @param array $options 鏍煎紡 + * @api + */ + public function __construct($foreground = null, $background = null, array $options = []) + { + if (null !== $foreground) { + $this->setForeground($foreground); + } + if (null !== $background) { + $this->setBackground($background); + } + if (count($options)) { + $this->setOptions($options); + } + } + + /** + * 璁剧疆瀛椾綋棰滆壊 + * @param string|null $color 棰滆壊鍚 + * @throws \InvalidArgumentException + * @api + */ + public function setForeground($color = null) + { + if (null === $color) { + $this->foreground = null; + + return; + } + + if (!isset(static::$availableForegroundColors[$color])) { + throw new \InvalidArgumentException(sprintf('Invalid foreground color specified: "%s". Expected one of (%s)', $color, implode(', ', array_keys(static::$availableForegroundColors)))); + } + + $this->foreground = static::$availableForegroundColors[$color]; + } + + /** + * 璁剧疆鑳屾櫙鑹 + * @param string|null $color 棰滆壊鍚 + * @throws \InvalidArgumentException + * @api + */ + public function setBackground($color = null) + { + if (null === $color) { + $this->background = null; + + return; + } + + if (!isset(static::$availableBackgroundColors[$color])) { + throw new \InvalidArgumentException(sprintf('Invalid background color specified: "%s". Expected one of (%s)', $color, implode(', ', array_keys(static::$availableBackgroundColors)))); + } + + $this->background = static::$availableBackgroundColors[$color]; + } + + /** + * 璁剧疆瀛椾綋鏍煎紡 + * @param string $option 鏍煎紡鍚 + * @throws \InvalidArgumentException When the option name isn't defined + * @api + */ + public function setOption($option) + { + if (!isset(static::$availableOptions[$option])) { + throw new \InvalidArgumentException(sprintf('Invalid option specified: "%s". Expected one of (%s)', $option, implode(', ', array_keys(static::$availableOptions)))); + } + + if (!in_array(static::$availableOptions[$option], $this->options)) { + $this->options[] = static::$availableOptions[$option]; + } + } + + /** + * 閲嶇疆瀛椾綋鏍煎紡 + * @param string $option 鏍煎紡鍚 + * @throws \InvalidArgumentException + */ + public function unsetOption($option) + { + if (!isset(static::$availableOptions[$option])) { + throw new \InvalidArgumentException(sprintf('Invalid option specified: "%s". Expected one of (%s)', $option, implode(', ', array_keys(static::$availableOptions)))); + } + + $pos = array_search(static::$availableOptions[$option], $this->options); + if (false !== $pos) { + unset($this->options[$pos]); + } + } + + /** + * 鎵归噺璁剧疆瀛椾綋鏍煎紡 + * @param array $options + */ + public function setOptions(array $options) + { + $this->options = []; + + foreach ($options as $option) { + $this->setOption($option); + } + } + + /** + * 搴旂敤鏍峰紡鍒版枃瀛 + * @param string $text 鏂囧瓧 + * @return string + */ + public function apply($text) + { + $setCodes = []; + $unsetCodes = []; + + if (null !== $this->foreground) { + $setCodes[] = $this->foreground['set']; + $unsetCodes[] = $this->foreground['unset']; + } + if (null !== $this->background) { + $setCodes[] = $this->background['set']; + $unsetCodes[] = $this->background['unset']; + } + if (count($this->options)) { + foreach ($this->options as $option) { + $setCodes[] = $option['set']; + $unsetCodes[] = $option['unset']; + } + } + + if (0 === count($setCodes)) { + return $text; + } + + return sprintf("\033[%sm%s\033[%sm", implode(';', $setCodes), $text, implode(';', $unsetCodes)); + } +} diff --git a/thinkphp/library/think/console/output/question/Choice.php b/thinkphp/library/think/console/output/question/Choice.php new file mode 100644 index 000000000..f6760e5ef --- /dev/null +++ b/thinkphp/library/think/console/output/question/Choice.php @@ -0,0 +1,163 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\question; + +use think\console\output\Question; + +class Choice extends Question +{ + + private $choices; + private $multiselect = false; + private $prompt = ' > '; + private $errorMessage = 'Value "%s" is invalid'; + + /** + * 鏋勯犳柟娉 + * @param string $question 闂 + * @param array $choices 閫夐」 + * @param mixed $default 榛樿绛旀 + */ + public function __construct($question, array $choices, $default = null) + { + parent::__construct($question, $default); + + $this->choices = $choices; + $this->setValidator($this->getDefaultValidator()); + $this->setAutocompleterValues($choices); + } + + /** + * 鍙夐」 + * @return array + */ + public function getChoices() + { + return $this->choices; + } + + /** + * 璁剧疆鍙惁澶氶 + * @param bool $multiselect + * @return self + */ + public function setMultiselect($multiselect) + { + $this->multiselect = $multiselect; + $this->setValidator($this->getDefaultValidator()); + + return $this; + } + + public function isMultiselect() + { + return $this->multiselect; + } + + /** + * 鑾峰彇鎻愮ず + * @return string + */ + public function getPrompt() + { + return $this->prompt; + } + + /** + * 璁剧疆鎻愮ず + * @param string $prompt + * @return self + */ + public function setPrompt($prompt) + { + $this->prompt = $prompt; + + return $this; + } + + /** + * 璁剧疆閿欒鎻愮ず淇℃伅 + * @param string $errorMessage + * @return self + */ + public function setErrorMessage($errorMessage) + { + $this->errorMessage = $errorMessage; + $this->setValidator($this->getDefaultValidator()); + + return $this; + } + + /** + * 鑾峰彇榛樿鐨勯獙璇佹柟娉 + * @return callable + */ + private function getDefaultValidator() + { + $choices = $this->choices; + $errorMessage = $this->errorMessage; + $multiselect = $this->multiselect; + $isAssoc = $this->isAssoc($choices); + + return function ($selected) use ($choices, $errorMessage, $multiselect, $isAssoc) { + // Collapse all spaces. + $selectedChoices = str_replace(' ', '', $selected); + + if ($multiselect) { + // Check for a separated comma values + if (!preg_match('/^[a-zA-Z0-9_-]+(?:,[a-zA-Z0-9_-]+)*$/', $selectedChoices, $matches)) { + throw new \InvalidArgumentException(sprintf($errorMessage, $selected)); + } + $selectedChoices = explode(',', $selectedChoices); + } else { + $selectedChoices = [$selected]; + } + + $multiselectChoices = []; + foreach ($selectedChoices as $value) { + $results = []; + foreach ($choices as $key => $choice) { + if ($choice === $value) { + $results[] = $key; + } + } + + if (count($results) > 1) { + throw new \InvalidArgumentException(sprintf('The provided answer is ambiguous. Value should be one of %s.', implode(' or ', $results))); + } + + $result = array_search($value, $choices); + + if (!$isAssoc) { + if (!empty($result)) { + $result = $choices[$result]; + } elseif (isset($choices[$value])) { + $result = $choices[$value]; + } + } elseif (empty($result) && array_key_exists($value, $choices)) { + $result = $value; + } + + if (empty($result)) { + throw new \InvalidArgumentException(sprintf($errorMessage, $value)); + } + array_push($multiselectChoices, $result); + } + + if ($multiselect) { + return $multiselectChoices; + } + + return current($multiselectChoices); + }; + } +} diff --git a/thinkphp/library/think/console/output/question/Confirmation.php b/thinkphp/library/think/console/output/question/Confirmation.php new file mode 100644 index 000000000..6598f9b3e --- /dev/null +++ b/thinkphp/library/think/console/output/question/Confirmation.php @@ -0,0 +1,57 @@ + +// +---------------------------------------------------------------------- + +namespace think\console\output\question; + +use think\console\output\Question; + +class Confirmation extends Question +{ + + private $trueAnswerRegex; + + /** + * 鏋勯犳柟娉 + * @param string $question 闂 + * @param bool $default 榛樿绛旀 + * @param string $trueAnswerRegex 楠岃瘉姝e垯 + */ + public function __construct($question, $default = true, $trueAnswerRegex = '/^y/i') + { + parent::__construct($question, (bool) $default); + + $this->trueAnswerRegex = $trueAnswerRegex; + $this->setNormalizer($this->getDefaultNormalizer()); + } + + /** + * 鑾峰彇榛樿鐨勭瓟妗堝洖璋 + * @return callable + */ + private function getDefaultNormalizer() + { + $default = $this->getDefault(); + $regex = $this->trueAnswerRegex; + + return function ($answer) use ($default, $regex) { + if (is_bool($answer)) { + return $answer; + } + + $answerIsTrue = (bool) preg_match($regex, $answer); + if (false === $default) { + return $answer && $answerIsTrue; + } + + return !$answer || $answerIsTrue; + }; + } +} diff --git a/thinkphp/library/think/controller/Rest.php b/thinkphp/library/think/controller/Rest.php new file mode 100644 index 000000000..c297f4eb1 --- /dev/null +++ b/thinkphp/library/think/controller/Rest.php @@ -0,0 +1,99 @@ + +// +---------------------------------------------------------------------- + +namespace think\controller; + +use think\App; +use think\Request; +use think\Response; + +abstract class Rest +{ + + protected $method; // 褰撳墠璇锋眰绫诲瀷 + protected $type; // 褰撳墠璧勬簮绫诲瀷 + // 杈撳嚭绫诲瀷 + protected $restMethodList = 'get|post|put|delete'; + protected $restDefaultMethod = 'get'; + protected $restTypeList = 'html|xml|json|rss'; + protected $restDefaultType = 'html'; + protected $restOutputType = [ // REST鍏佽杈撳嚭鐨勮祫婧愮被鍨嬪垪琛 + 'xml' => 'application/xml', + 'json' => 'application/json', + 'html' => 'text/html', + ]; + + /** + * 鏋舵瀯鍑芥暟 鍙栧緱妯℃澘瀵硅薄瀹炰緥 + * @access public + */ + public function __construct() + { + // 璧勬簮绫诲瀷妫娴 + $request = Request::instance(); + $ext = $request->ext(); + if ('' == $ext) { + // 鑷姩妫娴嬭祫婧愮被鍨 + $this->type = $request->type(); + } elseif (!preg_match('/\(' . $this->restTypeList . '\)$/i', $ext)) { + // 璧勬簮绫诲瀷闈炴硶 鍒欑敤榛樿璧勬簮绫诲瀷璁块棶 + $this->type = $this->restDefaultType; + } else { + $this->type = $ext; + } + // 璇锋眰鏂瑰紡妫娴 + $method = strtolower($request->method()); + if (false === stripos($this->restMethodList, $method)) { + // 璇锋眰鏂瑰紡闈炴硶 鍒欑敤榛樿璇锋眰鏂规硶 + $method = $this->restDefaultMethod; + } + $this->method = $method; + } + + /** + * REST 璋冪敤 + * @access public + * @param string $method 鏂规硶鍚 + * @return mixed + * @throws \Exception + */ + public function _empty($method) + { + if (method_exists($this, $method . '_' . $this->method . '_' . $this->type)) { + // RESTFul鏂规硶鏀寔 + $fun = $method . '_' . $this->method . '_' . $this->type; + } elseif ($this->method == $this->restDefaultMethod && method_exists($this, $method . '_' . $this->type)) { + $fun = $method . '_' . $this->type; + } elseif ($this->type == $this->restDefaultType && method_exists($this, $method . '_' . $this->method)) { + $fun = $method . '_' . $this->method; + } + if (isset($fun)) { + return App::invokeMethod([$this, $fun]); + } else { + // 鎶涘嚭寮傚父 + throw new \Exception('error action :' . $method); + } + } + + /** + * 杈撳嚭杩斿洖鏁版嵁 + * @access protected + * @param mixed $data 瑕佽繑鍥炵殑鏁版嵁 + * @param String $type 杩斿洖绫诲瀷 JSON XML + * @param integer $code HTTP鐘舵佺爜 + * @return Response + */ + protected function response($data, $type = 'json', $code = 200) + { + return Response::create($data, $type)->code($code); + } + +} diff --git a/thinkphp/library/think/controller/Yar.php b/thinkphp/library/think/controller/Yar.php new file mode 100644 index 000000000..fcf5ced1b --- /dev/null +++ b/thinkphp/library/think/controller/Yar.php @@ -0,0 +1,51 @@ + +// +---------------------------------------------------------------------- + +namespace think\controller; + +/** + * ThinkPHP Yar鎺у埗鍣ㄧ被 + */ +abstract class Yar +{ + + /** + * 鏋舵瀯鍑芥暟 + * @access public + */ + public function __construct() + { + //鎺у埗鍣ㄥ垵濮嬪寲 + if (method_exists($this, '_initialize')) { + $this->_initialize(); + } + + //鍒ゆ柇鎵╁睍鏄惁瀛樺湪 + if (!extension_loaded('yar')) { + throw new \Exception('not support yar'); + } + + //瀹炰緥鍖朰ar_Server + $server = new \Yar_Server($this); + // 鍚姩server + $server->handle(); + } + + /** + * 榄旀湳鏂规硶 鏈変笉瀛樺湪鐨勬搷浣滅殑鏃跺欐墽琛 + * @access public + * @param string $method 鏂规硶鍚 + * @param array $args 鍙傛暟 + * @return mixed + */ + public function __call($method, $args) + {} +} diff --git a/thinkphp/library/think/db/Builder.php b/thinkphp/library/think/db/Builder.php new file mode 100644 index 000000000..380fe2391 --- /dev/null +++ b/thinkphp/library/think/db/Builder.php @@ -0,0 +1,826 @@ + +// +---------------------------------------------------------------------- + +namespace think\db; + +use PDO; +use think\Exception; + +abstract class Builder +{ + // connection瀵硅薄瀹炰緥 + protected $connection; + // 鏌ヨ瀵硅薄瀹炰緥 + protected $query; + + // 鏁版嵁搴撹〃杈惧紡 + protected $exp = ['eq' => '=', 'neq' => '<>', 'gt' => '>', 'egt' => '>=', 'lt' => '<', 'elt' => '<=', 'notlike' => 'NOT LIKE', 'like' => 'LIKE', 'in' => 'IN', 'exp' => 'EXP', 'notin' => 'NOT IN', 'not in' => 'NOT IN', 'between' => 'BETWEEN', 'not between' => 'NOT BETWEEN', 'notbetween' => 'NOT BETWEEN', 'exists' => 'EXISTS', 'notexists' => 'NOT EXISTS', 'not exists' => 'NOT EXISTS', 'null' => 'NULL', 'notnull' => 'NOT NULL', 'not null' => 'NOT NULL', '> time' => '> TIME', '< time' => '< TIME', '>= time' => '>= TIME', '<= time' => '<= TIME', 'between time' => 'BETWEEN TIME', 'not between time' => 'NOT BETWEEN TIME', 'notbetween time' => 'NOT BETWEEN TIME']; + + // SQL琛ㄨ揪寮 + protected $selectSql = 'SELECT%DISTINCT% %FIELD% FROM %TABLE%%FORCE%%JOIN%%WHERE%%GROUP%%HAVING%%ORDER%%LIMIT% %UNION%%LOCK%%COMMENT%'; + protected $insertSql = '%INSERT% INTO %TABLE% (%FIELD%) VALUES (%DATA%) %COMMENT%'; + protected $insertAllSql = 'INSERT INTO %TABLE% (%FIELD%) %DATA% %COMMENT%'; + protected $updateSql = 'UPDATE %TABLE% SET %SET% %JOIN% %WHERE% %ORDER%%LIMIT% %LOCK%%COMMENT%'; + protected $deleteSql = 'DELETE FROM %TABLE% %USING% %JOIN% %WHERE% %ORDER%%LIMIT% %LOCK%%COMMENT%'; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Connection $connection 鏁版嵁搴撹繛鎺ュ璞″疄渚 + * @param Query $query 鏁版嵁搴撴煡璇㈠璞″疄渚 + */ + public function __construct(Connection $connection, Query $query) + { + $this->connection = $connection; + $this->query = $query; + } + + /** + * 鑾峰彇褰撳墠鐨勮繛鎺ュ璞″疄渚 + * @access public + * @return void + */ + public function getConnection() + { + return $this->connection; + } + + /** + * 鑾峰彇褰撳墠鐨凲uery瀵硅薄瀹炰緥 + * @access public + * @return void + */ + public function getQuery() + { + return $this->query; + } + + /** + * 灏哠QL璇彞涓殑__TABLE_NAME__瀛楃涓叉浛鎹㈡垚甯﹀墠缂鐨勮〃鍚嶏紙灏忓啓锛 + * @access protected + * @param string $sql sql璇彞 + * @return string + */ + protected function parseSqlTable($sql) + { + return $this->query->parseSqlTable($sql); + } + + /** + * 鏁版嵁鍒嗘瀽 + * @access protected + * @param array $data 鏁版嵁 + * @param array $options 鏌ヨ鍙傛暟 + * @return array + */ + protected function parseData($data, $options) + { + if (empty($data)) { + return []; + } + + // 鑾峰彇缁戝畾淇℃伅 + $bind = $this->query->getFieldsBind($options); + if ('*' == $options['field']) { + $fields = array_keys($bind); + } else { + $fields = $options['field']; + } + + $result = []; + foreach ($data as $key => $val) { + $item = $this->parseKey($key, $options); + if (false === strpos($key, '.') && !in_array($key, $fields, true)) { + if ($options['strict']) { + throw new Exception('fields not exists:[' . $key . ']'); + } + } elseif (isset($val[0]) && 'exp' == $val[0]) { + $result[$item] = $val[1]; + } elseif (is_null($val)) { + $result[$item] = 'NULL'; + } elseif (is_scalar($val)) { + // 杩囨护闈炴爣閲忔暟鎹 + if (0 === strpos($val, ':') && $this->query->isBind(substr($val, 1))) { + $result[$item] = $val; + } else { + $key = str_replace('.', '_', $key); + $this->query->bind('__data__' . $key, $val, isset($bind[$key]) ? $bind[$key] : PDO::PARAM_STR); + $result[$item] = ':__data__' . $key; + } + } elseif (is_object($val) && method_exists($val, '__toString')) { + // 瀵硅薄鏁版嵁鍐欏叆 + $result[$item] = $val->__toString(); + } + } + return $result; + } + + /** + * 瀛楁鍚嶅垎鏋 + * @access protected + * @param string $key + * @param array $options + * @return string + */ + protected function parseKey($key, $options = []) + { + return $key; + } + + /** + * value鍒嗘瀽 + * @access protected + * @param mixed $value + * @param string $field + * @return string|array + */ + protected function parseValue($value, $field = '') + { + if (is_string($value)) { + $value = strpos($value, ':') === 0 && $this->query->isBind(substr($value, 1)) ? $value : $this->connection->quote($value); + } elseif (is_array($value)) { + $value = array_map([$this, 'parseValue'], $value); + } elseif (is_bool($value)) { + $value = $value ? '1' : '0'; + } elseif (is_null($value)) { + $value = 'null'; + } + return $value; + } + + /** + * field鍒嗘瀽 + * @access protected + * @param mixed $fields + * @param array $options + * @return string + */ + protected function parseField($fields, $options = []) + { + if ('*' == $fields || empty($fields)) { + $fieldsStr = '*'; + } elseif (is_array($fields)) { + // 鏀寔 'field1'=>'field2' 杩欐牱鐨勫瓧娈靛埆鍚嶅畾涔 + $array = []; + foreach ($fields as $key => $field) { + if (!is_numeric($key)) { + $array[] = $this->parseKey($key, $options) . ' AS ' . $this->parseKey($field, $options); + } else { + $array[] = $this->parseKey($field, $options); + } + } + $fieldsStr = implode(',', $array); + } + return $fieldsStr; + } + + /** + * table鍒嗘瀽 + * @access protected + * @param mixed $tables + * @param array $options + * @return string + */ + protected function parseTable($tables, $options = []) + { + $item = []; + foreach ((array) $tables as $key => $table) { + if (!is_numeric($key)) { + if (strpos($key, '@think')) { + $key = strstr($key, '@think', true); + } + $key = $this->parseSqlTable($key); + $item[] = $this->parseKey($key) . ' ' . $this->parseKey($table); + } else { + $table = $this->parseSqlTable($table); + if (isset($options['alias'][$table])) { + $item[] = $this->parseKey($table) . ' ' . $this->parseKey($options['alias'][$table]); + } else { + $item[] = $this->parseKey($table); + } + } + } + return implode(',', $item); + } + + /** + * where鍒嗘瀽 + * @access protected + * @param mixed $where 鏌ヨ鏉′欢 + * @param array $options 鏌ヨ鍙傛暟 + * @return string + */ + protected function parseWhere($where, $options) + { + $whereStr = $this->buildWhere($where, $options); + return empty($whereStr) ? '' : ' WHERE ' . $whereStr; + } + + /** + * 鐢熸垚鏌ヨ鏉′欢SQL + * @access public + * @param mixed $where + * @param array $options + * @return string + */ + public function buildWhere($where, $options) + { + if (empty($where)) { + $where = []; + } + + if ($where instanceof Query) { + return $this->buildWhere($where->getOptions('where'), $options); + } + + $whereStr = ''; + $binds = $this->query->getFieldsBind($options); + foreach ($where as $key => $val) { + $str = []; + foreach ($val as $field => $value) { + if ($value instanceof \Closure) { + // 浣跨敤闂寘鏌ヨ + $query = new Query($this->connection); + call_user_func_array($value, [ & $query]); + $str[] = ' ' . $key . ' ( ' . $this->buildWhere($query->getOptions('where'), $options) . ' )'; + } elseif (strpos($field, '|')) { + // 涓嶅悓瀛楁浣跨敤鐩稿悓鏌ヨ鏉′欢锛圤R锛 + $array = explode('|', $field); + $item = []; + foreach ($array as $k) { + $item[] = $this->parseWhereItem($k, $value, '', $options, $binds); + } + $str[] = ' ' . $key . ' ( ' . implode(' OR ', $item) . ' )'; + } elseif (strpos($field, '&')) { + // 涓嶅悓瀛楁浣跨敤鐩稿悓鏌ヨ鏉′欢锛圓ND锛 + $array = explode('&', $field); + $item = []; + foreach ($array as $k) { + $item[] = $this->parseWhereItem($k, $value, '', $options, $binds); + } + $str[] = ' ' . $key . ' ( ' . implode(' AND ', $item) . ' )'; + } else { + // 瀵瑰瓧娈典娇鐢ㄨ〃杈惧紡鏌ヨ + $field = is_string($field) ? $field : ''; + $str[] = ' ' . $key . ' ' . $this->parseWhereItem($field, $value, $key, $options, $binds); + } + } + + $whereStr .= empty($whereStr) ? substr(implode(' ', $str), strlen($key) + 1) : implode(' ', $str); + } + return $whereStr; + } + + // where瀛愬崟鍏冨垎鏋 + protected function parseWhereItem($field, $val, $rule = '', $options = [], $binds = [], $bindName = null) + { + // 瀛楁鍒嗘瀽 + $key = $field ? $this->parseKey($field, $options) : ''; + + // 鏌ヨ瑙勫垯鍜屾潯浠 + if (!is_array($val)) { + $val = ['=', $val]; + } + list($exp, $value) = $val; + + // 瀵逛竴涓瓧娈典娇鐢ㄥ涓煡璇㈡潯浠 + if (is_array($exp)) { + $item = array_pop($val); + // 浼犲叆 or 鎴栬 and + if (is_string($item) && in_array($item, ['AND', 'and', 'OR', 'or'])) { + $rule = $item; + } else { + array_push($val, $item); + } + foreach ($val as $k => $item) { + $bindName = 'where_' . str_replace('.', '_', $field) . '_' . $k; + $str[] = $this->parseWhereItem($field, $item, $rule, $options, $binds, $bindName); + } + return '( ' . implode(' ' . $rule . ' ', $str) . ' )'; + } + + // 妫娴嬫搷浣滅 + if (!in_array($exp, $this->exp)) { + $exp = strtolower($exp); + if (isset($this->exp[$exp])) { + $exp = $this->exp[$exp]; + } else { + throw new Exception('where express error:' . $exp); + } + } + $bindName = $bindName ?: 'where_' . str_replace(['.', '-'], '_', $field); + if (preg_match('/\W/', $bindName)) { + // 澶勭悊甯﹂潪鍗曡瘝瀛楃鐨勫瓧娈靛悕 + $bindName = md5($bindName); + } + + $bindType = isset($binds[$field]) ? $binds[$field] : PDO::PARAM_STR; + if (is_scalar($value) && array_key_exists($field, $binds) && !in_array($exp, ['EXP', 'NOT NULL', 'NULL', 'IN', 'NOT IN', 'BETWEEN', 'NOT BETWEEN']) && strpos($exp, 'TIME') === false) { + if (strpos($value, ':') !== 0 || !$this->query->isBind(substr($value, 1))) { + if ($this->query->isBind($bindName)) { + $bindName .= '_' . str_replace('.', '_', uniqid('', true)); + } + $this->query->bind($bindName, $value, $bindType); + $value = ':' . $bindName; + } + } + + $whereStr = ''; + if (in_array($exp, ['=', '<>', '>', '>=', '<', '<='])) { + // 姣旇緝杩愮畻 鍙 妯$硦鍖归厤 + $whereStr .= $key . ' ' . $exp . ' ' . $this->parseValue($value, $field); + } elseif ('LIKE' == $exp || 'NOT LIKE' == $exp) { + if (is_array($value)) { + foreach ($value as $item) { + $array[] = $key . ' ' . $exp . ' ' . $this->parseValue($item, $field); + } + $logic = isset($val[2]) ? $val[2] : 'AND'; + $whereStr .= '(' . implode($array, ' ' . strtoupper($logic) . ' ') . ')'; + } else { + $whereStr .= $key . ' ' . $exp . ' ' . $this->parseValue($value, $field); + } + } elseif ('EXP' == $exp) { + // 琛ㄨ揪寮忔煡璇 + $whereStr .= '( ' . $key . ' ' . $value . ' )'; + } elseif (in_array($exp, ['NOT NULL', 'NULL'])) { + // NULL 鏌ヨ + $whereStr .= $key . ' IS ' . $exp; + } elseif (in_array($exp, ['NOT IN', 'IN'])) { + // IN 鏌ヨ + if ($value instanceof \Closure) { + $whereStr .= $key . ' ' . $exp . ' ' . $this->parseClosure($value); + } else { + $value = is_array($value) ? $value : explode(',', $value); + if (array_key_exists($field, $binds)) { + $bind = []; + $array = []; + foreach ($value as $k => $v) { + if ($this->query->isBind($bindName . '_in_' . $k)) { + $bindKey = $bindName . '_in_' . uniqid() . '_' . $k; + } else { + $bindKey = $bindName . '_in_' . $k; + } + $bind[$bindKey] = [$v, $bindType]; + $array[] = ':' . $bindKey; + } + $this->query->bind($bind); + $zone = implode(',', $array); + } else { + $zone = implode(',', $this->parseValue($value, $field)); + } + $whereStr .= $key . ' ' . $exp . ' (' . $zone . ')'; + } + } elseif (in_array($exp, ['NOT BETWEEN', 'BETWEEN'])) { + // BETWEEN 鏌ヨ + $data = is_array($value) ? $value : explode(',', $value); + if (array_key_exists($field, $binds)) { + if ($this->query->isBind($bindName . '_between_1')) { + $bindKey1 = $bindName . '_between_1' . uniqid(); + $bindKey2 = $bindName . '_between_2' . uniqid(); + } else { + $bindKey1 = $bindName . '_between_1'; + $bindKey2 = $bindName . '_between_2'; + } + $bind = [ + $bindKey1 => [$data[0], $bindType], + $bindKey2 => [$data[1], $bindType], + ]; + $this->query->bind($bind); + $between = ':' . $bindKey1 . ' AND :' . $bindKey2; + } else { + $between = $this->parseValue($data[0], $field) . ' AND ' . $this->parseValue($data[1], $field); + } + $whereStr .= $key . ' ' . $exp . ' ' . $between; + } elseif (in_array($exp, ['NOT EXISTS', 'EXISTS'])) { + // EXISTS 鏌ヨ + if ($value instanceof \Closure) { + $whereStr .= $exp . ' ' . $this->parseClosure($value); + } else { + $whereStr .= $exp . ' (' . $value . ')'; + } + } elseif (in_array($exp, ['< TIME', '> TIME', '<= TIME', '>= TIME'])) { + $whereStr .= $key . ' ' . substr($exp, 0, 2) . ' ' . $this->parseDateTime($value, $field, $options, $bindName, $bindType); + } elseif (in_array($exp, ['BETWEEN TIME', 'NOT BETWEEN TIME'])) { + if (is_string($value)) { + $value = explode(',', $value); + } + + $whereStr .= $key . ' ' . substr($exp, 0, -4) . $this->parseDateTime($value[0], $field, $options, $bindName . '_between_1', $bindType) . ' AND ' . $this->parseDateTime($value[1], $field, $options, $bindName . '_between_2', $bindType); + } + return $whereStr; + } + + // 鎵ц闂寘瀛愭煡璇 + protected function parseClosure($call, $show = true) + { + $query = new Query($this->connection); + call_user_func_array($call, [ & $query]); + return $query->buildSql($show); + } + + /** + * 鏃ユ湡鏃堕棿鏉′欢瑙f瀽 + * @access protected + * @param string $value + * @param string $key + * @param array $options + * @param string $bindName + * @param integer $bindType + * @return string + */ + protected function parseDateTime($value, $key, $options = [], $bindName = null, $bindType = null) + { + // 鑾峰彇鏃堕棿瀛楁绫诲瀷 + if (strpos($key, '.')) { + list($table, $key) = explode('.', $key); + if (isset($options['alias']) && $pos = array_search($table, $options['alias'])) { + $table = $pos; + } + } else { + $table = $options['table']; + } + $type = $this->query->getTableInfo($table, 'type'); + if (isset($type[$key])) { + $info = $type[$key]; + } + if (isset($info)) { + if (is_string($value)) { + $value = strtotime($value) ?: $value; + } + + if (preg_match('/(datetime|timestamp)/is', $info)) { + // 鏃ユ湡鍙婃椂闂存埑绫诲瀷 + $value = date('Y-m-d H:i:s', $value); + } elseif (preg_match('/(date)/is', $info)) { + // 鏃ユ湡鍙婃椂闂存埑绫诲瀷 + $value = date('Y-m-d', $value); + } + } + $bindName = $bindName ?: $key; + $this->query->bind($bindName, $value, $bindType); + return ':' . $bindName; + } + + /** + * limit鍒嗘瀽 + * @access protected + * @param mixed $lmit + * @return string + */ + protected function parseLimit($limit) + { + return (!empty($limit) && false === strpos($limit, '(')) ? ' LIMIT ' . $limit . ' ' : ''; + } + + /** + * join鍒嗘瀽 + * @access protected + * @param array $join + * @param array $options 鏌ヨ鏉′欢 + * @return string + */ + protected function parseJoin($join, $options = []) + { + $joinStr = ''; + if (!empty($join)) { + foreach ($join as $item) { + list($table, $type, $on) = $item; + $condition = []; + foreach ((array) $on as $val) { + if (strpos($val, '=')) { + list($val1, $val2) = explode('=', $val, 2); + $condition[] = $this->parseKey($val1, $options) . '=' . $this->parseKey($val2, $options); + } else { + $condition[] = $val; + } + } + + $table = $this->parseTable($table, $options); + $joinStr .= ' ' . $type . ' JOIN ' . $table . ' ON ' . implode(' AND ', $condition); + } + } + return $joinStr; + } + + /** + * order鍒嗘瀽 + * @access protected + * @param mixed $order + * @param array $options 鏌ヨ鏉′欢 + * @return string + */ + protected function parseOrder($order, $options = []) + { + if (is_array($order)) { + $array = []; + foreach ($order as $key => $val) { + if (is_numeric($key)) { + if ('[rand]' == $val) { + $array[] = $this->parseRand(); + } elseif (false === strpos($val, '(')) { + $array[] = $this->parseKey($val, $options); + } else { + $array[] = $val; + } + } else { + $sort = in_array(strtolower(trim($val)), ['asc', 'desc']) ? ' ' . $val : ''; + $array[] = $this->parseKey($key, $options) . ' ' . $sort; + } + } + $order = implode(',', $array); + } + return !empty($order) ? ' ORDER BY ' . $order : ''; + } + + /** + * group鍒嗘瀽 + * @access protected + * @param mixed $group + * @return string + */ + protected function parseGroup($group) + { + return !empty($group) ? ' GROUP BY ' . $group : ''; + } + + /** + * having鍒嗘瀽 + * @access protected + * @param string $having + * @return string + */ + protected function parseHaving($having) + { + return !empty($having) ? ' HAVING ' . $having : ''; + } + + /** + * comment鍒嗘瀽 + * @access protected + * @param string $comment + * @return string + */ + protected function parseComment($comment) + { + return !empty($comment) ? ' /* ' . $comment . ' */' : ''; + } + + /** + * distinct鍒嗘瀽 + * @access protected + * @param mixed $distinct + * @return string + */ + protected function parseDistinct($distinct) + { + return !empty($distinct) ? ' DISTINCT ' : ''; + } + + /** + * union鍒嗘瀽 + * @access protected + * @param mixed $union + * @return string + */ + protected function parseUnion($union) + { + if (empty($union)) { + return ''; + } + $type = $union['type']; + unset($union['type']); + foreach ($union as $u) { + if ($u instanceof \Closure) { + $sql[] = $type . ' ' . $this->parseClosure($u, false); + } elseif (is_string($u)) { + $sql[] = $type . ' ' . $this->parseSqlTable($u); + } + } + return implode(' ', $sql); + } + + /** + * index鍒嗘瀽锛屽彲鍦ㄦ搷浣滈摼涓寚瀹氶渶瑕佸己鍒朵娇鐢ㄧ殑绱㈠紩 + * @access protected + * @param mixed $index + * @return string + */ + protected function parseForce($index) + { + if (empty($index)) { + return ''; + } + + if (is_array($index)) { + $index = join(",", $index); + } + + return sprintf(" FORCE INDEX ( %s ) ", $index); + } + + /** + * 璁剧疆閿佹満鍒 + * @access protected + * @param bool $locl + * @return string + */ + protected function parseLock($lock = false) + { + return $lock ? ' FOR UPDATE ' : ''; + } + + /** + * 鐢熸垚鏌ヨSQL + * @access public + * @param array $options 琛ㄨ揪寮 + * @return string + */ + public function select($options = []) + { + $sql = str_replace( + ['%TABLE%', '%DISTINCT%', '%FIELD%', '%JOIN%', '%WHERE%', '%GROUP%', '%HAVING%', '%ORDER%', '%LIMIT%', '%UNION%', '%LOCK%', '%COMMENT%', '%FORCE%'], + [ + $this->parseTable($options['table'], $options), + $this->parseDistinct($options['distinct']), + $this->parseField($options['field'], $options), + $this->parseJoin($options['join'], $options), + $this->parseWhere($options['where'], $options), + $this->parseGroup($options['group']), + $this->parseHaving($options['having']), + $this->parseOrder($options['order'], $options), + $this->parseLimit($options['limit']), + $this->parseUnion($options['union']), + $this->parseLock($options['lock']), + $this->parseComment($options['comment']), + $this->parseForce($options['force']), + ], $this->selectSql); + return $sql; + } + + /** + * 鐢熸垚insert SQL + * @access public + * @param array $data 鏁版嵁 + * @param array $options 琛ㄨ揪寮 + * @param bool $replace 鏄惁replace + * @return string + */ + public function insert(array $data, $options = [], $replace = false) + { + // 鍒嗘瀽骞跺鐞嗘暟鎹 + $data = $this->parseData($data, $options); + if (empty($data)) { + return 0; + } + $fields = array_keys($data); + $values = array_values($data); + + $sql = str_replace( + ['%INSERT%', '%TABLE%', '%FIELD%', '%DATA%', '%COMMENT%'], + [ + $replace ? 'REPLACE' : 'INSERT', + $this->parseTable($options['table'], $options), + implode(' , ', $fields), + implode(' , ', $values), + $this->parseComment($options['comment']), + ], $this->insertSql); + + return $sql; + } + + /** + * 鐢熸垚insertall SQL + * @access public + * @param array $dataSet 鏁版嵁闆 + * @param array $options 琛ㄨ揪寮 + * @return string + */ + public function insertAll($dataSet, $options) + { + // 鑾峰彇鍚堟硶鐨勫瓧娈 + if ('*' == $options['field']) { + $fields = array_keys($this->query->getFieldsType($options)); + } else { + $fields = $options['field']; + } + + foreach ($dataSet as &$data) { + foreach ($data as $key => $val) { + if (!in_array($key, $fields, true)) { + if ($options['strict']) { + throw new Exception('fields not exists:[' . $key . ']'); + } + unset($data[$key]); + } elseif (is_null($val)) { + $data[$key] = 'NULL'; + } elseif (is_scalar($val)) { + $data[$key] = $this->parseValue($val, $key); + } elseif (is_object($val) && method_exists($val, '__toString')) { + // 瀵硅薄鏁版嵁鍐欏叆 + $data[$key] = $val->__toString(); + } else { + // 杩囨护鎺夐潪鏍囬噺鏁版嵁 + unset($data[$key]); + } + } + $value = array_values($data); + $values[] = 'SELECT ' . implode(',', $value); + } + $fields = array_map([$this, 'parseKey'], array_keys(reset($dataSet))); + $sql = str_replace( + ['%TABLE%', '%FIELD%', '%DATA%', '%COMMENT%'], + [ + $this->parseTable($options['table'], $options), + implode(' , ', $fields), + implode(' UNION ALL ', $values), + $this->parseComment($options['comment']), + ], $this->insertAllSql); + + return $sql; + } + + /** + * 鐢熸垚slectinsert SQL + * @access public + * @param array $fields 鏁版嵁 + * @param string $table 鏁版嵁琛 + * @param array $options 琛ㄨ揪寮 + * @return string + */ + public function selectInsert($fields, $table, $options) + { + if (is_string($fields)) { + $fields = explode(',', $fields); + } + + $fields = array_map([$this, 'parseKey'], $fields); + $sql = 'INSERT INTO ' . $this->parseTable($table, $options) . ' (' . implode(',', $fields) . ') ' . $this->select($options); + return $sql; + } + + /** + * 鐢熸垚update SQL + * @access public + * @param array $fields 鏁版嵁 + * @param array $options 琛ㄨ揪寮 + * @return string + */ + public function update($data, $options) + { + $table = $this->parseTable($options['table'], $options); + $data = $this->parseData($data, $options); + if (empty($data)) { + return ''; + } + foreach ($data as $key => $val) { + $set[] = $key . '=' . $val; + } + + $sql = str_replace( + ['%TABLE%', '%SET%', '%JOIN%', '%WHERE%', '%ORDER%', '%LIMIT%', '%LOCK%', '%COMMENT%'], + [ + $this->parseTable($options['table'], $options), + implode(',', $set), + $this->parseJoin($options['join'], $options), + $this->parseWhere($options['where'], $options), + $this->parseOrder($options['order'], $options), + $this->parseLimit($options['limit']), + $this->parseLock($options['lock']), + $this->parseComment($options['comment']), + ], $this->updateSql); + + return $sql; + } + + /** + * 鐢熸垚delete SQL + * @access public + * @param array $options 琛ㄨ揪寮 + * @return string + */ + public function delete($options) + { + $sql = str_replace( + ['%TABLE%', '%USING%', '%JOIN%', '%WHERE%', '%ORDER%', '%LIMIT%', '%LOCK%', '%COMMENT%'], + [ + $this->parseTable($options['table'], $options), + !empty($options['using']) ? ' USING ' . $this->parseTable($options['using'], $options) . ' ' : '', + $this->parseJoin($options['join'], $options), + $this->parseWhere($options['where'], $options), + $this->parseOrder($options['order'], $options), + $this->parseLimit($options['limit']), + $this->parseLock($options['lock']), + $this->parseComment($options['comment']), + ], $this->deleteSql); + + return $sql; + } +} diff --git a/thinkphp/library/think/db/Connection.php b/thinkphp/library/think/db/Connection.php new file mode 100644 index 000000000..6198f9249 --- /dev/null +++ b/thinkphp/library/think/db/Connection.php @@ -0,0 +1,961 @@ + +// +---------------------------------------------------------------------- + +namespace think\db; + +use PDO; +use PDOStatement; +use think\Db; +use think\db\exception\BindParamException; +use think\Debug; +use think\Exception; +use think\exception\PDOException; +use think\Log; + +/** + * Class Connection + * @package think + * @method Query table(string $table) 鎸囧畾鏁版嵁琛紙鍚墠缂锛 + * @method Query name(string $name) 鎸囧畾鏁版嵁琛紙涓嶅惈鍓嶇紑锛 + * + */ +abstract class Connection +{ + + /** @var PDOStatement PDO鎿嶄綔瀹炰緥 */ + protected $PDOStatement; + + /** @var string 褰撳墠SQL鎸囦护 */ + protected $queryStr = ''; + // 杩斿洖鎴栬呭奖鍝嶈褰曟暟 + protected $numRows = 0; + // 浜嬪姟鎸囦护鏁 + protected $transTimes = 0; + // 閿欒淇℃伅 + protected $error = ''; + + /** @var PDO[] 鏁版嵁搴撹繛鎺D 鏀寔澶氫釜杩炴帴 */ + protected $links = []; + + /** @var PDO 褰撳墠杩炴帴ID */ + protected $linkID; + protected $linkRead; + protected $linkWrite; + + // 鏌ヨ缁撴灉绫诲瀷 + protected $resultSetType = 'array'; + // 鏌ヨ缁撴灉绫诲瀷 + protected $fetchType = PDO::FETCH_ASSOC; + // 瀛楁灞炴уぇ灏忓啓 + protected $attrCase = PDO::CASE_LOWER; + // 鐩戝惉鍥炶皟 + protected static $event = []; + // 鏌ヨ瀵硅薄 + protected $query = []; + // 鏁版嵁搴撹繛鎺ュ弬鏁伴厤缃 + protected $config = [ + // 鏁版嵁搴撶被鍨 + 'type' => '', + // 鏈嶅姟鍣ㄥ湴鍧 + 'hostname' => '', + // 鏁版嵁搴撳悕 + 'database' => '', + // 鐢ㄦ埛鍚 + 'username' => '', + // 瀵嗙爜 + 'password' => '', + // 绔彛 + 'hostport' => '', + // 杩炴帴dsn + 'dsn' => '', + // 鏁版嵁搴撹繛鎺ュ弬鏁 + 'params' => [], + // 鏁版嵁搴撶紪鐮侀粯璁ら噰鐢╱tf8 + 'charset' => 'utf8', + // 鏁版嵁搴撹〃鍓嶇紑 + 'prefix' => '', + // 鏁版嵁搴撹皟璇曟ā寮 + 'debug' => false, + // 鏁版嵁搴撻儴缃叉柟寮:0 闆嗕腑寮(鍗曚竴鏈嶅姟鍣),1 鍒嗗竷寮(涓讳粠鏈嶅姟鍣) + 'deploy' => 0, + // 鏁版嵁搴撹鍐欐槸鍚﹀垎绂 涓讳粠寮忔湁鏁 + 'rw_separate' => false, + // 璇诲啓鍒嗙鍚 涓绘湇鍔″櫒鏁伴噺 + 'master_num' => 1, + // 鎸囧畾浠庢湇鍔″櫒搴忓彿 + 'slave_no' => '', + // 鏄惁涓ユ牸妫鏌ュ瓧娈垫槸鍚﹀瓨鍦 + 'fields_strict' => true, + // 鏁版嵁杩斿洖绫诲瀷 + 'result_type' => PDO::FETCH_ASSOC, + // 鏁版嵁闆嗚繑鍥炵被鍨 + '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', + ]; + + // PDO杩炴帴鍙傛暟 + protected $params = [ + PDO::ATTR_CASE => PDO::CASE_NATURAL, + PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, + PDO::ATTR_ORACLE_NULLS => PDO::NULL_NATURAL, + PDO::ATTR_STRINGIFY_FETCHES => false, + PDO::ATTR_EMULATE_PREPARES => false, + ]; + + // 缁戝畾鍙傛暟 + protected $bind = []; + + /** + * 鏋舵瀯鍑芥暟 璇诲彇鏁版嵁搴撻厤缃俊鎭 + * @access public + * @param array $config 鏁版嵁搴撻厤缃暟缁 + */ + public function __construct(array $config = []) + { + if (!empty($config)) { + $this->config = array_merge($this->config, $config); + } + } + + /** + * 鍒涘缓鎸囧畾妯″瀷鐨勬煡璇㈠璞 + * @access public + * @param string $model 妯″瀷绫诲悕绉 + * @param string $queryClass 鏌ヨ瀵硅薄绫诲悕 + * @return Query + */ + public function getQuery($model = 'db', $queryClass = '') + { + if (!isset($this->query[$model])) { + $class = $queryClass ?: $this->config['query']; + $this->query[$model] = new $class($this, 'db' == $model ? '' : $model); + } + return $this->query[$model]; + } + + /** + * 璋冪敤Query绫荤殑鏌ヨ鏂规硶 + * @access public + * @param string $method 鏂规硶鍚嶇О + * @param array $args 璋冪敤鍙傛暟 + * @return mixed + */ + public function __call($method, $args) + { + return call_user_func_array([$this->getQuery(), $method], $args); + } + + /** + * 瑙f瀽pdo杩炴帴鐨刣sn淇℃伅 + * @access protected + * @param array $config 杩炴帴淇℃伅 + * @return string + */ + abstract protected function parseDsn($config); + + /** + * 鍙栧緱鏁版嵁琛ㄧ殑瀛楁淇℃伅 + * @access public + * @param string $tableName + * @return array + */ + abstract public function getFields($tableName); + + /** + * 鍙栧緱鏁版嵁搴撶殑琛ㄤ俊鎭 + * @access public + * @param string $dbName + * @return array + */ + abstract public function getTables($dbName); + + /** + * SQL鎬ц兘鍒嗘瀽 + * @access protected + * @param string $sql + * @return array + */ + abstract protected function getExplain($sql); + + /** + * 瀵硅繑鏁版嵁琛ㄥ瓧娈典俊鎭繘琛屽ぇ灏忓啓杞崲鍑烘潵 + * @access public + * @param array $info 瀛楁淇℃伅 + * @return array + */ + public function fieldCase($info) + { + // 瀛楁澶у皬鍐欒浆鎹 + switch ($this->attrCase) { + case PDO::CASE_LOWER: + $info = array_change_key_case($info); + break; + case PDO::CASE_UPPER: + $info = array_change_key_case($info, CASE_UPPER); + break; + case PDO::CASE_NATURAL: + default: + // 涓嶅仛杞崲 + } + return $info; + } + + /** + * 鑾峰彇鏁版嵁搴撶殑閰嶇疆鍙傛暟 + * @access public + * @param string $config 閰嶇疆鍚嶇О + * @return mixed + */ + public function getConfig($config = '') + { + return $config ? $this->config[$config] : $this->config; + } + + /** + * 璁剧疆鏁版嵁搴撶殑閰嶇疆鍙傛暟 + * @access public + * @param string|array $config 閰嶇疆鍚嶇О + * @param mixed $value 閰嶇疆鍊 + * @return void + */ + public function setConfig($config, $value = '') + { + if (is_array($config)) { + $this->config = array_merge($this->config, $config); + } else { + $this->config[$config] = $value; + } + } + + /** + * 杩炴帴鏁版嵁搴撴柟娉 + * @access public + * @param array $config 杩炴帴鍙傛暟 + * @param integer $linkNum 杩炴帴搴忓彿 + * @param array|bool $autoConnection 鏄惁鑷姩杩炴帴涓绘暟鎹簱锛堢敤浜庡垎甯冨紡锛 + * @return PDO + * @throws Exception + */ + public function connect(array $config = [], $linkNum = 0, $autoConnection = false) + { + if (!isset($this->links[$linkNum])) { + if (!$config) { + $config = $this->config; + } else { + $config = array_merge($this->config, $config); + } + // 杩炴帴鍙傛暟 + if (isset($config['params']) && is_array($config['params'])) { + $params = $config['params'] + $this->params; + } else { + $params = $this->params; + } + // 璁板綍褰撳墠瀛楁灞炴уぇ灏忓啓璁剧疆 + $this->attrCase = $params[PDO::ATTR_CASE]; + // 璁板綍鏁版嵁闆嗚繑鍥炵被鍨 + if (isset($config['resultset_type'])) { + $this->resultSetType = $config['resultset_type']; + } + // 鏁版嵁杩斿洖绫诲瀷 + if (isset($config['result_type'])) { + $this->fetchType = $config['result_type']; + } + try { + if (empty($config['dsn'])) { + $config['dsn'] = $this->parseDsn($config); + } + if ($config['debug']) { + $startTime = microtime(true); + } + $this->links[$linkNum] = new PDO($config['dsn'], $config['username'], $config['password'], $params); + if ($config['debug']) { + // 璁板綍鏁版嵁搴撹繛鎺ヤ俊鎭 + Log::record('[ DB ] CONNECT:[ UseTime:' . number_format(microtime(true) - $startTime, 6) . 's ] ' . $config['dsn'], 'sql'); + } + } catch (\PDOException $e) { + if ($autoConnection) { + Log::record($e->getMessage(), 'error'); + return $this->connect($autoConnection, $linkNum); + } else { + throw $e; + } + } + } + return $this->links[$linkNum]; + } + + /** + * 閲婃斁鏌ヨ缁撴灉 + * @access public + */ + public function free() + { + $this->PDOStatement = null; + } + + /** + * 鑾峰彇PDO瀵硅薄 + * @access public + * @return \PDO|false + */ + public function getPdo() + { + if (!$this->linkID) { + return false; + } else { + return $this->linkID; + } + } + + /** + * 鎵ц鏌ヨ 杩斿洖鏁版嵁闆 + * @access public + * @param string $sql sql鎸囦护 + * @param array $bind 鍙傛暟缁戝畾 + * @param bool $master 鏄惁鍦ㄤ富鏈嶅姟鍣ㄨ鎿嶄綔 + * @param bool $class 鏄惁杩斿洖PDO瀵硅薄 + * @param string $sql sql鎸囦护 + * @param array $bind 鍙傛暟缁戝畾 + * @param boolean $master 鏄惁鍦ㄤ富鏈嶅姟鍣ㄨ鎿嶄綔 + * @param bool $pdo 鏄惁杩斿洖PDO瀵硅薄 + * @return mixed + * @throws BindParamException + * @throws PDOException + */ + public function query($sql, $bind = [], $master = false, $pdo = false) + { + $this->initConnect($master); + if (!$this->linkID) { + return false; + } + + // 璁板綍SQL璇彞 + $this->queryStr = $sql; + if ($bind) { + $this->bind = $bind; + } + + //閲婃斁鍓嶆鐨勬煡璇㈢粨鏋 + if (!empty($this->PDOStatement) && $this->PDOStatement->queryString != $sql) { + $this->free(); + } + + Db::$queryTimes++; + try { + // 璋冭瘯寮濮 + $this->debug(true); + // 棰勫鐞 + if (empty($this->PDOStatement)) { + $this->PDOStatement = $this->linkID->prepare($sql); + } + // 鏄惁涓哄瓨鍌ㄨ繃绋嬭皟鐢 + $procedure = in_array(strtolower(substr(trim($sql), 0, 4)), ['call', 'exec']); + // 鍙傛暟缁戝畾 + if ($procedure) { + $this->bindParam($bind); + } else { + $this->bindValue($bind); + } + // 鎵ц鏌ヨ + $this->PDOStatement->execute(); + // 璋冭瘯缁撴潫 + $this->debug(false); + // 杩斿洖缁撴灉闆 + return $this->getResult($pdo, $procedure); + } catch (\PDOException $e) { + throw new PDOException($e, $this->config, $this->getLastsql()); + } + } + + /** + * 鎵ц璇彞 + * @access public + * @param string $sql sql鎸囦护 + * @param array $bind 鍙傛暟缁戝畾 + * @return int + * @throws BindParamException + * @throws PDOException + */ + public function execute($sql, $bind = []) + { + $this->initConnect(true); + if (!$this->linkID) { + return false; + } + + // 璁板綍SQL璇彞 + $this->queryStr = $sql; + if ($bind) { + $this->bind = $bind; + } + + //閲婃斁鍓嶆鐨勬煡璇㈢粨鏋 + if (!empty($this->PDOStatement) && $this->PDOStatement->queryString != $sql) { + $this->free(); + } + + Db::$executeTimes++; + try { + // 璋冭瘯寮濮 + $this->debug(true); + // 棰勫鐞 + if (empty($this->PDOStatement)) { + $this->PDOStatement = $this->linkID->prepare($sql); + } + // 鏄惁涓哄瓨鍌ㄨ繃绋嬭皟鐢 + $procedure = in_array(strtolower(substr(trim($sql), 0, 4)), ['call', 'exec']); + // 鍙傛暟缁戝畾 + if ($procedure) { + $this->bindParam($bind); + } else { + $this->bindValue($bind); + } + // 鎵ц璇彞 + $this->PDOStatement->execute(); + // 璋冭瘯缁撴潫 + $this->debug(false); + + $this->numRows = $this->PDOStatement->rowCount(); + return $this->numRows; + } catch (\PDOException $e) { + throw new PDOException($e, $this->config, $this->getLastsql()); + } + } + + /** + * 鏍规嵁鍙傛暟缁戝畾缁勮鏈缁堢殑SQL璇彞 渚夸簬璋冭瘯 + * @access public + * @param string $sql 甯﹀弬鏁扮粦瀹氱殑sql璇彞 + * @param array $bind 鍙傛暟缁戝畾鍒楄〃 + * @return string + */ + public function getRealSql($sql, array $bind = []) + { + foreach ($bind as $key => $val) { + $value = is_array($val) ? $val[0] : $val; + $type = is_array($val) ? $val[1] : PDO::PARAM_STR; + if (PDO::PARAM_STR == $type) { + $value = $this->quote($value); + } elseif (PDO::PARAM_INT == $type) { + $value = (float) $value; + } + // 鍒ゆ柇鍗犱綅绗 + $sql = is_numeric($key) ? + substr_replace($sql, $value, strpos($sql, '?'), 1) : + str_replace( + [':' . $key . ')', ':' . $key . ',', ':' . $key . ' '], + [$value . ')', $value . ',', $value . ' '], + $sql . ' '); + } + return rtrim($sql); + } + + /** + * 鍙傛暟缁戝畾 + * 鏀寔 ['name'=>'value','id'=>123] 瀵瑰簲鍛藉悕鍗犱綅绗 + * 鎴栬 ['value',123] 瀵瑰簲闂彿鍗犱綅绗 + * @access public + * @param array $bind 瑕佺粦瀹氱殑鍙傛暟鍒楄〃 + * @return void + * @throws BindParamException + */ + protected function bindValue(array $bind = []) + { + foreach ($bind as $key => $val) { + // 鍗犱綅绗 + $param = is_numeric($key) ? $key + 1 : ':' . $key; + if (is_array($val)) { + if (PDO::PARAM_INT == $val[1] && '' === $val[0]) { + $val[0] = 0; + } + $result = $this->PDOStatement->bindValue($param, $val[0], $val[1]); + } else { + $result = $this->PDOStatement->bindValue($param, $val); + } + if (!$result) { + throw new BindParamException( + "Error occurred when binding parameters '{$param}'", + $this->config, + $this->getLastsql(), + $bind + ); + } + } + } + + /** + * 瀛樺偍杩囩▼鐨勮緭鍏ヨ緭鍑哄弬鏁扮粦瀹 + * @access public + * @param array $bind 瑕佺粦瀹氱殑鍙傛暟鍒楄〃 + * @return void + * @throws BindParamException + */ + protected function bindParam($bind) + { + foreach ($bind as $key => $val) { + if (is_numeric($key)) { + $key = $key + 1; + } + array_unshift($val, $key); + $result = call_user_func_array([$this->PDOStatement, 'bindParam'], $val); + if (!$result) { + $param = array_shift($val); + throw new BindParamException( + "Error occurred when binding parameters '{$param}'", + $this->config, + $this->getLastsql(), + $bind + ); + } + } + } + + /** + * 鑾峰緱鏁版嵁闆嗘暟缁 + * @access protected + * @param bool $pdo 鏄惁杩斿洖PDOStatement + * @param bool $procedure 鏄惁瀛樺偍杩囩▼ + * @return array + */ + protected function getResult($pdo = false, $procedure = false) + { + if ($pdo) { + // 杩斿洖PDOStatement瀵硅薄澶勭悊 + return $this->PDOStatement; + } + if ($procedure) { + // 瀛樺偍杩囩▼杩斿洖缁撴灉 + return $this->procedure(); + } + $result = $this->PDOStatement->fetchAll($this->fetchType); + $this->numRows = count($result); + return $result; + } + + /** + * 鑾峰緱瀛樺偍杩囩▼鏁版嵁闆 + * @access protected + * @return array + */ + protected function procedure() + { + $item = []; + do { + $result = $this->getResult(); + if ($result) { + $item[] = $result; + } + } while ($this->PDOStatement->nextRowset()); + $this->numRows = count($item); + return $item; + } + + /** + * 鎵ц鏁版嵁搴撲簨鍔 + * @access public + * @param callable $callback 鏁版嵁鎿嶄綔鏂规硶鍥炶皟 + * @return mixed + * @throws PDOException + * @throws \Exception + * @throws \Throwable + */ + public function transaction($callback) + { + $this->startTrans(); + try { + $result = null; + if (is_callable($callback)) { + $result = call_user_func_array($callback, [$this]); + } + $this->commit(); + return $result; + } catch (\Exception $e) { + $this->rollback(); + throw $e; + } catch (\Throwable $e) { + $this->rollback(); + throw $e; + } + } + + /** + * 鍚姩浜嬪姟 + * @access public + * @return void + */ + public function startTrans() + { + $this->initConnect(true); + if (!$this->linkID) { + return false; + } + + ++$this->transTimes; + + if (1 == $this->transTimes) { + $this->linkID->beginTransaction(); + } elseif ($this->transTimes > 1 && $this->supportSavepoint()) { + $this->linkID->exec( + $this->parseSavepoint('trans' . $this->transTimes) + ); + } + } + + /** + * 鐢ㄤ簬闈炶嚜鍔ㄦ彁浜ょ姸鎬佷笅闈㈢殑鏌ヨ鎻愪氦 + * @access public + * @return void + * @throws PDOException + */ + public function commit() + { + $this->initConnect(true); + + if (1 == $this->transTimes) { + $this->linkID->commit(); + } + + --$this->transTimes; + } + + /** + * 浜嬪姟鍥炴粴 + * @access public + * @return void + * @throws PDOException + */ + public function rollback() + { + $this->initConnect(true); + + if (1 == $this->transTimes) { + $this->linkID->rollBack(); + } elseif ($this->transTimes > 1 && $this->supportSavepoint()) { + $this->linkID->exec( + $this->parseSavepointRollBack('trans' . $this->transTimes) + ); + } + + $this->transTimes = max(0, $this->transTimes - 1); + } + + /** + * 鏄惁鏀寔浜嬪姟宓屽 + * @return bool + */ + protected function supportSavepoint() + { + return false; + } + + /** + * 鐢熸垚瀹氫箟淇濆瓨鐐圭殑SQL + * @param $name + * @return string + */ + protected function parseSavepoint($name) + { + return 'SAVEPOINT ' . $name; + } + + /** + * 鐢熸垚鍥炴粴鍒颁繚瀛樼偣鐨凷QL + * @param $name + * @return string + */ + protected function parseSavepointRollBack($name) + { + return 'ROLLBACK TO SAVEPOINT ' . $name; + } + + /** + * 鎵瑰鐞嗘墽琛孲QL璇彞 + * 鎵瑰鐞嗙殑鎸囦护閮借涓烘槸execute鎿嶄綔 + * @access public + * @param array $sqlArray SQL鎵瑰鐞嗘寚浠 + * @return boolean + */ + public function batchQuery($sqlArray = []) + { + if (!is_array($sqlArray)) { + return false; + } + // 鑷姩鍚姩浜嬪姟鏀寔 + $this->startTrans(); + try { + foreach ($sqlArray as $sql) { + $this->execute($sql); + } + // 鎻愪氦浜嬪姟 + $this->commit(); + } catch (\Exception $e) { + $this->rollback(); + throw $e; + } + return true; + } + + /** + * 鑾峰緱鏌ヨ娆℃暟 + * @access public + * @param boolean $execute 鏄惁鍖呭惈鎵鏈夋煡璇 + * @return integer + */ + public function getQueryTimes($execute = false) + { + return $execute ? Db::$queryTimes + Db::$executeTimes : Db::$queryTimes; + } + + /** + * 鑾峰緱鎵ц娆℃暟 + * @access public + * @return integer + */ + public function getExecuteTimes() + { + return Db::$executeTimes; + } + + /** + * 鍏抽棴鏁版嵁搴 + * @access public + */ + public function close() + { + $this->linkID = null; + } + + /** + * 鑾峰彇鏈杩戜竴娆℃煡璇㈢殑sql璇彞 + * @access public + * @return string + */ + public function getLastSql() + { + return $this->getRealSql($this->queryStr, $this->bind); + } + + /** + * 鑾峰彇鏈杩戞彃鍏ョ殑ID + * @access public + * @param string $sequence 鑷搴忓垪鍚 + * @return string + */ + public function getLastInsID($sequence = null) + { + return $this->linkID->lastInsertId($sequence); + } + + /** + * 鑾峰彇杩斿洖鎴栬呭奖鍝嶇殑璁板綍鏁 + * @access public + * @return integer + */ + public function getNumRows() + { + return $this->numRows; + } + + /** + * 鑾峰彇鏈杩戠殑閿欒淇℃伅 + * @access public + * @return string + */ + public function getError() + { + if ($this->PDOStatement) { + $error = $this->PDOStatement->errorInfo(); + $error = $error[1] . ':' . $error[2]; + } else { + $error = ''; + } + if ('' != $this->queryStr) { + $error .= "\n [ SQL璇彞 ] : " . $this->getLastsql(); + } + return $error; + } + + /** + * SQL鎸囦护瀹夊叏杩囨护 + * @access public + * @param string $str SQL瀛楃涓 + * @param bool $master 鏄惁涓诲簱鏌ヨ + * @return string + */ + public function quote($str, $master = true) + { + $this->initConnect($master); + return $this->linkID ? $this->linkID->quote($str) : $str; + } + + /** + * 鏁版嵁搴撹皟璇 璁板綍褰撳墠SQL鍙婂垎鏋愭ц兘 + * @access protected + * @param boolean $start 璋冭瘯寮濮嬫爣璁 true 寮濮 false 缁撴潫 + * @param string $sql 鎵ц鐨凷QL璇彞 鐣欑┖鑷姩鑾峰彇 + * @return void + */ + protected function debug($start, $sql = '') + { + if (!empty($this->config['debug'])) { + // 寮鍚暟鎹簱璋冭瘯妯″紡 + if ($start) { + Debug::remark('queryStartTime', 'time'); + } else { + // 璁板綍鎿嶄綔缁撴潫鏃堕棿 + Debug::remark('queryEndTime', 'time'); + $runtime = Debug::getRangeTime('queryStartTime', 'queryEndTime'); + $sql = $sql ?: $this->getLastsql(); + $log = $sql . ' [ RunTime:' . $runtime . 's ]'; + $result = []; + // SQL鎬ц兘鍒嗘瀽 + if ($this->config['sql_explain'] && 0 === stripos(trim($sql), 'select')) { + $result = $this->getExplain($sql); + } + // SQL鐩戝惉 + $this->trigger($sql, $runtime, $result); + } + } + } + + /** + * 鐩戝惉SQL鎵ц + * @access public + * @param callable $callback 鍥炶皟鏂规硶 + * @return void + */ + public function listen($callback) + { + self::$event[] = $callback; + } + + /** + * 瑙﹀彂SQL浜嬩欢 + * @access protected + * @param string $sql SQL璇彞 + * @param float $runtime SQL杩愯鏃堕棿 + * @param mixed $explain SQL鍒嗘瀽 + * @return bool + */ + protected function trigger($sql, $runtime, $explain = []) + { + if (!empty(self::$event)) { + foreach (self::$event as $callback) { + if (is_callable($callback)) { + call_user_func_array($callback, [$sql, $runtime, $explain]); + } + } + } else { + // 鏈敞鍐岀洃鍚垯璁板綍鍒版棩蹇椾腑 + Log::record('[ SQL ] ' . $sql . ' [ RunTime:' . $runtime . 's ]', 'sql'); + if (!empty($explain)) { + Log::record('[ EXPLAIN : ' . var_export($explain, true) . ' ]', 'sql'); + } + } + } + + /** + * 鍒濆鍖栨暟鎹簱杩炴帴 + * @access protected + * @param boolean $master 鏄惁涓绘湇鍔″櫒 + * @return void + */ + protected function initConnect($master = true) + { + if (!empty($this->config['deploy'])) { + // 閲囩敤鍒嗗竷寮忔暟鎹簱 + if ($master) { + if (!$this->linkWrite) { + $this->linkWrite = $this->multiConnect(true); + } + $this->linkID = $this->linkWrite; + } else { + if (!$this->linkRead) { + $this->linkRead = $this->multiConnect(false); + } + $this->linkID = $this->linkRead; + } + } elseif (!$this->linkID) { + // 榛樿鍗曟暟鎹簱 + $this->linkID = $this->connect(); + } + } + + /** + * 杩炴帴鍒嗗竷寮忔湇鍔″櫒 + * @access protected + * @param boolean $master 涓绘湇鍔″櫒 + * @return PDO + */ + protected function multiConnect($master = false) + { + $_config = []; + // 鍒嗗竷寮忔暟鎹簱閰嶇疆瑙f瀽 + foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) { + $_config[$name] = explode(',', $this->config[$name]); + } + + // 涓绘湇鍔″櫒搴忓彿 + $m = floor(mt_rand(0, $this->config['master_num'] - 1)); + + if ($this->config['rw_separate']) { + // 涓讳粠寮忛噰鐢ㄨ鍐欏垎绂 + if ($master) // 涓绘湇鍔″櫒鍐欏叆 + { + $r = $m; + } elseif (is_numeric($this->config['slave_no'])) { + // 鎸囧畾鏈嶅姟鍣ㄨ + $r = $this->config['slave_no']; + } else { + // 璇绘搷浣滆繛鎺ヤ粠鏈嶅姟鍣 姣忔闅忔満杩炴帴鐨勬暟鎹簱 + $r = floor(mt_rand($this->config['master_num'], count($_config['hostname']) - 1)); + } + } else { + // 璇诲啓鎿嶄綔涓嶅尯鍒嗘湇鍔″櫒 姣忔闅忔満杩炴帴鐨勬暟鎹簱 + $r = floor(mt_rand(0, count($_config['hostname']) - 1)); + } + $dbMaster = false; + if ($m != $r) { + $dbMaster = []; + foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) { + $dbMaster[$name] = isset($_config[$name][$m]) ? $_config[$name][$m] : $_config[$name][0]; + } + } + $dbConfig = []; + foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) { + $dbConfig[$name] = isset($_config[$name][$r]) ? $_config[$name][$r] : $_config[$name][0]; + } + return $this->connect($dbConfig, $r, $r == $m ? false : $dbMaster); + } + + /** + * 鏋愭瀯鏂规硶 + * @access public + */ + public function __destruct() + { + // 閲婃斁鏌ヨ + if ($this->PDOStatement) { + $this->free(); + } + // 鍏抽棴杩炴帴 + $this->close(); + } +} diff --git a/thinkphp/library/think/db/Query.php b/thinkphp/library/think/db/Query.php new file mode 100644 index 000000000..45cdd4326 --- /dev/null +++ b/thinkphp/library/think/db/Query.php @@ -0,0 +1,2727 @@ + +// +---------------------------------------------------------------------- + +namespace think\db; + +use PDO; +use think\Cache; +use think\Collection; +use think\Config; +use think\Db; +use think\db\exception\BindParamException; +use think\db\exception\DataNotFoundException; +use think\db\exception\ModelNotFoundException; +use think\Exception; +use think\exception\DbException; +use think\exception\PDOException; +use think\Loader; +use think\Model; +use think\model\Relation; +use think\model\relation\OneToOne; +use think\Paginator; + +class Query +{ + // 鏁版嵁搴揅onnection瀵硅薄瀹炰緥 + protected $connection; + // 鏁版嵁搴揃uilder瀵硅薄瀹炰緥 + protected $builder; + // 褰撳墠妯″瀷绫诲悕绉 + protected $model; + // 褰撳墠鏁版嵁琛ㄥ悕绉帮紙鍚墠缂锛 + protected $table = ''; + // 褰撳墠鏁版嵁琛ㄥ悕绉帮紙涓嶅惈鍓嶇紑锛 + protected $name = ''; + // 褰撳墠鏁版嵁琛ㄤ富閿 + protected $pk; + // 褰撳墠鏁版嵁琛ㄥ墠缂 + protected $prefix = ''; + // 鏌ヨ鍙傛暟 + protected $options = []; + // 鍙傛暟缁戝畾 + protected $bind = []; + // 鏁版嵁琛ㄤ俊鎭 + protected static $info = []; + // 鍥炶皟浜嬩欢 + private static $event = []; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Connection $connection 鏁版嵁搴撳璞″疄渚 + * @param string $model 妯″瀷鍚 + */ + public function __construct(Connection $connection = null, $model = '') + { + $this->connection = $connection ?: Db::connect([], true); + $this->prefix = $this->connection->getConfig('prefix'); + $this->model = $model; + // 璁剧疆褰撳墠杩炴帴鐨凚uilder瀵硅薄 + $this->setBuilder(); + } + + /** + * 鍒╃敤__call鏂规硶瀹炵幇涓浜涚壒娈婄殑Model鏂规硶 + * @access public + * @param string $method 鏂规硶鍚嶇О + * @param array $args 璋冪敤鍙傛暟 + * @return mixed + * @throws DbException + * @throws Exception + */ + public function __call($method, $args) + { + if (strtolower(substr($method, 0, 5)) == 'getby') { + // 鏍规嵁鏌愪釜瀛楁鑾峰彇璁板綍 + $field = Loader::parseName(substr($method, 5)); + $where[$field] = $args[0]; + return $this->where($where)->find(); + } elseif (strtolower(substr($method, 0, 10)) == 'getfieldby') { + // 鏍规嵁鏌愪釜瀛楁鑾峰彇璁板綍鐨勬煇涓 + $name = Loader::parseName(substr($method, 10)); + $where[$name] = $args[0]; + return $this->where($where)->value($args[1]); + } else { + throw new Exception('method not exist:' . __CLASS__ . '->' . $method); + } + } + + /** + * 鑾峰彇褰撳墠鐨勬暟鎹簱Connection瀵硅薄 + * @access public + * @return Connection + */ + public function getConnection() + { + return $this->connection; + } + + /** + * 鍒囨崲褰撳墠鐨勬暟鎹簱杩炴帴 + * @access public + * @param mixed $config + * @return $this + */ + public function connect($config) + { + $this->connection = Db::connect($config); + $this->setBuilder(); + return $this; + } + + /** + * 璁剧疆褰撳墠鐨勬暟鎹簱Builder瀵硅薄 + * @access protected + * @return void + */ + protected function setBuilder() + { + $builder = $this->connection->getConfig('builder') ?: $this->connection->getConfig('type'); + $class = false !== strpos($builder, '\\') ? $builder : '\\think\\db\\builder\\' . ucfirst($builder); + $this->builder = new $class($this->connection, $this); + } + + /** + * 鑾峰彇褰撳墠鐨勬ā鍨嬪璞″悕 + * @access public + * @return string + */ + public function getModel() + { + return $this->model; + } + + /** + * 鑾峰彇褰撳墠鐨刡uilder瀹炰緥瀵硅薄 + * @access public + * @return Builder + */ + public function getBuilder() + { + return $this->builder; + } + + /** + * 鎸囧畾榛樿鐨勬暟鎹〃鍚嶏紙涓嶅惈鍓嶇紑锛 + * @access public + * @param string $name + * @return $this + */ + public function name($name) + { + $this->name = $name; + return $this; + } + + /** + * 鎸囧畾榛樿鏁版嵁琛ㄥ悕锛堝惈鍓嶇紑锛 + * @access public + * @param string $table 琛ㄥ悕 + * @return $this + */ + public function setTable($table) + { + $this->table = $table; + return $this; + } + + /** + * 寰楀埌褰撳墠鎴栬呮寚瀹氬悕绉扮殑鏁版嵁琛 + * @access public + * @param string $name + * @return string + */ + public function getTable($name = '') + { + if ($name || empty($this->table)) { + $name = $name ?: $this->name; + $tableName = $this->prefix; + if ($name) { + $tableName .= Loader::parseName($name); + } + } else { + $tableName = $this->table; + } + return $tableName; + } + + /** + * 灏哠QL璇彞涓殑__TABLE_NAME__瀛楃涓叉浛鎹㈡垚甯﹀墠缂鐨勮〃鍚嶏紙灏忓啓锛 + * @access public + * @param string $sql sql璇彞 + * @return string + */ + public function parseSqlTable($sql) + { + if (false !== strpos($sql, '__')) { + $prefix = $this->prefix; + $sql = preg_replace_callback("/__([A-Z0-9_-]+)__/sU", function ($match) use ($prefix) { + return $prefix . strtolower($match[1]); + }, $sql); + } + return $sql; + } + + /** + * 鎵ц鏌ヨ 杩斿洖鏁版嵁闆 + * @access public + * @param string $sql sql鎸囦护 + * @param array $bind 鍙傛暟缁戝畾 + * @param boolean $master 鏄惁鍦ㄤ富鏈嶅姟鍣ㄨ鎿嶄綔 + * @param bool|string $class 鎸囧畾杩斿洖鐨勬暟鎹泦瀵硅薄 + * @return mixed + * @throws BindParamException + * @throws PDOException + */ + public function query($sql, $bind = [], $master = false, $class = false) + { + return $this->connection->query($sql, $bind, $master, $class); + } + + /** + * 鎵ц璇彞 + * @access public + * @param string $sql sql鎸囦护 + * @param array $bind 鍙傛暟缁戝畾 + * @return int + * @throws BindParamException + * @throws PDOException + */ + public function execute($sql, $bind = []) + { + return $this->connection->execute($sql, $bind); + } + + /** + * 鑾峰彇鏈杩戞彃鍏ョ殑ID + * @access public + * @param string $sequence 鑷搴忓垪鍚 + * @return string + */ + public function getLastInsID($sequence = null) + { + return $this->connection->getLastInsID($sequence); + } + + /** + * 鑾峰彇鏈杩戜竴娆℃煡璇㈢殑sql璇彞 + * @access public + * @return string + */ + public function getLastSql() + { + return $this->connection->getLastSql(); + } + + /** + * 鎵ц鏁版嵁搴撲簨鍔 + * @access public + * @param callable $callback 鏁版嵁鎿嶄綔鏂规硶鍥炶皟 + * @return mixed + */ + public function transaction($callback) + { + return $this->connection->transaction($callback); + } + + /** + * 鍚姩浜嬪姟 + * @access public + * @return void + */ + public function startTrans() + { + $this->connection->startTrans(); + } + + /** + * 鐢ㄤ簬闈炶嚜鍔ㄦ彁浜ょ姸鎬佷笅闈㈢殑鏌ヨ鎻愪氦 + * @access public + * @return void + * @throws PDOException + */ + public function commit() + { + $this->connection->commit(); + } + + /** + * 浜嬪姟鍥炴粴 + * @access public + * @return void + * @throws PDOException + */ + public function rollback() + { + $this->connection->rollback(); + } + + /** + * 鎵瑰鐞嗘墽琛孲QL璇彞 + * 鎵瑰鐞嗙殑鎸囦护閮借涓烘槸execute鎿嶄綔 + * @access public + * @param array $sql SQL鎵瑰鐞嗘寚浠 + * @return boolean + */ + public function batchQuery($sql = []) + { + return $this->connection->batchQuery($sql); + } + + /** + * 鑾峰彇鏁版嵁搴撶殑閰嶇疆鍙傛暟 + * @access public + * @param string $name 鍙傛暟鍚嶇О + * @return boolean + */ + public function getConfig($name = '') + { + return $this->connection->getConfig($name); + } + + /** + * 寰楀埌鍒嗚〃鐨勭殑鏁版嵁琛ㄥ悕 + * @access public + * @param array $data 鎿嶄綔鐨勬暟鎹 + * @param string $field 鍒嗚〃渚濇嵁鐨勫瓧娈 + * @param array $rule 鍒嗚〃瑙勫垯 + * @return string + */ + public function getPartitionTableName($data, $field, $rule = []) + { + // 瀵规暟鎹〃杩涜鍒嗗尯 + if ($field && isset($data[$field])) { + $value = $data[$field]; + $type = $rule['type']; + switch ($type) { + case 'id': + // 鎸夌収id鑼冨洿鍒嗚〃 + $step = $rule['expr']; + $seq = floor($value / $step) + 1; + break; + case 'year': + // 鎸夌収骞翠唤鍒嗚〃 + if (!is_numeric($value)) { + $value = strtotime($value); + } + $seq = date('Y', $value) - $rule['expr'] + 1; + break; + case 'mod': + // 鎸夌収id鐨勬ā鏁板垎琛 + $seq = ($value % $rule['num']) + 1; + break; + case 'md5': + // 鎸夌収md5鐨勫簭鍒楀垎琛 + $seq = (ord(substr(md5($value), 0, 1)) % $rule['num']) + 1; + break; + default: + if (function_exists($type)) { + // 鏀寔鎸囧畾鍑芥暟鍝堝笇 + $seq = (ord(substr($type($value), 0, 1)) % $rule['num']) + 1; + } else { + // 鎸夌収瀛楁鐨勯瀛楁瘝鐨勫煎垎琛 + $seq = (ord($value{0}) % $rule['num']) + 1; + } + } + return $this->getTable() . '_' . $seq; + } else { + // 褰撹缃殑鍒嗚〃瀛楁涓嶅湪鏌ヨ鏉′欢鎴栬呮暟鎹腑 + // 杩涜鑱斿悎鏌ヨ锛屽繀椤昏瀹 partition['num'] + $tableName = []; + for ($i = 0; $i < $rule['num']; $i++) { + $tableName[] = 'SELECT * FROM ' . $this->getTable() . '_' . ($i + 1); + } + + $tableName = '( ' . implode(" UNION ", $tableName) . ') AS ' . $this->name; + return $tableName; + } + } + + /** + * 寰楀埌鏌愪釜瀛楁鐨勫 + * @access public + * @param string $field 瀛楁鍚 + * @param mixed $default 榛樿鍊 + * @param bool $force 寮哄埗杞负鏁板瓧绫诲瀷 + * @return mixed + */ + public function value($field, $default = null, $force = false) + { + $result = false; + if (empty($options['fetch_sql']) && !empty($this->options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $this->options['cache']; + if (empty($this->options['table'])) { + $this->options['table'] = $this->getTable(); + } + $key = is_string($cache['key']) ? $cache['key'] : md5($field . serialize($this->options)); + $result = Cache::get($key); + } + if (false === $result) { + if (isset($this->options['field'])) { + unset($this->options['field']); + } + $pdo = $this->field($field)->limit(1)->getPdo(); + if (is_string($pdo)) { + // 杩斿洖SQL璇彞 + return $pdo; + } + $result = $pdo->fetchColumn(); + if ($force) { + $result = is_numeric($result) ? $result + 0 : $result; + } + if (isset($cache)) { + // 缂撳瓨鏁版嵁 + if (isset($cache['tag'])) { + Cache::tag($cache['tag'])->set($key, $result, $cache['expire']); + } else { + Cache::set($key, $result, $cache['expire']); + } + } + } else { + // 娓呯┖鏌ヨ鏉′欢 + $this->options = []; + } + return false !== $result ? $result : $default; + } + + /** + * 寰楀埌鏌愪釜鍒楃殑鏁扮粍 + * @access public + * @param string $field 瀛楁鍚 澶氫釜瀛楁鐢ㄩ楀彿鍒嗛殧 + * @param string $key 绱㈠紩 + * @return array + */ + public function column($field, $key = '') + { + $result = false; + if (empty($options['fetch_sql']) && !empty($this->options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $this->options['cache']; + if (empty($this->options['table'])) { + $this->options['table'] = $this->getTable(); + } + $guid = is_string($cache['key']) ? $cache['key'] : md5($field . serialize($this->options)); + $result = Cache::get($guid); + } + if (false === $result) { + if (isset($this->options['field'])) { + unset($this->options['field']); + } + if ($key && '*' != $field) { + $field = $key . ',' . $field; + } + $pdo = $this->field($field)->getPdo(); + if (is_string($pdo)) { + // 杩斿洖SQL璇彞 + return $pdo; + } + if (1 == $pdo->columnCount()) { + $result = $pdo->fetchAll(PDO::FETCH_COLUMN); + } else { + $resultSet = $pdo->fetchAll(PDO::FETCH_ASSOC); + if ($resultSet) { + $fields = array_keys($resultSet[0]); + $count = count($fields); + $key1 = array_shift($fields); + $key2 = $fields ? array_shift($fields) : ''; + $key = $key ?: $key1; + if (strpos($key, '.')) { + list($alias, $key) = explode('.', $key); + } + foreach ($resultSet as $val) { + if ($count > 2) { + $result[$val[$key]] = $val; + } elseif (2 == $count) { + $result[$val[$key]] = $val[$key2]; + } elseif (1 == $count) { + $result[$val[$key]] = $val[$key1]; + } + } + } else { + $result = []; + } + } + if (isset($cache) && isset($guid)) { + // 缂撳瓨鏁版嵁 + if (isset($cache['tag'])) { + Cache::tag($cache['tag'])->set($guid, $result, $cache['expire']); + } else { + Cache::set($guid, $result, $cache['expire']); + } + } + } else { + // 娓呯┖鏌ヨ鏉′欢 + $this->options = []; + } + return $result; + } + + /** + * COUNT鏌ヨ + * @access public + * @param string $field 瀛楁鍚 + * @return integer|string + */ + public function count($field = '*') + { + if (isset($this->options['group'])) { + // 鏀寔GROUP + $options = $this->getOptions(); + $subSql = $this->options($options)->field('count(' . $field . ')')->bind($this->bind)->buildSql(); + return $this->table([$subSql => '_group_count_'])->value('COUNT(*) AS tp_count', 0, true); + } + + return $this->value('COUNT(' . $field . ') AS tp_count', 0, true); + } + + /** + * SUM鏌ヨ + * @access public + * @param string $field 瀛楁鍚 + * @return float|int + */ + public function sum($field) + { + return $this->value('SUM(' . $field . ') AS tp_sum', 0, true); + } + + /** + * MIN鏌ヨ + * @access public + * @param string $field 瀛楁鍚 + * @return mixed + */ + public function min($field) + { + return $this->value('MIN(' . $field . ') AS tp_min', 0, true); + } + + /** + * MAX鏌ヨ + * @access public + * @param string $field 瀛楁鍚 + * @return mixed + */ + public function max($field) + { + return $this->value('MAX(' . $field . ') AS tp_max', 0, true); + } + + /** + * AVG鏌ヨ + * @access public + * @param string $field 瀛楁鍚 + * @return float|int + */ + public function avg($field) + { + return $this->value('AVG(' . $field . ') AS tp_avg', 0, true); + } + + /** + * 璁剧疆璁板綍鐨勬煇涓瓧娈靛 + * 鏀寔浣跨敤鏁版嵁搴撳瓧娈靛拰鏂规硶 + * @access public + * @param string|array $field 瀛楁鍚 + * @param mixed $value 瀛楁鍊 + * @return integer + */ + public function setField($field, $value = '') + { + if (is_array($field)) { + $data = $field; + } else { + $data[$field] = $value; + } + return $this->update($data); + } + + /** + * 瀛楁鍊(寤惰繜)澧為暱 + * @access public + * @param string $field 瀛楁鍚 + * @param integer $step 澧為暱鍊 + * @param integer $lazyTime 寤舵椂鏃堕棿(s) + * @return integer|true + * @throws Exception + */ + public function setInc($field, $step = 1, $lazyTime = 0) + { + $condition = !empty($this->options['where']) ? $this->options['where'] : []; + if (empty($condition)) { + // 娌℃湁鏉′欢涓嶅仛浠讳綍鏇存柊 + throw new Exception('no data to update'); + } + if ($lazyTime > 0) { + // 寤惰繜鍐欏叆 + $guid = md5($this->getTable() . '_' . $field . '_' . serialize($condition)); + $step = $this->lazyWrite('inc', $guid, $step, $lazyTime); + if (false === $step) { + // 娓呯┖鏌ヨ鏉′欢 + $this->options = []; + return true; + } + } + return $this->setField($field, ['exp', $field . '+' . $step]); + } + + /** + * 瀛楁鍊硷紙寤惰繜锛夊噺灏 + * @access public + * @param string $field 瀛楁鍚 + * @param integer $step 鍑忓皯鍊 + * @param integer $lazyTime 寤舵椂鏃堕棿(s) + * @return integer|true + * @throws Exception + */ + public function setDec($field, $step = 1, $lazyTime = 0) + { + $condition = !empty($this->options['where']) ? $this->options['where'] : []; + if (empty($condition)) { + // 娌℃湁鏉′欢涓嶅仛浠讳綍鏇存柊 + throw new Exception('no data to update'); + } + if ($lazyTime > 0) { + // 寤惰繜鍐欏叆 + $guid = md5($this->getTable() . '_' . $field . '_' . serialize($condition)); + $step = $this->lazyWrite('dec', $guid, $step, $lazyTime); + if (false === $step) { + // 娓呯┖鏌ヨ鏉′欢 + $this->options = []; + return true; + } + } + return $this->setField($field, ['exp', $field . '-' . $step]); + } + + /** + * 寤舵椂鏇存柊妫鏌 杩斿洖false琛ㄧず闇瑕佸欢鏃 + * 鍚﹀垯杩斿洖瀹為檯鍐欏叆鐨勬暟鍊 + * @access protected + * @param string $type 鑷鎴栬呰嚜鍑 + * @param string $guid 鍐欏叆鏍囪瘑 + * @param integer $step 鍐欏叆姝ヨ繘鍊 + * @param integer $lazyTime 寤舵椂鏃堕棿(s) + * @return false|integer + */ + protected function lazyWrite($type, $guid, $step, $lazyTime) + { + if (!Cache::has($guid . '_time')) { + // 璁℃椂寮濮 + Cache::set($guid . '_time', $_SERVER['REQUEST_TIME'], 0); + Cache::$type($guid, $step, 0); + } elseif ($_SERVER['REQUEST_TIME'] > Cache::get($guid . '_time') + $lazyTime) { + // 鍒犻櫎缂撳瓨 + $value = Cache::$type($guid, $step, 0); + Cache::rm($guid); + Cache::rm($guid . '_time'); + return 0 === $value ? false : $value; + } else { + // 鏇存柊缂撳瓨 + Cache::$type($guid, $step, 0); + } + return false; + } + + /** + * 鏌ヨSQL缁勮 join + * @access public + * @param mixed $join 鍏宠仈鐨勮〃鍚 + * @param mixed $condition 鏉′欢 + * @param string $type JOIN绫诲瀷 + * @return $this + */ + public function join($join, $condition = null, $type = 'INNER') + { + if (empty($condition)) { + // 濡傛灉涓虹粍鏁帮紝鍒欏惊鐜皟鐢╦oin + foreach ($join as $key => $value) { + if (is_array($value) && 2 <= count($value)) { + $this->join($value[0], $value[1], isset($value[2]) ? $value[2] : $type); + } + } + } else { + $table = $this->getJoinTable($join); + + $this->options['join'][] = [$table, strtoupper($type), $condition]; + } + return $this; + } + + /** + * 鑾峰彇Join琛ㄥ悕鍙婂埆鍚 鏀寔 + * ['prefix_table鎴栬呭瓙鏌ヨ'=>'alias'] 'prefix_table alias' 'table alias' + * @access public + * @param array|string $join + * @return array|string + */ + protected function getJoinTable($join, &$alias = null) + { + // 浼犲叆鐨勮〃鍚嶄负鏁扮粍 + if (is_array($join)) { + list($table, $alias) = each($join); + } else { + $join = trim($join); + if (false !== strpos($join, '(')) { + // 浣跨敤瀛愭煡璇 + $table = $join; + } else { + $prefix = $this->prefix; + if (strpos($join, ' ')) { + // 浣跨敤鍒悕 + list($table, $alias) = explode(' ', $join); + } else { + $table = $join; + if (false === strpos($join, '.') && 0 !== strpos($join, '__')) { + $alias = $join; + } + } + if ($prefix && false === strpos($table, '.') && 0 !== strpos($table, $prefix) && 0 !== strpos($table, '__')) { + $table = $this->getTable($table); + } + } + } + if (isset($alias)) { + if (isset($this->options['alias'][$table])) { + $table = $table . '@think' . uniqid(); + } + $table = [$table => $alias]; + $this->alias($table); + } + return $table; + } + + /** + * 鏌ヨSQL缁勮 union + * @access public + * @param mixed $union + * @param boolean $all + * @return $this + */ + public function union($union, $all = false) + { + $this->options['union']['type'] = $all ? 'UNION ALL' : 'UNION'; + + if (is_array($union)) { + $this->options['union'] = array_merge($this->options['union'], $union); + } else { + $this->options['union'][] = $union; + } + return $this; + } + + /** + * 鎸囧畾鏌ヨ瀛楁 鏀寔瀛楁鎺掗櫎鍜屾寚瀹氭暟鎹〃 + * @access public + * @param mixed $field + * @param boolean $except 鏄惁鎺掗櫎 + * @param string $tableName 鏁版嵁琛ㄥ悕 + * @param string $prefix 瀛楁鍓嶇紑 + * @param string $alias 鍒悕鍓嶇紑 + * @return $this + */ + public function field($field, $except = false, $tableName = '', $prefix = '', $alias = '') + { + if (empty($field)) { + return $this; + } + if (is_string($field)) { + $field = array_map('trim', explode(',', $field)); + } + if (true === $field) { + // 鑾峰彇鍏ㄩ儴瀛楁 + $fields = $this->getTableInfo($tableName ?: (isset($this->options['table']) ? $this->options['table'] : ''), 'fields'); + $field = $fields ?: ['*']; + } elseif ($except) { + // 瀛楁鎺掗櫎 + $fields = $this->getTableInfo($tableName ?: (isset($this->options['table']) ? $this->options['table'] : ''), 'fields'); + $field = $fields ? array_diff($fields, $field) : $field; + } + if ($tableName) { + // 娣诲姞缁熶竴鐨勫墠缂 + $prefix = $prefix ?: $tableName; + foreach ($field as $key => $val) { + if (is_numeric($key)) { + $val = $prefix . '.' . $val . ($alias ? ' AS ' . $alias . $val : ''); + } + $field[$key] = $val; + } + } + + if (isset($this->options['field'])) { + $field = array_merge($this->options['field'], $field); + } + $this->options['field'] = array_unique($field); + return $this; + } + + /** + * 璁剧疆鏁版嵁 + * @access public + * @param mixed $field 瀛楁鍚嶆垨鑰呮暟鎹 + * @param mixed $value 瀛楁鍊 + * @return $this + */ + public function data($field, $value = null) + { + if (is_array($field)) { + $this->options['data'] = isset($this->options['data']) ? array_merge($this->options['data'], $field) : $field; + } else { + $this->options['data'][$field] = $value; + } + return $this; + } + + /** + * 瀛楁鍊煎闀 + * @access public + * @param string|array $field 瀛楁鍚 + * @param integer $step 澧為暱鍊 + * @return $this + */ + public function inc($field, $step = 1) + { + $fields = is_string($field) ? explode(',', $field) : $field; + foreach ($fields as $field) { + $this->data($field, ['exp', $field . '+' . $step]); + } + return $this; + } + + /** + * 瀛楁鍊煎噺灏 + * @access public + * @param string|array $field 瀛楁鍚 + * @param integer $step 澧為暱鍊 + * @return $this + */ + public function dec($field, $step = 1) + { + $fields = is_string($field) ? explode(',', $field) : $field; + foreach ($fields as $field) { + $this->data($field, ['exp', $field . '-' . $step]); + } + return $this; + } + + /** + * 浣跨敤琛ㄨ揪寮忚缃暟鎹 + * @access public + * @param string $field 瀛楁鍚 + * @param string $value 瀛楁鍊 + * @return $this + */ + public function exp($field, $value) + { + $this->data($field, ['exp', $value]); + return $this; + } + + /** + * 鎸囧畾JOIN鏌ヨ瀛楁 + * @access public + * @param string|array $table 鏁版嵁琛 + * @param string|array $field 鏌ヨ瀛楁 + * @param string|array $on JOIN鏉′欢 + * @param string $type JOIN绫诲瀷 + * @return $this + */ + public function view($join, $field = true, $on = null, $type = 'INNER') + { + $this->options['view'] = true; + if (is_array($join) && is_null($field)) { + foreach ($join as $key => $val) { + $this->view($key, $val[0], isset($val[1]) ? $val[1] : null, isset($val[2]) ? $val[2] : 'INNER'); + } + } else { + $fields = []; + $table = $this->getJoinTable($join, $alias); + + if (true === $field) { + $fields = $alias . '.*'; + } else { + if (is_string($field)) { + $field = explode(',', $field); + } + foreach ($field as $key => $val) { + if (is_numeric($key)) { + $fields[] = $alias . '.' . $val; + $this->options['map'][$val] = $alias . '.' . $val; + } else { + if (preg_match('/[,=\.\'\"\(\s]/', $key)) { + $name = $key; + } else { + $name = $alias . '.' . $key; + } + $fields[] = $name . ' AS ' . $val; + $this->options['map'][$val] = $name; + } + } + } + $this->field($fields); + if ($on) { + $this->join($table, $on, $type); + } else { + $this->table($table); + } + } + return $this; + } + + /** + * 璁剧疆鍒嗚〃瑙勫垯 + * @access public + * @param array $data 鎿嶄綔鐨勬暟鎹 + * @param string $field 鍒嗚〃渚濇嵁鐨勫瓧娈 + * @param array $rule 鍒嗚〃瑙勫垯 + * @return $this + */ + public function partition($data, $field, $rule = []) + { + $this->options['table'] = $this->getPartitionTableName($data, $field, $rule); + return $this; + } + + /** + * 鎸囧畾AND鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @return $this + */ + public function where($field, $op = null, $condition = null) + { + $param = func_get_args(); + array_shift($param); + $this->parseWhereExp('AND', $field, $op, $condition, $param); + return $this; + } + + /** + * 鎸囧畾OR鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @return $this + */ + public function whereOr($field, $op = null, $condition = null) + { + $param = func_get_args(); + array_shift($param); + $this->parseWhereExp('OR', $field, $op, $condition, $param); + return $this; + } + + /** + * 鎸囧畾XOR鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @return $this + */ + public function whereXor($field, $op = null, $condition = null) + { + $param = func_get_args(); + array_shift($param); + $this->parseWhereExp('XOR', $field, $op, $condition, $param); + return $this; + } + + /** + * 鎸囧畾Null鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNull($field, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'null', null); + return $this; + } + + /** + * 鎸囧畾NotNull鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotNull($field, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'notnull', null); + return $this; + } + + /** + * 鎸囧畾Exists鏌ヨ鏉′欢 + * @access public + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereExists($condition, $logic = 'AND') + { + $this->options['where'][strtoupper($logic)][] = ['exists', $condition]; + return $this; + } + + /** + * 鎸囧畾NotExists鏌ヨ鏉′欢 + * @access public + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotExists($condition, $logic = 'AND') + { + $this->options['where'][strtoupper($logic)][] = ['not exists', $condition]; + return $this; + } + + /** + * 鎸囧畾In鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereIn($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'in', $condition); + return $this; + } + + /** + * 鎸囧畾NotIn鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotIn($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'not in', $condition); + return $this; + } + + /** + * 鎸囧畾Like鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereLike($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'like', $condition); + return $this; + } + + /** + * 鎸囧畾NotLike鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotLike($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'not like', $condition); + return $this; + } + + /** + * 鎸囧畾Between鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereBetween($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'between', $condition); + return $this; + } + + /** + * 鎸囧畾NotBetween鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotBetween($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'not between', $condition); + return $this; + } + + /** + * 鎸囧畾Exp鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereExp($field, $condition, $logic = 'AND') + { + $this->parseWhereExp($logic, $field, 'exp', $condition); + return $this; + } + + /** + * 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + * @access public + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @param string|array|\Closure $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param array $param 鏌ヨ鍙傛暟 + * @return void + */ + protected function parseWhereExp($logic, $field, $op, $condition, $param = []) + { + $logic = strtoupper($logic); + if ($field instanceof \Closure) { + $this->options['where'][$logic][] = is_string($op) ? [$op, $field] : $field; + return; + } + + if (is_string($field) && !empty($this->options['via']) && !strpos($field, '.')) { + $field = $this->options['via'] . '.' . $field; + } + if (is_string($field) && preg_match('/[,=\>\<\'\"\(\s]/', $field)) { + $where[] = ['exp', $field]; + if (is_array($op)) { + // 鍙傛暟缁戝畾 + $this->bind($op); + } + } elseif (is_null($op) && is_null($condition)) { + if (is_array($field)) { + // 鏁扮粍鎵归噺鏌ヨ + $where = $field; + foreach ($where as $k => $val) { + $this->options['multi'][$logic][$k][] = $val; + } + } elseif ($field && is_string($field)) { + // 瀛楃涓叉煡璇 + $where[$field] = ['null', '']; + $this->options['multi'][$logic][$field][] = $where[$field]; + } + } elseif (is_array($op)) { + $where[$field] = $param; + } elseif (in_array(strtolower($op), ['null', 'notnull', 'not null'])) { + // null鏌ヨ + $where[$field] = [$op, '']; + $this->options['multi'][$logic][$field][] = $where[$field]; + } elseif (is_null($condition)) { + // 瀛楁鐩哥瓑鏌ヨ + $where[$field] = ['eq', $op]; + if ('AND' != $logic) { + $this->options['multi'][$logic][$field][] = $where[$field]; + } + } else { + $where[$field] = [$op, $condition, isset($param[2]) ? $param[2] : null]; + if ('exp' == strtolower($op) && isset($param[2]) && is_array($param[2])) { + // 鍙傛暟缁戝畾 + $this->bind($param[2]); + } + // 璁板綍涓涓瓧娈靛娆℃煡璇㈡潯浠 + $this->options['multi'][$logic][$field][] = $where[$field]; + } + if (!empty($where)) { + if (!isset($this->options['where'][$logic])) { + $this->options['where'][$logic] = []; + } + if (is_string($field) && $this->checkMultiField($field, $logic)) { + $where[$field] = $this->options['multi'][$logic][$field]; + } elseif (is_array($field)) { + foreach ($field as $key => $val) { + if ($this->checkMultiField($key, $logic)) { + $where[$key] = $this->options['multi'][$logic][$key]; + } + } + } + $this->options['where'][$logic] = array_merge($this->options['where'][$logic], $where); + } + } + + /** + * 妫鏌ユ槸鍚﹀瓨鍦ㄤ竴涓瓧娈靛娆℃煡璇㈡潯浠 + * @access public + * @param string $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return bool + */ + private function checkMultiField($field, $logic) + { + return isset($this->options['multi'][$logic][$field]) && count($this->options['multi'][$logic][$field]) > 1; + } + + /** + * 鍘婚櫎鏌愪釜鏌ヨ鏉′欢 + * @access public + * @param string $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function removeWhereField($field, $logic = 'AND') + { + $logic = strtoupper($logic); + if (isset($this->options['where'][$logic][$field])) { + unset($this->options['where'][$logic][$field]); + } + return $this; + } + + /** + * 鍘婚櫎鏌ヨ鍙傛暟 + * @access public + * @param string|bool $option 鍙傛暟鍚 true 琛ㄧず鍘婚櫎鎵鏈夊弬鏁 + * @return $this + */ + public function removeOption($option = true) + { + if (true === $option) { + $this->options = []; + } elseif (is_string($option) && isset($this->options[$option])) { + unset($this->options[$option]); + } + return $this; + } + + /** + * 鎸囧畾鏌ヨ鏁伴噺 + * @access public + * @param mixed $offset 璧峰浣嶇疆 + * @param mixed $length 鏌ヨ鏁伴噺 + * @return $this + */ + public function limit($offset, $length = null) + { + if (is_null($length) && strpos($offset, ',')) { + list($offset, $length) = explode(',', $offset); + } + $this->options['limit'] = intval($offset) . ($length ? ',' . intval($length) : ''); + return $this; + } + + /** + * 鎸囧畾鍒嗛〉 + * @access public + * @param mixed $page 椤垫暟 + * @param mixed $listRows 姣忛〉鏁伴噺 + * @return $this + */ + public function page($page, $listRows = null) + { + if (is_null($listRows) && strpos($page, ',')) { + list($page, $listRows) = explode(',', $page); + } + $this->options['page'] = [intval($page), intval($listRows)]; + return $this; + } + + /** + * 鍒嗛〉鏌ヨ + * @param int|array $listRows 姣忛〉鏁伴噺 鏁扮粍琛ㄧず閰嶇疆鍙傛暟 + * @param int|bool $simple 鏄惁绠娲佹ā寮忔垨鑰呮昏褰曟暟 + * @param array $config 閰嶇疆鍙傛暟 + * page:褰撳墠椤, + * path:url璺緞, + * query:url棰濆鍙傛暟, + * fragment:url閿氱偣, + * var_page:鍒嗛〉鍙橀噺, + * list_rows:姣忛〉鏁伴噺 + * type:鍒嗛〉绫诲悕 + * @return \think\Paginator + * @throws DbException + */ + public function paginate($listRows = null, $simple = false, $config = []) + { + if (is_int($simple)) { + $total = $simple; + $simple = false; + } + if (is_array($listRows)) { + $config = array_merge(Config::get('paginate'), $listRows); + $listRows = $config['list_rows']; + } else { + $config = array_merge(Config::get('paginate'), $config); + $listRows = $listRows ?: $config['list_rows']; + } + + /** @var Paginator $class */ + $class = false !== strpos($config['type'], '\\') ? $config['type'] : '\\think\\paginator\\driver\\' . ucwords($config['type']); + $page = isset($config['page']) ? (int) $config['page'] : call_user_func([ + $class, + 'getCurrentPage', + ], $config['var_page']); + + $page = $page < 1 ? 1 : $page; + + $config['path'] = isset($config['path']) ? $config['path'] : call_user_func([$class, 'getCurrentPath']); + + if (!isset($total) && !$simple) { + $options = $this->getOptions(); + + unset($this->options['order'], $this->options['limit'], $this->options['page'], $this->options['field']); + + $bind = $this->bind; + $total = $this->count(); + $results = $this->options($options)->bind($bind)->page($page, $listRows)->select(); + } elseif ($simple) { + $results = $this->limit(($page - 1) * $listRows, $listRows + 1)->select(); + $total = null; + } else { + $results = $this->page($page, $listRows)->select(); + } + return $class::make($results, $listRows, $page, $total, $simple, $config); + } + + /** + * 鎸囧畾褰撳墠鎿嶄綔鐨勬暟鎹〃 + * @access public + * @param mixed $table 琛ㄥ悕 + * @return $this + */ + public function table($table) + { + if (is_string($table)) { + if (strpos($table, ')')) { + // 瀛愭煡璇 + } elseif (strpos($table, ',')) { + $tables = explode(',', $table); + $table = []; + foreach ($tables as $item) { + list($item, $alias) = explode(' ', trim($item)); + if ($alias) { + $this->alias([$item => $alias]); + $table[$item] = $alias; + } else { + $table[] = $item; + } + } + } elseif (strpos($table, ' ')) { + list($table, $alias) = explode(' ', $table); + + $table = [$table => $alias]; + $this->alias($table); + } + } else { + $tables = $table; + $table = []; + foreach ($tables as $key => $val) { + if (is_numeric($key)) { + $table[] = $val; + } else { + $this->alias([$key => $val]); + $table[$key] = $val; + } + } + } + $this->options['table'] = $table; + return $this; + } + + /** + * USING鏀寔 鐢ㄤ簬澶氳〃鍒犻櫎 + * @access public + * @param mixed $using + * @return $this + */ + public function using($using) + { + $this->options['using'] = $using; + return $this; + } + + /** + * 鎸囧畾鎺掑簭 order('id','desc') 鎴栬 order(['id'=>'desc','create_time'=>'desc']) + * @access public + * @param string|array $field 鎺掑簭瀛楁 + * @param string $order 鎺掑簭 + * @return $this + */ + public function order($field, $order = null) + { + if (!empty($field)) { + if (is_string($field)) { + if (!empty($this->options['via'])) { + $field = $this->options['via'] . '.' . $field; + } + $field = empty($order) ? $field : [$field => $order]; + } elseif (!empty($this->options['via'])) { + foreach ($field as $key => $val) { + if (is_numeric($key)) { + $field[$key] = $this->options['via'] . '.' . $val; + } else { + $field[$this->options['via'] . '.' . $key] = $val; + unset($field[$key]); + } + } + } + if (!isset($this->options['order'])) { + $this->options['order'] = []; + } + if (is_array($field)) { + $this->options['order'] = array_merge($this->options['order'], $field); + } else { + $this->options['order'][] = $field; + } + } + return $this; + } + + /** + * 鏌ヨ缂撳瓨 + * @access public + * @param mixed $key 缂撳瓨key + * @param integer $expire 缂撳瓨鏈夋晥鏈 + * @param string $tag 缂撳瓨鏍囩 + * @return $this + */ + public function cache($key = true, $expire = null, $tag = null) + { + // 澧炲姞蹇嵎璋冪敤鏂瑰紡 cache(10) 绛夊悓浜 cache(true, 10) + if (is_numeric($key) && is_null($expire)) { + $expire = $key; + $key = true; + } + if (false !== $key) { + $this->options['cache'] = ['key' => $key, 'expire' => $expire, 'tag' => $tag]; + } + return $this; + } + + /** + * 鎸囧畾group鏌ヨ + * @access public + * @param string $group GROUP + * @return $this + */ + public function group($group) + { + $this->options['group'] = $group; + return $this; + } + + /** + * 鎸囧畾having鏌ヨ + * @access public + * @param string $having having + * @return $this + */ + public function having($having) + { + $this->options['having'] = $having; + return $this; + } + + /** + * 鎸囧畾鏌ヨlock + * @access public + * @param boolean $lock 鏄惁lock + * @return $this + */ + public function lock($lock = false) + { + $this->options['lock'] = $lock; + $this->options['master'] = true; + return $this; + } + + /** + * 鎸囧畾distinct鏌ヨ + * @access public + * @param string $distinct 鏄惁鍞竴 + * @return $this + */ + public function distinct($distinct) + { + $this->options['distinct'] = $distinct; + return $this; + } + + /** + * 鎸囧畾鏁版嵁琛ㄥ埆鍚 + * @access public + * @param mixed $alias 鏁版嵁琛ㄥ埆鍚 + * @return $this + */ + public function alias($alias) + { + if (is_array($alias)) { + foreach ($alias as $key => $val) { + $this->options['alias'][$key] = $val; + } + } else { + if (isset($this->options['table'])) { + $table = is_array($this->options['table']) ? key($this->options['table']) : $this->options['table']; + if (false !== strpos($table, '__')) { + $table = $this->parseSqlTable($table); + } + } else { + $table = $this->getTable(); + } + + $this->options['alias'][$table] = $alias; + } + return $this; + } + + /** + * 鎸囧畾寮哄埗绱㈠紩 + * @access public + * @param string $force 绱㈠紩鍚嶇О + * @return $this + */ + public function force($force) + { + $this->options['force'] = $force; + return $this; + } + + /** + * 鏌ヨ娉ㄩ噴 + * @access public + * @param string $comment 娉ㄩ噴 + * @return $this + */ + public function comment($comment) + { + $this->options['comment'] = $comment; + return $this; + } + + /** + * 鑾峰彇鎵ц鐨凷QL璇彞 + * @access public + * @param boolean $fetch 鏄惁杩斿洖sql + * @return $this + */ + public function fetchSql($fetch = true) + { + $this->options['fetch_sql'] = $fetch; + return $this; + } + + /** + * 涓嶄富鍔ㄨ幏鍙栨暟鎹泦 + * @access public + * @param bool $pdo 鏄惁杩斿洖 PDOStatement 瀵硅薄 + * @return $this + */ + public function fetchPdo($pdo = true) + { + $this->options['fetch_pdo'] = $pdo; + return $this; + } + + /** + * 璁剧疆浠庝富鏈嶅姟鍣ㄨ鍙栨暟鎹 + * @access public + * @return $this + */ + public function master() + { + $this->options['master'] = true; + return $this; + } + + /** + * 璁剧疆鏄惁涓ユ牸妫鏌ュ瓧娈靛悕 + * @access public + * @param bool $strict 鏄惁涓ユ牸妫鏌ュ瓧娈 + * @return $this + */ + public function strict($strict = true) + { + $this->options['strict'] = $strict; + return $this; + } + + /** + * 璁剧疆鏌ヨ鏁版嵁涓嶅瓨鍦ㄦ槸鍚︽姏鍑哄紓甯 + * @access public + * @param bool $fail 鏁版嵁涓嶅瓨鍦ㄦ槸鍚︽姏鍑哄紓甯 + * @return $this + */ + public function failException($fail = true) + { + $this->options['fail'] = $fail; + return $this; + } + + /** + * 璁剧疆鑷搴忓垪鍚 + * @access public + * @param string $sequence 鑷搴忓垪鍚 + * @return $this + */ + public function sequence($sequence = null) + { + $this->options['sequence'] = $sequence; + return $this; + } + + /** + * 鎸囧畾鏁版嵁琛ㄤ富閿 + * @access public + * @param string $pk 涓婚敭 + * @return $this + */ + public function pk($pk) + { + $this->pk = $pk; + return $this; + } + + /** + * 鏌ヨ鏃ユ湡鎴栬呮椂闂 + * @access public + * @param string $field 鏃ユ湡瀛楁鍚 + * @param string $op 姣旇緝杩愮畻绗︽垨鑰呰〃杈惧紡 + * @param string|array $range 姣旇緝鑼冨洿 + * @return $this + */ + public function whereTime($field, $op, $range = null) + { + if (is_null($range)) { + // 浣跨敤鏃ユ湡琛ㄨ揪寮 + $date = getdate(); + switch (strtolower($op)) { + case 'today': + case 'd': + $range = ['today', 'tomorrow']; + break; + case 'week': + case 'w': + $range = 'this week 00:00:00'; + break; + case 'month': + case 'm': + $range = mktime(0, 0, 0, $date['mon'], 1, $date['year']); + break; + case 'year': + case 'y': + $range = mktime(0, 0, 0, 1, 1, $date['year']); + break; + case 'yesterday': + $range = ['yesterday', 'today']; + break; + case 'last week': + $range = ['last week 00:00:00', 'this week 00:00:00']; + break; + case 'last month': + $range = [date('y-m-01', strtotime('-1 month')), mktime(0, 0, 0, $date['mon'], 1, $date['year'])]; + break; + case 'last year': + $range = [mktime(0, 0, 0, 1, 1, $date['year'] - 1), mktime(0, 0, 0, 1, 1, $date['year'])]; + break; + default: + $range = $op; + } + $op = is_array($range) ? 'between' : '>'; + } + $this->where($field, strtolower($op) . ' time', $range); + return $this; + } + + /** + * 鑾峰彇鏁版嵁琛ㄤ俊鎭 + * @access public + * @param mixed $tableName 鏁版嵁琛ㄥ悕 鐣欑┖鑷姩鑾峰彇 + * @param string $fetch 鑾峰彇淇℃伅绫诲瀷 鍖呮嫭 fields type bind pk + * @return mixed + */ + public function getTableInfo($tableName = '', $fetch = '') + { + if (!$tableName) { + $tableName = $this->getTable(); + } + if (is_array($tableName)) { + $tableName = key($tableName) ?: current($tableName); + } + + if (strpos($tableName, ',')) { + // 澶氳〃涓嶈幏鍙栧瓧娈典俊鎭 + return false; + } else { + $tableName = $this->parseSqlTable($tableName); + } + + // 淇瀛愭煡璇綔涓鸿〃鍚嶇殑闂 + if (strpos($tableName, ')')) { + return []; + } + + list($guid) = explode(' ', $tableName); + $db = $this->getConfig('database'); + if (!isset(self::$info[$db . '.' . $guid])) { + if (!strpos($guid, '.')) { + $schema = $db . '.' . $guid; + } else { + $schema = $guid; + } + // 璇诲彇缂撳瓨 + if (is_file(RUNTIME_PATH . 'schema/' . $schema . '.php')) { + $info = include RUNTIME_PATH . 'schema/' . $schema . '.php'; + } else { + $info = $this->connection->getFields($guid); + } + $fields = array_keys($info); + $bind = $type = []; + foreach ($info as $key => $val) { + // 璁板綍瀛楁绫诲瀷 + $type[$key] = $val['type']; + $bind[$key] = $this->getFieldBindType($val['type']); + if (!empty($val['primary'])) { + $pk[] = $key; + } + } + if (isset($pk)) { + // 璁剧疆涓婚敭 + $pk = count($pk) > 1 ? $pk : $pk[0]; + } else { + $pk = null; + } + self::$info[$db . '.' . $guid] = ['fields' => $fields, 'type' => $type, 'bind' => $bind, 'pk' => $pk]; + } + return $fetch ? self::$info[$db . '.' . $guid][$fetch] : self::$info[$db . '.' . $guid]; + } + + /** + * 鑾峰彇褰撳墠鏁版嵁琛ㄧ殑涓婚敭 + * @access public + * @param string|array $options 鏁版嵁琛ㄥ悕鎴栬呮煡璇㈠弬鏁 + * @return string|array + */ + public function getPk($options = '') + { + if (!empty($this->pk)) { + $pk = $this->pk; + } else { + $pk = $this->getTableInfo(is_array($options) ? $options['table'] : $options, 'pk'); + } + return $pk; + } + + // 鑾峰彇褰撳墠鏁版嵁琛ㄥ瓧娈典俊鎭 + public function getTableFields($options) + { + return $this->getTableInfo($options['table'], 'fields'); + } + + // 鑾峰彇褰撳墠鏁版嵁琛ㄥ瓧娈电被鍨 + public function getFieldsType($options) + { + return $this->getTableInfo($options['table'], 'type'); + } + + // 鑾峰彇褰撳墠鏁版嵁琛ㄧ粦瀹氫俊鎭 + public function getFieldsBind($options) + { + $types = $this->getFieldsType($options); + $bind = []; + if ($types) { + foreach ($types as $key => $type) { + $bind[$key] = $this->getFieldBindType($type); + } + } + return $bind; + } + + /** + * 鑾峰彇瀛楁缁戝畾绫诲瀷 + * @access public + * @param string $type 瀛楁绫诲瀷 + * @return integer + */ + protected function getFieldBindType($type) + { + if (preg_match('/(int|double|float|decimal|real|numeric|serial|bit)/is', $type)) { + $bind = PDO::PARAM_INT; + } elseif (preg_match('/bool/is', $type)) { + $bind = PDO::PARAM_BOOL; + } else { + $bind = PDO::PARAM_STR; + } + return $bind; + } + + /** + * 鍙傛暟缁戝畾 + * @access public + * @param mixed $key 鍙傛暟鍚 + * @param mixed $value 缁戝畾鍙橀噺鍊 + * @param integer $type 缁戝畾绫诲瀷 + * @return $this + */ + public function bind($key, $value = false, $type = PDO::PARAM_STR) + { + if (is_array($key)) { + $this->bind = array_merge($this->bind, $key); + } else { + $this->bind[$key] = [$value, $type]; + } + return $this; + } + + /** + * 妫娴嬪弬鏁版槸鍚﹀凡缁忕粦瀹 + * @access public + * @param string $key 鍙傛暟鍚 + * @return bool + */ + public function isBind($key) + { + return isset($this->bind[$key]); + } + + /** + * 鏌ヨ鍙傛暟璧嬪 + * @access protected + * @param array $options 琛ㄨ揪寮忓弬鏁 + * @return $this + */ + protected function options(array $options) + { + $this->options = $options; + return $this; + } + + /** + * 鑾峰彇褰撳墠鐨勬煡璇㈠弬鏁 + * @access public + * @param string $name 鍙傛暟鍚 + * @return mixed + */ + public function getOptions($name = '') + { + if ('' === $name) { + return $this->options; + } else { + return isset($this->options[$name]) ? $this->options[$name] : null; + } + } + + /** + * 璁剧疆鍏宠仈鏌ヨJOIN棰勬煡璇 + * @access public + * @param string|array $with 鍏宠仈鏂规硶鍚嶇О + * @return $this + */ + public function with($with) + { + if (empty($with)) { + return $this; + } + + if (is_string($with)) { + $with = explode(',', $with); + } + + $first = true; + $currentModel = $this->model; + + /** @var Model $class */ + $class = new $currentModel; + foreach ($with as $key => $relation) { + $subRelation = ''; + $closure = false; + if ($relation instanceof \Closure) { + // 鏀寔闂寘鏌ヨ杩囨护鍏宠仈鏉′欢 + $closure = $relation; + $relation = $key; + $with[$key] = $key; + } elseif (is_string($relation) && strpos($relation, '.')) { + $with[$key] = $relation; + list($relation, $subRelation) = explode('.', $relation, 2); + } + + /** @var Relation $model */ + $relation = Loader::parseName($relation, 1, false); + $model = $class->$relation(); + if ($model instanceof OneToOne && 0 == $model->getEagerlyType()) { + $model->eagerly($this, $relation, $subRelation, $closure, $first); + $first = false; + } elseif ($closure) { + $with[$key] = $closure; + } + } + $this->via(); + if (isset($this->options['with'])) { + $this->options['with'] = array_merge($this->options['with'], $with); + } else { + $this->options['with'] = $with; + } + return $this; + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param string|array $relation 鍏宠仈鏂规硶鍚 + * @param bool $subQuery 鏄惁浣跨敤瀛愭煡璇 + * @return $this + */ + public function withCount($relation, $subQuery = true) + { + if (!$subQuery) { + $this->options['with_count'] = $relation; + } else { + $relations = is_string($relation) ? explode(',', $relation) : $relation; + if (!isset($this->options['field'])) { + $this->field('*'); + } + foreach ($relations as $key => $relation) { + $closure = false; + if ($relation instanceof \Closure) { + $closure = $relation; + $relation = $key; + } + $relation = Loader::parseName($relation, 1, false); + $count = '(' . (new $this->model)->$relation()->getRelationCountQuery($closure) . ')'; + $this->field([$count => Loader::parseName($relation) . '_count']); + } + } + return $this; + } + + /** + * 鍏宠仈棰勫姞杞戒腑 鑾峰彇鍏宠仈鎸囧畾瀛楁鍊 + * example: + * Model::with(['relation' => function($query){ + * $query->withField("id,name"); + * }]) + * + * @param string | array $field 鎸囧畾鑾峰彇鐨勫瓧娈 + * @return $this + */ + public function withField($field) + { + $this->options['with_field'] = $field; + return $this; + } + + /** + * 璁剧疆褰撳墠瀛楁娣诲姞鐨勮〃鍒悕 + * @access public + * @param string $via + * @return $this + */ + public function via($via = '') + { + $this->options['via'] = $via; + return $this; + } + + /** + * 璁剧疆鍏宠仈鏌ヨ + * @access public + * @param string|array $relation 鍏宠仈鍚嶇О + * @return $this + */ + public function relation($relation) + { + if (empty($relation)) { + return $this; + } + if (is_string($relation)) { + $relation = explode(',', $relation); + } + if (isset($this->options['relation'])) { + $this->options['relation'] = array_mrege($this->options['relation'], $relation); + } else { + $this->options['relation'] = $relation; + } + return $this; + } + + /** + * 鎶婁富閿艰浆鎹负鏌ヨ鏉′欢 鏀寔澶嶅悎涓婚敭 + * @access public + * @param array|string $data 涓婚敭鏁版嵁 + * @param mixed $options 琛ㄨ揪寮忓弬鏁 + * @return void + * @throws Exception + */ + protected function parsePkWhere($data, &$options) + { + $pk = $this->getPk($options); + // 鑾峰彇褰撳墠鏁版嵁琛 + $table = is_array($options['table']) ? key($options['table']) : $options['table']; + if (!empty($options['alias'][$table])) { + $alias = $options['alias'][$table]; + } + if (is_string($pk)) { + $key = isset($alias) ? $alias . '.' . $pk : $pk; + // 鏍规嵁涓婚敭鏌ヨ + if (is_array($data)) { + $where[$key] = isset($data[$pk]) ? $data[$pk] : ['in', $data]; + } else { + $where[$key] = strpos($data, ',') ? ['IN', $data] : $data; + } + } elseif (is_array($pk) && is_array($data) && !empty($data)) { + // 鏍规嵁澶嶅悎涓婚敭鏌ヨ + foreach ($pk as $key) { + if (isset($data[$key])) { + $attr = isset($alias) ? $alias . '.' . $key : $key; + $where[$attr] = $data[$key]; + } else { + throw new Exception('miss complex primary data'); + } + } + } + + if (!empty($where)) { + if (isset($options['where']['AND'])) { + $options['where']['AND'] = array_merge($options['where']['AND'], $where); + } else { + $options['where']['AND'] = $where; + } + } + return; + } + + /** + * 鎻掑叆璁板綍 + * @access public + * @param mixed $data 鏁版嵁 + * @param boolean $replace 鏄惁replace + * @param boolean $getLastInsID 杩斿洖鑷涓婚敭 + * @param string $sequence 鑷搴忓垪鍚 + * @return integer|string + */ + public function insert(array $data = [], $replace = false, $getLastInsID = false, $sequence = null) + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + $data = array_merge($options['data'], $data); + // 鐢熸垚SQL璇彞 + $sql = $this->builder->insert($data, $options, $replace); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } + + // 鎵ц鎿嶄綔 + $result = $this->execute($sql, $bind); + if ($result) { + $this->trigger('after_insert', $options); + } + if ($getLastInsID) { + $sequence = $sequence ?: (isset($options['sequence']) ? $options['sequence'] : null); + return $this->getLastInsID($sequence); + } + return $result; + } + + /** + * 鎻掑叆璁板綍骞惰幏鍙栬嚜澧濱D + * @access public + * @param mixed $data 鏁版嵁 + * @param boolean $replace 鏄惁replace + * @param string $sequence 鑷搴忓垪鍚 + * @return integer|string + */ + public function insertGetId(array $data, $replace = false, $sequence = null) + { + return $this->insert($data, $replace, true, $sequence); + } + + /** + * 鎵归噺鎻掑叆璁板綍 + * @access public + * @param mixed $dataSet 鏁版嵁闆 + * @return integer|string + */ + public function insertAll(array $dataSet) + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + if (!is_array(reset($dataSet))) { + return false; + } + // 鐢熸垚SQL璇彞 + $sql = $this->builder->insertAll($dataSet, $options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } else { + // 鎵ц鎿嶄綔 + return $this->execute($sql, $bind); + } + } + + /** + * 閫氳繃Select鏂瑰紡鎻掑叆璁板綍 + * @access public + * @param string $fields 瑕佹彃鍏ョ殑鏁版嵁琛ㄥ瓧娈靛悕 + * @param string $table 瑕佹彃鍏ョ殑鏁版嵁琛ㄥ悕 + * @return integer|string + * @throws PDOException + */ + public function selectInsert($fields, $table) + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + // 鐢熸垚SQL璇彞 + $table = $this->parseSqlTable($table); + $sql = $this->builder->selectInsert($fields, $table, $options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } else { + // 鎵ц鎿嶄綔 + return $this->execute($sql, $bind); + } + } + + /** + * 鏇存柊璁板綍 + * @access public + * @param mixed $data 鏁版嵁 + * @return integer|string + * @throws Exception + * @throws PDOException + */ + public function update(array $data = []) + { + $options = $this->parseExpress(); + $data = array_merge($options['data'], $data); + $pk = $this->getPk($options); + if (isset($options['cache']) && is_string($options['cache'])) { + $key = $options['cache']; + } + + if (empty($options['where'])) { + // 濡傛灉瀛樺湪涓婚敭鏁版嵁 鍒欒嚜鍔ㄤ綔涓烘洿鏂版潯浠 + if (is_string($pk) && isset($data[$pk])) { + $where[$pk] = $data[$pk]; + if (!isset($key)) { + $key = 'think:' . $options['table'] . '|' . $data[$pk]; + } + unset($data[$pk]); + } elseif (is_array($pk)) { + // 澧炲姞澶嶅悎涓婚敭鏀寔 + foreach ($pk as $field) { + if (isset($data[$field])) { + $where[$field] = $data[$field]; + } else { + // 濡傛灉缂哄皯澶嶅悎涓婚敭鏁版嵁鍒欎笉鎵ц + throw new Exception('miss complex primary data'); + } + unset($data[$field]); + } + } + if (!isset($where)) { + // 濡傛灉娌℃湁浠讳綍鏇存柊鏉′欢鍒欎笉鎵ц + throw new Exception('miss update condition'); + } else { + $options['where']['AND'] = $where; + } + } elseif (is_string($pk) && isset($options['where']['AND'][$pk]) && is_scalar($options['where']['AND'][$pk])) { + $key = 'think:' . $options['table'] . '|' . $options['where']['AND'][$pk]; + } + // 鐢熸垚UPDATE SQL璇彞 + $sql = $this->builder->update($data, $options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } else { + // 妫娴嬬紦瀛 + if (isset($key) && Cache::get($key)) { + // 鍒犻櫎缂撳瓨 + Cache::rm($key); + } + // 鎵ц鎿嶄綔 + $result = '' == $sql ? 0 : $this->execute($sql, $bind); + if ($result) { + $this->trigger('after_update', $options); + } + return $result; + } + } + + /** + * 鎵ц鏌ヨ浣嗗彧杩斿洖PDOStatement瀵硅薄 + * @access public + * @return \PDOStatement|string + */ + public function getPdo() + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + // 鐢熸垚鏌ヨSQL + $sql = $this->builder->select($options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } + // 鎵ц鏌ヨ鎿嶄綔 + return $this->query($sql, $bind, $options['master'], true); + } + + /** + * 鏌ユ壘璁板綍 + * @access public + * @param array|string|Query|\Closure $data + * @return Collection|false|\PDOStatement|string + * @throws DbException + * @throws ModelNotFoundException + * @throws DataNotFoundException + */ + public function select($data = null) + { + if ($data instanceof Query) { + return $data->select(); + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [ & $this]); + $data = null; + } + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + + if (false === $data) { + // 鐢ㄤ簬瀛愭煡璇 涓嶆煡璇㈠彧杩斿洖SQL + $options['fetch_sql'] = true; + } elseif (!is_null($data)) { + // 涓婚敭鏉′欢鍒嗘瀽 + $this->parsePkWhere($data, $options); + } + + $resultSet = false; + if (empty($options['fetch_sql']) && !empty($options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $options['cache']; + unset($options['cache']); + $key = is_string($cache['key']) ? $cache['key'] : md5(serialize($options)); + $resultSet = Cache::get($key); + } + if (!$resultSet) { + // 鐢熸垚鏌ヨSQL + $sql = $this->builder->select($options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } + if ($resultSet = $this->trigger('before_select', $options)) { + } else { + // 鎵ц鏌ヨ鎿嶄綔 + $resultSet = $this->query($sql, $bind, $options['master'], $options['fetch_pdo']); + + if ($resultSet instanceof \PDOStatement) { + // 杩斿洖PDOStatement瀵硅薄 + return $resultSet; + } + } + + if (isset($cache)) { + // 缂撳瓨鏁版嵁闆 + if (isset($cache['tag'])) { + Cache::tag($cache['tag'])->set($key, $resultSet, $cache['expire']); + } else { + Cache::set($key, $resultSet, $cache['expire']); + } + } + } + + // 鏁版嵁鍒楄〃璇诲彇鍚庣殑澶勭悊 + if (!empty($this->model)) { + // 鐢熸垚妯″瀷瀵硅薄 + $modelName = $this->model; + if (count($resultSet) > 0) { + foreach ($resultSet as $key => $result) { + /** @var Model $result */ + $model = new $modelName($result); + $model->isUpdate(true); + + // 鍏宠仈鏌ヨ + if (!empty($options['relation'])) { + $model->relationQuery($options['relation']); + } + // 鍏宠仈缁熻 + if (!empty($options['with_count'])) { + $model->relationCount($model, $options['with_count']); + } + $resultSet[$key] = $model; + } + if (!empty($options['with'])) { + // 棰勮浇鍏 + $model->eagerlyResultSet($resultSet, $options['with']); + } + // 妯″瀷鏁版嵁闆嗚浆鎹 + $resultSet = $model->toCollection($resultSet); + } else { + $resultSet = (new $modelName)->toCollection($resultSet); + } + } elseif ('collection' == $this->connection->getConfig('resultset_type')) { + // 杩斿洖Collection瀵硅薄 + $resultSet = new Collection($resultSet); + } + // 杩斿洖缁撴灉澶勭悊 + if (!empty($options['fail']) && count($resultSet) == 0) { + $this->throwNotFound($options); + } + return $resultSet; + } + + /** + * 鏌ユ壘鍗曟潯璁板綍 + * @access public + * @param array|string|Query|\Closure $data + * @return array|false|\PDOStatement|string|Model + * @throws DbException + * @throws ModelNotFoundException + * @throws DataNotFoundException + */ + public function find($data = null) + { + if ($data instanceof Query) { + return $data->find(); + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [ & $this]); + $data = null; + } + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + + if (!is_null($data)) { + // AR妯″紡鍒嗘瀽涓婚敭鏉′欢 + $this->parsePkWhere($data, $options); + } + + $options['limit'] = 1; + $result = false; + if (empty($options['fetch_sql']) && !empty($options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $options['cache']; + if (true === $cache['key'] && !is_null($data) && !is_array($data)) { + $key = 'think:' . (is_array($options['table']) ? key($options['table']) : $options['table']) . '|' . $data; + } else { + $key = is_string($cache['key']) ? $cache['key'] : md5(serialize($options)); + } + $result = Cache::get($key); + } + if (!$result) { + // 鐢熸垚鏌ヨSQL + $sql = $this->builder->select($options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } + + // 浜嬩欢鍥炶皟 + if ($result = $this->trigger('before_find', $options)) { + } else { + // 鎵ц鏌ヨ + $result = $this->query($sql, $bind, $options['master'], $options['fetch_pdo']); + + if ($result instanceof \PDOStatement) { + // 杩斿洖PDOStatement瀵硅薄 + return $result; + } + } + + if (isset($cache)) { + // 缂撳瓨鏁版嵁 + if (isset($cache['tag'])) { + Cache::tag($cache['tag'])->set($key, $result, $cache['expire']); + } else { + Cache::set($key, $result, $cache['expire']); + } + } + } + + // 鏁版嵁澶勭悊 + if (!empty($result[0])) { + $data = $result[0]; + if (!empty($this->model)) { + // 杩斿洖妯″瀷瀵硅薄 + $model = $this->model; + $data = new $model($data); + $data->isUpdate(true, isset($options['where']['AND']) ? $options['where']['AND'] : null); + // 鍏宠仈鏌ヨ + if (!empty($options['relation'])) { + $data->relationQuery($options['relation']); + } + // 棰勮浇鍏ユ煡璇 + if (!empty($options['with'])) { + $data->eagerlyResult($data, $options['with']); + } + // 鍏宠仈缁熻 + if (!empty($options['with_count'])) { + $data->relationCount($data, $options['with_count']); + } + } + } elseif (!empty($options['fail'])) { + $this->throwNotFound($options); + } else { + $data = null; + } + return $data; + } + + /** + * 鏌ヨ澶辫触 鎶涘嚭寮傚父 + * @access public + * @param array $options 鏌ヨ鍙傛暟 + * @throws ModelNotFoundException + * @throws DataNotFoundException + */ + protected function throwNotFound($options = []) + { + if (!empty($this->model)) { + throw new ModelNotFoundException('model data Not Found:' . $this->model, $this->model, $options); + } else { + $table = is_array($options['table']) ? key($options['table']) : $options['table']; + throw new DataNotFoundException('table data not Found:' . $table, $table, $options); + } + } + + /** + * 鏌ユ壘澶氭潯璁板綍 濡傛灉涓嶅瓨鍦ㄥ垯鎶涘嚭寮傚父 + * @access public + * @param array|string|Query|\Closure $data + * @return array|\PDOStatement|string|Model + * @throws DbException + * @throws ModelNotFoundException + * @throws DataNotFoundException + */ + public function selectOrFail($data = null) + { + return $this->failException(true)->select($data); + } + + /** + * 鏌ユ壘鍗曟潯璁板綍 濡傛灉涓嶅瓨鍦ㄥ垯鎶涘嚭寮傚父 + * @access public + * @param array|string|Query|\Closure $data + * @return array|\PDOStatement|string|Model + * @throws DbException + * @throws ModelNotFoundException + * @throws DataNotFoundException + */ + public function findOrFail($data = null) + { + return $this->failException(true)->find($data); + } + + /** + * 鍒嗘壒鏁版嵁杩斿洖澶勭悊 + * @access public + * @param integer $count 姣忔澶勭悊鐨勬暟鎹暟閲 + * @param callable $callback 澶勭悊鍥炶皟鏂规硶 + * @param string $column 鍒嗘壒澶勭悊鐨勫瓧娈靛悕 + * @return boolean + */ + public function chunk($count, $callback, $column = null) + { + $options = $this->getOptions(); + if (isset($options['table'])) { + $table = is_array($options['table']) ? key($options['table']) : $options['table']; + } else { + $table = ''; + } + $column = $column ?: $this->getPk($table); + $bind = $this->bind; + $resultSet = $this->limit($count)->order($column, 'asc')->select(); + if (strpos($column, '.')) { + list($alias, $key) = explode('.', $column); + } else { + $key = $column; + } + if ($resultSet instanceof Collection) { + $resultSet = $resultSet->all(); + } + + while (!empty($resultSet)) { + if (false === call_user_func($callback, $resultSet)) { + return false; + } + $end = end($resultSet); + $lastId = is_array($end) ? $end[$key] : $end->$key; + $resultSet = $this->options($options) + ->limit($count) + ->bind($bind) + ->where($column, '>', $lastId) + ->order($column, 'asc') + ->select(); + if ($resultSet instanceof Collection) { + $resultSet = $resultSet->all(); + } + } + return true; + } + + /** + * 鑾峰彇缁戝畾鐨勫弬鏁 骞舵竻绌 + * @access public + * @return array + */ + public function getBind() + { + $bind = $this->bind; + $this->bind = []; + return $bind; + } + + /** + * 鍒涘缓瀛愭煡璇QL + * @access public + * @param bool $sub + * @return string + * @throws DbException + */ + public function buildSql($sub = true) + { + return $sub ? '( ' . $this->select(false) . ' )' : $this->select(false); + } + + /** + * 鍒犻櫎璁板綍 + * @access public + * @param mixed $data 琛ㄨ揪寮 true 琛ㄧず寮哄埗鍒犻櫎 + * @return int + * @throws Exception + * @throws PDOException + */ + public function delete($data = null) + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + if (isset($options['cache']) && is_string($options['cache'])) { + $key = $options['cache']; + } + + if (!is_null($data) && true !== $data) { + if (!isset($key) && !is_array($data)) { + // 缂撳瓨鏍囪瘑 + $key = 'think:' . $options['table'] . '|' . $data; + } + // AR妯″紡鍒嗘瀽涓婚敭鏉′欢 + $this->parsePkWhere($data, $options); + } + + if (true !== $data && empty($options['where'])) { + // 濡傛灉鏉′欢涓虹┖ 涓嶈繘琛屽垹闄ゆ搷浣 闄ら潪璁剧疆 1=1 + throw new Exception('delete without condition'); + } + // 鐢熸垚鍒犻櫎SQL璇彞 + $sql = $this->builder->delete($options); + // 鑾峰彇鍙傛暟缁戝畾 + $bind = $this->getBind(); + if ($options['fetch_sql']) { + // 鑾峰彇瀹為檯鎵ц鐨凷QL璇彞 + return $this->connection->getRealSql($sql, $bind); + } + + // 妫娴嬬紦瀛 + if (isset($key) && Cache::get($key)) { + // 鍒犻櫎缂撳瓨 + Cache::rm($key); + } + // 鎵ц鎿嶄綔 + $result = $this->execute($sql, $bind); + if ($result) { + $this->trigger('after_delete', $options); + } + return $result; + } + + /** + * 鍒嗘瀽琛ㄨ揪寮忥紙鍙敤浜庢煡璇㈡垨鑰呭啓鍏ユ搷浣滐級 + * @access protected + * @return array + */ + protected function parseExpress() + { + $options = $this->options; + + // 鑾峰彇鏁版嵁琛 + if (empty($options['table'])) { + $options['table'] = $this->getTable(); + } + + if (!isset($options['where'])) { + $options['where'] = []; + } elseif (isset($options['view'])) { + // 瑙嗗浘鏌ヨ鏉′欢澶勭悊 + foreach (['AND', 'OR'] as $logic) { + if (isset($options['where'][$logic])) { + foreach ($options['where'][$logic] as $key => $val) { + if (array_key_exists($key, $options['map'])) { + $options['where'][$logic][$options['map'][$key]] = $val; + unset($options['where'][$logic][$key]); + } + } + } + } + + if (isset($options['order'])) { + // 瑙嗗浘鏌ヨ鎺掑簭澶勭悊 + if (is_string($options['order'])) { + $options['order'] = explode(',', $options['order']); + } + foreach ($options['order'] as $key => $val) { + if (is_numeric($key)) { + if (strpos($val, ' ')) { + list($field, $sort) = explode(' ', $val); + if (array_key_exists($field, $options['map'])) { + $options['order'][$options['map'][$field]] = $sort; + unset($options['order'][$key]); + } + } elseif (array_key_exists($val, $options['map'])) { + $options['order'][$options['map'][$val]] = 'asc'; + unset($options['order'][$key]); + } + } elseif (array_key_exists($key, $options['map'])) { + $options['order'][$options['map'][$key]] = $val; + unset($options['order'][$key]); + } + } + } + } + + if (!isset($options['field'])) { + $options['field'] = '*'; + } + + if (!isset($options['data'])) { + $options['data'] = []; + } + + if (!isset($options['strict'])) { + $options['strict'] = $this->getConfig('fields_strict'); + } + + foreach (['master', 'lock', 'fetch_pdo', 'fetch_sql', 'distinct'] as $name) { + if (!isset($options[$name])) { + $options[$name] = false; + } + } + + foreach (['join', 'union', 'group', 'having', 'limit', 'order', 'force', 'comment'] as $name) { + if (!isset($options[$name])) { + $options[$name] = ''; + } + } + + if (isset($options['page'])) { + // 鏍规嵁椤垫暟璁$畻limit + list($page, $listRows) = $options['page']; + $page = $page > 0 ? $page : 1; + $listRows = $listRows > 0 ? $listRows : (is_numeric($options['limit']) ? $options['limit'] : 20); + $offset = $listRows * ($page - 1); + $options['limit'] = $offset . ',' . $listRows; + } + + $this->options = []; + return $options; + } + + /** + * 娉ㄥ唽鍥炶皟鏂规硶 + * @access public + * @param string $event 浜嬩欢鍚 + * @param callable $callback 鍥炶皟鏂规硶 + * @return void + */ + public static function event($event, $callback) + { + self::$event[$event] = $callback; + } + + /** + * 瑙﹀彂浜嬩欢 + * @access protected + * @param string $event 浜嬩欢鍚 + * @param mixed $options 褰撳墠鏌ヨ鍙傛暟 + * @return bool + */ + protected function trigger($event, $options = []) + { + $result = false; + if (isset(self::$event[$event])) { + $callback = self::$event[$event]; + $result = call_user_func_array($callback, [$options, $this]); + } + return $result; + } +} diff --git a/thinkphp/library/think/db/builder/Mysql.php b/thinkphp/library/think/db/builder/Mysql.php new file mode 100644 index 000000000..de38fac5f --- /dev/null +++ b/thinkphp/library/think/db/builder/Mysql.php @@ -0,0 +1,62 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\builder; + +use think\db\Builder; + +/** + * mysql鏁版嵁搴撻┍鍔 + */ +class Mysql extends Builder +{ + protected $updateSql = 'UPDATE %TABLE% %JOIN% SET %SET% %WHERE% %ORDER%%LIMIT% %LOCK%%COMMENT%'; + + /** + * 瀛楁鍜岃〃鍚嶅鐞 + * @access protected + * @param string $key + * @param array $options + * @return string + */ + protected function parseKey($key, $options = []) + { + $key = trim($key); + if (strpos($key, '$.') && false === strpos($key, '(')) { + // JSON瀛楁鏀寔 + list($field, $name) = explode('$.', $key); + $key = 'json_extract(' . $field . ', \'$.' . $name . '\')'; + } elseif (strpos($key, '.') && !preg_match('/[,\'\"\(\)`\s]/', $key)) { + list($table, $key) = explode('.', $key, 2); + if (isset($options['alias'][$table])) { + $table = $options['alias'][$table]; + } + } + if (!preg_match('/[,\'\"\*\(\)`.\s]/', $key)) { + $key = '`' . $key . '`'; + } + if (isset($table)) { + $key = '`' . $table . '`.' . $key; + } + return $key; + } + + /** + * 闅忔満鎺掑簭 + * @access protected + * @return string + */ + protected function parseRand() + { + return 'rand()'; + } + +} diff --git a/thinkphp/library/think/db/builder/Pgsql.php b/thinkphp/library/think/db/builder/Pgsql.php new file mode 100644 index 000000000..8b853a2f1 --- /dev/null +++ b/thinkphp/library/think/db/builder/Pgsql.php @@ -0,0 +1,78 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\builder; + +use think\db\Builder; + +/** + * Pgsql鏁版嵁搴撻┍鍔 + */ +class Pgsql extends Builder +{ + + /** + * limit鍒嗘瀽 + * @access protected + * @param mixed $limit + * @return string + */ + public function parseLimit($limit) + { + $limitStr = ''; + if (!empty($limit)) { + $limit = explode(',', $limit); + if (count($limit) > 1) { + $limitStr .= ' LIMIT ' . $limit[1] . ' OFFSET ' . $limit[0] . ' '; + } else { + $limitStr .= ' LIMIT ' . $limit[0] . ' '; + } + } + return $limitStr; + } + + /** + * 瀛楁鍜岃〃鍚嶅鐞 + * @access protected + * @param string $key + * @param array $options + * @return string + */ + protected function parseKey($key, $options = []) + { + $key = trim($key); + if (strpos($key, '$.') && false === strpos($key, '(')) { + // JSON瀛楁鏀寔 + list($field, $name) = explode('$.', $key); + $key = $field . '->>\'' . $name . '\''; + } elseif (strpos($key, '.')) { + list($table, $key) = explode('.', $key, 2); + if (isset($options['alias'][$table])) { + $table = $options['alias'][$table]; + } + } + if (isset($table)) { + $key = $table . '.' . $key; + } + return $key; + } + + /** + * 闅忔満鎺掑簭 + * @access protected + * @return string + */ + protected function parseRand() + { + return 'RANDOM()'; + } + +} diff --git a/thinkphp/library/think/db/builder/Sqlite.php b/thinkphp/library/think/db/builder/Sqlite.php new file mode 100644 index 000000000..02d1bf2e0 --- /dev/null +++ b/thinkphp/library/think/db/builder/Sqlite.php @@ -0,0 +1,72 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\builder; + +use think\db\Builder; + +/** + * Sqlite鏁版嵁搴撻┍鍔 + */ +class Sqlite extends Builder +{ + + /** + * limit + * @access public + * @return string + */ + public function parseLimit($limit) + { + $limitStr = ''; + if (!empty($limit)) { + $limit = explode(',', $limit); + if (count($limit) > 1) { + $limitStr .= ' LIMIT ' . $limit[1] . ' OFFSET ' . $limit[0] . ' '; + } else { + $limitStr .= ' LIMIT ' . $limit[0] . ' '; + } + } + return $limitStr; + } + + /** + * 闅忔満鎺掑簭 + * @access protected + * @return string + */ + protected function parseRand() + { + return 'RANDOM()'; + } + + /** + * 瀛楁鍜岃〃鍚嶅鐞 + * @access protected + * @param string $key + * @param array $options + * @return string + */ + protected function parseKey($key, $options = []) + { + $key = trim($key); + if (strpos($key, '.')) { + list($table, $key) = explode('.', $key, 2); + if (isset($options['alias'][$table])) { + $table = $options['alias'][$table]; + } + } + if (isset($table)) { + $key = $table . '.' . $key; + } + return $key; + } +} diff --git a/thinkphp/library/think/db/builder/Sqlsrv.php b/thinkphp/library/think/db/builder/Sqlsrv.php new file mode 100644 index 000000000..d2f418f38 --- /dev/null +++ b/thinkphp/library/think/db/builder/Sqlsrv.php @@ -0,0 +1,116 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\builder; + +use think\db\Builder; + +/** + * Sqlsrv鏁版嵁搴撻┍鍔 + */ +class Sqlsrv extends Builder +{ + protected $selectSql = 'SELECT T1.* FROM (SELECT thinkphp.*, ROW_NUMBER() OVER (%ORDER%) AS ROW_NUMBER FROM (SELECT %DISTINCT% %FIELD% FROM %TABLE%%JOIN%%WHERE%%GROUP%%HAVING%) AS thinkphp) AS T1 %LIMIT%%COMMENT%'; + protected $selectInsertSql = 'SELECT %DISTINCT% %FIELD% FROM %TABLE%%JOIN%%WHERE%%GROUP%%HAVING%'; + protected $updateSql = 'UPDATE %TABLE% SET %SET% FROM %TABLE% %JOIN% %WHERE% %LIMIT% %LOCK%%COMMENT%'; + protected $deleteSql = 'DELETE FROM %TABLE% %USING% %JOIN% %WHERE% %LIMIT% %LOCK%%COMMENT%'; + + /** + * order鍒嗘瀽 + * @access protected + * @param mixed $order + * @param array $options + * @return string + */ + protected function parseOrder($order, $options = []) + { + if (is_array($order)) { + $array = []; + foreach ($order as $key => $val) { + if (is_numeric($key)) { + if (false === strpos($val, '(')) { + $array[] = $this->parseKey($val, $options); + } elseif ('[rand]' == $val) { + $array[] = $this->parseRand(); + } + } else { + $sort = in_array(strtolower(trim($val)), ['asc', 'desc']) ? ' ' . $val : ''; + $array[] = $this->parseKey($key, $options) . ' ' . $sort; + } + } + $order = implode(',', $array); + } + return !empty($order) ? ' ORDER BY ' . $order : ' ORDER BY rand()'; + } + + /** + * 闅忔満鎺掑簭 + * @access protected + * @return string + */ + protected function parseRand() + { + return 'rand()'; + } + + /** + * 瀛楁鍜岃〃鍚嶅鐞 + * @access protected + * @param string $key + * @param array $options + * @return string + */ + protected function parseKey($key, $options = []) + { + $key = trim($key); + if (strpos($key, '.') && !preg_match('/[,\'\"\(\)\[\s]/', $key)) { + list($table, $key) = explode('.', $key, 2); + if (isset($options['alias'][$table])) { + $table = $options['alias'][$table]; + } + } + if (!is_numeric($key) && !preg_match('/[,\'\"\*\(\)\[.\s]/', $key)) { + $key = '[' . $key . ']'; + } + if (isset($table)) { + $key = '[' . $table . '].' . $key; + } + return $key; + } + + /** + * limit + * @access protected + * @param mixed $limit + * @return string + */ + protected function parseLimit($limit) + { + if (empty($limit)) { + return ''; + } + + $limit = explode(',', $limit); + if (count($limit) > 1) { + $limitStr = '(T1.ROW_NUMBER BETWEEN ' . $limit[0] . ' + 1 AND ' . $limit[0] . ' + ' . $limit[1] . ')'; + } else { + $limitStr = '(T1.ROW_NUMBER BETWEEN 1 AND ' . $limit[0] . ")"; + } + return 'WHERE ' . $limitStr; + } + + public function selectInsert($fields, $table, $options) + { + $this->selectSql = $this->selectInsertSql; + return parent::selectInsert($fields, $table, $options); + } + +} diff --git a/thinkphp/library/think/db/connector/Mysql.php b/thinkphp/library/think/db/connector/Mysql.php new file mode 100644 index 000000000..31435694f --- /dev/null +++ b/thinkphp/library/think/db/connector/Mysql.php @@ -0,0 +1,130 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\connector; + +use PDO; +use think\db\Connection; +use think\Log; + +/** + * mysql鏁版嵁搴撻┍鍔 + */ +class Mysql extends Connection +{ + + /** + * 瑙f瀽pdo杩炴帴鐨刣sn淇℃伅 + * @access protected + * @param array $config 杩炴帴淇℃伅 + * @return string + */ + protected function parseDsn($config) + { + $dsn = 'mysql:dbname=' . $config['database'] . ';host=' . $config['hostname']; + if (!empty($config['hostport'])) { + $dsn .= ';port=' . $config['hostport']; + } elseif (!empty($config['socket'])) { + $dsn .= ';unix_socket=' . $config['socket']; + } + if (!empty($config['charset'])) { + $dsn .= ';charset=' . $config['charset']; + } + return $dsn; + } + + /** + * 鍙栧緱鏁版嵁琛ㄧ殑瀛楁淇℃伅 + * @access public + * @param string $tableName + * @return array + */ + public function getFields($tableName) + { + $this->initConnect(true); + list($tableName) = explode(' ', $tableName); + if (false === strpos($tableName, '`')) { + if (strpos($tableName, '.')) { + $tableName = str_replace('.', '`.`', $tableName); + } + $tableName = '`' . $tableName . '`'; + } + $sql = 'SHOW COLUMNS FROM ' . $tableName; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + if ($result) { + foreach ($result as $key => $val) { + $val = array_change_key_case($val); + $info[$val['field']] = [ + 'name' => $val['field'], + 'type' => $val['type'], + 'notnull' => (bool) ('' === $val['null']), // not null is empty, null is yes + 'default' => $val['default'], + 'primary' => (strtolower($val['key']) == 'pri'), + 'autoinc' => (strtolower($val['extra']) == 'auto_increment'), + ]; + } + } + return $this->fieldCase($info); + } + + /** + * 鍙栧緱鏁版嵁搴撶殑琛ㄤ俊鎭 + * @access public + * @param string $dbName + * @return array + */ + public function getTables($dbName = '') + { + $this->initConnect(true); + $sql = !empty($dbName) ? 'SHOW TABLES FROM ' . $dbName : 'SHOW TABLES '; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + foreach ($result as $key => $val) { + $info[$key] = current($val); + } + return $info; + } + + /** + * SQL鎬ц兘鍒嗘瀽 + * @access protected + * @param string $sql + * @return array + */ + protected function getExplain($sql) + { + $pdo = $this->linkID->query("EXPLAIN " . $sql); + $result = $pdo->fetch(PDO::FETCH_ASSOC); + $result = array_change_key_case($result); + if (isset($result['extra'])) { + if (strpos($result['extra'], 'filesort') || strpos($result['extra'], 'temporary')) { + Log::record('SQL:' . $this->queryStr . '[' . $result['extra'] . ']', 'warn'); + } + } + return $result; + } + + protected function supportSavepoint() + { + return true; + } +} diff --git a/thinkphp/library/think/db/connector/Pgsql.php b/thinkphp/library/think/db/connector/Pgsql.php new file mode 100644 index 000000000..4f8756cab --- /dev/null +++ b/thinkphp/library/think/db/connector/Pgsql.php @@ -0,0 +1,110 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\connector; + +use PDO; +use think\db\Connection; + +/** + * Pgsql鏁版嵁搴撻┍鍔 + */ +class Pgsql extends Connection +{ + + /** + * 瑙f瀽pdo杩炴帴鐨刣sn淇℃伅 + * @access protected + * @param array $config 杩炴帴淇℃伅 + * @return string + */ + protected function parseDsn($config) + { + $dsn = 'pgsql:dbname=' . $config['database'] . ';host=' . $config['hostname']; + if (!empty($config['hostport'])) { + $dsn .= ';port=' . $config['hostport']; + } + return $dsn; + } + + /** + * 鍙栧緱鏁版嵁琛ㄧ殑瀛楁淇℃伅 + * @access public + * @param string $tableName + * @return array + */ + public function getFields($tableName) + { + $this->initConnect(true); + list($tableName) = explode(' ', $tableName); + $sql = 'select fields_name as "field",fields_type as "type",fields_not_null as "null",fields_key_name as "key",fields_default as "default",fields_default as "extra" from table_msg(\'' . $tableName . '\');'; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + if ($result) { + foreach ($result as $key => $val) { + $val = array_change_key_case($val); + $info[$val['field']] = [ + 'name' => $val['field'], + 'type' => $val['type'], + 'notnull' => (bool) ('' !== $val['null']), + 'default' => $val['default'], + 'primary' => !empty($val['key']), + 'autoinc' => (0 === strpos($val['extra'], 'nextval(')), + ]; + } + } + return $this->fieldCase($info); + } + + /** + * 鍙栧緱鏁版嵁搴撶殑琛ㄤ俊鎭 + * @access public + * @param string $dbName + * @return array + */ + public function getTables($dbName = '') + { + $this->initConnect(true); + $sql = "select tablename as Tables_in_test from pg_tables where schemaname ='public'"; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + foreach ($result as $key => $val) { + $info[$key] = current($val); + } + return $info; + } + + /** + * SQL鎬ц兘鍒嗘瀽 + * @access protected + * @param string $sql + * @return array + */ + protected function getExplain($sql) + { + return []; + } + + protected function supportSavepoint() + { + return true; + } +} diff --git a/thinkphp/library/think/db/connector/Sqlite.php b/thinkphp/library/think/db/connector/Sqlite.php new file mode 100644 index 000000000..4a08c7400 --- /dev/null +++ b/thinkphp/library/think/db/connector/Sqlite.php @@ -0,0 +1,109 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\connector; + +use PDO; +use think\db\Connection; + +/** + * Sqlite鏁版嵁搴撻┍鍔 + */ +class Sqlite extends Connection +{ + + /** + * 瑙f瀽pdo杩炴帴鐨刣sn淇℃伅 + * @access protected + * @param array $config 杩炴帴淇℃伅 + * @return string + */ + protected function parseDsn($config) + { + $dsn = 'sqlite:' . $config['database']; + return $dsn; + } + + /** + * 鍙栧緱鏁版嵁琛ㄧ殑瀛楁淇℃伅 + * @access public + * @param string $tableName + * @return array + */ + public function getFields($tableName) + { + $this->initConnect(true); + list($tableName) = explode(' ', $tableName); + $sql = 'PRAGMA table_info( ' . $tableName . ' )'; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + if ($result) { + foreach ($result as $key => $val) { + $val = array_change_key_case($val); + $info[$val['name']] = [ + 'name' => $val['name'], + 'type' => $val['type'], + 'notnull' => 1 === $val['notnull'], + 'default' => $val['dflt_value'], + 'primary' => '1' == $val['pk'], + 'autoinc' => '1' == $val['pk'], + ]; + } + } + return $this->fieldCase($info); + } + + /** + * 鍙栧緱鏁版嵁搴撶殑琛ㄤ俊鎭 + * @access public + * @param string $dbName + * @return array + */ + public function getTables($dbName = '') + { + $this->initConnect(true); + $sql = "SELECT name FROM sqlite_master WHERE type='table' " + . "UNION ALL SELECT name FROM sqlite_temp_master " + . "WHERE type='table' ORDER BY name"; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + foreach ($result as $key => $val) { + $info[$key] = current($val); + } + return $info; + } + + /** + * SQL鎬ц兘鍒嗘瀽 + * @access protected + * @param string $sql + * @return array + */ + protected function getExplain($sql) + { + return []; + } + + protected function supportSavepoint() + { + return true; + } +} diff --git a/thinkphp/library/think/db/connector/Sqlsrv.php b/thinkphp/library/think/db/connector/Sqlsrv.php new file mode 100644 index 000000000..181480513 --- /dev/null +++ b/thinkphp/library/think/db/connector/Sqlsrv.php @@ -0,0 +1,130 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\connector; + +use PDO; +use think\db\Connection; + +/** + * Sqlsrv鏁版嵁搴撻┍鍔 + */ +class Sqlsrv extends Connection +{ + // PDO杩炴帴鍙傛暟 + protected $params = [ + PDO::ATTR_CASE => PDO::CASE_NATURAL, + PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, + PDO::ATTR_STRINGIFY_FETCHES => false, + ]; + + /** + * 瑙f瀽pdo杩炴帴鐨刣sn淇℃伅 + * @access protected + * @param array $config 杩炴帴淇℃伅 + * @return string + */ + protected function parseDsn($config) + { + $dsn = 'sqlsrv:Database=' . $config['database'] . ';Server=' . $config['hostname']; + if (!empty($config['hostport'])) { + $dsn .= ',' . $config['hostport']; + } + return $dsn; + } + + /** + * 鍙栧緱鏁版嵁琛ㄧ殑瀛楁淇℃伅 + * @access public + * @param string $tableName + * @return array + */ + public function getFields($tableName) + { + $this->initConnect(true); + list($tableName) = explode(' ', $tableName); + $sql = "SELECT column_name, data_type, column_default, is_nullable + FROM information_schema.tables AS t + JOIN information_schema.columns AS c + ON t.table_catalog = c.table_catalog + AND t.table_schema = c.table_schema + AND t.table_name = c.table_name + WHERE t.table_name = '$tableName'"; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + if ($result) { + foreach ($result as $key => $val) { + $val = array_change_key_case($val); + $info[$val['column_name']] = [ + 'name' => $val['column_name'], + 'type' => $val['data_type'], + 'notnull' => (bool) ('' === $val['is_nullable']), // not null is empty, null is yes + 'default' => $val['column_default'], + 'primary' => false, + 'autoinc' => false, + ]; + } + } + $sql = "SELECT column_name FROM information_schema.key_column_usage WHERE table_name='$tableName'"; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetch(PDO::FETCH_ASSOC); + if ($result) { + $info[$result['column_name']]['primary'] = true; + } + return $this->fieldCase($info); + } + + /** + * 鍙栧緱鏁版嵁琛ㄧ殑瀛楁淇℃伅 + * @access public + * @param string $dbName + * @return array + */ + public function getTables($dbName = '') + { + $this->initConnect(true); + $sql = "SELECT TABLE_NAME + FROM INFORMATION_SCHEMA.TABLES + WHERE TABLE_TYPE = 'BASE TABLE' + "; + // 璋冭瘯寮濮 + $this->debug(true); + $pdo = $this->linkID->query($sql); + // 璋冭瘯缁撴潫 + $this->debug(false, $sql); + $result = $pdo->fetchAll(PDO::FETCH_ASSOC); + $info = []; + foreach ($result as $key => $val) { + $info[$key] = current($val); + } + return $info; + } + + /** + * SQL鎬ц兘鍒嗘瀽 + * @access protected + * @param string $sql + * @return array + */ + protected function getExplain($sql) + { + return []; + } +} diff --git a/thinkphp/library/think/db/connector/pgsql.sql b/thinkphp/library/think/db/connector/pgsql.sql new file mode 100644 index 000000000..e1a09a30c --- /dev/null +++ b/thinkphp/library/think/db/connector/pgsql.sql @@ -0,0 +1,117 @@ +CREATE OR REPLACE FUNCTION pgsql_type(a_type varchar) RETURNS varchar AS +$BODY$ +DECLARE + v_type varchar; +BEGIN + IF a_type='int8' THEN + v_type:='bigint'; + ELSIF a_type='int4' THEN + v_type:='integer'; + ELSIF a_type='int2' THEN + v_type:='smallint'; + ELSIF a_type='bpchar' THEN + v_type:='char'; + ELSE + v_type:=a_type; + END IF; + RETURN v_type; +END; +$BODY$ +LANGUAGE PLPGSQL; + +CREATE TYPE "public"."tablestruct" AS ( + "fields_key_name" varchar(100), + "fields_name" VARCHAR(200), + "fields_type" VARCHAR(20), + "fields_length" BIGINT, + "fields_not_null" VARCHAR(10), + "fields_default" VARCHAR(500), + "fields_comment" VARCHAR(1000) +); + +CREATE OR REPLACE FUNCTION "public"."table_msg" (a_schema_name varchar, a_table_name varchar) RETURNS SETOF "public"."tablestruct" AS +$body$ +DECLARE + v_ret tablestruct; + v_oid oid; + v_sql varchar; + v_rec RECORD; + v_key varchar; +BEGIN + SELECT + pg_class.oid INTO v_oid + FROM + pg_class + INNER JOIN pg_namespace ON (pg_class.relnamespace = pg_namespace.oid AND lower(pg_namespace.nspname) = a_schema_name) + WHERE + pg_class.relname=a_table_name; + IF NOT FOUND THEN + RETURN; + END IF; + + v_sql=' + SELECT + pg_attribute.attname AS fields_name, + pg_attribute.attnum AS fields_index, + pgsql_type(pg_type.typname::varchar) AS fields_type, + pg_attribute.atttypmod-4 as fields_length, + CASE WHEN pg_attribute.attnotnull THEN ''not null'' + ELSE '''' + END AS fields_not_null, + pg_attrdef.adsrc AS fields_default, + pg_description.description AS fields_comment + FROM + pg_attribute + INNER JOIN pg_class ON pg_attribute.attrelid = pg_class.oid + INNER JOIN pg_type ON pg_attribute.atttypid = pg_type.oid + LEFT OUTER JOIN pg_attrdef ON pg_attrdef.adrelid = pg_class.oid AND pg_attrdef.adnum = pg_attribute.attnum + LEFT OUTER JOIN pg_description ON pg_description.objoid = pg_class.oid AND pg_description.objsubid = pg_attribute.attnum + WHERE + pg_attribute.attnum > 0 + AND attisdropped <> ''t'' + AND pg_class.oid = ' || v_oid || ' + ORDER BY pg_attribute.attnum' ; + + FOR v_rec IN EXECUTE v_sql LOOP + v_ret.fields_name=v_rec.fields_name; + v_ret.fields_type=v_rec.fields_type; + IF v_rec.fields_length > 0 THEN + v_ret.fields_length:=v_rec.fields_length; + ELSE + v_ret.fields_length:=NULL; + END IF; + v_ret.fields_not_null=v_rec.fields_not_null; + v_ret.fields_default=v_rec.fields_default; + v_ret.fields_comment=v_rec.fields_comment; + SELECT constraint_name INTO v_key FROM information_schema.key_column_usage WHERE table_schema=a_schema_name AND table_name=a_table_name AND column_name=v_rec.fields_name; + IF FOUND THEN + v_ret.fields_key_name=v_key; + ELSE + v_ret.fields_key_name=''; + END IF; + RETURN NEXT v_ret; + END LOOP; + RETURN ; +END; +$body$ +LANGUAGE 'plpgsql' VOLATILE CALLED ON NULL INPUT SECURITY INVOKER; + +COMMENT ON FUNCTION "public"."table_msg"(a_schema_name varchar, a_table_name varchar) +IS '鑾峰緱琛ㄤ俊鎭'; + +---閲嶈浇涓涓嚱鏁 +CREATE OR REPLACE FUNCTION "public"."table_msg" (a_table_name varchar) RETURNS SETOF "public"."tablestruct" AS +$body$ +DECLARE + v_ret tablestruct; +BEGIN + FOR v_ret IN SELECT * FROM table_msg('public',a_table_name) LOOP + RETURN NEXT v_ret; + END LOOP; + RETURN; +END; +$body$ +LANGUAGE 'plpgsql' VOLATILE CALLED ON NULL INPUT SECURITY INVOKER; + +COMMENT ON FUNCTION "public"."table_msg"(a_table_name varchar) +IS '鑾峰緱琛ㄤ俊鎭'; \ No newline at end of file diff --git a/thinkphp/library/think/db/exception/BindParamException.php b/thinkphp/library/think/db/exception/BindParamException.php new file mode 100644 index 000000000..d0e2387bc --- /dev/null +++ b/thinkphp/library/think/db/exception/BindParamException.php @@ -0,0 +1,35 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\exception; + +use think\exception\DbException; + +/** + * PDO鍙傛暟缁戝畾寮傚父 + */ +class BindParamException extends DbException +{ + + /** + * BindParamException constructor. + * @param string $message + * @param array $config + * @param string $sql + * @param array $bind + * @param int $code + */ + public function __construct($message, $config, $sql, $bind, $code = 10502) + { + $this->setData('Bind Param', $bind); + parent::__construct($message, $config, $sql, $code); + } +} diff --git a/thinkphp/library/think/db/exception/DataNotFoundException.php b/thinkphp/library/think/db/exception/DataNotFoundException.php new file mode 100644 index 000000000..e399b0637 --- /dev/null +++ b/thinkphp/library/think/db/exception/DataNotFoundException.php @@ -0,0 +1,43 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\exception; + +use think\exception\DbException; + +class DataNotFoundException extends DbException +{ + protected $table; + + /** + * DbException constructor. + * @param string $message + * @param string $table + * @param array $config + */ + public function __construct($message, $table = '', array $config = []) + { + $this->message = $message; + $this->table = $table; + + $this->setData('Database Config', $config); + } + + /** + * 鑾峰彇鏁版嵁琛ㄥ悕 + * @access public + * @return string + */ + public function getTable() + { + return $this->table; + } +} diff --git a/thinkphp/library/think/db/exception/ModelNotFoundException.php b/thinkphp/library/think/db/exception/ModelNotFoundException.php new file mode 100644 index 000000000..2180ab072 --- /dev/null +++ b/thinkphp/library/think/db/exception/ModelNotFoundException.php @@ -0,0 +1,43 @@ + +// +---------------------------------------------------------------------- + +namespace think\db\exception; + +use think\exception\DbException; + +class ModelNotFoundException extends DbException +{ + protected $model; + + /** + * 鏋勯犳柟娉 + * @param string $message + * @param string $model + */ + public function __construct($message, $model = '', array $config = []) + { + $this->message = $message; + $this->model = $model; + + $this->setData('Database Config', $config); + } + + /** + * 鑾峰彇妯″瀷绫诲悕 + * @access public + * @return string + */ + public function getModel() + { + return $this->model; + } + +} diff --git a/thinkphp/library/think/debug/Console.php b/thinkphp/library/think/debug/Console.php new file mode 100644 index 000000000..8a232c8c1 --- /dev/null +++ b/thinkphp/library/think/debug/Console.php @@ -0,0 +1,160 @@ + +// +---------------------------------------------------------------------- + +namespace think\debug; + +use think\Cache; +use think\Config; +use think\Db; +use think\Debug; +use think\Request; +use think\Response; + +/** + * 娴忚鍣ㄨ皟璇曡緭鍑 + */ +class Console +{ + protected $config = [ + 'trace_tabs' => ['base' => '鍩烘湰', 'file' => '鏂囦欢', 'info' => '娴佺▼', 'notice|error' => '閿欒', 'sql' => 'SQL', 'debug|log' => '璋冭瘯'], + ]; + + // 瀹炰緥鍖栧苟浼犲叆鍙傛暟 + public function __construct($config = []) + { + if (is_array($config)) { + $this->config = array_merge($this->config, $config); + } + } + + /** + * 璋冭瘯杈撳嚭鎺ュ彛 + * @access public + * @param Response $response Response瀵硅薄 + * @param array $log 鏃ュ織淇℃伅 + * @return bool + */ + public function output(Response $response, array $log = []) + { + $request = Request::instance(); + $contentType = $response->getHeader('Content-Type'); + $accept = $request->header('accept'); + if (strpos($accept, 'application/json') === 0 || $request->isAjax()) { + return false; + } elseif (!empty($contentType) && strpos($contentType, 'html') === false) { + return false; + } + // 鑾峰彇鍩烘湰淇℃伅 + $runtime = number_format(microtime(true) - THINK_START_TIME, 10); + $reqs = $runtime > 0 ? number_format(1 / $runtime, 2) : '鈭'; + $mem = number_format((memory_get_usage() - THINK_START_MEM) / 1024, 2); + + if (isset($_SERVER['HTTP_HOST'])) { + $uri = $_SERVER['SERVER_PROTOCOL'] . ' ' . $_SERVER['REQUEST_METHOD'] . ' : ' . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']; + } else { + $uri = 'cmd:' . implode(' ', $_SERVER['argv']); + } + + // 椤甸潰Trace淇℃伅 + $base = [ + '璇锋眰淇℃伅' => date('Y-m-d H:i:s', $_SERVER['REQUEST_TIME']) . ' ' . $uri, + '杩愯鏃堕棿' => number_format($runtime, 6) . 's [ 鍚炲悙鐜囷細' . $reqs . 'req/s ] 鍐呭瓨娑堣楋細' . $mem . 'kb 鏂囦欢鍔犺浇锛' . count(get_included_files()), + '鏌ヨ淇℃伅' => Db::$queryTimes . ' queries ' . Db::$executeTimes . ' writes ', + '缂撳瓨淇℃伅' => Cache::$readTimes . ' reads,' . Cache::$writeTimes . ' writes', + '閰嶇疆鍔犺浇' => count(Config::get()), + ]; + + if (session_id()) { + $base['浼氳瘽淇℃伅'] = 'SESSION_ID=' . session_id(); + } + + $info = Debug::getFile(true); + + // 椤甸潰Trace淇℃伅 + $trace = []; + foreach ($this->config['trace_tabs'] as $name => $title) { + $name = strtolower($name); + switch ($name) { + case 'base': // 鍩烘湰淇℃伅 + $trace[$title] = $base; + break; + case 'file': // 鏂囦欢淇℃伅 + $trace[$title] = $info; + break; + default: // 璋冭瘯淇℃伅 + if (strpos($name, '|')) { + // 澶氱粍淇℃伅 + $names = explode('|', $name); + $result = []; + foreach ($names as $name) { + $result = array_merge($result, isset($log[$name]) ? $log[$name] : []); + } + $trace[$title] = $result; + } else { + $trace[$title] = isset($log[$name]) ? $log[$name] : ''; + } + } + } + + //杈撳嚭鍒版帶鍒跺彴 + $lines = ''; + foreach ($trace as $type => $msg) { + $lines .= $this->console($type, $msg); + } + $js = << +{$lines} + +JS; + return $js; + } + + protected function console($type, $msg) + { + $type = strtolower($type); + $trace_tabs = array_values($this->config['trace_tabs']); + $line[] = ($type == $trace_tabs[0] || '璋冭瘯' == $type || '閿欒' == $type) + ? "console.group('{$type}');" + : "console.groupCollapsed('{$type}');"; + + foreach ((array) $msg as $key => $m) { + switch ($type) { + case '璋冭瘯': + $var_type = gettype($m); + if (in_array($var_type, ['array', 'string'])) { + $line[] = "console.log(" . json_encode($m) . ");"; + } else { + $line[] = "console.log(" . json_encode(var_export($m, 1)) . ");"; + } + break; + case '閿欒': + $msg = str_replace("\n", '\n', $m); + $style = 'color:#F4006B;font-size:14px;'; + $line[] = "console.error(\"%c{$msg}\", \"{$style}\");"; + break; + case 'sql': + $msg = str_replace("\n", '\n', $m); + $style = "color:#009bb4;"; + $line[] = "console.log(\"%c{$msg}\", \"{$style}\");"; + break; + default: + $m = is_string($key) ? $key . ' ' . $m : $key + 1 . ' ' . $m; + $msg = json_encode($m); + $line[] = "console.log({$msg});"; + break; + } + } + $line[] = "console.groupEnd();"; + return implode(PHP_EOL, $line); + } + +} diff --git a/thinkphp/library/think/debug/Html.php b/thinkphp/library/think/debug/Html.php new file mode 100644 index 000000000..f8651aa92 --- /dev/null +++ b/thinkphp/library/think/debug/Html.php @@ -0,0 +1,111 @@ + +// +---------------------------------------------------------------------- + +namespace think\debug; + +use think\Cache; +use think\Config; +use think\Db; +use think\Debug; +use think\Request; +use think\Response; + +/** + * 椤甸潰Trace璋冭瘯 + */ +class Html +{ + protected $config = [ + 'trace_file' => '', + 'trace_tabs' => ['base' => '鍩烘湰', 'file' => '鏂囦欢', 'info' => '娴佺▼', 'notice|error' => '閿欒', 'sql' => 'SQL', 'debug|log' => '璋冭瘯'], + ]; + + // 瀹炰緥鍖栧苟浼犲叆鍙傛暟 + public function __construct(array $config = []) + { + $this->config['trace_file'] = THINK_PATH . 'tpl/page_trace.tpl'; + $this->config = array_merge($this->config, $config); + } + + /** + * 璋冭瘯杈撳嚭鎺ュ彛 + * @access public + * @param Response $response Response瀵硅薄 + * @param array $log 鏃ュ織淇℃伅 + * @return bool + */ + public function output(Response $response, array $log = []) + { + $request = Request::instance(); + $contentType = $response->getHeader('Content-Type'); + $accept = $request->header('accept'); + if (strpos($accept, 'application/json') === 0 || $request->isAjax()) { + return false; + } elseif (!empty($contentType) && strpos($contentType, 'html') === false) { + return false; + } + // 鑾峰彇鍩烘湰淇℃伅 + $runtime = number_format(microtime(true) - THINK_START_TIME, 10); + $reqs = $runtime > 0 ? number_format(1 / $runtime, 2) : '鈭'; + $mem = number_format((memory_get_usage() - THINK_START_MEM) / 1024, 2); + + // 椤甸潰Trace淇℃伅 + if (isset($_SERVER['HTTP_HOST'])) { + $uri = $_SERVER['SERVER_PROTOCOL'] . ' ' . $_SERVER['REQUEST_METHOD'] . ' : ' . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']; + } else { + $uri = 'cmd:' . implode(' ', $_SERVER['argv']); + } + $base = [ + '璇锋眰淇℃伅' => date('Y-m-d H:i:s', $_SERVER['REQUEST_TIME']) . ' ' . $uri, + '杩愯鏃堕棿' => number_format($runtime, 6) . 's [ 鍚炲悙鐜囷細' . $reqs . 'req/s ] 鍐呭瓨娑堣楋細' . $mem . 'kb 鏂囦欢鍔犺浇锛' . count(get_included_files()), + '鏌ヨ淇℃伅' => Db::$queryTimes . ' queries ' . Db::$executeTimes . ' writes ', + '缂撳瓨淇℃伅' => Cache::$readTimes . ' reads,' . Cache::$writeTimes . ' writes', + '閰嶇疆鍔犺浇' => count(Config::get()), + ]; + + if (session_id()) { + $base['浼氳瘽淇℃伅'] = 'SESSION_ID=' . session_id(); + } + + $info = Debug::getFile(true); + + // 椤甸潰Trace淇℃伅 + $trace = []; + foreach ($this->config['trace_tabs'] as $name => $title) { + $name = strtolower($name); + switch ($name) { + case 'base': // 鍩烘湰淇℃伅 + $trace[$title] = $base; + break; + case 'file': // 鏂囦欢淇℃伅 + $trace[$title] = $info; + break; + default: // 璋冭瘯淇℃伅 + if (strpos($name, '|')) { + // 澶氱粍淇℃伅 + $names = explode('|', $name); + $result = []; + foreach ($names as $name) { + $result = array_merge($result, isset($log[$name]) ? $log[$name] : []); + } + $trace[$title] = $result; + } else { + $trace[$title] = isset($log[$name]) ? $log[$name] : ''; + } + } + } + // 璋冪敤Trace椤甸潰妯℃澘 + ob_start(); + include $this->config['trace_file']; + return ob_get_clean(); + } + +} diff --git a/thinkphp/library/think/exception/ClassNotFoundException.php b/thinkphp/library/think/exception/ClassNotFoundException.php new file mode 100644 index 000000000..eb22e7301 --- /dev/null +++ b/thinkphp/library/think/exception/ClassNotFoundException.php @@ -0,0 +1,32 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +class ClassNotFoundException extends \RuntimeException +{ + protected $class; + public function __construct($message, $class = '') + { + $this->message = $message; + $this->class = $class; + } + + /** + * 鑾峰彇绫诲悕 + * @access public + * @return string + */ + public function getClass() + { + return $this->class; + } +} diff --git a/thinkphp/library/think/exception/DbException.php b/thinkphp/library/think/exception/DbException.php new file mode 100644 index 000000000..656d69132 --- /dev/null +++ b/thinkphp/library/think/exception/DbException.php @@ -0,0 +1,42 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +use think\Exception; + +/** + * Database鐩稿叧寮傚父澶勭悊绫 + */ +class DbException extends Exception +{ + /** + * DbException constructor. + * @param string $message + * @param array $config + * @param string $sql + * @param int $code + */ + public function __construct($message, array $config, $sql, $code = 10500) + { + $this->message = $message; + $this->code = $code; + + $this->setData('Database Status', [ + 'Error Code' => $code, + 'Error Message' => $message, + 'Error SQL' => $sql, + ]); + + $this->setData('Database Config', $config); + } + +} diff --git a/thinkphp/library/think/exception/ErrorException.php b/thinkphp/library/think/exception/ErrorException.php new file mode 100644 index 000000000..e3f18375d --- /dev/null +++ b/thinkphp/library/think/exception/ErrorException.php @@ -0,0 +1,57 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +use think\Exception; + +/** + * ThinkPHP閿欒寮傚父 + * 涓昏鐢ㄤ簬灏佽 set_error_handler 鍜 register_shutdown_function 寰楀埌鐨勯敊璇 + * 闄ゅ紑浠 think\Exception 缁ф壙鐨勫姛鑳 + * 鍏朵粬鍜孭HP绯荤粺\ErrorException鍔熻兘鍩烘湰涓鏍 + */ +class ErrorException extends Exception +{ + /** + * 鐢ㄤ簬淇濆瓨閿欒绾у埆 + * @var integer + */ + protected $severity; + + /** + * 閿欒寮傚父鏋勯犲嚱鏁 + * @param integer $severity 閿欒绾у埆 + * @param string $message 閿欒璇︾粏淇℃伅 + * @param string $file 鍑洪敊鏂囦欢璺緞 + * @param integer $line 鍑洪敊琛屽彿 + * @param array $context 閿欒涓婁笅鏂囷紝浼氬寘鍚敊璇Е鍙戝浣滅敤鍩熷唴鎵鏈夊彉閲忕殑鏁扮粍 + */ + public function __construct($severity, $message, $file, $line, array $context = []) + { + $this->severity = $severity; + $this->message = $message; + $this->file = $file; + $this->line = $line; + $this->code = 0; + + empty($context) || $this->setData('Error Context', $context); + } + + /** + * 鑾峰彇閿欒绾у埆 + * @return integer 閿欒绾у埆 + */ + final public function getSeverity() + { + return $this->severity; + } +} diff --git a/thinkphp/library/think/exception/Handle.php b/thinkphp/library/think/exception/Handle.php new file mode 100644 index 000000000..8b6e6c40d --- /dev/null +++ b/thinkphp/library/think/exception/Handle.php @@ -0,0 +1,266 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +use Exception; +use think\App; +use think\Config; +use think\console\Output; +use think\Lang; +use think\Log; +use think\Response; + +class Handle +{ + + protected $ignoreReport = [ + '\\think\\exception\\HttpException', + ]; + + /** + * Report or log an exception. + * + * @param \Exception $exception + * @return void + */ + public function report(Exception $exception) + { + if (!$this->isIgnoreReport($exception)) { + // 鏀堕泦寮傚父鏁版嵁 + if (App::$debug) { + $data = [ + 'file' => $exception->getFile(), + 'line' => $exception->getLine(), + 'message' => $this->getMessage($exception), + 'code' => $this->getCode($exception), + ]; + $log = "[{$data['code']}]{$data['message']}[{$data['file']}:{$data['line']}]"; + } else { + $data = [ + 'code' => $this->getCode($exception), + 'message' => $this->getMessage($exception), + ]; + $log = "[{$data['code']}]{$data['message']}"; + } + + Log::record($log, 'error'); + } + } + + protected function isIgnoreReport(Exception $exception) + { + foreach ($this->ignoreReport as $class) { + if ($exception instanceof $class) { + return true; + } + } + return false; + } + + /** + * Render an exception into an HTTP response. + * + * @param \Exception $e + * @return Response + */ + public function render(Exception $e) + { + if ($e instanceof HttpException) { + return $this->renderHttpException($e); + } else { + return $this->convertExceptionToResponse($e); + } + } + + /** + * @param Output $output + * @param Exception $e + */ + public function renderForConsole(Output $output, Exception $e) + { + if (App::$debug) { + $output->setVerbosity(Output::VERBOSITY_DEBUG); + } + $output->renderException($e); + } + + /** + * @param HttpException $e + * @return Response + */ + protected function renderHttpException(HttpException $e) + { + $status = $e->getStatusCode(); + $template = Config::get('http_exception_template'); + if (!App::$debug && !empty($template[$status])) { + return Response::create($template[$status], 'view', $status)->assign(['e' => $e]); + } else { + return $this->convertExceptionToResponse($e); + } + } + + /** + * @param Exception $exception + * @return Response + */ + protected function convertExceptionToResponse(Exception $exception) + { + // 鏀堕泦寮傚父鏁版嵁 + if (App::$debug) { + // 璋冭瘯妯″紡锛岃幏鍙栬缁嗙殑閿欒淇℃伅 + $data = [ + 'name' => get_class($exception), + 'file' => $exception->getFile(), + 'line' => $exception->getLine(), + 'message' => $this->getMessage($exception), + 'trace' => $exception->getTrace(), + 'code' => $this->getCode($exception), + 'source' => $this->getSourceCode($exception), + 'datas' => $this->getExtendData($exception), + 'tables' => [ + 'GET Data' => $_GET, + 'POST Data' => $_POST, + 'Files' => $_FILES, + 'Cookies' => $_COOKIE, + 'Session' => isset($_SESSION) ? $_SESSION : [], + 'Server/Request Data' => $_SERVER, + 'Environment Variables' => $_ENV, + 'ThinkPHP Constants' => $this->getConst(), + ], + ]; + } else { + // 閮ㄧ讲妯″紡浠呮樉绀 Code 鍜 Message + $data = [ + 'code' => $this->getCode($exception), + 'message' => $this->getMessage($exception), + ]; + + if (!Config::get('show_error_msg')) { + // 涓嶆樉绀鸿缁嗛敊璇俊鎭 + $data['message'] = Config::get('error_message'); + } + } + + //淇濈暀涓灞 + while (ob_get_level() > 1) { + ob_end_clean(); + } + + $data['echo'] = ob_get_clean(); + + ob_start(); + extract($data); + include Config::get('exception_tmpl'); + // 鑾峰彇骞舵竻绌虹紦瀛 + $content = ob_get_clean(); + $response = new Response($content, 'html'); + + if ($exception instanceof HttpException) { + $statusCode = $exception->getStatusCode(); + $response->header($exception->getHeaders()); + } + + if (!isset($statusCode)) { + $statusCode = 500; + } + $response->code($statusCode); + return $response; + } + + /** + * 鑾峰彇閿欒缂栫爜 + * ErrorException鍒欎娇鐢ㄩ敊璇骇鍒綔涓洪敊璇紪鐮 + * @param \Exception $exception + * @return integer 閿欒缂栫爜 + */ + protected function getCode(Exception $exception) + { + $code = $exception->getCode(); + if (!$code && $exception instanceof ErrorException) { + $code = $exception->getSeverity(); + } + return $code; + } + + /** + * 鑾峰彇閿欒淇℃伅 + * ErrorException鍒欎娇鐢ㄩ敊璇骇鍒綔涓洪敊璇紪鐮 + * @param \Exception $exception + * @return string 閿欒淇℃伅 + */ + protected function getMessage(Exception $exception) + { + $message = $exception->getMessage(); + if (IS_CLI) { + return $message; + } + + if (strpos($message, ':')) { + $name = strstr($message, ':', true); + $message = Lang::has($name) ? Lang::get($name) . strstr($message, ':') : $message; + } elseif (strpos($message, ',')) { + $name = strstr($message, ',', true); + $message = Lang::has($name) ? Lang::get($name) . ':' . substr(strstr($message, ','), 1) : $message; + } elseif (Lang::has($message)) { + $message = Lang::get($message); + } + return $message; + } + + /** + * 鑾峰彇鍑洪敊鏂囦欢鍐呭 + * 鑾峰彇閿欒鐨勫墠9琛屽拰鍚9琛 + * @param \Exception $exception + * @return array 閿欒鏂囦欢鍐呭 + */ + protected function getSourceCode(Exception $exception) + { + // 璇诲彇鍓9琛屽拰鍚9琛 + $line = $exception->getLine(); + $first = ($line - 9 > 0) ? $line - 9 : 1; + + try { + $contents = file($exception->getFile()); + $source = [ + 'first' => $first, + 'source' => array_slice($contents, $first - 1, 19), + ]; + } catch (Exception $e) { + $source = []; + } + return $source; + } + + /** + * 鑾峰彇寮傚父鎵╁睍淇℃伅 + * 鐢ㄤ簬闈炶皟璇曟ā寮廻tml杩斿洖绫诲瀷鏄剧ず + * @param \Exception $exception + * @return array 寮傚父绫诲畾涔夌殑鎵╁睍鏁版嵁 + */ + protected function getExtendData(Exception $exception) + { + $data = []; + if ($exception instanceof \think\Exception) { + $data = $exception->getData(); + } + return $data; + } + + /** + * 鑾峰彇甯搁噺鍒楄〃 + * @return array 甯搁噺鍒楄〃 + */ + private static function getConst() + { + return get_defined_constants(true)['user']; + } +} diff --git a/thinkphp/library/think/exception/HttpException.php b/thinkphp/library/think/exception/HttpException.php new file mode 100644 index 000000000..01a27fc23 --- /dev/null +++ b/thinkphp/library/think/exception/HttpException.php @@ -0,0 +1,36 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +class HttpException extends \RuntimeException +{ + private $statusCode; + private $headers; + + public function __construct($statusCode, $message = null, \Exception $previous = null, array $headers = [], $code = 0) + { + $this->statusCode = $statusCode; + $this->headers = $headers; + + parent::__construct($message, $code, $previous); + } + + public function getStatusCode() + { + return $this->statusCode; + } + + public function getHeaders() + { + return $this->headers; + } +} diff --git a/thinkphp/library/think/exception/HttpResponseException.php b/thinkphp/library/think/exception/HttpResponseException.php new file mode 100644 index 000000000..52972867b --- /dev/null +++ b/thinkphp/library/think/exception/HttpResponseException.php @@ -0,0 +1,33 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +use think\Response; + +class HttpResponseException extends \RuntimeException +{ + /** + * @var Response + */ + protected $response; + + public function __construct(Response $response) + { + $this->response = $response; + } + + public function getResponse() + { + return $this->response; + } + +} diff --git a/thinkphp/library/think/exception/PDOException.php b/thinkphp/library/think/exception/PDOException.php new file mode 100644 index 000000000..ebd53dff9 --- /dev/null +++ b/thinkphp/library/think/exception/PDOException.php @@ -0,0 +1,39 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +/** + * PDO寮傚父澶勭悊绫 + * 閲嶆柊灏佽浜嗙郴缁熺殑\PDOException绫 + */ +class PDOException extends DbException +{ + /** + * PDOException constructor. + * @param \PDOException $exception + * @param array $config + * @param string $sql + * @param int $code + */ + public function __construct(\PDOException $exception, array $config, $sql, $code = 10501) + { + $error = $exception->errorInfo; + + $this->setData('PDO Error Info', [ + 'SQLSTATE' => $error[0], + 'Driver Error Code' => isset($error[1]) ? $error[1] : 0, + 'Driver Error Message' => isset($error[2]) ? $error[2] : '', + ]); + + parent::__construct($exception->getMessage(), $config, $sql, $code); + } +} diff --git a/thinkphp/library/think/exception/RouteNotFoundException.php b/thinkphp/library/think/exception/RouteNotFoundException.php new file mode 100644 index 000000000..f85a3c4bb --- /dev/null +++ b/thinkphp/library/think/exception/RouteNotFoundException.php @@ -0,0 +1,22 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +class RouteNotFoundException extends HttpException +{ + + public function __construct() + { + parent::__construct(404); + } + +} diff --git a/thinkphp/library/think/exception/TemplateNotFoundException.php b/thinkphp/library/think/exception/TemplateNotFoundException.php new file mode 100644 index 000000000..420206931 --- /dev/null +++ b/thinkphp/library/think/exception/TemplateNotFoundException.php @@ -0,0 +1,33 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +class TemplateNotFoundException extends \RuntimeException +{ + protected $template; + + public function __construct($message, $template = '') + { + $this->message = $message; + $this->template = $template; + } + + /** + * 鑾峰彇妯℃澘鏂囦欢 + * @access public + * @return string + */ + public function getTemplate() + { + return $this->template; + } +} diff --git a/thinkphp/library/think/exception/ThrowableError.php b/thinkphp/library/think/exception/ThrowableError.php new file mode 100644 index 000000000..87b6b9d74 --- /dev/null +++ b/thinkphp/library/think/exception/ThrowableError.php @@ -0,0 +1,47 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +class ThrowableError extends \ErrorException +{ + public function __construct(\Throwable $e) + { + + if ($e instanceof \ParseError) { + $message = 'Parse error: ' . $e->getMessage(); + $severity = E_PARSE; + } elseif ($e instanceof \TypeError) { + $message = 'Type error: ' . $e->getMessage(); + $severity = E_RECOVERABLE_ERROR; + } else { + $message = 'Fatal error: ' . $e->getMessage(); + $severity = E_ERROR; + } + + parent::__construct( + $message, + $e->getCode(), + $severity, + $e->getFile(), + $e->getLine() + ); + + $this->setTrace($e->getTrace()); + } + + protected function setTrace($trace) + { + $traceReflector = new \ReflectionProperty('Exception', 'trace'); + $traceReflector->setAccessible(true); + $traceReflector->setValue($this, $trace); + } +} diff --git a/thinkphp/library/think/exception/ValidateException.php b/thinkphp/library/think/exception/ValidateException.php new file mode 100644 index 000000000..b3684169c --- /dev/null +++ b/thinkphp/library/think/exception/ValidateException.php @@ -0,0 +1,33 @@ + +// +---------------------------------------------------------------------- + +namespace think\exception; + +class ValidateException extends \RuntimeException +{ + protected $error; + + public function __construct($error) + { + $this->error = $error; + $this->message = is_array($error) ? implode("\n\r", $error) : $error; + } + + /** + * 鑾峰彇楠岃瘉閿欒淇℃伅 + * @access public + * @return array|string + */ + public function getError() + { + return $this->error; + } +} diff --git a/thinkphp/library/think/log/driver/File.php b/thinkphp/library/think/log/driver/File.php new file mode 100644 index 000000000..b639fd043 --- /dev/null +++ b/thinkphp/library/think/log/driver/File.php @@ -0,0 +1,101 @@ + +// +---------------------------------------------------------------------- + +namespace think\log\driver; + +use think\App; + +/** + * 鏈湴鍖栬皟璇曡緭鍑哄埌鏂囦欢 + */ +class File +{ + protected $config = [ + 'time_format' => ' c ', + 'file_size' => 2097152, + 'path' => LOG_PATH, + 'apart_level' => [], + ]; + + // 瀹炰緥鍖栧苟浼犲叆鍙傛暟 + public function __construct($config = []) + { + if (is_array($config)) { + $this->config = array_merge($this->config, $config); + } + } + + /** + * 鏃ュ織鍐欏叆鎺ュ彛 + * @access public + * @param array $log 鏃ュ織淇℃伅 + * @param bool $depr 鏄惁鍐欏叆鍒嗗壊绾 + * @return bool + */ + public function save(array $log = [], $depr = true) + { + $now = date($this->config['time_format']); + $destination = $this->config['path'] . date('Ym') . DS . date('d') . '.log'; + + $path = dirname($destination); + !is_dir($path) && mkdir($path, 0755, true); + + //妫娴嬫棩蹇楁枃浠跺ぇ灏忥紝瓒呰繃閰嶇疆澶у皬鍒欏浠芥棩蹇楁枃浠堕噸鏂扮敓鎴 + if (is_file($destination) && floor($this->config['file_size']) <= filesize($destination)) { + rename($destination, dirname($destination) . DS . $_SERVER['REQUEST_TIME'] . '-' . basename($destination)); + } + + $depr = $depr ? "---------------------------------------------------------------\r\n" : ''; + $info = ''; + if (App::$debug) { + // 鑾峰彇鍩烘湰淇℃伅 + if (isset($_SERVER['HTTP_HOST'])) { + $current_uri = $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']; + } else { + $current_uri = "cmd:" . implode(' ', $_SERVER['argv']); + } + + $runtime = round(microtime(true) - THINK_START_TIME, 10); + $reqs = $runtime > 0 ? number_format(1 / $runtime, 2) : '鈭'; + $time_str = ' [杩愯鏃堕棿锛' . number_format($runtime, 6) . 's][鍚炲悙鐜囷細' . $reqs . 'req/s]'; + $memory_use = number_format((memory_get_usage() - THINK_START_MEM) / 1024, 2); + $memory_str = ' [鍐呭瓨娑堣楋細' . $memory_use . 'kb]'; + $file_load = ' [鏂囦欢鍔犺浇锛' . count(get_included_files()) . ']'; + + $info = '[ log ] ' . $current_uri . $time_str . $memory_str . $file_load . "\r\n"; + $server = isset($_SERVER['SERVER_ADDR']) ? $_SERVER['SERVER_ADDR'] : '0.0.0.0'; + $remote = isset($_SERVER['REMOTE_ADDR']) ? $_SERVER['REMOTE_ADDR'] : '0.0.0.0'; + $method = isset($_SERVER['REQUEST_METHOD']) ? $_SERVER['REQUEST_METHOD'] : 'CLI'; + $uri = isset($_SERVER['REQUEST_URI']) ? $_SERVER['REQUEST_URI'] : ''; + } + foreach ($log as $type => $val) { + $level = ''; + foreach ($val as $msg) { + if (!is_string($msg)) { + $msg = var_export($msg, true); + } + $level .= '[ ' . $type . ' ] ' . $msg . "\r\n"; + } + if (in_array($type, $this->config['apart_level'])) { + // 鐙珛璁板綍鐨勬棩蹇楃骇鍒 + $filename = $path . DS . date('d') . '_' . $type . '.log'; + error_log("[{$now}] {$level}\r\n{$depr}", 3, $filename); + } else { + $info .= $level; + } + } + if (App::$debug) { + $info = "{$server} {$remote} {$method} {$uri}\r\n" . $info; + } + return error_log("[{$now}] {$info}\r\n{$depr}", 3, $destination); + } + +} diff --git a/thinkphp/library/think/log/driver/Socket.php b/thinkphp/library/think/log/driver/Socket.php new file mode 100644 index 000000000..08fd8ace9 --- /dev/null +++ b/thinkphp/library/think/log/driver/Socket.php @@ -0,0 +1,250 @@ + +// +---------------------------------------------------------------------- + +namespace think\log\driver; + +use think\App; + +/** + * github: https://github.com/luofei614/SocketLog + * @author luofei614 + */ +class Socket +{ + public $port = 1116; //SocketLog 鏈嶅姟鐨刪ttp鐨勭鍙e彿 + + protected $config = [ + // socket鏈嶅姟鍣ㄥ湴鍧 + 'host' => 'localhost', + // 鏄惁鏄剧ず鍔犺浇鐨勬枃浠跺垪琛 + 'show_included_files' => false, + // 鏃ュ織寮哄埗璁板綍鍒伴厤缃殑client_id + 'force_client_ids' => [], + // 闄愬埗鍏佽璇诲彇鏃ュ織鐨刢lient_id + 'allow_client_ids' => [], + ]; + + protected $css = [ + 'sql' => 'color:#009bb4;', + 'sql_warn' => 'color:#009bb4;font-size:14px;', + 'error' => 'color:#f4006b;font-size:14px;', + 'page' => 'color:#40e2ff;background:#171717;', + 'big' => 'font-size:20px;color:red;', + ]; + + protected $allowForceClientIds = []; //閰嶇疆寮哄埗鎺ㄩ佷笖琚巿鏉冪殑client_id + + /** + * 鏋舵瀯鍑芥暟 + * @param array $config 缂撳瓨鍙傛暟 + * @access public + */ + public function __construct(array $config = []) + { + if (!empty($config)) { + $this->config = array_merge($this->config, $config); + } + } + + /** + * 璋冭瘯杈撳嚭鎺ュ彛 + * @access public + * @param array $log 鏃ュ織淇℃伅 + * @return bool + */ + public function save(array $log = []) + { + if (!$this->check()) { + return false; + } + $trace = []; + if (App::$debug) { + $runtime = round(microtime(true) - THINK_START_TIME, 10); + $reqs = $runtime > 0 ? number_format(1 / $runtime, 2) : '鈭'; + $time_str = ' [杩愯鏃堕棿锛' . number_format($runtime, 6) . 's][鍚炲悙鐜囷細' . $reqs . 'req/s]'; + $memory_use = number_format((memory_get_usage() - THINK_START_MEM) / 1024, 2); + $memory_str = ' [鍐呭瓨娑堣楋細' . $memory_use . 'kb]'; + $file_load = ' [鏂囦欢鍔犺浇锛' . count(get_included_files()) . ']'; + + if (isset($_SERVER['HTTP_HOST'])) { + $current_uri = $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']; + } else { + $current_uri = 'cmd:' . implode(' ', $_SERVER['argv']); + } + // 鍩烘湰淇℃伅 + $trace[] = [ + 'type' => 'group', + 'msg' => $current_uri . $time_str . $memory_str . $file_load, + 'css' => $this->css['page'], + ]; + } + + foreach ($log as $type => $val) { + $trace[] = [ + 'type' => 'groupCollapsed', + 'msg' => '[ ' . $type . ' ]', + 'css' => isset($this->css[$type]) ? $this->css[$type] : '', + ]; + foreach ($val as $msg) { + if (!is_string($msg)) { + $msg = var_export($msg, true); + } + $trace[] = [ + 'type' => 'log', + 'msg' => $msg, + 'css' => '', + ]; + } + $trace[] = [ + 'type' => 'groupEnd', + 'msg' => '', + 'css' => '', + ]; + } + + if ($this->config['show_included_files']) { + $trace[] = [ + 'type' => 'groupCollapsed', + 'msg' => '[ file ]', + 'css' => '', + ]; + $trace[] = [ + 'type' => 'log', + 'msg' => implode("\n", get_included_files()), + 'css' => '', + ]; + $trace[] = [ + 'type' => 'groupEnd', + 'msg' => '', + 'css' => '', + ]; + } + + $trace[] = [ + 'type' => 'groupEnd', + 'msg' => '', + 'css' => '', + ]; + + $tabid = $this->getClientArg('tabid'); + if (!$client_id = $this->getClientArg('client_id')) { + $client_id = ''; + } + + if (!empty($this->allowForceClientIds)) { + //寮哄埗鎺ㄩ佸埌澶氫釜client_id + foreach ($this->allowForceClientIds as $force_client_id) { + $client_id = $force_client_id; + $this->sendToClient($tabid, $client_id, $trace, $force_client_id); + } + } else { + $this->sendToClient($tabid, $client_id, $trace, ''); + } + return true; + } + + /** + * 鍙戦佺粰鎸囧畾瀹㈡埛绔 + * @author Zjmainstay + * @param $tabid + * @param $client_id + * @param $logs + * @param $force_client_id + */ + protected function sendToClient($tabid, $client_id, $logs, $force_client_id) + { + $logs = [ + 'tabid' => $tabid, + 'client_id' => $client_id, + 'logs' => $logs, + 'force_client_id' => $force_client_id, + ]; + $msg = @json_encode($logs); + $address = '/' . $client_id; //灏哻lient_id浣滀负鍦板潃锛 server绔氳繃鍦板潃鍒ゆ柇灏嗘棩蹇楀彂甯冪粰璋 + $this->send($this->config['host'], $msg, $address); + } + + protected function check() + { + $tabid = $this->getClientArg('tabid'); + //鏄惁璁板綍鏃ュ織鐨勬鏌 + if (!$tabid && !$this->config['force_client_ids']) { + return false; + } + //鐢ㄦ埛璁よ瘉 + $allow_client_ids = $this->config['allow_client_ids']; + if (!empty($allow_client_ids)) { + //閫氳繃鏁扮粍浜ら泦寰楀嚭鎺堟潈寮哄埗鎺ㄩ佺殑client_id + $this->allowForceClientIds = array_intersect($allow_client_ids, $this->config['force_client_ids']); + if (!$tabid && count($this->allowForceClientIds)) { + return true; + } + + $client_id = $this->getClientArg('client_id'); + if (!in_array($client_id, $allow_client_ids)) { + return false; + } + } else { + $this->allowForceClientIds = $this->config['force_client_ids']; + } + return true; + } + + protected function getClientArg($name) + { + static $args = []; + + $key = 'HTTP_USER_AGENT'; + + if (isset($_SERVER['HTTP_SOCKETLOG'])) { + $key = 'HTTP_SOCKETLOG'; + } + + if (!isset($_SERVER[$key])) { + return; + } + if (empty($args)) { + if (!preg_match('/SocketLog\((.*?)\)/', $_SERVER[$key], $match)) { + $args = ['tabid' => null]; + return; + } + parse_str($match[1], $args); + } + if (isset($args[$name])) { + return $args[$name]; + } + return; + } + + /** + * @param string $host - $host of socket server + * @param string $message - 鍙戦佺殑娑堟伅 + * @param string $address - 鍦板潃 + * @return bool + */ + protected function send($host, $message = '', $address = '/') + { + $url = 'http://' . $host . ':' . $this->port . $address; + $ch = curl_init(); + curl_setopt($ch, CURLOPT_URL, $url); + curl_setopt($ch, CURLOPT_POST, true); + curl_setopt($ch, CURLOPT_POSTFIELDS, $message); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); + curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 1); + curl_setopt($ch, CURLOPT_TIMEOUT, 10); + $headers = [ + "Content-Type: application/json;charset=UTF-8", + ]; + curl_setopt($ch, CURLOPT_HTTPHEADER, $headers); //璁剧疆header + return curl_exec($ch); + } + +} diff --git a/thinkphp/library/think/log/driver/Test.php b/thinkphp/library/think/log/driver/Test.php new file mode 100644 index 000000000..7f663384c --- /dev/null +++ b/thinkphp/library/think/log/driver/Test.php @@ -0,0 +1,30 @@ + +// +---------------------------------------------------------------------- + +namespace think\log\driver; + +/** + * 妯℃嫙娴嬭瘯杈撳嚭 + */ +class Test +{ + /** + * 鏃ュ織鍐欏叆鎺ュ彛 + * @access public + * @param array $log 鏃ュ織淇℃伅 + * @return bool + */ + public function save(array $log = []) + { + return true; + } + +} diff --git a/thinkphp/library/think/model/Collection.php b/thinkphp/library/think/model/Collection.php new file mode 100644 index 000000000..08e11ad82 --- /dev/null +++ b/thinkphp/library/think/model/Collection.php @@ -0,0 +1,79 @@ + +// +---------------------------------------------------------------------- + +namespace think\model; + +use think\Collection as BaseCollection; +use think\Model; + +class Collection extends BaseCollection +{ + /** + * 寤惰繜棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param mixed $relation 鍏宠仈 + * @return $this + */ + public function load($relation) + { + $item = current($this->items); + $item->eagerlyResultSet($this->items, $relation); + return $this; + } + + /** + * 璁剧疆闇瑕侀殣钘忕殑杈撳嚭灞炴 + * @access public + * @param array $hidden 灞炴у垪琛 + * @param bool $override 鏄惁瑕嗙洊 + * @return $this + */ + public function hidden($hidden = [], $override = false) + { + $this->each(function ($model) use ($hidden, $override) { + /** @var Model $model */ + $model->hidden($hidden, $override); + }); + return $this; + } + + /** + * 璁剧疆闇瑕佽緭鍑虹殑灞炴 + * @param array $visible + * @param bool $override 鏄惁瑕嗙洊 + * @return $this + */ + public function visible($visible = [], $override = false) + { + $this->each(function ($model) use ($visible, $override) { + /** @var Model $model */ + $model->visible($visible, $override); + }); + return $this; + } + + /** + * 璁剧疆闇瑕佽拷鍔犵殑杈撳嚭灞炴 + * @access public + * @param array $append 灞炴у垪琛 + * @param bool $override 鏄惁瑕嗙洊 + * @return $this + */ + public function append($append = [], $override = false) + { + $this->each(function ($model) use ($append, $override) { + /** @var Model $model */ + $model->append($append, $override); + }); + return $this; + } + +} diff --git a/thinkphp/library/think/model/Merge.php b/thinkphp/library/think/model/Merge.php new file mode 100644 index 000000000..47d54923d --- /dev/null +++ b/thinkphp/library/think/model/Merge.php @@ -0,0 +1,315 @@ + +// +---------------------------------------------------------------------- + +namespace think\model; + +use think\db\Query; +use think\Model; + +class Merge extends Model +{ + + protected $relationModel = []; // HAS ONE 鍏宠仈鐨勬ā鍨嬪垪琛 + protected $fk = ''; // 澶栭敭鍚 榛樿涓轰富琛ㄥ悕_id + protected $mapFields = []; // 闇瑕佸鐞嗙殑妯″瀷鏄犲皠瀛楁锛岄伩鍏嶆贩娣 array( id => 'user.id' ) + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param array|object $data 鏁版嵁 + */ + public function __construct($data = []) + { + parent::__construct($data); + + // 璁剧疆榛樿澶栭敭鍚 浠呮敮鎸佸崟涓澶栭敭 + if (empty($this->fk)) { + $this->fk = strtolower($this->name) . '_id'; + } + } + + /** + * 鏌ユ壘鍗曟潯璁板綍 + * @access public + * @param mixed $data 涓婚敭鍊兼垨鑰呮煡璇㈡潯浠讹紙闂寘锛 + * @param string|array $with 鍏宠仈棰勬煡璇 + * @param bool $cache 鏄惁缂撳瓨 + * @return \think\Model + */ + public static function get($data = null, $with = [], $cache = false) + { + $query = self::parseQuery($data, $with, $cache); + $query = self::attachQuery($query); + return $query->find($data); + } + + /** + * 闄勫姞鏌ヨ琛ㄨ揪寮 + * @access protected + * @param \think\db\Query $query 鏌ヨ瀵硅薄 + * @return \think\db\Query + */ + protected static function attachQuery($query) + { + $class = new static(); + $master = $class->name; + $fields = self::getModelField($query, $master, '', $class->mapFields, $class->field); + $query->alias($master)->field($fields); + + foreach ($class->relationModel as $key => $model) { + $name = is_int($key) ? $model : $key; + $table = is_int($key) ? $query->getTable($name) : $model; + $query->join($table . ' ' . $name, $name . '.' . $class->fk . '=' . $master . '.' . $class->getPk()); + $fields = self::getModelField($query, $name, $table, $class->mapFields, $class->field); + $query->field($fields); + } + return $query; + } + + /** + * 鑾峰彇鍏宠仈妯″瀷鐨勫瓧娈 骞惰В鍐虫贩娣 + * @access protected + * @param \think\db\Query $query 鏌ヨ瀵硅薄 + * @param string $name 妯″瀷鍚嶇О + * @param string $table 鍏宠仈琛ㄥ悕绉 + * @param array $map 瀛楁鏄犲皠 + * @param array $fields 鏌ヨ瀛楁 + * @return array + */ + protected static function getModelField($query, $name, $table = '', $map = [], $fields = []) + { + // 鑾峰彇妯″瀷鐨勫瓧娈典俊鎭 + $fields = $fields ?: $query->getTableInfo($table, 'fields'); + $array = []; + foreach ($fields as $field) { + if ($key = array_search($name . '.' . $field, $map)) { + // 闇瑕佸鐞嗘槧灏勫瓧娈 + $array[] = $name . '.' . $field . ' AS ' . $key; + } else { + $array[] = $field; + } + } + return $array; + } + + /** + * 鏌ユ壘鎵鏈夎褰 + * @access public + * @param mixed $data 涓婚敭鍒楄〃鎴栬呮煡璇㈡潯浠讹紙闂寘锛 + * @param array|string $with 鍏宠仈棰勬煡璇 + * @param bool $cache + * @return array|false|string + */ + public static function all($data = null, $with = [], $cache = false) + { + $query = self::parseQuery($data, $with, $cache); + $query = self::attachQuery($query); + return $query->select($data); + } + + /** + * 澶勭悊鍐欏叆鐨勬ā鍨嬫暟鎹 + * @access public + * @param string $model 妯″瀷鍚嶇О + * @param array $data 鏁版嵁 + * @param bool $insert 鏄惁鏂板 + * @return array + */ + protected function parseData($model, $data, $insert = false) + { + $item = []; + foreach ($data as $key => $val) { + if ($insert || in_array($key, $this->change) || $this->isPk($key)) { + if ($this->fk != $key && array_key_exists($key, $this->mapFields)) { + list($name, $key) = explode('.', $this->mapFields[$key]); + if ($model == $name) { + $item[$key] = $val; + } + } else { + $item[$key] = $val; + } + } + } + return $item; + } + + /** + * 淇濆瓨妯″瀷鏁版嵁 浠ュ強鍏宠仈鏁版嵁 + * @access public + * @param mixed $data 鏁版嵁 + * @param array $where 鏇存柊鏉′欢 + * @param string $sequence 鑷搴忓垪鍚 + * @return false|int + * @throws \Exception + */ + public function save($data = [], $where = [], $sequence = null) + { + if (!empty($data)) { + // 鏁版嵁鑷姩楠岃瘉 + if (!$this->validateData($data)) { + return false; + } + // 鏁版嵁瀵硅薄璧嬪 + foreach ($data as $key => $value) { + $this->setAttr($key, $value, $data); + } + if (!empty($where)) { + $this->isUpdate = true; + } + } + + // 鏁版嵁鑷姩瀹屾垚 + $this->autoCompleteData($this->auto); + + // 鑷姩鍐欏叆鏇存柊鏃堕棿 + if ($this->autoWriteTimestamp && $this->updateTime && !isset($this->data[$this->updateTime])) { + $this->setAttr($this->updateTime, null); + } + + $db = $this->db(); + $db->startTrans(); + $pk = $this->getPk(); + try { + if ($this->isUpdate) { + // 鑷姩鍐欏叆 + $this->autoCompleteData($this->update); + + if (false === $this->trigger('before_update', $this)) { + return false; + } + + if (empty($where) && !empty($this->updateWhere)) { + $where = $this->updateWhere; + } + + // 澶勭悊妯″瀷鏁版嵁 + $data = $this->parseData($this->name, $this->data); + if (is_string($pk) && isset($data[$pk])) { + if (!isset($where[$pk])) { + unset($where); + $where[$pk] = $data[$pk]; + } + unset($data[$pk]); + } + // 鍐欏叆涓昏〃鏁版嵁 + $result = $db->strict(false)->where($where)->update($data); + + // 鍐欏叆闄勮〃鏁版嵁 + foreach ($this->relationModel as $key => $model) { + $name = is_int($key) ? $model : $key; + $table = is_int($key) ? $db->getTable($model) : $model; + // 澶勭悊鍏宠仈妯″瀷鏁版嵁 + $data = $this->parseData($name, $this->data); + $query = new Query; + if ($query->table($table)->strict(false)->where($this->fk, $this->data[$this->getPk()])->update($data)) { + $result = 1; + } + } + // 娓呯┖change + $this->change = []; + // 鏂板鍥炶皟 + $this->trigger('after_update', $this); + } else { + // 鑷姩鍐欏叆 + $this->autoCompleteData($this->insert); + + // 鑷姩鍐欏叆鍒涘缓鏃堕棿 + if ($this->autoWriteTimestamp && $this->createTime && !isset($this->data[$this->createTime])) { + $this->setAttr($this->createTime, null); + } + + if (false === $this->trigger('before_insert', $this)) { + return false; + } + + // 澶勭悊妯″瀷鏁版嵁 + $data = $this->parseData($this->name, $this->data, true); + // 鍐欏叆涓昏〃鏁版嵁 + $result = $db->name($this->name)->strict(false)->insert($data); + if ($result) { + $insertId = $db->getLastInsID($sequence); + // 鍐欏叆澶栭敭鏁版嵁 + if ($insertId) { + if (is_string($pk)) { + $this->data[$pk] = $insertId; + if ($this->fk == $pk) { + $this->change[] = $pk; + } + } + $this->data[$this->fk] = $insertId; + } + + // 鍐欏叆闄勮〃鏁版嵁 + $source = $this->data; + if ($insertId && is_string($pk) && isset($source[$pk]) && $this->fk != $pk) { + unset($source[$pk]); + } + foreach ($this->relationModel as $key => $model) { + $name = is_int($key) ? $model : $key; + $table = is_int($key) ? $db->getTable($model) : $model; + // 澶勭悊鍏宠仈妯″瀷鏁版嵁 + $data = $this->parseData($name, $source, true); + $query = new Query; + $query->table($table)->strict(false)->insert($data); + } + } + // 鏍囪涓烘洿鏂 + $this->isUpdate = true; + // 娓呯┖change + $this->change = []; + // 鏂板鍥炶皟 + $this->trigger('after_insert', $this); + } + $db->commit(); + return $result; + } catch (\Exception $e) { + $db->rollback(); + throw $e; + } + } + + /** + * 鍒犻櫎褰撳墠鐨勮褰 骞跺垹闄ゅ叧鑱旀暟鎹 + * @access public + * @return int + * @throws \Exception + */ + public function delete() + { + if (false === $this->trigger('before_delete', $this)) { + return false; + } + + $db = $this->db(); + $db->startTrans(); + try { + $result = $db->delete($this->data); + if ($result) { + // 鑾峰彇涓婚敭鏁版嵁 + $pk = $this->data[$this->getPk()]; + + // 鍒犻櫎鍏宠仈鏁版嵁 + foreach ($this->relationModel as $key => $model) { + $table = is_int($key) ? $db->getTable($model) : $model; + $query = new Query; + $query->table($table)->where($this->fk, $pk)->delete(); + } + } + $this->trigger('after_delete', $this); + $db->commit(); + return $result; + } catch (\Exception $e) { + $db->rollback(); + throw $e; + } + } + +} diff --git a/thinkphp/library/think/model/Pivot.php b/thinkphp/library/think/model/Pivot.php new file mode 100644 index 000000000..7823370c4 --- /dev/null +++ b/thinkphp/library/think/model/Pivot.php @@ -0,0 +1,36 @@ + +// +---------------------------------------------------------------------- + +namespace think\model; + +use think\Model; + +class Pivot extends Model +{ + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param array|object $data 鏁版嵁 + * @param string $table 涓棿鏁版嵁琛ㄥ悕 + */ + public function __construct($data = [], $table = '') + { + if (is_object($data)) { + $this->data = get_object_vars($data); + } else { + $this->data = $data; + } + + $this->table = $table; + } + +} diff --git a/thinkphp/library/think/model/Relation.php b/thinkphp/library/think/model/Relation.php new file mode 100644 index 000000000..3d56091b2 --- /dev/null +++ b/thinkphp/library/think/model/Relation.php @@ -0,0 +1,119 @@ + +// +---------------------------------------------------------------------- + +namespace think\model; + +use think\db\Query; +use think\Exception; +use think\Model; + +/** + * Class Relation + * @package think\model + * + * @mixin Query + */ +abstract class Relation +{ + // 鐖舵ā鍨嬪璞 + protected $parent; + /** @var Model 褰撳墠鍏宠仈鐨勬ā鍨嬬被 */ + protected $model; + /** @var Query 鍏宠仈妯″瀷鏌ヨ瀵硅薄 */ + protected $query; + // 鍏宠仈琛ㄥ閿 + protected $foreignKey; + // 鍏宠仈琛ㄤ富閿 + protected $localKey; + // 鍏宠仈鏌ヨ鍙傛暟 + protected $option; + // 鍩虹鏌ヨ + protected $baseQuery; + + /** + * 鑾峰彇鍏宠仈鐨勬墍灞炴ā鍨 + * @access public + * @return Model + */ + public function getParent() + { + return $this->parent; + } + + /** + * 鑾峰彇褰撳墠鐨勫叧鑱旀ā鍨嬬被 + * @access public + * @return string + */ + public function getModel() + { + return $this->model; + } + + /** + * 鑾峰彇鍏宠仈鐨勬煡璇㈠璞 + * @access public + * @return Query + */ + public function getQuery() + { + return $this->query; + } + + /** + * 灏佽鍏宠仈鏁版嵁闆 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @return mixed + */ + protected function resultSetBuild($resultSet) + { + return (new $this->model)->toCollection($resultSet); + } + + /** + * 绉婚櫎鍏宠仈鏌ヨ鍙傛暟 + * @access public + * @return $this + */ + public function removeOption() + { + $this->query->removeOption(); + return $this; + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + abstract protected function baseQuery(); + + public function __call($method, $args) + { + if ($this->query) { + // 鎵ц鍩虹鏌ヨ + $this->baseQuery(); + + $result = call_user_func_array([$this->query, $method], $args); + if ($result instanceof Query) { + $this->option = $result->getOptions(); + return $this; + } else { + $this->option = []; + $this->baseQuery = false; + return $result; + } + } else { + throw new Exception('method not exists:' . __CLASS__ . '->' . $method); + } + } +} diff --git a/thinkphp/library/think/model/relation/BelongsTo.php b/thinkphp/library/think/model/relation/BelongsTo.php new file mode 100644 index 000000000..d1c4b9bde --- /dev/null +++ b/thinkphp/library/think/model/relation/BelongsTo.php @@ -0,0 +1,132 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\Loader; +use think\Model; + +class BelongsTo extends OneToOne +{ + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $model 妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + * @param string $joinType JOIN绫诲瀷 + */ + public function __construct(Model $parent, $model, $foreignKey, $localKey, $joinType = 'INNER') + { + $this->parent = $parent; + $this->model = $model; + $this->foreignKey = $foreignKey; + $this->localKey = $localKey; + $this->joinType = $joinType; + $this->query = (new $model)->db(); + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @access public + * @return array|false|\PDOStatement|string|Model + */ + public function getRelation($subRelation = '', $closure = null) + { + $foreignKey = $this->foreignKey; + if ($closure) { + call_user_func_array($closure, [ & $this->query]); + } + return $this->query->where($this->localKey, $this->parent->$foreignKey)->relation($subRelation)->find(); + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁闆嗭級 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + protected function eagerlySet(&$resultSet, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + $foreignKey = $this->foreignKey; + + $range = []; + foreach ($resultSet as $result) { + // 鑾峰彇鍏宠仈澶栭敭鍒楄〃 + if (isset($result->$foreignKey)) { + $range[] = $result->$foreignKey; + } + } + + if (!empty($range)) { + $data = $this->eagerlyWhere($this, [ + $localKey => [ + 'in', + $range, + ], + ], $localKey, $relation, $subRelation, $closure); + // 鍏宠仈灞炴у悕 + $attr = Loader::parseName($relation); + // 鍏宠仈鏁版嵁灏佽 + foreach ($resultSet as $result) { + // 鍏宠仈妯″瀷 + if (!isset($data[$result->$localKey])) { + $relationModel = null; + } else { + $relationModel = $data[$result->$localKey]; + } + + if ($relationModel && !empty($this->bindAttr)) { + // 缁戝畾鍏宠仈灞炴 + $this->bindAttr($relationModel, $result, $this->bindAttr); + } + // 璁剧疆鍏宠仈灞炴 + $result->setAttr($attr, $relationModel); + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁锛 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + protected function eagerlyOne(&$result, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + $foreignKey = $this->foreignKey; + $data = $this->eagerlyWhere($this, [$localKey => $result->$foreignKey], $localKey, $relation, $subRelation, $closure); + // 鍏宠仈妯″瀷 + if (!isset($data[$result->$localKey])) { + $relationModel = null; + } else { + $relationModel = $data[$result->$localKey]; + } + if ($relationModel && !empty($this->bindAttr)) { + // 缁戝畾鍏宠仈灞炴 + $this->bindAttr($relationModel, $result, $this->bindAttr); + } + // 璁剧疆鍏宠仈灞炴 + $result->setAttr(Loader::parseName($relation), $relationModel); + } + +} diff --git a/thinkphp/library/think/model/relation/BelongsToMany.php b/thinkphp/library/think/model/relation/BelongsToMany.php new file mode 100644 index 000000000..c85a8c0c3 --- /dev/null +++ b/thinkphp/library/think/model/relation/BelongsToMany.php @@ -0,0 +1,360 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\db\Query; +use think\Exception; +use think\Loader; +use think\Model; +use think\model\Pivot; +use think\model\Relation; + +class BelongsToMany extends Relation +{ + // 涓棿琛ㄦā鍨 + protected $middle; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $model 妯″瀷鍚 + * @param string $table 涓棿琛ㄥ悕 + * @param string $foreignKey 鍏宠仈妯″瀷澶栭敭 + * @param string $localKey 褰撳墠妯″瀷鍏宠仈閿 + */ + public function __construct(Model $parent, $model, $table, $foreignKey, $localKey) + { + $this->parent = $parent; + $this->model = $model; + $this->foreignKey = $foreignKey; + $this->localKey = $localKey; + $this->middle = $table; + $this->query = (new $model)->db(); + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @return false|\PDOStatement|string|\think\Collection + */ + public function getRelation($subRelation = '', $closure = null) + { + $foreignKey = $this->foreignKey; + $localKey = $this->localKey; + $middle = $this->middle; + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + // 鍏宠仈鏌ヨ + $pk = $this->parent->getPk(); + $condition['pivot.' . $localKey] = $this->parent->$pk; + $result = $this->belongsToManyQuery($middle, $foreignKey, $localKey, $condition)->relation($subRelation)->select(); + foreach ($result as $set) { + $pivot = []; + foreach ($set->getData() as $key => $val) { + if (strpos($key, '__')) { + list($name, $attr) = explode('__', $key, 2); + if ('pivot' == $name) { + $pivot[$attr] = $val; + unset($set->$key); + } + } + } + $set->pivot = new Pivot($pivot, $this->middle); + } + return $result; + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁闆嗭級 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResultSet(&$resultSet, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + $foreignKey = $this->foreignKey; + + $pk = $resultSet[0]->getPk(); + $range = []; + foreach ($resultSet as $result) { + // 鑾峰彇鍏宠仈澶栭敭鍒楄〃 + if (isset($result->$pk)) { + $range[] = $result->$pk; + } + } + + if (!empty($range)) { + // 鏌ヨ鍏宠仈鏁版嵁 + $data = $this->eagerlyManyToMany([ + 'pivot.' . $localKey => [ + 'in', + $range, + ], + ], $relation, $subRelation); + // 鍏宠仈灞炴у悕 + $attr = Loader::parseName($relation); + // 鍏宠仈鏁版嵁灏佽 + foreach ($resultSet as $result) { + if (!isset($data[$result->$pk])) { + $data[$result->$pk] = []; + } + + $result->setAttr($attr, $this->resultSetBuild($data[$result->$pk])); + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鍗曚釜鏁版嵁锛 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResult(&$result, $relation, $subRelation, $closure) + { + $pk = $result->getPk(); + if (isset($result->$pk)) { + $pk = $result->$pk; + // 鏌ヨ绠$悊鏁版嵁 + $data = $this->eagerlyManyToMany(['pivot.' . $this->localKey => $pk], $relation, $subRelation); + + // 鍏宠仈鏁版嵁灏佽 + if (!isset($data[$pk])) { + $data[$pk] = []; + } + $result->setAttr(Loader::parseName($relation), $this->resultSetBuild($data[$pk])); + } + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param \Closure $closure 闂寘 + * @return integer + */ + public function relationCount($result, $closure) + { + $pk = $result->getPk(); + $count = 0; + if (isset($result->$pk)) { + $pk = $result->$pk; + $count = $this->belongsToManyQuery($this->middle, $this->foreignKey, $this->localKey, ['pivot.' . $this->localKey => $pk])->count(); + } + return $count; + } + + /** + * 鑾峰彇鍏宠仈缁熻瀛愭煡璇 + * @access public + * @param \Closure $closure 闂寘 + * @return string + */ + public function getRelationCountQuery($closure) + { + return $this->belongsToManyQuery($this->middle, $this->foreignKey, $this->localKey, [ + 'pivot.' . $this->localKey => [ + 'exp', + '=' . $this->parent->getTable() . '.' . $this->parent->getPk() + ] + ])->fetchSql()->count(); + } + + /** + * 澶氬澶 鍏宠仈妯″瀷棰勬煡璇 + * @access public + * @param array $where 鍏宠仈棰勬煡璇㈡潯浠 + * @param string $relation 鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱 + * @return array + */ + protected function eagerlyManyToMany($where, $relation, $subRelation = '') + { + // 棰勮浇鍏ュ叧鑱旀煡璇 鏀寔宓屽棰勮浇鍏 + $list = $this->belongsToManyQuery($this->middle, $this->foreignKey, $this->localKey, $where)->with($subRelation)->select(); + + // 缁勮妯″瀷鏁版嵁 + $data = []; + foreach ($list as $set) { + $pivot = []; + foreach ($set->getData() as $key => $val) { + if (strpos($key, '__')) { + list($name, $attr) = explode('__', $key, 2); + if ('pivot' == $name) { + $pivot[$attr] = $val; + unset($set->$key); + } + } + } + $set->pivot = new Pivot($pivot, $this->middle); + $data[$pivot[$this->localKey]][] = $set; + } + return $data; + } + + /** + * BELONGS TO MANY 鍏宠仈鏌ヨ + * @access public + * @param string $table 涓棿琛ㄥ悕 + * @param string $foreignKey 鍏宠仈妯″瀷鍏宠仈閿 + * @param string $localKey 褰撳墠妯″瀷鍏宠仈閿 + * @param array $condition 鍏宠仈鏌ヨ鏉′欢 + * @return Query + */ + protected function belongsToManyQuery($table, $foreignKey, $localKey, $condition = []) + { + // 鍏宠仈鏌ヨ灏佽 + $tableName = $this->query->getTable(); + $relationFk = $this->query->getPk(); + return $this->query->field($tableName . '.*') + ->field(true, false, $table, 'pivot', 'pivot__') + ->join($table . ' pivot', 'pivot.' . $foreignKey . '=' . $tableName . '.' . $relationFk) + ->where($condition); + } + + /** + * 淇濆瓨锛堟柊澧烇級褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param mixed $data 鏁版嵁 鍙互浣跨敤鏁扮粍 鍏宠仈妯″瀷瀵硅薄 鍜 鍏宠仈瀵硅薄鐨勪富閿 + * @param array $pivot 涓棿琛ㄩ澶栨暟鎹 + * @return integer + */ + public function save($data, array $pivot = []) + { + // 淇濆瓨鍏宠仈琛/涓棿琛ㄦ暟鎹 + return $this->attach($data, $pivot); + } + + /** + * 鎵归噺淇濆瓨褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param array $dataSet 鏁版嵁闆 + * @param array $pivot 涓棿琛ㄩ澶栨暟鎹 + * @param bool $samePivot 棰濆鏁版嵁鏄惁鐩稿悓 + * @return integer + */ + public function saveAll(array $dataSet, array $pivot = [], $samePivot = false) + { + $result = false; + foreach ($dataSet as $key => $data) { + if (!$samePivot) { + $pivotData = isset($pivot[$key]) ? $pivot[$key] : []; + } else { + $pivotData = $pivot; + } + $result = $this->attach($data, $pivotData); + } + return $result; + } + + /** + * 闄勫姞鍏宠仈鐨勪竴涓腑闂磋〃鏁版嵁 + * @access public + * @param mixed $data 鏁版嵁 鍙互浣跨敤鏁扮粍銆佸叧鑱旀ā鍨嬪璞 鎴栬 鍏宠仈瀵硅薄鐨勪富閿 + * @param array $pivot 涓棿琛ㄩ澶栨暟鎹 + * @return int + * @throws Exception + */ + public function attach($data, $pivot = []) + { + if (is_array($data)) { + if (key($data) === 0) { + $id = $data; + } else { + // 淇濆瓨鍏宠仈琛ㄦ暟鎹 + $model = new $this->model; + $model->save($data); + $id = $model->getLastInsID(); + } + } elseif (is_numeric($data) || is_string($data)) { + // 鏍规嵁鍏宠仈琛ㄤ富閿洿鎺ュ啓鍏ヤ腑闂磋〃 + $id = $data; + } elseif ($data instanceof Model) { + // 鏍规嵁鍏宠仈琛ㄤ富閿洿鎺ュ啓鍏ヤ腑闂磋〃 + $relationFk = $data->getPk(); + $id = $data->$relationFk; + } + + if ($id) { + // 淇濆瓨涓棿琛ㄦ暟鎹 + $pk = $this->parent->getPk(); + $pivot[$this->localKey] = $this->parent->$pk; + $ids = (array) $id; + foreach ($ids as $id) { + $pivot[$this->foreignKey] = $id; + $result = $this->query->table($this->middle)->insert($pivot, true); + } + return $result; + } else { + throw new Exception('miss relation data'); + } + } + + /** + * 瑙i櫎鍏宠仈鐨勪竴涓腑闂磋〃鏁版嵁 + * @access public + * @param integer|array $data 鏁版嵁 鍙互浣跨敤鍏宠仈瀵硅薄鐨勪富閿 + * @param bool $relationDel 鏄惁鍚屾椂鍒犻櫎鍏宠仈琛ㄦ暟鎹 + * @return integer + */ + public function detach($data = null, $relationDel = false) + { + if (is_array($data)) { + $id = $data; + } elseif (is_numeric($data) || is_string($data)) { + // 鏍规嵁鍏宠仈琛ㄤ富閿洿鎺ュ啓鍏ヤ腑闂磋〃 + $id = $data; + } elseif ($data instanceof Model) { + // 鏍规嵁鍏宠仈琛ㄤ富閿洿鎺ュ啓鍏ヤ腑闂磋〃 + $relationFk = $data->getPk(); + $id = $data->$relationFk; + } + // 鍒犻櫎涓棿琛ㄦ暟鎹 + $pk = $this->parent->getPk(); + $pivot[$this->localKey] = $this->parent->$pk; + if (isset($id)) { + $pivot[$this->foreignKey] = is_array($id) ? ['in', $id] : $id; + } + $this->query->table($this->middle)->where($pivot)->delete(); + + // 鍒犻櫎鍏宠仈琛ㄦ暟鎹 + if (isset($id) && $relationDel) { + $model = $this->model; + $model::destroy($id); + } + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + protected function baseQuery() + { + if (empty($this->baseQuery)) { + $pk = $this->parent->getPk(); + $this->query->join($this->middle . ' pivot', 'pivot.' . $this->foreignKey . '=' . $this->query->getTable() . '.' . $this->query->getPk())->where('pivot.' . $this->localKey, $this->parent->$pk); + $this->baseQuery = true; + } + } + +} diff --git a/thinkphp/library/think/model/relation/HasMany.php b/thinkphp/library/think/model/relation/HasMany.php new file mode 100644 index 000000000..3094bb55f --- /dev/null +++ b/thinkphp/library/think/model/relation/HasMany.php @@ -0,0 +1,272 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\Db; +use think\db\Query; +use think\Loader; +use think\Model; +use think\model\Relation; + +class HasMany extends Relation +{ + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $model 妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + */ + public function __construct(Model $parent, $model, $foreignKey, $localKey) + { + $this->parent = $parent; + $this->model = $model; + $this->foreignKey = $foreignKey; + $this->localKey = $localKey; + $this->query = (new $model)->db(); + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @return false|\PDOStatement|string|\think\Collection + */ + public function getRelation($subRelation = '', $closure = null) + { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + return $this->relation($subRelation)->select(); + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResultSet(&$resultSet, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + $range = []; + foreach ($resultSet as $result) { + // 鑾峰彇鍏宠仈澶栭敭鍒楄〃 + if (isset($result->$localKey)) { + $range[] = $result->$localKey; + } + } + + if (!empty($range)) { + $data = $this->eagerlyOneToMany($this, [ + $this->foreignKey => [ + 'in', + $range, + ], + ], $relation, $subRelation, $closure); + // 鍏宠仈灞炴у悕 + $attr = Loader::parseName($relation); + // 鍏宠仈鏁版嵁灏佽 + foreach ($resultSet as $result) { + if (!isset($data[$result->$localKey])) { + $data[$result->$localKey] = []; + } + $result->setAttr($attr, $this->resultSetBuild($data[$result->$localKey])); + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResult(&$result, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + + if (isset($result->$localKey)) { + $data = $this->eagerlyOneToMany($this, [$this->foreignKey => $result->$localKey], $relation, $subRelation, $closure); + // 鍏宠仈鏁版嵁灏佽 + if (!isset($data[$result->$localKey])) { + $data[$result->$localKey] = []; + } + $result->setAttr(Loader::parseName($relation), $this->resultSetBuild($data[$result->$localKey])); + } + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param \Closure $closure 闂寘 + * @return integer + */ + public function relationCount($result, $closure) + { + $localKey = $this->localKey; + $count = 0; + if (isset($result->$localKey)) { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + $count = $this->query->where([$this->foreignKey => $result->$localKey])->count(); + } + return $count; + } + + /** + * 鍒涘缓鍏宠仈缁熻瀛愭煡璇 + * @access public + * @param \Closure $closure 闂寘 + * @return string + */ + public function getRelationCountQuery($closure) + { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + + return $this->query->where([ + $this->foreignKey => [ + 'exp', + '=' . $this->parent->getTable() . '.' . $this->parent->getPk() + ] + ])->fetchSql()->count(); + } + + /** + * 涓瀵瑰 鍏宠仈妯″瀷棰勬煡璇 + * @access public + * @param object $model 鍏宠仈妯″瀷瀵硅薄 + * @param array $where 鍏宠仈棰勬煡璇㈡潯浠 + * @param string $relation 鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱 + * @param bool $closure + * @return array + */ + protected function eagerlyOneToMany($model, $where, $relation, $subRelation = '', $closure = false) + { + $foreignKey = $this->foreignKey; + // 棰勮浇鍏ュ叧鑱旀煡璇 鏀寔宓屽棰勮浇鍏 + if ($closure) { + call_user_func_array($closure, [& $model]); + } + $list = $model->where($where)->with($subRelation)->select(); + + // 缁勮妯″瀷鏁版嵁 + $data = []; + foreach ($list as $set) { + $data[$set->$foreignKey][] = $set; + } + return $data; + } + + /** + * 淇濆瓨锛堟柊澧烇級褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param mixed $data 鏁版嵁 鍙互浣跨敤鏁扮粍 鍏宠仈妯″瀷瀵硅薄 鍜 鍏宠仈瀵硅薄鐨勪富閿 + * @return integer + */ + public function save($data) + { + if ($data instanceof Model) { + $data = $data->getData(); + } + // 淇濆瓨鍏宠仈琛ㄦ暟鎹 + $data[$this->foreignKey] = $this->parent->{$this->localKey}; + $model = new $this->model; + return $model->save($data); + } + + /** + * 鎵归噺淇濆瓨褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param array $dataSet 鏁版嵁闆 + * @return integer + */ + public function saveAll(array $dataSet) + { + $result = false; + foreach ($dataSet as $key => $data) { + $result = $this->save($data); + } + return $result; + } + + /** + * 鏍规嵁鍏宠仈鏉′欢鏌ヨ褰撳墠妯″瀷 + * @access public + * @param string $operator 姣旇緝鎿嶄綔绗 + * @param integer $count 涓暟 + * @param string $id 鍏宠仈琛ㄧ殑缁熻瀛楁 + * @return Query + */ + public function has($operator = '>=', $count = 1, $id = '*') + { + $table = $this->query->getTable(); + return $this->parent->db()->alias('a') + ->join($table . ' b', 'a.' . $this->localKey . '=b.' . $this->foreignKey, $this->joinType) + ->group('b.' . $this->foreignKey) + ->having('count(' . $id . ')' . $operator . $count); + } + + /** + * 鏍规嵁鍏宠仈鏉′欢鏌ヨ褰撳墠妯″瀷 + * @access public + * @param mixed $where 鏌ヨ鏉′欢锛堟暟缁勬垨鑰呴棴鍖咃級 + * @return Query + */ + public function hasWhere($where = []) + { + $table = $this->query->getTable(); + $model = basename(str_replace('\\', '/', get_class($this->parent))); + $relation = basename(str_replace('\\', '/', $this->model)); + if (is_array($where)) { + foreach ($where as $key => $val) { + if (false === strpos($key, '.')) { + $where[$relation . '.' . $key] = $val; + unset($where[$key]); + } + } + } + return $this->parent->db()->alias($model) + ->field($model . '.*') + ->join($table . ' ' . $relation, $model . '.' . $this->localKey . '=' . $relation . '.' . $this->foreignKey) + ->where($where); + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + protected function baseQuery() + { + if (empty($this->baseQuery)) { + if (isset($this->parent->{$this->localKey})) { + // 鍏宠仈鏌ヨ甯﹀叆鍏宠仈鏉′欢 + $this->query->where($this->foreignKey, $this->parent->{$this->localKey}); + } + $this->baseQuery = true; + } + } + +} diff --git a/thinkphp/library/think/model/relation/HasManyThrough.php b/thinkphp/library/think/model/relation/HasManyThrough.php new file mode 100644 index 000000000..e1518972f --- /dev/null +++ b/thinkphp/library/think/model/relation/HasManyThrough.php @@ -0,0 +1,124 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\Db; +use think\db\Query; +use think\Loader; +use think\Model; +use think\model\Relation; + +class HasManyThrough extends Relation +{ + // 涓棿鍏宠仈琛ㄥ閿 + protected $throughKey; + // 涓棿琛ㄦā鍨 + protected $through; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $model 妯″瀷鍚 + * @param string $through 涓棿妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $throughKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + */ + public function __construct(Model $parent, $model, $through, $foreignKey, $throughKey, $localKey) + { + $this->parent = $parent; + $this->model = $model; + $this->through = $through; + $this->foreignKey = $foreignKey; + $this->throughKey = $throughKey; + $this->localKey = $localKey; + $this->query = (new $model)->db(); + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @return false|\PDOStatement|string|\think\Collection + */ + public function getRelation($subRelation = '', $closure = null) + { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + return $this->relation($subRelation)->select(); + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @param string $class 鏁版嵁闆嗗璞″悕 涓虹┖琛ㄧず鏁扮粍 + * @return void + */ + public function eagerlyResultSet(&$resultSet, $relation, $subRelation, $closure, $class) + { + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 杩斿洖妯″瀷瀵硅薄 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @param string $class 鏁版嵁闆嗗璞″悕 涓虹┖琛ㄧず鏁扮粍 + * @return void + */ + public function eagerlyResult(&$result, $relation, $subRelation, $closure, $class) + { + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param \Closure $closure 闂寘 + * @return integer + */ + public function relationCount($result, $closure) + { + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + protected function baseQuery() + { + if (empty($this->baseQuery)) { + $through = $this->through; + $model = $this->model; + $alias = Loader::parseName(basename(str_replace('\\', '/', $model))); + $throughTable = $through::getTable(); + $pk = (new $this->model)->getPk(); + $throughKey = $this->throughKey; + $modelTable = $this->parent->getTable(); + $this->query->field($alias . '.*')->alias($alias) + ->join($throughTable, $throughTable . '.' . $pk . '=' . $alias . '.' . $throughKey) + ->join($modelTable, $modelTable . '.' . $this->localKey . '=' . $throughTable . '.' . $this->foreignKey) + ->where($throughTable . '.' . $this->foreignKey, $this->parent->{$this->localKey}); + $this->baseQuery = true; + } + } + +} diff --git a/thinkphp/library/think/model/relation/HasOne.php b/thinkphp/library/think/model/relation/HasOne.php new file mode 100644 index 000000000..2c0cf3509 --- /dev/null +++ b/thinkphp/library/think/model/relation/HasOne.php @@ -0,0 +1,159 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\db\Query; +use think\Loader; +use think\Model; + +class HasOne extends OneToOne +{ + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $model 妯″瀷鍚 + * @param string $foreignKey 鍏宠仈澶栭敭 + * @param string $localKey 鍏宠仈涓婚敭 + * @param string $joinType JOIN绫诲瀷 + */ + public function __construct(Model $parent, $model, $foreignKey, $localKey, $joinType = 'INNER') + { + $this->parent = $parent; + $this->model = $model; + $this->foreignKey = $foreignKey; + $this->localKey = $localKey; + $this->joinType = $joinType; + $this->query = (new $model)->db(); + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @return array|false|\PDOStatement|string|Model + */ + public function getRelation($subRelation = '', $closure = null) + { + // 鎵ц鍏宠仈瀹氫箟鏂规硶 + $localKey = $this->localKey; + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + // 鍒ゆ柇鍏宠仈绫诲瀷鎵ц鏌ヨ + return $this->query->where($this->foreignKey, $this->parent->$localKey)->relation($subRelation)->find(); + } + + /** + * 鏍规嵁鍏宠仈鏉′欢鏌ヨ褰撳墠妯″瀷 + * @access public + * @param mixed $where 鏌ヨ鏉′欢锛堟暟缁勬垨鑰呴棴鍖咃級 + * @return Query + */ + public function hasWhere($where = []) + { + $table = $this->query->getTable(); + $model = basename(str_replace('\\', '/', get_class($this->parent))); + $relation = basename(str_replace('\\', '/', $this->model)); + if (is_array($where)) { + foreach ($where as $key => $val) { + if (false === strpos($key, '.')) { + $where[$relation . '.' . $key] = $val; + unset($where[$key]); + } + } + } + return $this->parent->db()->alias($model) + ->field($model . '.*') + ->join($table . ' ' . $relation, $model . '.' . $this->localKey . '=' . $relation . '.' . $this->foreignKey, $this->joinType) + ->where($where); + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁闆嗭級 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + protected function eagerlySet(&$resultSet, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + $foreignKey = $this->foreignKey; + + $range = []; + foreach ($resultSet as $result) { + // 鑾峰彇鍏宠仈澶栭敭鍒楄〃 + if (isset($result->$localKey)) { + $range[] = $result->$localKey; + } + } + + if (!empty($range)) { + $data = $this->eagerlyWhere($this, [ + $foreignKey => [ + 'in', + $range, + ], + ], $foreignKey, $relation, $subRelation, $closure); + // 鍏宠仈灞炴у悕 + $attr = Loader::parseName($relation); + // 鍏宠仈鏁版嵁灏佽 + foreach ($resultSet as $result) { + // 鍏宠仈妯″瀷 + if (!isset($data[$result->$localKey])) { + $relationModel = null; + } else { + $relationModel = $data[$result->$localKey]; + } + if ($relationModel && !empty($this->bindAttr)) { + // 缁戝畾鍏宠仈灞炴 + $this->bindAttr($relationModel, $result, $this->bindAttr); + } + // 璁剧疆鍏宠仈灞炴 + $result->setAttr($attr, $relationModel); + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁锛 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + protected function eagerlyOne(&$result, $relation, $subRelation, $closure) + { + $localKey = $this->localKey; + $foreignKey = $this->foreignKey; + $data = $this->eagerlyWhere($this, [$foreignKey => $result->$localKey], $foreignKey, $relation, $subRelation, $closure); + + // 鍏宠仈妯″瀷 + if (!isset($data[$result->$localKey])) { + $relationModel = null; + } else { + $relationModel = $data[$result->$localKey]; + } + + if ($relationModel && !empty($this->bindAttr)) { + // 缁戝畾鍏宠仈灞炴 + $this->bindAttr($relationModel, $result, $this->bindAttr); + } + $result->setAttr(Loader::parseName($relation), $relationModel); + } + +} diff --git a/thinkphp/library/think/model/relation/MorphMany.php b/thinkphp/library/think/model/relation/MorphMany.php new file mode 100644 index 000000000..3fc8179c4 --- /dev/null +++ b/thinkphp/library/think/model/relation/MorphMany.php @@ -0,0 +1,239 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\Db; +use think\db\Query; +use think\Loader; +use think\Model; +use think\model\Relation; + +class MorphMany extends Relation +{ + // 澶氭佸瓧娈 + protected $morphKey; + protected $morphType; + // 澶氭佺被鍨 + protected $type; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $model 妯″瀷鍚 + * @param string $morphKey 鍏宠仈澶栭敭 + * @param string $morphType 澶氭佸瓧娈靛悕 + * @param string $type 澶氭佺被鍨 + */ + public function __construct(Model $parent, $model, $morphKey, $morphType, $type) + { + $this->parent = $parent; + $this->model = $model; + $this->type = $type; + $this->morphKey = $morphKey; + $this->morphType = $morphType; + $this->query = (new $model)->db(); + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @return false|\PDOStatement|string|\think\Collection + */ + public function getRelation($subRelation = '', $closure = null) + { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + return $this->relation($subRelation)->select(); + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResultSet(&$resultSet, $relation, $subRelation, $closure) + { + $morphType = $this->morphType; + $morphKey = $this->morphKey; + $type = $this->type; + $range = []; + foreach ($resultSet as $result) { + $pk = $result->getPk(); + // 鑾峰彇鍏宠仈澶栭敭鍒楄〃 + if (isset($result->$pk)) { + $range[] = $result->$pk; + } + } + + if (!empty($range)) { + $data = $this->eagerlyMorphToMany([ + $morphKey => ['in', $range], + $morphType => $type, + ], $relation, $subRelation, $closure); + // 鍏宠仈灞炴у悕 + $attr = Loader::parseName($relation); + // 鍏宠仈鏁版嵁灏佽 + foreach ($resultSet as $result) { + if (!isset($data[$result->$pk])) { + $data[$result->$pk] = []; + } + $result->setAttr($attr, $this->resultSetBuild($data[$result->$pk])); + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResult(&$result, $relation, $subRelation, $closure) + { + $pk = $result->getPk(); + if (isset($result->$pk)) { + $data = $this->eagerlyMorphToMany([ + $this->morphKey => $result->$pk, + $this->morphType => $this->type + ], $relation, $subRelation, $closure); + $result->setAttr(Loader::parseName($relation), $this->resultSetBuild($data[$result->$pk])); + } + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param \Closure $closure 闂寘 + * @return integer + */ + public function relationCount($result, $closure) + { + $pk = $result->getPk(); + $count = 0; + if (isset($result->$pk)) { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + $count = $this->query->where([$this->morphKey => $result->$pk, $this->morphType => $this->type])->count(); + } + return $count; + } + + /** + * 鑾峰彇鍏宠仈缁熻瀛愭煡璇 + * @access public + * @param \Closure $closure 闂寘 + * @return string + */ + public function getRelationCountQuery($closure) + { + if ($closure) { + call_user_func_array($closure, [& $this->query]); + } + + return $this->query->where([ + $this->morphKey => [ + 'exp', + '=' . $this->parent->getTable() . '.' . $this->parent->getPk() + ], + $this->morphType => $this->type + ])->fetchSql()->count(); + } + + /** + * 澶氭佷竴瀵瑰 鍏宠仈妯″瀷棰勬煡璇 + * @access public + * @param array $where 鍏宠仈棰勬煡璇㈡潯浠 + * @param string $relation 鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱 + * @param bool|\Closure $closure 闂寘 + * @return array + */ + protected function eagerlyMorphToMany($where, $relation, $subRelation = '', $closure = false) + { + // 棰勮浇鍏ュ叧鑱旀煡璇 鏀寔宓屽棰勮浇鍏 + if ($closure) { + call_user_func_array($closure, [& $this]); + } + $list = $this->query->where($where)->with($subRelation)->select(); + $morphKey = $this->morphKey; + // 缁勮妯″瀷鏁版嵁 + $data = []; + foreach ($list as $set) { + $data[$set->$morphKey][] = $set; + } + return $data; + } + + /** + * 淇濆瓨锛堟柊澧烇級褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param mixed $data 鏁版嵁 鍙互浣跨敤鏁扮粍 鍏宠仈妯″瀷瀵硅薄 鍜 鍏宠仈瀵硅薄鐨勪富閿 + * @return integer + */ + public function save($data) + { + if ($data instanceof Model) { + $data = $data->getData(); + } + // 淇濆瓨鍏宠仈琛ㄦ暟鎹 + $pk = $this->parent->getPk(); + + $data[$this->morphKey] = $this->parent->$pk; + $data[$this->morphType] = $this->type; + $model = new $this->model; + return $model->save($data); + } + + /** + * 鎵归噺淇濆瓨褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param array $dataSet 鏁版嵁闆 + * @return integer + */ + public function saveAll(array $dataSet) + { + $result = false; + foreach ($dataSet as $key => $data) { + $result = $this->save($data); + } + return $result; + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + protected function baseQuery() + { + if (empty($this->baseQuery)) { + $pk = $this->parent->getPk(); + $map[$this->morphKey] = $this->parent->$pk; + $map[$this->morphType] = $this->type; + $this->query->where($map); + $this->baseQuery = true; + } + } + +} diff --git a/thinkphp/library/think/model/relation/MorphTo.php b/thinkphp/library/think/model/relation/MorphTo.php new file mode 100644 index 000000000..5e1867f2b --- /dev/null +++ b/thinkphp/library/think/model/relation/MorphTo.php @@ -0,0 +1,198 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\Exception; +use think\Loader; +use think\Model; +use think\model\Relation; + +class MorphTo extends Relation +{ + // 澶氭佸瓧娈 + protected $morphKey; + protected $morphType; + // 澶氭佸埆鍚 + protected $alias; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Model $parent 涓婄骇妯″瀷瀵硅薄 + * @param string $morphType 澶氭佸瓧娈靛悕 + * @param string $morphKey 澶栭敭鍚 + * @param array $alias 澶氭佸埆鍚嶅畾涔 + */ + public function __construct(Model $parent, $morphType, $morphKey, $alias = []) + { + $this->parent = $parent; + $this->morphType = $morphType; + $this->morphKey = $morphKey; + $this->alias = $alias; + } + + /** + * 寤惰繜鑾峰彇鍏宠仈鏁版嵁 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘鏌ヨ鏉′欢 + * @return mixed + */ + public function getRelation($subRelation = '', $closure = null) + { + $morphKey = $this->morphKey; + $morphType = $this->morphType; + // 澶氭佹ā鍨 + $model = $this->parseModel($this->parent->$morphType); + // 涓婚敭鏁版嵁 + $pk = $this->parent->$morphKey; + return (new $model)->relation($subRelation)->find($pk); + } + + /** + * 瑙f瀽妯″瀷鐨勫畬鏁村懡鍚嶇┖闂 + * @access public + * @param string $model 妯″瀷鍚嶏紙鎴栬呭畬鏁寸被鍚嶏級 + * @return string + */ + protected function parseModel($model) + { + if (isset($this->alias[$model])) { + $model = $this->alias[$model]; + } + if (false === strpos($model, '\\')) { + $path = explode('\\', get_class($this->parent)); + array_pop($path); + array_push($path, Loader::parseName($model, 1)); + $model = implode('\\', $path); + } + return $model; + } + + /** + * 璁剧疆澶氭佸埆鍚 + * @access public + * @param array $alias 鍒悕瀹氫箟 + * @return $this + */ + public function setAlias($alias) + { + $this->alias = $alias; + return $this; + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + * @throws Exception + */ + public function eagerlyResultSet(&$resultSet, $relation, $subRelation, $closure) + { + $morphKey = $this->morphKey; + $morphType = $this->morphType; + $range = []; + foreach ($resultSet as $result) { + // 鑾峰彇鍏宠仈澶栭敭鍒楄〃 + if (!empty($result->$morphKey)) { + $range[$result->$morphType][] = $result->$morphKey; + } + } + + if (!empty($range)) { + // 鍏宠仈灞炴у悕 + $attr = Loader::parseName($relation); + foreach ($range as $key => $val) { + // 澶氭佺被鍨嬫槧灏 + $model = $this->parseModel($key); + $obj = new $model; + $pk = $obj->getPk(); + $list = $obj->all($val, $subRelation); + $data = []; + foreach ($list as $k => $vo) { + $data[$vo->$pk] = $vo; + } + foreach ($resultSet as $result) { + if ($key == $result->$morphType) { + // 鍏宠仈妯″瀷 + if (!isset($data[$result->$morphKey])) { + throw new Exception('relation data not exists :' . $this->model); + } else { + $result->setAttr($attr, $data[$result->$morphKey]); + } + } + } + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResult(&$result, $relation, $subRelation, $closure) + { + $morphKey = $this->morphKey; + $morphType = $this->morphType; + // 澶氭佺被鍨嬫槧灏 + $model = $this->parseModel($result->{$this->morphType}); + $this->eagerlyMorphToOne($model, $relation, $result, $subRelation); + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param \Closure $closure 闂寘 + * @return integer + */ + public function relationCount($result, $closure) + { + } + + /** + * 澶氭丮orphTo 鍏宠仈妯″瀷棰勬煡璇 + * @access public + * @param object $model 鍏宠仈妯″瀷瀵硅薄 + * @param string $relation 鍏宠仈鍚 + * @param $result + * @param string $subRelation 瀛愬叧鑱 + * @return void + */ + protected function eagerlyMorphToOne($model, $relation, &$result, $subRelation = '') + { + // 棰勮浇鍏ュ叧鑱旀煡璇 鏀寔宓屽棰勮浇鍏 + $pk = $this->parent->{$this->morphKey}; + $data = (new $model)->with($subRelation)->find($pk); + if ($data) { + $data->isUpdate(true); + } + $result->setAttr(Loader::parseName($relation), $data ?: null); + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + protected function baseQuery() + { + } +} diff --git a/thinkphp/library/think/model/relation/OneToOne.php b/thinkphp/library/think/model/relation/OneToOne.php new file mode 100644 index 000000000..18b922256 --- /dev/null +++ b/thinkphp/library/think/model/relation/OneToOne.php @@ -0,0 +1,316 @@ + +// +---------------------------------------------------------------------- + +namespace think\model\relation; + +use think\db\Query; +use think\Exception; +use think\Loader; +use think\Model; +use think\model\Relation; + +/** + * Class OneToOne + * @package think\model\relation + * + */ +abstract class OneToOne extends Relation +{ + // 棰勮浇鍏ユ柟寮 0 -JOIN 1 -IN + protected $eagerlyType = 1; + // 褰撳墠鍏宠仈鐨凧OIN绫诲瀷 + protected $joinType; + // 瑕佺粦瀹氱殑灞炴 + protected $bindAttr = []; + + /** + * 璁剧疆join绫诲瀷 + * @access public + * @param string $type JOIN绫诲瀷 + * @return $this + */ + public function joinType($type) + { + $this->joinType = $type; + return $this; + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙JOIN鏂瑰紡锛 + * @access public + * @param Query $query 鏌ヨ瀵硅薄 + * @param string $relation 鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱 + * @param \Closure $closure 闂寘鏉′欢 + * @param bool $first + * @return void + */ + public function eagerly(Query $query, $relation, $subRelation, $closure, $first) + { + $name = Loader::parseName(basename(str_replace('\\', '/', $query->getModel()))); + $alias = $name; + if ($first) { + $table = $query->getTable(); + $query->table([$table => $alias]); + if ($query->getOptions('field')) { + $field = $query->getOptions('field'); + $query->removeOption('field'); + } else { + $field = true; + } + $query->field($field, false, $table, $alias); + } + + // 棰勮浇鍏ュ皝瑁 + $joinTable = $this->query->getTable(); + $joinAlias = $relation; + $query->via($joinAlias); + + if ($this instanceof BelongsTo) { + $query->join($joinTable . ' ' . $joinAlias, $alias . '.' . $this->foreignKey . '=' . $joinAlias . '.' . $this->localKey, $this->joinType); + } else { + $query->join($joinTable . ' ' . $joinAlias, $alias . '.' . $this->localKey . '=' . $joinAlias . '.' . $this->foreignKey, $this->joinType); + } + + if ($closure) { + // 鎵ц闂寘鏌ヨ + call_user_func_array($closure, [& $query]); + // 浣跨敤withField鎸囧畾鑾峰彇鍏宠仈鐨勫瓧娈碉紝濡 + // $query->where(['id'=>1])->withField('id,name'); + if ($query->getOptions('with_field')) { + $field = $query->getOptions('with_field'); + $query->removeOption('with_field'); + } + } elseif (isset($this->option['field'])) { + $field = $this->option['field']; + } else { + $field = true; + } + $query->field($field, false, $joinTable, $joinAlias, $relation . '__'); + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁闆嗭級 + * @param array $resultSet + * @param string $relation + * @param string $subRelation + * @param \Closure $closure + * @return mixed + */ + abstract protected function eagerlySet(&$resultSet, $relation, $subRelation, $closure); + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁锛 + * @param Model $result + * @param string $relation + * @param string $subRelation + * @param \Closure $closure + * @return mixed + */ + abstract protected function eagerlyOne(&$result, $relation, $subRelation, $closure); + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁闆嗭級 + * @access public + * @param array $resultSet 鏁版嵁闆 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResultSet(&$resultSet, $relation, $subRelation, $closure) + { + if (1 == $this->eagerlyType) { + // IN鏌ヨ + $this->eagerlySet($resultSet, $relation, $subRelation, $closure); + } else { + // 妯″瀷鍏宠仈缁勮 + foreach ($resultSet as $result) { + $this->match($this->model, $relation, $result); + } + } + } + + /** + * 棰勮浇鍏ュ叧鑱旀煡璇紙鏁版嵁锛 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param string $relation 褰撳墠鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱斿悕 + * @param \Closure $closure 闂寘 + * @return void + */ + public function eagerlyResult(&$result, $relation, $subRelation, $closure) + { + if (1 == $this->eagerlyType) { + // IN鏌ヨ + $this->eagerlyOne($result, $relation, $subRelation, $closure); + } else { + // 妯″瀷鍏宠仈缁勮 + $this->match($this->model, $relation, $result); + } + } + + /** + * 淇濆瓨锛堟柊澧烇級褰撳墠鍏宠仈鏁版嵁瀵硅薄 + * @access public + * @param mixed $data 鏁版嵁 鍙互浣跨敤鏁扮粍 鍏宠仈妯″瀷瀵硅薄 鍜 鍏宠仈瀵硅薄鐨勪富閿 + * @return integer + */ + public function save($data) + { + if ($data instanceof Model) { + $data = $data->getData(); + } + // 淇濆瓨鍏宠仈琛ㄦ暟鎹 + $data[$this->foreignKey] = $this->parent->{$this->localKey}; + $model = new $this->model; + return $model->save($data); + } + + /** + * 璁剧疆棰勮浇鍏ユ柟寮 + * @access public + * @param integer $type 棰勮浇鍏ユ柟寮 0 JOIN鏌ヨ 1 IN鏌ヨ + * @return $this + */ + public function setEagerlyType($type) + { + $this->eagerlyType = $type; + return $this; + } + + /** + * 鑾峰彇棰勮浇鍏ユ柟寮 + * @access public + * @return integer + */ + public function getEagerlyType() + { + $this->removeOption(); + return $this->eagerlyType; + } + + /** + * 缁戝畾鍏宠仈琛ㄧ殑灞炴у埌鐖舵ā鍨嬪睘鎬 + * @access public + * @param mixed $attr 瑕佺粦瀹氱殑灞炴у垪琛 + * @return $this + */ + public function bind($attr) + { + if (is_string($attr)) { + $attr = explode(',', $attr); + } + $this->bindAttr = $attr; + return $this; + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param Model $result 鏁版嵁瀵硅薄 + * @param \Closure $closure 闂寘 + * @return integer + */ + public function relationCount($result, $closure) + { + } + + /** + * 涓瀵逛竴 鍏宠仈妯″瀷棰勬煡璇㈡嫾瑁 + * @access public + * @param string $model 妯″瀷鍚嶇О + * @param string $relation 鍏宠仈鍚 + * @param Model $result 妯″瀷瀵硅薄瀹炰緥 + * @return void + */ + protected function match($model, $relation, &$result) + { + // 閲嶆柊缁勮妯″瀷鏁版嵁 + foreach ($result->getData() as $key => $val) { + if (strpos($key, '__')) { + list($name, $attr) = explode('__', $key, 2); + if ($name == $relation) { + $list[$name][$attr] = $val; + unset($result->$key); + } + } + } + if (isset($list[$relation])) { + $relationModel = new $model($list[$relation]); + if (!empty($this->bindAttr)) { + $this->bindAttr($relationModel, $result, $this->bindAttr); + } + } + $result->setAttr(Loader::parseName($relation), !isset($relationModel) ? null : $relationModel->isUpdate(true)); + } + + /** + * 缁戝畾鍏宠仈灞炴у埌鐖舵ā鍨 + * @access protected + * @param Model $model 鍏宠仈妯″瀷瀵硅薄 + * @param Model $result 鐖舵ā鍨嬪璞 + * @param array $bindAttr 缁戝畾灞炴 + * @return void + * @throws Exception + */ + protected function bindAttr($model, &$result, $bindAttr) + { + foreach ($bindAttr as $key => $attr) { + $key = is_numeric($key) ? $attr : $key; + if (isset($result->$key)) { + throw new Exception('bind attr has exists:' . $key); + } else { + $result->setAttr($key, $model->$attr); + } + } + } + + /** + * 涓瀵逛竴 鍏宠仈妯″瀷棰勬煡璇紙IN鏂瑰紡锛 + * @access public + * @param object $model 鍏宠仈妯″瀷瀵硅薄 + * @param array $where 鍏宠仈棰勬煡璇㈡潯浠 + * @param string $key 鍏宠仈閿悕 + * @param string $relation 鍏宠仈鍚 + * @param string $subRelation 瀛愬叧鑱 + * @param bool|\Closure $closure + * @return array + */ + protected function eagerlyWhere($model, $where, $key, $relation, $subRelation = '', $closure = false) + { + // 棰勮浇鍏ュ叧鑱旀煡璇 鏀寔宓屽棰勮浇鍏 + if ($closure) { + call_user_func_array($closure, [& $model]); + if ($field = $model->getOptions('with_field')) { + $model->field($field)->removeOption('with_field'); + } + } + $list = $model->where($where)->with($subRelation)->select(); + + // 缁勮妯″瀷鏁版嵁 + $data = []; + foreach ($list as $set) { + $data[$set->$key] = $set; + } + return $data; + } + + /** + * 鎵ц鍩虹鏌ヨ锛堣繘鎵ц涓娆★級 + * @access protected + * @return void + */ + protected function baseQuery() + { + } +} diff --git a/thinkphp/library/think/paginator/driver/Bootstrap.php b/thinkphp/library/think/paginator/driver/Bootstrap.php new file mode 100644 index 000000000..58fa94364 --- /dev/null +++ b/thinkphp/library/think/paginator/driver/Bootstrap.php @@ -0,0 +1,205 @@ + +// +---------------------------------------------------------------------- + +namespace think\paginator\driver; + +use think\Paginator; + +class Bootstrap extends Paginator +{ + + /** + * 涓婁竴椤垫寜閽 + * @param string $text + * @return string + */ + protected function getPreviousButton($text = "«") + { + + if ($this->currentPage() <= 1) { + return $this->getDisabledTextWrapper($text); + } + + $url = $this->url( + $this->currentPage() - 1 + ); + + return $this->getPageLinkWrapper($url, $text); + } + + /** + * 涓嬩竴椤垫寜閽 + * @param string $text + * @return string + */ + protected function getNextButton($text = '»') + { + if (!$this->hasMore) { + return $this->getDisabledTextWrapper($text); + } + + $url = $this->url($this->currentPage() + 1); + + return $this->getPageLinkWrapper($url, $text); + } + + /** + * 椤电爜鎸夐挳 + * @return string + */ + protected function getLinks() + { + if ($this->simple) + return ''; + + $block = [ + 'first' => null, + 'slider' => null, + 'last' => null + ]; + + $side = 3; + $window = $side * 2; + + if ($this->lastPage < $window + 6) { + $block['first'] = $this->getUrlRange(1, $this->lastPage); + } elseif ($this->currentPage <= $window) { + $block['first'] = $this->getUrlRange(1, $window + 2); + $block['last'] = $this->getUrlRange($this->lastPage - 1, $this->lastPage); + } elseif ($this->currentPage > ($this->lastPage - $window)) { + $block['first'] = $this->getUrlRange(1, 2); + $block['last'] = $this->getUrlRange($this->lastPage - ($window + 2), $this->lastPage); + } else { + $block['first'] = $this->getUrlRange(1, 2); + $block['slider'] = $this->getUrlRange($this->currentPage - $side, $this->currentPage + $side); + $block['last'] = $this->getUrlRange($this->lastPage - 1, $this->lastPage); + } + + $html = ''; + + if (is_array($block['first'])) { + $html .= $this->getUrlLinks($block['first']); + } + + if (is_array($block['slider'])) { + $html .= $this->getDots(); + $html .= $this->getUrlLinks($block['slider']); + } + + if (is_array($block['last'])) { + $html .= $this->getDots(); + $html .= $this->getUrlLinks($block['last']); + } + + return $html; + } + + /** + * 娓叉煋鍒嗛〉html + * @return mixed + */ + public function render() + { + if ($this->hasPages()) { + if ($this->simple) { + return sprintf( + '
    %s %s
', + $this->getPreviousButton(), + $this->getNextButton() + ); + } else { + return sprintf( + '
    %s %s %s
', + $this->getPreviousButton(), + $this->getLinks(), + $this->getNextButton() + ); + } + } + } + + /** + * 鐢熸垚涓涓彲鐐瑰嚮鐨勬寜閽 + * + * @param string $url + * @param int $page + * @return string + */ + protected function getAvailablePageWrapper($url, $page) + { + return '
  • ' . $page . '
  • '; + } + + /** + * 鐢熸垚涓涓鐢ㄧ殑鎸夐挳 + * + * @param string $text + * @return string + */ + protected function getDisabledTextWrapper($text) + { + return '
  • ' . $text . '
  • '; + } + + /** + * 鐢熸垚涓涓縺娲荤殑鎸夐挳 + * + * @param string $text + * @return string + */ + protected function getActivePageWrapper($text) + { + return '
  • ' . $text . '
  • '; + } + + /** + * 鐢熸垚鐪佺暐鍙锋寜閽 + * + * @return string + */ + protected function getDots() + { + return $this->getDisabledTextWrapper('...'); + } + + /** + * 鎵归噺鐢熸垚椤电爜鎸夐挳. + * + * @param array $urls + * @return string + */ + protected function getUrlLinks(array $urls) + { + $html = ''; + + foreach ($urls as $page => $url) { + $html .= $this->getPageLinkWrapper($url, $page); + } + + return $html; + } + + /** + * 鐢熸垚鏅氶〉鐮佹寜閽 + * + * @param string $url + * @param int $page + * @return string + */ + protected function getPageLinkWrapper($url, $page) + { + if ($page == $this->currentPage()) { + return $this->getActivePageWrapper($page); + } + + return $this->getAvailablePageWrapper($url, $page); + } +} diff --git a/thinkphp/library/think/process/Builder.php b/thinkphp/library/think/process/Builder.php new file mode 100644 index 000000000..da5616397 --- /dev/null +++ b/thinkphp/library/think/process/Builder.php @@ -0,0 +1,233 @@ + +// +---------------------------------------------------------------------- + +namespace think\process; + +use think\Process; + +class Builder +{ + private $arguments; + private $cwd; + private $env = null; + private $input; + private $timeout = 60; + private $options = []; + private $inheritEnv = true; + private $prefix = []; + private $outputDisabled = false; + + /** + * 鏋勯犳柟娉 + * @param string[] $arguments 鍙傛暟 + */ + public function __construct(array $arguments = []) + { + $this->arguments = $arguments; + } + + /** + * 鍒涘缓涓涓疄渚 + * @param string[] $arguments 鍙傛暟 + * @return self + */ + public static function create(array $arguments = []) + { + return new static($arguments); + } + + /** + * 娣诲姞涓涓弬鏁 + * @param string $argument 鍙傛暟 + * @return self + */ + public function add($argument) + { + $this->arguments[] = $argument; + + return $this; + } + + /** + * 娣诲姞涓涓墠缂 + * @param string|array $prefix + * @return self + */ + public function setPrefix($prefix) + { + $this->prefix = is_array($prefix) ? $prefix : [$prefix]; + + return $this; + } + + /** + * 璁剧疆鍙傛暟 + * @param string[] $arguments + * @return self + */ + public function setArguments(array $arguments) + { + $this->arguments = $arguments; + + return $this; + } + + /** + * 璁剧疆宸ヤ綔鐩綍 + * @param null|string $cwd + * @return self + */ + public function setWorkingDirectory($cwd) + { + $this->cwd = $cwd; + + return $this; + } + + /** + * 鏄惁鍒濆鍖栫幆澧冨彉閲 + * @param bool $inheritEnv + * @return self + */ + public function inheritEnvironmentVariables($inheritEnv = true) + { + $this->inheritEnv = $inheritEnv; + + return $this; + } + + /** + * 璁剧疆鐜鍙橀噺 + * @param string $name + * @param null|string $value + * @return self + */ + public function setEnv($name, $value) + { + $this->env[$name] = $value; + + return $this; + } + + /** + * 娣诲姞鐜鍙橀噺 + * @param array $variables + * @return self + */ + public function addEnvironmentVariables(array $variables) + { + $this->env = array_replace($this->env, $variables); + + return $this; + } + + /** + * 璁剧疆杈撳叆 + * @param mixed $input + * @return self + */ + public function setInput($input) + { + $this->input = Utils::validateInput(sprintf('%s::%s', __CLASS__, __FUNCTION__), $input); + + return $this; + } + + /** + * 璁剧疆瓒呮椂鏃堕棿 + * @param float|null $timeout + * @return self + */ + public function setTimeout($timeout) + { + if (null === $timeout) { + $this->timeout = null; + + return $this; + } + + $timeout = (float) $timeout; + + if ($timeout < 0) { + throw new \InvalidArgumentException('The timeout value must be a valid positive integer or float number.'); + } + + $this->timeout = $timeout; + + return $this; + } + + /** + * 璁剧疆proc_open閫夐」 + * @param string $name + * @param string $value + * @return self + */ + public function setOption($name, $value) + { + $this->options[$name] = $value; + + return $this; + } + + /** + * 绂佹杈撳嚭 + * @return self + */ + public function disableOutput() + { + $this->outputDisabled = true; + + return $this; + } + + /** + * 寮鍚緭鍑 + * @return self + */ + public function enableOutput() + { + $this->outputDisabled = false; + + return $this; + } + + /** + * 鍒涘缓涓涓狿rocess瀹炰緥 + * @return Process + */ + public function getProcess() + { + if (0 === count($this->prefix) && 0 === count($this->arguments)) { + throw new \LogicException('You must add() command arguments before calling getProcess().'); + } + + $options = $this->options; + + $arguments = array_merge($this->prefix, $this->arguments); + $script = implode(' ', array_map([__NAMESPACE__ . '\\Utils', 'escapeArgument'], $arguments)); + + if ($this->inheritEnv) { + // include $_ENV for BC purposes + $env = array_replace($_ENV, $_SERVER, $this->env); + } else { + $env = $this->env; + } + + $process = new Process($script, $this->cwd, $env, $this->input, $this->timeout, $options); + + if ($this->outputDisabled) { + $process->disableOutput(); + } + + return $process; + } +} diff --git a/thinkphp/library/think/process/Utils.php b/thinkphp/library/think/process/Utils.php new file mode 100644 index 000000000..f94c6488e --- /dev/null +++ b/thinkphp/library/think/process/Utils.php @@ -0,0 +1,75 @@ + +// +---------------------------------------------------------------------- + +namespace think\process; + +class Utils +{ + + /** + * 杞箟瀛楃涓 + * @param string $argument + * @return string + */ + public static function escapeArgument($argument) + { + + if ('' === $argument) { + return escapeshellarg($argument); + } + $escapedArgument = ''; + $quote = false; + foreach (preg_split('/(")/i', $argument, -1, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE) as $part) { + if ('"' === $part) { + $escapedArgument .= '\\"'; + } elseif (self::isSurroundedBy($part, '%')) { + // Avoid environment variable expansion + $escapedArgument .= '^%"' . substr($part, 1, -1) . '"^%'; + } else { + // escape trailing backslash + if ('\\' === substr($part, -1)) { + $part .= '\\'; + } + $quote = true; + $escapedArgument .= $part; + } + } + if ($quote) { + $escapedArgument = '"' . $escapedArgument . '"'; + } + return $escapedArgument; + } + + /** + * 楠岃瘉骞惰繘琛岃鑼冨寲Process杈撳叆銆 + * @param string $caller + * @param mixed $input + * @return string + * @throws \InvalidArgumentException + */ + public static function validateInput($caller, $input) + { + if (null !== $input) { + if (is_resource($input)) { + return $input; + } + if (is_scalar($input)) { + return (string) $input; + } + throw new \InvalidArgumentException(sprintf('%s only accepts strings or stream resources.', $caller)); + } + return $input; + } + + private static function isSurroundedBy($arg, $char) + { + return 2 < strlen($arg) && $char === $arg[0] && $char === $arg[strlen($arg) - 1]; + } + +} diff --git a/thinkphp/library/think/process/exception/Failed.php b/thinkphp/library/think/process/exception/Failed.php new file mode 100644 index 000000000..529508232 --- /dev/null +++ b/thinkphp/library/think/process/exception/Failed.php @@ -0,0 +1,42 @@ + +// +---------------------------------------------------------------------- + +namespace think\process\exception; + +use think\Process; + +class Failed extends \RuntimeException +{ + + private $process; + + public function __construct(Process $process) + { + if ($process->isSuccessful()) { + throw new \InvalidArgumentException('Expected a failed process, but the given process was successful.'); + } + + $error = sprintf('The command "%s" failed.' . "\nExit Code: %s(%s)", $process->getCommandLine(), $process->getExitCode(), $process->getExitCodeText()); + + if (!$process->isOutputDisabled()) { + $error .= sprintf("\n\nOutput:\n================\n%s\n\nError Output:\n================\n%s", $process->getOutput(), $process->getErrorOutput()); + } + + parent::__construct($error); + + $this->process = $process; + } + + public function getProcess() + { + return $this->process; + } +} diff --git a/thinkphp/library/think/process/exception/Timeout.php b/thinkphp/library/think/process/exception/Timeout.php new file mode 100644 index 000000000..d5f1162f4 --- /dev/null +++ b/thinkphp/library/think/process/exception/Timeout.php @@ -0,0 +1,61 @@ + +// +---------------------------------------------------------------------- + +namespace think\process\exception; + +use think\Process; + +class Timeout extends \RuntimeException +{ + + const TYPE_GENERAL = 1; + const TYPE_IDLE = 2; + + private $process; + private $timeoutType; + + public function __construct(Process $process, $timeoutType) + { + $this->process = $process; + $this->timeoutType = $timeoutType; + + parent::__construct(sprintf('The process "%s" exceeded the timeout of %s seconds.', $process->getCommandLine(), $this->getExceededTimeout())); + } + + public function getProcess() + { + return $this->process; + } + + public function isGeneralTimeout() + { + return $this->timeoutType === self::TYPE_GENERAL; + } + + public function isIdleTimeout() + { + return $this->timeoutType === self::TYPE_IDLE; + } + + public function getExceededTimeout() + { + switch ($this->timeoutType) { + case self::TYPE_GENERAL: + return $this->process->getTimeout(); + + case self::TYPE_IDLE: + return $this->process->getIdleTimeout(); + + default: + throw new \LogicException(sprintf('Unknown timeout type "%d".', $this->timeoutType)); + } + } +} diff --git a/thinkphp/library/think/process/pipes/Pipes.php b/thinkphp/library/think/process/pipes/Pipes.php new file mode 100644 index 000000000..82396b8ff --- /dev/null +++ b/thinkphp/library/think/process/pipes/Pipes.php @@ -0,0 +1,93 @@ + +// +---------------------------------------------------------------------- + +namespace think\process\pipes; + +abstract class Pipes +{ + + /** @var array */ + public $pipes = []; + + /** @var string */ + protected $inputBuffer = ''; + /** @var resource|null */ + protected $input; + + /** @var bool */ + private $blocked = true; + + const CHUNK_SIZE = 16384; + + /** + * 杩斿洖鐢ㄤ簬 proc_open 鎻忚堪绗︾殑鏁扮粍 + * @return array + */ + abstract public function getDescriptors(); + + /** + * 杩斿洖涓涓暟缁勭殑绱㈠紩鐢卞叾鐩稿叧鐨勬祦锛屼互闃茶繖浜涚閬撲娇鐢ㄧ殑涓存椂鏂囦欢鐨勬枃浠跺悕銆 + * @return string[] + */ + abstract public function getFiles(); + + /** + * 鏂囦欢鍙ユ焺鍜岀閬撲腑璇诲彇鏁版嵁銆 + * @param bool $blocking 鏄惁浣跨敤闃诲璋冪敤 + * @param bool $close 鏄惁瑕佸叧闂閬擄紝濡傛灉浠栦滑宸茬粡鍒拌揪 EOF銆 + * @return string[] + */ + abstract public function readAndWrite($blocking, $close = false); + + /** + * 杩斿洖褰撳墠鐘舵佸鏋滄湁鎵撳紑鐨勬枃浠跺彞鏌勬垨绠¢亾銆 + * @return bool + */ + abstract public function areOpen(); + + /** + * {@inheritdoc} + */ + public function close() + { + foreach ($this->pipes as $pipe) { + fclose($pipe); + } + $this->pipes = []; + } + + /** + * 妫鏌ョ郴缁熻皟鐢ㄥ凡琚腑鏂 + * @return bool + */ + protected function hasSystemCallBeenInterrupted() + { + $lastError = error_get_last(); + + return isset($lastError['message']) && false !== stripos($lastError['message'], 'interrupted system call'); + } + + protected function unblock() + { + if (!$this->blocked) { + return; + } + + foreach ($this->pipes as $pipe) { + stream_set_blocking($pipe, 0); + } + if (null !== $this->input) { + stream_set_blocking($this->input, 0); + } + + $this->blocked = false; + } +} diff --git a/thinkphp/library/think/process/pipes/Unix.php b/thinkphp/library/think/process/pipes/Unix.php new file mode 100644 index 000000000..fd99a5d67 --- /dev/null +++ b/thinkphp/library/think/process/pipes/Unix.php @@ -0,0 +1,196 @@ + +// +---------------------------------------------------------------------- + +namespace think\process\pipes; + +use think\Process; + +class Unix extends Pipes +{ + + /** @var bool */ + private $ttyMode; + /** @var bool */ + private $ptyMode; + /** @var bool */ + private $disableOutput; + + public function __construct($ttyMode, $ptyMode, $input, $disableOutput) + { + $this->ttyMode = (bool) $ttyMode; + $this->ptyMode = (bool) $ptyMode; + $this->disableOutput = (bool) $disableOutput; + + if (is_resource($input)) { + $this->input = $input; + } else { + $this->inputBuffer = (string) $input; + } + } + + public function __destruct() + { + $this->close(); + } + + /** + * {@inheritdoc} + */ + public function getDescriptors() + { + if ($this->disableOutput) { + $nullstream = fopen('/dev/null', 'c'); + + return [ + ['pipe', 'r'], + $nullstream, + $nullstream, + ]; + } + + if ($this->ttyMode) { + return [ + ['file', '/dev/tty', 'r'], + ['file', '/dev/tty', 'w'], + ['file', '/dev/tty', 'w'], + ]; + } + + if ($this->ptyMode && Process::isPtySupported()) { + return [ + ['pty'], + ['pty'], + ['pty'], + ]; + } + + return [ + ['pipe', 'r'], + ['pipe', 'w'], // stdout + ['pipe', 'w'], // stderr + ]; + } + + /** + * {@inheritdoc} + */ + public function getFiles() + { + return []; + } + + /** + * {@inheritdoc} + */ + public function readAndWrite($blocking, $close = false) + { + + if (1 === count($this->pipes) && [0] === array_keys($this->pipes)) { + fclose($this->pipes[0]); + unset($this->pipes[0]); + } + + if (empty($this->pipes)) { + return []; + } + + $this->unblock(); + + $read = []; + + if (null !== $this->input) { + $r = array_merge($this->pipes, ['input' => $this->input]); + } else { + $r = $this->pipes; + } + + unset($r[0]); + + $w = isset($this->pipes[0]) ? [$this->pipes[0]] : null; + $e = null; + + if (false === $n = @stream_select($r, $w, $e, 0, $blocking ? Process::TIMEOUT_PRECISION * 1E6 : 0)) { + + if (!$this->hasSystemCallBeenInterrupted()) { + $this->pipes = []; + } + + return $read; + } + + if (0 === $n) { + return $read; + } + + foreach ($r as $pipe) { + + $type = (false !== $found = array_search($pipe, $this->pipes)) ? $found : 'input'; + $data = ''; + while ('' !== $dataread = (string) fread($pipe, self::CHUNK_SIZE)) { + $data .= $dataread; + } + + if ('' !== $data) { + if ('input' === $type) { + $this->inputBuffer .= $data; + } else { + $read[$type] = $data; + } + } + + if (false === $data || (true === $close && feof($pipe) && '' === $data)) { + if ('input' === $type) { + $this->input = null; + } else { + fclose($this->pipes[$type]); + unset($this->pipes[$type]); + } + } + } + + if (null !== $w && 0 < count($w)) { + while (strlen($this->inputBuffer)) { + $written = fwrite($w[0], $this->inputBuffer, 2 << 18); // write 512k + if ($written > 0) { + $this->inputBuffer = (string) substr($this->inputBuffer, $written); + } else { + break; + } + } + } + + if ('' === $this->inputBuffer && null === $this->input && isset($this->pipes[0])) { + fclose($this->pipes[0]); + unset($this->pipes[0]); + } + + return $read; + } + + /** + * {@inheritdoc} + */ + public function areOpen() + { + return (bool) $this->pipes; + } + + /** + * 鍒涘缓涓涓柊鐨 UnixPipes 瀹炰緥 + * @param Process $process + * @param string|resource $input + * @return self + */ + public static function create(Process $process, $input) + { + return new static($process->isTty(), $process->isPty(), $input, $process->isOutputDisabled()); + } +} diff --git a/thinkphp/library/think/process/pipes/Windows.php b/thinkphp/library/think/process/pipes/Windows.php new file mode 100644 index 000000000..bba7e9b86 --- /dev/null +++ b/thinkphp/library/think/process/pipes/Windows.php @@ -0,0 +1,228 @@ + +// +---------------------------------------------------------------------- + +namespace think\process\pipes; + +use think\Process; + +class Windows extends Pipes +{ + + /** @var array */ + private $files = []; + /** @var array */ + private $fileHandles = []; + /** @var array */ + private $readBytes = [ + Process::STDOUT => 0, + Process::STDERR => 0, + ]; + /** @var bool */ + private $disableOutput; + + public function __construct($disableOutput, $input) + { + $this->disableOutput = (bool) $disableOutput; + + if (!$this->disableOutput) { + + $this->files = [ + Process::STDOUT => tempnam(sys_get_temp_dir(), 'sf_proc_stdout'), + Process::STDERR => tempnam(sys_get_temp_dir(), 'sf_proc_stderr'), + ]; + foreach ($this->files as $offset => $file) { + $this->fileHandles[$offset] = fopen($this->files[$offset], 'rb'); + if (false === $this->fileHandles[$offset]) { + throw new \RuntimeException('A temporary file could not be opened to write the process output to, verify that your TEMP environment variable is writable'); + } + } + } + + if (is_resource($input)) { + $this->input = $input; + } else { + $this->inputBuffer = $input; + } + } + + public function __destruct() + { + $this->close(); + $this->removeFiles(); + } + + /** + * {@inheritdoc} + */ + public function getDescriptors() + { + if ($this->disableOutput) { + $nullstream = fopen('NUL', 'c'); + + return [ + ['pipe', 'r'], + $nullstream, + $nullstream, + ]; + } + + return [ + ['pipe', 'r'], + ['file', 'NUL', 'w'], + ['file', 'NUL', 'w'], + ]; + } + + /** + * {@inheritdoc} + */ + public function getFiles() + { + return $this->files; + } + + /** + * {@inheritdoc} + */ + public function readAndWrite($blocking, $close = false) + { + $this->write($blocking, $close); + + $read = []; + $fh = $this->fileHandles; + foreach ($fh as $type => $fileHandle) { + if (0 !== fseek($fileHandle, $this->readBytes[$type])) { + continue; + } + $data = ''; + $dataread = null; + while (!feof($fileHandle)) { + if (false !== $dataread = fread($fileHandle, self::CHUNK_SIZE)) { + $data .= $dataread; + } + } + if (0 < $length = strlen($data)) { + $this->readBytes[$type] += $length; + $read[$type] = $data; + } + + if (false === $dataread || (true === $close && feof($fileHandle) && '' === $data)) { + fclose($this->fileHandles[$type]); + unset($this->fileHandles[$type]); + } + } + + return $read; + } + + /** + * {@inheritdoc} + */ + public function areOpen() + { + return (bool) $this->pipes && (bool) $this->fileHandles; + } + + /** + * {@inheritdoc} + */ + public function close() + { + parent::close(); + foreach ($this->fileHandles as $handle) { + fclose($handle); + } + $this->fileHandles = []; + } + + /** + * 鍒涘缓涓涓柊鐨 WindowsPipes 瀹炰緥銆 + * @param Process $process + * @param $input + * @return self + */ + public static function create(Process $process, $input) + { + return new static($process->isOutputDisabled(), $input); + } + + /** + * 鍒犻櫎涓存椂鏂囦欢 + */ + private function removeFiles() + { + foreach ($this->files as $filename) { + if (file_exists($filename)) { + @unlink($filename); + } + } + $this->files = []; + } + + /** + * 鍐欏叆鍒 stdin 杈撳叆 + * @param bool $blocking + * @param bool $close + */ + private function write($blocking, $close) + { + if (empty($this->pipes)) { + return; + } + + $this->unblock(); + + $r = null !== $this->input ? ['input' => $this->input] : null; + $w = isset($this->pipes[0]) ? [$this->pipes[0]] : null; + $e = null; + + if (false === $n = @stream_select($r, $w, $e, 0, $blocking ? Process::TIMEOUT_PRECISION * 1E6 : 0)) { + if (!$this->hasSystemCallBeenInterrupted()) { + $this->pipes = []; + } + + return; + } + + if (0 === $n) { + return; + } + + if (null !== $w && 0 < count($r)) { + $data = ''; + while ($dataread = fread($r['input'], self::CHUNK_SIZE)) { + $data .= $dataread; + } + + $this->inputBuffer .= $data; + + if (false === $data || (true === $close && feof($r['input']) && '' === $data)) { + $this->input = null; + } + } + + if (null !== $w && 0 < count($w)) { + while (strlen($this->inputBuffer)) { + $written = fwrite($w[0], $this->inputBuffer, 2 << 18); + if ($written > 0) { + $this->inputBuffer = (string) substr($this->inputBuffer, $written); + } else { + break; + } + } + } + + if ('' === $this->inputBuffer && null === $this->input && isset($this->pipes[0])) { + fclose($this->pipes[0]); + unset($this->pipes[0]); + } + } +} diff --git a/thinkphp/library/think/response/Json.php b/thinkphp/library/think/response/Json.php new file mode 100644 index 000000000..538fc5a0d --- /dev/null +++ b/thinkphp/library/think/response/Json.php @@ -0,0 +1,51 @@ + +// +---------------------------------------------------------------------- + +namespace think\response; + +use think\Response; + +class Json extends Response +{ + // 杈撳嚭鍙傛暟 + protected $options = [ + 'json_encode_param' => JSON_UNESCAPED_UNICODE, + ]; + + protected $contentType = 'application/json'; + + /** + * 澶勭悊鏁版嵁 + * @access protected + * @param mixed $data 瑕佸鐞嗙殑鏁版嵁 + * @return mixed + * @throws \Exception + */ + protected function output($data) + { + try { + // 杩斿洖JSON鏁版嵁鏍煎紡鍒板鎴风 鍖呭惈鐘舵佷俊鎭 + $data = json_encode($data, $this->options['json_encode_param']); + + if ($data === false) { + throw new \InvalidArgumentException(json_last_error_msg()); + } + + return $data; + } catch (\Exception $e) { + if ($e->getPrevious()) { + throw $e->getPrevious(); + } + throw $e; + } + } + +} diff --git a/thinkphp/library/think/response/Jsonp.php b/thinkphp/library/think/response/Jsonp.php new file mode 100644 index 000000000..de8fb3041 --- /dev/null +++ b/thinkphp/library/think/response/Jsonp.php @@ -0,0 +1,58 @@ + +// +---------------------------------------------------------------------- + +namespace think\response; + +use think\Request; +use think\Response; + +class Jsonp extends Response +{ + // 杈撳嚭鍙傛暟 + protected $options = [ + 'var_jsonp_handler' => 'callback', + 'default_jsonp_handler' => 'jsonpReturn', + 'json_encode_param' => JSON_UNESCAPED_UNICODE, + ]; + + protected $contentType = 'application/javascript'; + + /** + * 澶勭悊鏁版嵁 + * @access protected + * @param mixed $data 瑕佸鐞嗙殑鏁版嵁 + * @return mixed + * @throws \Exception + */ + protected function output($data) + { + try { + // 杩斿洖JSON鏁版嵁鏍煎紡鍒板鎴风 鍖呭惈鐘舵佷俊鎭 [褰搖rl_common_param涓篺alse鏃舵槸鏃犳硶鑾峰彇鍒$_GET鐨勬暟鎹殑锛屾晠浣跨敤Request鏉ヨ幏鍙] + $var_jsonp_handler = Request::instance()->param($this->options['var_jsonp_handler'], ""); + $handler = !empty($var_jsonp_handler) ? $var_jsonp_handler : $this->options['default_jsonp_handler']; + + $data = json_encode($data, $this->options['json_encode_param']); + + if ($data === false) { + throw new \InvalidArgumentException(json_last_error_msg()); + } + + $data = $handler . '(' . $data . ');'; + return $data; + } catch (\Exception $e) { + if ($e->getPrevious()) { + throw $e->getPrevious(); + } + throw $e; + } + } + +} diff --git a/thinkphp/library/think/response/Redirect.php b/thinkphp/library/think/response/Redirect.php new file mode 100644 index 000000000..f20027790 --- /dev/null +++ b/thinkphp/library/think/response/Redirect.php @@ -0,0 +1,101 @@ + +// +---------------------------------------------------------------------- + +namespace think\response; + +use think\Request; +use think\Response; +use think\Session; +use think\Url; + +class Redirect extends Response +{ + + protected $options = []; + + // URL鍙傛暟 + protected $params = []; + + public function __construct($data = '', $code = 302, array $header = [], array $options = []) + { + parent::__construct($data, $code, $header, $options); + $this->cacheControl('no-cache,must-revalidate'); + } + + /** + * 澶勭悊鏁版嵁 + * @access protected + * @param mixed $data 瑕佸鐞嗙殑鏁版嵁 + * @return mixed + */ + protected function output($data) + { + $this->header['Location'] = $this->getTargetUrl(); + return; + } + + /** + * 閲嶅畾鍚戜紶鍊硷紙閫氳繃Session锛 + * @access protected + * @param string|array $name 鍙橀噺鍚嶆垨鑰呮暟缁 + * @param mixed $value 鍊 + * @return $this + */ + public function with($name, $value = null) + { + if (is_array($name)) { + foreach ($name as $key => $val) { + Session::flash($key, $val); + } + } else { + Session::flash($name, $value); + } + return $this; + } + + /** + * 鑾峰彇璺宠浆鍦板潃 + * @return string + */ + public function getTargetUrl() + { + return (strpos($this->data, '://') || 0 === strpos($this->data, '/')) ? $this->data : Url::build($this->data, $this->params); + } + + public function params($params = []) + { + $this->params = $params; + return $this; + } + + /** + * 璁颁綇褰撳墠url鍚庤烦杞 + * @return $this + */ + public function remember() + { + Session::set('redirect_url', Request::instance()->url()); + return $this; + } + + /** + * 璺宠浆鍒颁笂娆¤浣忕殑url + * @return $this + */ + public function restore() + { + if (Session::has('redirect_url')) { + $this->data = Session::get('redirect_url'); + Session::delete('redirect_url'); + } + return $this; + } +} diff --git a/thinkphp/library/think/response/View.php b/thinkphp/library/think/response/View.php new file mode 100644 index 000000000..de75515a9 --- /dev/null +++ b/thinkphp/library/think/response/View.php @@ -0,0 +1,89 @@ + +// +---------------------------------------------------------------------- + +namespace think\response; + +use think\Config; +use think\Response; +use think\View as ViewTemplate; + +class View extends Response +{ + // 杈撳嚭鍙傛暟 + protected $options = []; + protected $vars = []; + protected $replace = []; + protected $contentType = 'text/html'; + + /** + * 澶勭悊鏁版嵁 + * @access protected + * @param mixed $data 瑕佸鐞嗙殑鏁版嵁 + * @return mixed + */ + protected function output($data) + { + // 娓叉煋妯℃澘杈撳嚭 + return ViewTemplate::instance(Config::get('template'), Config::get('view_replace_str')) + ->fetch($data, $this->vars, $this->replace); + } + + /** + * 鑾峰彇瑙嗗浘鍙橀噺 + * @access public + * @param string $name 妯℃澘鍙橀噺 + * @return mixed + */ + public function getVars($name = null) + { + if (is_null($name)) { + return $this->vars; + } else { + return isset($this->vars[$name]) ? $this->vars[$name] : null; + } + } + + /** + * 妯℃澘鍙橀噺璧嬪 + * @access public + * @param mixed $name 鍙橀噺鍚 + * @param mixed $value 鍙橀噺鍊 + * @return $this + */ + public function assign($name, $value = '') + { + if (is_array($name)) { + $this->vars = array_merge($this->vars, $name); + return $this; + } else { + $this->vars[$name] = $value; + } + return $this; + } + + /** + * 瑙嗗浘鍐呭鏇挎崲 + * @access public + * @param string|array $content 琚浛鎹㈠唴瀹癸紙鏀寔鎵归噺鏇挎崲锛 + * @param string $replace 鏇挎崲鍐呭 + * @return $this + */ + public function replace($content, $replace = '') + { + if (is_array($content)) { + $this->replace = array_merge($this->replace, $content); + } else { + $this->replace[$content] = $replace; + } + return $this; + } + +} diff --git a/thinkphp/library/think/response/Xml.php b/thinkphp/library/think/response/Xml.php new file mode 100644 index 000000000..ca12e295e --- /dev/null +++ b/thinkphp/library/think/response/Xml.php @@ -0,0 +1,95 @@ + +// +---------------------------------------------------------------------- + +namespace think\response; + +use think\Response; + +class Xml extends Response +{ + // 杈撳嚭鍙傛暟 + protected $options = [ + // 鏍硅妭鐐瑰悕 + 'root_node' => 'think', + // 鏍硅妭鐐瑰睘鎬 + 'root_attr' => '', + //鏁板瓧绱㈠紩鐨勫瓙鑺傜偣鍚 + 'item_node' => 'item', + // 鏁板瓧绱㈠紩瀛愯妭鐐筴ey杞崲鐨勫睘鎬у悕 + 'item_key' => 'id', + // 鏁版嵁缂栫爜 + 'encoding' => 'utf-8', + ]; + + protected $contentType = 'text/xml'; + + /** + * 澶勭悊鏁版嵁 + * @access protected + * @param mixed $data 瑕佸鐞嗙殑鏁版嵁 + * @return mixed + */ + protected function output($data) + { + // XML鏁版嵁杞崲 + return $this->xmlEncode($data, $this->options['root_node'], $this->options['item_node'], $this->options['root_attr'], $this->options['item_key'], $this->options['encoding']); + } + + /** + * XML缂栫爜 + * @param mixed $data 鏁版嵁 + * @param string $root 鏍硅妭鐐瑰悕 + * @param string $item 鏁板瓧绱㈠紩鐨勫瓙鑺傜偣鍚 + * @param string $attr 鏍硅妭鐐瑰睘鎬 + * @param string $id 鏁板瓧绱㈠紩瀛愯妭鐐筴ey杞崲鐨勫睘鎬у悕 + * @param string $encoding 鏁版嵁缂栫爜 + * @return string + */ + protected function xmlEncode($data, $root, $item, $attr, $id, $encoding) + { + if (is_array($attr)) { + $array = []; + foreach ($attr as $key => $value) { + $array[] = "{$key}=\"{$value}\""; + } + $attr = implode(' ', $array); + } + $attr = trim($attr); + $attr = empty($attr) ? '' : " {$attr}"; + $xml = ""; + $xml .= "<{$root}{$attr}>"; + $xml .= $this->dataToXml($data, $item, $id); + $xml .= ""; + return $xml; + } + + /** + * 鏁版嵁XML缂栫爜 + * @param mixed $data 鏁版嵁 + * @param string $item 鏁板瓧绱㈠紩鏃剁殑鑺傜偣鍚嶇О + * @param string $id 鏁板瓧绱㈠紩key杞崲涓虹殑灞炴у悕 + * @return string + */ + protected function dataToXml($data, $item, $id) + { + $xml = $attr = ''; + foreach ($data as $key => $val) { + if (is_numeric($key)) { + $id && $attr = " {$id}=\"{$key}\""; + $key = $item; + } + $xml .= "<{$key}{$attr}>"; + $xml .= (is_array($val) || is_object($val)) ? $this->dataToXml($val, $item, $id) : $val; + $xml .= ""; + } + return $xml; + } +} diff --git a/thinkphp/library/think/session/driver/Memcache.php b/thinkphp/library/think/session/driver/Memcache.php new file mode 100644 index 000000000..0c02e23ec --- /dev/null +++ b/thinkphp/library/think/session/driver/Memcache.php @@ -0,0 +1,118 @@ + +// +---------------------------------------------------------------------- + +namespace think\session\driver; + +use SessionHandler; +use think\Exception; + +class Memcache extends SessionHandler +{ + protected $handler = null; + protected $config = [ + 'host' => '127.0.0.1', // memcache涓绘満 + 'port' => 11211, // memcache绔彛 + 'expire' => 3600, // session鏈夋晥鏈 + 'timeout' => 0, // 杩炴帴瓒呮椂鏃堕棿锛堝崟浣嶏細姣锛 + 'persistent' => true, // 闀胯繛鎺 + 'session_name' => '', // memcache key鍓嶇紑 + ]; + + public function __construct($config = []) + { + $this->config = array_merge($this->config, $config); + } + + /** + * 鎵撳紑Session + * @access public + * @param string $savePath + * @param mixed $sessName + */ + public function open($savePath, $sessName) + { + // 妫娴媝hp鐜 + if (!extension_loaded('memcache')) { + throw new Exception('not support:memcache'); + } + $this->handler = new \Memcache; + // 鏀寔闆嗙兢 + $hosts = explode(',', $this->config['host']); + $ports = explode(',', $this->config['port']); + if (empty($ports[0])) { + $ports[0] = 11211; + } + // 寤虹珛杩炴帴 + foreach ((array) $hosts as $i => $host) { + $port = isset($ports[$i]) ? $ports[$i] : $ports[0]; + $this->config['timeout'] > 0 ? + $this->handler->addServer($host, $port, $this->config['persistent'], 1, $this->config['timeout']) : + $this->handler->addServer($host, $port, $this->config['persistent'], 1); + } + return true; + } + + /** + * 鍏抽棴Session + * @access public + */ + public function close() + { + $this->gc(ini_get('session.gc_maxlifetime')); + $this->handler->close(); + $this->handler = null; + return true; + } + + /** + * 璇诲彇Session + * @access public + * @param string $sessID + */ + public function read($sessID) + { + return (string) $this->handler->get($this->config['session_name'] . $sessID); + } + + /** + * 鍐欏叆Session + * @access public + * @param string $sessID + * @param String $sessData + * @return bool + */ + public function write($sessID, $sessData) + { + return $this->handler->set($this->config['session_name'] . $sessID, $sessData, 0, $this->config['expire']); + } + + /** + * 鍒犻櫎Session + * @access public + * @param string $sessID + * @return bool + */ + public function destroy($sessID) + { + return $this->handler->delete($this->config['session_name'] . $sessID); + } + + /** + * Session 鍨冨溇鍥炴敹 + * @access public + * @param string $sessMaxLifeTime + * @return true + */ + public function gc($sessMaxLifeTime) + { + return true; + } +} diff --git a/thinkphp/library/think/session/driver/Memcached.php b/thinkphp/library/think/session/driver/Memcached.php new file mode 100644 index 000000000..bcaf8a1b5 --- /dev/null +++ b/thinkphp/library/think/session/driver/Memcached.php @@ -0,0 +1,126 @@ + +// +---------------------------------------------------------------------- + +namespace think\session\driver; + +use SessionHandler; +use think\Exception; + +class Memcached extends SessionHandler +{ + protected $handler = null; + protected $config = [ + 'host' => '127.0.0.1', // memcache涓绘満 + 'port' => 11211, // memcache绔彛 + 'expire' => 3600, // session鏈夋晥鏈 + 'timeout' => 0, // 杩炴帴瓒呮椂鏃堕棿锛堝崟浣嶏細姣锛 + 'session_name' => '', // memcache key鍓嶇紑 + 'username' => '', //璐﹀彿 + 'password' => '', //瀵嗙爜 + ]; + + public function __construct($config = []) + { + $this->config = array_merge($this->config, $config); + } + + /** + * 鎵撳紑Session + * @access public + * @param string $savePath + * @param mixed $sessName + */ + public function open($savePath, $sessName) + { + // 妫娴媝hp鐜 + if (!extension_loaded('memcached')) { + throw new Exception('not support:memcached'); + } + $this->handler = new \Memcached; + // 璁剧疆杩炴帴瓒呮椂鏃堕棿锛堝崟浣嶏細姣锛 + if ($this->config['timeout'] > 0) { + $this->handler->setOption(\Memcached::OPT_CONNECT_TIMEOUT, $this->config['timeout']); + } + // 鏀寔闆嗙兢 + $hosts = explode(',', $this->config['host']); + $ports = explode(',', $this->config['port']); + if (empty($ports[0])) { + $ports[0] = 11211; + } + // 寤虹珛杩炴帴 + $servers = []; + foreach ((array) $hosts as $i => $host) { + $servers[] = [$host, (isset($ports[$i]) ? $ports[$i] : $ports[0]), 1]; + } + $this->handler->addServers($servers); + if ('' != $this->config['username']) { + $this->handler->setOption(\Memcached::OPT_BINARY_PROTOCOL, true); + $this->handler->setSaslAuthData($this->config['username'], $this->config['password']); + } + return true; + } + + /** + * 鍏抽棴Session + * @access public + */ + public function close() + { + $this->gc(ini_get('session.gc_maxlifetime')); + $this->handler->quit(); + $this->handler = null; + return true; + } + + /** + * 璇诲彇Session + * @access public + * @param string $sessID + */ + public function read($sessID) + { + return (string) $this->handler->get($this->config['session_name'] . $sessID); + } + + /** + * 鍐欏叆Session + * @access public + * @param string $sessID + * @param String $sessData + * @return bool + */ + public function write($sessID, $sessData) + { + return $this->handler->set($this->config['session_name'] . $sessID, $sessData, $this->config['expire']); + } + + /** + * 鍒犻櫎Session + * @access public + * @param string $sessID + * @return bool + */ + public function destroy($sessID) + { + return $this->handler->delete($this->config['session_name'] . $sessID); + } + + /** + * Session 鍨冨溇鍥炴敹 + * @access public + * @param string $sessMaxLifeTime + * @return true + */ + public function gc($sessMaxLifeTime) + { + return true; + } +} diff --git a/thinkphp/library/think/session/driver/Redis.php b/thinkphp/library/think/session/driver/Redis.php new file mode 100644 index 000000000..a4c2b54a6 --- /dev/null +++ b/thinkphp/library/think/session/driver/Redis.php @@ -0,0 +1,128 @@ + +// +---------------------------------------------------------------------- + +namespace think\session\driver; + +use SessionHandler; +use think\Exception; + +class Redis extends SessionHandler +{ + /** @var \Redis */ + protected $handler = null; + protected $config = [ + 'host' => '127.0.0.1', // redis涓绘満 + 'port' => 6379, // redis绔彛 + 'password' => '', // 瀵嗙爜 + 'select' => 0, // 鎿嶄綔搴 + 'expire' => 3600, // 鏈夋晥鏈(绉) + 'timeout' => 0, // 瓒呮椂鏃堕棿(绉) + 'persistent' => true, // 鏄惁闀胯繛鎺 + 'session_name' => '', // sessionkey鍓嶇紑 + ]; + + public function __construct($config = []) + { + $this->config = array_merge($this->config, $config); + } + + /** + * 鎵撳紑Session + * @access public + * @param string $savePath + * @param mixed $sessName + * @return bool + * @throws Exception + */ + public function open($savePath, $sessName) + { + // 妫娴媝hp鐜 + if (!extension_loaded('redis')) { + throw new Exception('not support:redis'); + } + $this->handler = new \Redis; + + // 寤虹珛杩炴帴 + $func = $this->config['persistent'] ? 'pconnect' : 'connect'; + $this->handler->$func($this->config['host'], $this->config['port'], $this->config['timeout']); + + if ('' != $this->config['password']) { + $this->handler->auth($this->config['password']); + } + + if (0 != $this->config['select']) { + $this->handler->select($this->config['select']); + } + + return true; + } + + /** + * 鍏抽棴Session + * @access public + */ + public function close() + { + $this->gc(ini_get('session.gc_maxlifetime')); + $this->handler->close(); + $this->handler = null; + return true; + } + + /** + * 璇诲彇Session + * @access public + * @param string $sessID + * @return string + */ + public function read($sessID) + { + return (string) $this->handler->get($this->config['session_name'] . $sessID); + } + + /** + * 鍐欏叆Session + * @access public + * @param string $sessID + * @param String $sessData + * @return bool + */ + public function write($sessID, $sessData) + { + if ($this->config['expire'] > 0) { + return $this->handler->setex($this->config['session_name'] . $sessID, $this->config['expire'], $sessData); + } else { + return $this->handler->set($this->config['session_name'] . $sessID, $sessData); + } + } + + /** + * 鍒犻櫎Session + * @access public + * @param string $sessID + * @return bool + */ + public function destroy($sessID) + { + return $this->handler->delete($this->config['session_name'] . $sessID) > 0; + } + + /** + * Session 鍨冨溇鍥炴敹 + * @access public + * @param string $sessMaxLifeTime + * @return bool + */ + public function gc($sessMaxLifeTime) + { + return true; + } +} diff --git a/thinkphp/library/think/template/TagLib.php b/thinkphp/library/think/template/TagLib.php new file mode 100644 index 000000000..b33251916 --- /dev/null +++ b/thinkphp/library/think/template/TagLib.php @@ -0,0 +1,334 @@ + +// +---------------------------------------------------------------------- + +namespace think\template; + +use think\Exception; + +/** + * ThinkPHP鏍囩搴揟agLib瑙f瀽鍩虹被 + * @category Think + * @package Think + * @subpackage Template + * @author liu21st + */ +class TagLib +{ + + /** + * 鏍囩搴撳畾涔塜ML鏂囦欢 + * @var string + * @access protected + */ + protected $xml = ''; + protected $tags = []; // 鏍囩瀹氫箟 + /** + * 鏍囩搴撳悕绉 + * @var string + * @access protected + */ + protected $tagLib = ''; + + /** + * 鏍囩搴撴爣绛惧垪琛 + * @var array + * @access protected + */ + protected $tagList = []; + + /** + * 鏍囩搴撳垎鏋愭暟缁 + * @var array + * @access protected + */ + protected $parse = []; + + /** + * 鏍囩搴撴槸鍚︽湁鏁 + * @var bool + * @access protected + */ + protected $valid = false; + + /** + * 褰撳墠妯℃澘瀵硅薄 + * @var object + * @access protected + */ + protected $tpl; + + protected $comparison = [' nheq ' => ' !== ', ' heq ' => ' === ', ' neq ' => ' != ', ' eq ' => ' == ', ' egt ' => ' >= ', ' gt ' => ' > ', ' elt ' => ' <= ', ' lt ' => ' < ']; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param \stdClass $template 妯℃澘寮曟搸瀵硅薄 + */ + public function __construct($template) + { + $this->tpl = $template; + } + + /** + * 鎸夌鏍囧簱鏇挎崲椤甸潰涓殑鏍囩 + * @access public + * @param string $content 妯℃澘鍐呭 + * @param string $lib 鏍囩搴撳悕 + * @return void + */ + public function parseTag(&$content, $lib = '') + { + $tags = []; + $lib = $lib ? strtolower($lib) . ':' : ''; + foreach ($this->tags as $name => $val) { + $close = !isset($val['close']) || $val['close'] ? 1 : 0; + $tags[$close][$lib . $name] = $name; + if (isset($val['alias'])) { + // 鍒悕璁剧疆 + $array = (array) $val['alias']; + foreach (explode(',', $array[0]) as $v) { + $tags[$close][$lib . $v] = $name; + } + } + } + + // 闂悎鏍囩 + if (!empty($tags[1])) { + $nodes = []; + $regex = $this->getRegex(array_keys($tags[1]), 1); + if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER | PREG_OFFSET_CAPTURE)) { + $right = []; + foreach ($matches as $match) { + if ('' == $match[1][0]) { + $name = strtolower($match[2][0]); + // 濡傛灉鏈夋病闂悎鐨勬爣绛惧ご鍒欏彇鍑烘渶鍚庝竴涓 + if (!empty($right[$name])) { + // $match[0][1]涓烘爣绛剧粨鏉熺鍦ㄦā鏉夸腑鐨勪綅缃 + $nodes[$match[0][1]] = [ + 'name' => $name, + 'begin' => array_pop($right[$name]), // 鏍囩寮濮嬬 + 'end' => $match[0], // 鏍囩缁撴潫绗 + ]; + } + } else { + // 鏍囩澶村帇鍏ユ爤 + $right[strtolower($match[1][0])][] = $match[0]; + } + } + unset($right, $matches); + // 鎸夋爣绛惧湪妯℃澘涓殑浣嶇疆浠庡悗鍚戝墠鎺掑簭 + krsort($nodes); + } + + $break = ''; + if ($nodes) { + $beginArray = []; + // 鏍囩鏇挎崲 浠庡悗鍚戝墠 + foreach ($nodes as $pos => $node) { + // 瀵瑰簲鐨勬爣绛惧悕 + $name = $tags[1][$node['name']]; + $alias = $lib . $name != $node['name'] ? ($lib ? strstr($node['name'], $lib) : $node['name']) : ''; + // 瑙f瀽鏍囩灞炴 + $attrs = $this->parseAttr($node['begin'][0], $name, $alias); + $method = 'tag' . $name; + // 璇诲彇鏍囩搴撲腑瀵瑰簲鐨勬爣绛惧唴瀹 replace[0]鐢ㄦ潵鏇挎崲鏍囩澶达紝replace[1]鐢ㄦ潵鏇挎崲鏍囩灏 + $replace = explode($break, $this->$method($attrs, $break)); + if (count($replace) > 1) { + while ($beginArray) { + $begin = end($beginArray); + // 鍒ゆ柇褰撳墠鏍囩灏剧殑浣嶇疆鏄惁鍦ㄦ爤涓渶鍚庝竴涓爣绛惧ご鐨勫悗闈紝鏄垯涓哄瓙鏍囩 + if ($node['end'][1] > $begin['pos']) { + break; + } else { + // 涓嶄负瀛愭爣绛炬椂锛屽彇鍑烘爤涓渶鍚庝竴涓爣绛惧ご + $begin = array_pop($beginArray); + // 鏇挎崲鏍囩澶撮儴 + $content = substr_replace($content, $begin['str'], $begin['pos'], $begin['len']); + } + } + // 鏇挎崲鏍囩灏鹃儴 + $content = substr_replace($content, $replace[1], $node['end'][1], strlen($node['end'][0])); + // 鎶婃爣绛惧ご鍘嬪叆鏍 + $beginArray[] = ['pos' => $node['begin'][1], 'len' => strlen($node['begin'][0]), 'str' => $replace[0]]; + } + } + while ($beginArray) { + $begin = array_pop($beginArray); + // 鏇挎崲鏍囩澶撮儴 + $content = substr_replace($content, $begin['str'], $begin['pos'], $begin['len']); + } + } + } + // 鑷棴鍚堟爣绛 + if (!empty($tags[0])) { + $regex = $this->getRegex(array_keys($tags[0]), 0); + $content = preg_replace_callback($regex, function ($matches) use (&$tags, &$lib) { + // 瀵瑰簲鐨勬爣绛惧悕 + $name = $tags[0][strtolower($matches[1])]; + $alias = $lib . $name != $matches[1] ? ($lib ? strstr($matches[1], $lib) : $matches[1]) : ''; + // 瑙f瀽鏍囩灞炴 + $attrs = $this->parseAttr($matches[0], $name, $alias); + $method = 'tag' . $name; + return $this->$method($attrs, ''); + }, $content); + } + return; + } + + /** + * 鎸夋爣绛剧敓鎴愭鍒 + * @access private + * @param array|string $tags 鏍囩鍚 + * @param boolean $close 鏄惁涓洪棴鍚堟爣绛 + * @return string + */ + private function getRegex($tags, $close) + { + $begin = $this->tpl->config('taglib_begin'); + $end = $this->tpl->config('taglib_end'); + $single = strlen(ltrim($begin, '\\')) == 1 && strlen(ltrim($end, '\\')) == 1 ? true : false; + $tagName = is_array($tags) ? implode('|', $tags) : $tags; + if ($single) { + if ($close) { + // 濡傛灉鏄棴鍚堟爣绛 + $regex = $begin . '(?:(' . $tagName . ')\b(?>[^' . $end . ']*)|\/(' . $tagName . '))' . $end; + } else { + $regex = $begin . '(' . $tagName . ')\b(?>[^' . $end . ']*)' . $end; + } + } else { + if ($close) { + // 濡傛灉鏄棴鍚堟爣绛 + $regex = $begin . '(?:(' . $tagName . ')\b(?>(?:(?!' . $end . ').)*)|\/(' . $tagName . '))' . $end; + } else { + $regex = $begin . '(' . $tagName . ')\b(?>(?:(?!' . $end . ').)*)' . $end; + } + } + return '/' . $regex . '/is'; + } + + /** + * 鍒嗘瀽鏍囩灞炴 姝e垯鏂瑰紡 + * @access public + * @param string $str 鏍囩灞炴у瓧绗︿覆 + * @param string $name 鏍囩鍚 + * @param string $alias 鍒悕 + * @return array + */ + public function parseAttr($str, $name, $alias = '') + { + $regex = '/\s+(?>(?P[\w-]+)\s*)=(?>\s*)([\"\'])(?P(?:(?!\\2).)*)\\2/is'; + $result = []; + if (preg_match_all($regex, $str, $matches)) { + foreach ($matches['name'] as $key => $val) { + $result[$val] = $matches['value'][$key]; + } + if (!isset($this->tags[$name])) { + // 妫娴嬫槸鍚﹀瓨鍦ㄥ埆鍚嶅畾涔 + foreach ($this->tags as $key => $val) { + if (isset($val['alias'])) { + $array = (array) $val['alias']; + if (in_array($name, explode(',', $array[0]))) { + $tag = $val; + $type = !empty($array[1]) ? $array[1] : 'type'; + $result[$type] = $name; + break; + } + } + } + } else { + $tag = $this->tags[$name]; + // 璁剧疆浜嗘爣绛惧埆鍚 + if (!empty($alias) && isset($tag['alias'])) { + $type = !empty($tag['alias'][1]) ? $tag['alias'][1] : 'type'; + $result[$type] = $alias; + } + } + if (!empty($tag['must'])) { + $must = explode(',', $tag['must']); + foreach ($must as $name) { + if (!isset($result[$name])) { + throw new Exception('tag attr must:' . $name); + } + } + } + } else { + // 鍏佽鐩存帴浣跨敤琛ㄨ揪寮忕殑鏍囩 + if (!empty($this->tags[$name]['expression'])) { + static $_taglibs; + if (!isset($_taglibs[$name])) { + $_taglibs[$name][0] = strlen(ltrim($this->tpl->config('taglib_begin'), '\\') . $name); + $_taglibs[$name][1] = strlen(ltrim($this->tpl->config('taglib_end'), '\\')); + } + $result['expression'] = substr($str, $_taglibs[$name][0], -$_taglibs[$name][1]); + // 娓呴櫎鑷棴鍚堟爣绛惧熬閮/ + $result['expression'] = rtrim($result['expression'], '/'); + $result['expression'] = trim($result['expression']); + } elseif (empty($this->tags[$name]) || !empty($this->tags[$name]['attr'])) { + throw new Exception('tag error:' . $name); + } + } + return $result; + } + + /** + * 瑙f瀽鏉′欢琛ㄨ揪寮 + * @access public + * @param string $condition 琛ㄨ揪寮忔爣绛惧唴瀹 + * @return string + */ + public function parseCondition($condition) + { + if (strpos($condition, ':')) { + $condition = ' ' . substr(strstr($condition, ':'), 1); + } + $condition = str_ireplace(array_keys($this->comparison), array_values($this->comparison), $condition); + $this->tpl->parseVar($condition); + // $this->tpl->parseVarFunction($condition); // XXX: 姝ゅ彞鑳借В鏋愯〃杈惧紡涓敤|鍒嗛殧鐨勫嚱鏁帮紝浣嗚〃杈惧紡涓鏋滄湁|銆亅|杩欐牱鐨勯昏緫杩愮畻灏变骇鐢熶簡姝у紓 + return $condition; + } + + /** + * 鑷姩璇嗗埆鏋勫缓鍙橀噺 + * @access public + * @param string $name 鍙橀噺鎻忚堪 + * @return string + */ + public function autoBuildVar(&$name) + { + $flag = substr($name, 0, 1); + if (':' == $flag) { + // 浠:寮澶翠负鍑芥暟璋冪敤锛岃В鏋愬墠鍘绘帀: + $name = substr($name, 1); + } elseif ('$' != $flag && preg_match('/[a-zA-Z_]/', $flag)) { + // XXX: 杩欏彞鐨勫啓娉曞彲鑳借繕闇瑕佹敼杩 + // 甯搁噺涓嶉渶瑕佽В鏋 + if (defined($name)) { + return $name; + } + // 涓嶄互$寮澶村苟涓斾篃涓嶆槸甯搁噺锛岃嚜鍔ㄨˉ涓$鍓嶇紑 + $name = '$' . $name; + } + $this->tpl->parseVar($name); + $this->tpl->parseVarFunction($name); + return $name; + } + + /** + * 鑾峰彇鏍囩鍒楄〃 + * @access public + * @return array + */ + // 鑾峰彇鏍囩瀹氫箟 + public function getTags() + { + return $this->tags; + } +} diff --git a/thinkphp/library/think/template/driver/File.php b/thinkphp/library/think/template/driver/File.php new file mode 100644 index 000000000..b27e72659 --- /dev/null +++ b/thinkphp/library/think/template/driver/File.php @@ -0,0 +1,71 @@ + +// +---------------------------------------------------------------------- + +namespace think\template\driver; + +use think\Exception; + +class File +{ + /** + * 鍐欏叆缂栬瘧缂撳瓨 + * @param string $cacheFile 缂撳瓨鐨勬枃浠跺悕 + * @param string $content 缂撳瓨鐨勫唴瀹 + * @return void|array + */ + public function write($cacheFile, $content) + { + // 妫娴嬫ā鏉跨洰褰 + $dir = dirname($cacheFile); + if (!is_dir($dir)) { + mkdir($dir, 0755, true); + } + // 鐢熸垚妯℃澘缂撳瓨鏂囦欢 + if (false === file_put_contents($cacheFile, $content)) { + throw new Exception('cache write error:' . $cacheFile, 11602); + } + } + + /** + * 璇诲彇缂栬瘧缂栬瘧 + * @param string $cacheFile 缂撳瓨鐨勬枃浠跺悕 + * @param array $vars 鍙橀噺鏁扮粍 + * @return void + */ + public function read($cacheFile, $vars = []) + { + if (!empty($vars) && is_array($vars)) { + // 妯℃澘闃靛垪鍙橀噺鍒嗚В鎴愪负鐙珛鍙橀噺 + extract($vars, EXTR_OVERWRITE); + } + //杞藉叆妯$増缂撳瓨鏂囦欢 + include $cacheFile; + } + + /** + * 妫鏌ョ紪璇戠紦瀛樻槸鍚︽湁鏁 + * @param string $cacheFile 缂撳瓨鐨勬枃浠跺悕 + * @param int $cacheTime 缂撳瓨鏃堕棿 + * @return boolean + */ + public function check($cacheFile, $cacheTime) + { + // 缂撳瓨鏂囦欢涓嶅瓨鍦, 鐩存帴杩斿洖false + if (!file_exists($cacheFile)) { + return false; + } + if (0 != $cacheTime && $_SERVER['REQUEST_TIME'] > filemtime($cacheFile) + $cacheTime) { + // 缂撳瓨鏄惁鍦ㄦ湁鏁堟湡 + return false; + } + return true; + } +} diff --git a/thinkphp/library/think/template/taglib/Cx.php b/thinkphp/library/think/template/taglib/Cx.php new file mode 100644 index 000000000..af7a54c89 --- /dev/null +++ b/thinkphp/library/think/template/taglib/Cx.php @@ -0,0 +1,673 @@ + +// +---------------------------------------------------------------------- + +namespace think\template\taglib; + +use think\template\TagLib; + +/** + * CX鏍囩搴撹В鏋愮被 + * @category Think + * @package Think + * @subpackage Driver.Taglib + * @author liu21st + */ +class Cx extends Taglib +{ + + // 鏍囩瀹氫箟 + protected $tags = [ + // 鏍囩瀹氫箟锛 attr 灞炴у垪琛 close 鏄惁闂悎锛0 鎴栬1 榛樿1锛 alias 鏍囩鍒悕 level 宓屽灞傛 + 'php' => ['attr' => ''], + 'volist' => ['attr' => 'name,id,offset,length,key,mod', 'alias' => 'iterate'], + 'foreach' => ['attr' => 'name,id,item,key,offset,length,mod', 'expression' => true], + 'if' => ['attr' => 'condition', 'expression' => true], + 'elseif' => ['attr' => 'condition', 'close' => 0, 'expression' => true], + 'else' => ['attr' => '', 'close' => 0], + 'switch' => ['attr' => 'name', 'expression' => true], + 'case' => ['attr' => 'value,break', 'expression' => true], + 'default' => ['attr' => '', 'close' => 0], + 'compare' => ['attr' => 'name,value,type', 'alias' => ['eq,equal,notequal,neq,gt,lt,egt,elt,heq,nheq', 'type']], + 'range' => ['attr' => 'name,value,type', 'alias' => ['in,notin,between,notbetween', 'type']], + 'empty' => ['attr' => 'name'], + 'notempty' => ['attr' => 'name'], + 'present' => ['attr' => 'name'], + 'notpresent' => ['attr' => 'name'], + 'defined' => ['attr' => 'name'], + 'notdefined' => ['attr' => 'name'], + 'load' => ['attr' => 'file,href,type,value,basepath', 'close' => 0, 'alias' => ['import,css,js', 'type']], + 'assign' => ['attr' => 'name,value', 'close' => 0], + 'define' => ['attr' => 'name,value', 'close' => 0], + 'for' => ['attr' => 'start,end,name,comparison,step'], + 'url' => ['attr' => 'link,vars,suffix,domain', 'close' => 0, 'expression' => true], + 'function' => ['attr' => 'name,vars,use,call'], + ]; + + /** + * php鏍囩瑙f瀽 + * 鏍煎紡锛 + * {php}echo $name{/php} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagPhp($tag, $content) + { + $parseStr = ''; + return $parseStr; + } + + /** + * volist鏍囩瑙f瀽 寰幆杈撳嚭鏁版嵁闆 + * 鏍煎紡锛 + * {volist name="userList" id="user" empty=""} + * {user.username} + * {user.email} + * {/volist} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string|void + */ + public function tagVolist($tag, $content) + { + $name = $tag['name']; + $id = $tag['id']; + $empty = isset($tag['empty']) ? $tag['empty'] : ''; + $key = !empty($tag['key']) ? $tag['key'] : 'i'; + $mod = isset($tag['mod']) ? $tag['mod'] : '2'; + $offset = !empty($tag['offset']) && is_numeric($tag['offset']) ? intval($tag['offset']) : 0; + $length = !empty($tag['length']) && is_numeric($tag['length']) ? intval($tag['length']) : 'null'; + // 鍏佽浣跨敤鍑芥暟璁惧畾鏁版嵁闆 {$vo.name} + $parseStr = 'autoBuildVar($name); + $parseStr .= '$_result=' . $name . ';'; + $name = '$_result'; + } else { + $name = $this->autoBuildVar($name); + } + + $parseStr .= 'if(is_array(' . $name . ') || ' . $name . ' instanceof \think\Collection || ' . $name . ' instanceof \think\Paginator): $' . $key . ' = 0;'; + // 璁剧疆浜嗚緭鍑烘暟缁勯暱搴 + if (0 != $offset || 'null' != $length) { + $parseStr .= '$__LIST__ = is_array(' . $name . ') ? array_slice(' . $name . ',' . $offset . ',' . $length . ', true) : ' . $name . '->slice(' . $offset . ',' . $length . ', true); '; + } else { + $parseStr .= ' $__LIST__ = ' . $name . ';'; + } + $parseStr .= 'if( count($__LIST__)==0 ) : echo "' . $empty . '" ;'; + $parseStr .= 'else: '; + $parseStr .= 'foreach($__LIST__ as $key=>$' . $id . '): '; + $parseStr .= '$mod = ($' . $key . ' % ' . $mod . ' );'; + $parseStr .= '++$' . $key . ';?>'; + $parseStr .= $content; + $parseStr .= ''; + + if (!empty($parseStr)) { + return $parseStr; + } + return; + } + + /** + * foreach鏍囩瑙f瀽 寰幆杈撳嚭鏁版嵁闆 + * 鏍煎紡锛 + * {foreach name="userList" id="user" key="key" index="i" mod="2" offset="3" length="5" empty=""} + * {user.username} + * {/foreach} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string|void + */ + public function tagForeach($tag, $content) + { + // 鐩存帴浣跨敤琛ㄨ揪寮 + if (!empty($tag['expression'])) { + $expression = ltrim(rtrim($tag['expression'], ')'), '('); + $expression = $this->autoBuildVar($expression); + $parseStr = ''; + $parseStr .= $content; + $parseStr .= ''; + return $parseStr; + } + $name = $tag['name']; + $key = !empty($tag['key']) ? $tag['key'] : 'key'; + $item = !empty($tag['id']) ? $tag['id'] : $tag['item']; + $empty = isset($tag['empty']) ? $tag['empty'] : ''; + $offset = !empty($tag['offset']) && is_numeric($tag['offset']) ? intval($tag['offset']) : 0; + $length = !empty($tag['length']) && is_numeric($tag['length']) ? intval($tag['length']) : 'null'; + + $parseStr = 'autoBuildVar($name); + $parseStr .= $var . '=' . $name . '; '; + $name = $var; + } else { + $name = $this->autoBuildVar($name); + } + $parseStr .= 'if(is_array(' . $name . ') || ' . $name . ' instanceof \think\Collection || ' . $name . ' instanceof \think\Paginator): '; + // 璁剧疆浜嗚緭鍑烘暟缁勯暱搴 + if (0 != $offset || 'null' != $length) { + if (!isset($var)) { + $var = '$_' . uniqid(); + } + $parseStr .= $var . ' = is_array(' . $name . ') ? array_slice(' . $name . ',' . $offset . ',' . $length . ', true) : ' . $name . '->slice(' . $offset . ',' . $length . ', true); '; + } else { + $var = &$name; + } + + $parseStr .= 'if( count(' . $var . ')==0 ) : echo "' . $empty . '" ;'; + $parseStr .= 'else: '; + + // 璁剧疆浜嗙储寮曢」 + if (isset($tag['index'])) { + $index = $tag['index']; + $parseStr .= '$' . $index . '=0; '; + } + $parseStr .= 'foreach(' . $var . ' as $' . $key . '=>$' . $item . '): '; + // 璁剧疆浜嗙储寮曢」 + if (isset($tag['index'])) { + $index = $tag['index']; + if (isset($tag['mod'])) { + $mod = (int) $tag['mod']; + $parseStr .= '$mod = ($' . $index . ' % ' . $mod . '); '; + } + $parseStr .= '++$' . $index . '; '; + } + $parseStr .= '?>'; + // 寰幆浣撲腑鐨勫唴瀹 + $parseStr .= $content; + $parseStr .= ''; + + if (!empty($parseStr)) { + return $parseStr; + } + return; + } + + /** + * if鏍囩瑙f瀽 + * 鏍煎紡锛 + * {if condition=" $a eq 1"} + * {elseif condition="$a eq 2" /} + * {else /} + * {/if} + * 琛ㄨ揪寮忔敮鎸 eq neq gt egt lt elt == > >= < <= or and || && + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagIf($tag, $content) + { + $condition = !empty($tag['expression']) ? $tag['expression'] : $tag['condition']; + $condition = $this->parseCondition($condition); + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * elseif鏍囩瑙f瀽 + * 鏍煎紡锛氳if鏍囩 + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagElseif($tag, $content) + { + $condition = !empty($tag['expression']) ? $tag['expression'] : $tag['condition']; + $condition = $this->parseCondition($condition); + $parseStr = ''; + return $parseStr; + } + + /** + * else鏍囩瑙f瀽 + * 鏍煎紡锛氳if鏍囩 + * @access public + * @param array $tag 鏍囩灞炴 + * @return string + */ + public function tagElse($tag) + { + $parseStr = ''; + return $parseStr; + } + + /** + * switch鏍囩瑙f瀽 + * 鏍煎紡锛 + * {switch name="a.name"} + * {case value="1" break="false"}1{/case} + * {case value="2" }2{/case} + * {default /}other + * {/switch} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagSwitch($tag, $content) + { + $name = !empty($tag['expression']) ? $tag['expression'] : $tag['name']; + $name = $this->autoBuildVar($name); + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * case鏍囩瑙f瀽 闇瑕侀厤鍚坰witch鎵嶆湁鏁 + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagCase($tag, $content) + { + $value = !empty($tag['expression']) ? $tag['expression'] : $tag['value']; + $flag = substr($value, 0, 1); + if ('$' == $flag || ':' == $flag) { + $value = $this->autoBuildVar($value); + $value = 'case ' . $value . ':'; + } elseif (strpos($value, '|')) { + $values = explode('|', $value); + $value = ''; + foreach ($values as $val) { + $value .= 'case "' . addslashes($val) . '":'; + } + } else { + $value = 'case "' . $value . '":'; + } + $parseStr = '' . $content; + $isBreak = isset($tag['break']) ? $tag['break'] : ''; + if ('' == $isBreak || $isBreak) { + $parseStr .= ''; + } + return $parseStr; + } + + /** + * default鏍囩瑙f瀽 闇瑕侀厤鍚坰witch鎵嶆湁鏁 + * 浣跨敤锛 {default /}ddfdf + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagDefault($tag) + { + $parseStr = ''; + return $parseStr; + } + + /** + * compare鏍囩瑙f瀽 + * 鐢ㄤ簬鍊肩殑姣旇緝 鏀寔 eq neq gt lt egt elt heq nheq 榛樿鏄痚q + * 鏍煎紡锛 {compare name="" type="eq" value="" }content{/compare} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagCompare($tag, $content) + { + $name = $tag['name']; + $value = $tag['value']; + $type = isset($tag['type']) ? $tag['type'] : 'eq'; // 姣旇緝绫诲瀷 + $name = $this->autoBuildVar($name); + $flag = substr($value, 0, 1); + if ('$' == $flag || ':' == $flag) { + $value = $this->autoBuildVar($value); + } else { + $value = '\'' . $value . '\''; + } + switch ($type) { + case 'equal': + $type = 'eq'; + break; + case 'notequal': + $type = 'neq'; + break; + } + $type = $this->parseCondition(' ' . $type . ' '); + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * range鏍囩瑙f瀽 + * 濡傛灉鏌愪釜鍙橀噺瀛樺湪浜庢煇涓寖鍥 鍒欒緭鍑哄唴瀹 type= in 琛ㄧず鍦ㄨ寖鍥村唴 鍚﹀垯琛ㄧず鍦ㄨ寖鍥村 + * 鏍煎紡锛 {range name="var|function" value="val" type='in|notin' }content{/range} + * example: {range name="a" value="1,2,3" type='in' }content{/range} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagRange($tag, $content) + { + $name = $tag['name']; + $value = $tag['value']; + $type = isset($tag['type']) ? $tag['type'] : 'in'; // 姣旇緝绫诲瀷 + + $name = $this->autoBuildVar($name); + $flag = substr($value, 0, 1); + if ('$' == $flag || ':' == $flag) { + $value = $this->autoBuildVar($value); + $str = 'is_array(' . $value . ')?' . $value . ':explode(\',\',' . $value . ')'; + } else { + $value = '"' . $value . '"'; + $str = 'explode(\',\',' . $value . ')'; + } + if ('between' == $type) { + $parseStr = '= $_RANGE_VAR_[0] && ' . $name . '<= $_RANGE_VAR_[1]):?>' . $content . ''; + } elseif ('notbetween' == $type) { + $parseStr = '$_RANGE_VAR_[1]):?>' . $content . ''; + } else { + $fun = ('in' == $type) ? 'in_array' : '!in_array'; + $parseStr = '' . $content . ''; + } + return $parseStr; + } + + /** + * present鏍囩瑙f瀽 + * 濡傛灉鏌愪釜鍙橀噺宸茬粡璁剧疆 鍒欒緭鍑哄唴瀹 + * 鏍煎紡锛 {present name="" }content{/present} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagPresent($tag, $content) + { + $name = $tag['name']; + $name = $this->autoBuildVar($name); + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * notpresent鏍囩瑙f瀽 + * 濡傛灉鏌愪釜鍙橀噺娌℃湁璁剧疆锛屽垯杈撳嚭鍐呭 + * 鏍煎紡锛 {notpresent name="" }content{/notpresent} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagNotpresent($tag, $content) + { + $name = $tag['name']; + $name = $this->autoBuildVar($name); + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * empty鏍囩瑙f瀽 + * 濡傛灉鏌愪釜鍙橀噺涓篹mpty 鍒欒緭鍑哄唴瀹 + * 鏍煎紡锛 {empty name="" }content{/empty} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagEmpty($tag, $content) + { + $name = $tag['name']; + $name = $this->autoBuildVar($name); + $parseStr = 'isEmpty())): ?>' . $content . ''; + return $parseStr; + } + + /** + * notempty鏍囩瑙f瀽 + * 濡傛灉鏌愪釜鍙橀噺涓嶄负empty 鍒欒緭鍑哄唴瀹 + * 鏍煎紡锛 {notempty name="" }content{/notempty} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagNotempty($tag, $content) + { + $name = $tag['name']; + $name = $this->autoBuildVar($name); + $parseStr = 'isEmpty()))): ?>' . $content . ''; + return $parseStr; + } + + /** + * 鍒ゆ柇鏄惁宸茬粡瀹氫箟浜嗚甯搁噺 + * {defined name='TXT'}宸插畾涔墈/defined} + * @param array $tag + * @param string $content + * @return string + */ + public function tagDefined($tag, $content) + { + $name = $tag['name']; + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * 鍒ゆ柇鏄惁娌℃湁瀹氫箟浜嗚甯搁噺 + * {notdefined name='TXT'}宸插畾涔墈/notdefined} + * @param array $tag + * @param string $content + * @return string + */ + public function tagNotdefined($tag, $content) + { + $name = $tag['name']; + $parseStr = '' . $content . ''; + return $parseStr; + } + + /** + * load 鏍囩瑙f瀽 {load file="/static/js/base.js" /} + * 鏍煎紡锛歿load file="/static/css/base.css" /} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagLoad($tag, $content) + { + $file = isset($tag['file']) ? $tag['file'] : $tag['href']; + $type = isset($tag['type']) ? strtolower($tag['type']) : ''; + $parseStr = ''; + $endStr = ''; + // 鍒ゆ柇鏄惁瀛樺湪鍔犺浇鏉′欢 鍏佽浣跨敤鍑芥暟鍒ゆ柇(榛樿涓篿sset) + if (isset($tag['value'])) { + $name = $tag['value']; + $name = $this->autoBuildVar($name); + $name = 'isset(' . $name . ')'; + $parseStr .= ''; + $endStr = ''; + } + + // 鏂囦欢鏂瑰紡瀵煎叆 + $array = explode(',', $file); + foreach ($array as $val) { + $type = strtolower(substr(strrchr($val, '.'), 1)); + switch ($type) { + case 'js': + $parseStr .= ''; + break; + case 'css': + $parseStr .= ''; + break; + case 'php': + $parseStr .= ''; + break; + } + } + return $parseStr . $endStr; + } + + /** + * assign鏍囩瑙f瀽 + * 鍦ㄦā鏉夸腑缁欐煇涓彉閲忚祴鍊 鏀寔鍙橀噺璧嬪 + * 鏍煎紡锛 {assign name="" value="" /} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagAssign($tag, $content) + { + $name = $this->autoBuildVar($tag['name']); + $flag = substr($tag['value'], 0, 1); + if ('$' == $flag || ':' == $flag) { + $value = $this->autoBuildVar($tag['value']); + } else { + $value = '\'' . $tag['value'] . '\''; + } + $parseStr = ''; + return $parseStr; + } + + /** + * define鏍囩瑙f瀽 + * 鍦ㄦā鏉夸腑瀹氫箟甯搁噺 鏀寔鍙橀噺璧嬪 + * 鏍煎紡锛 {define name="" value="" /} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagDefine($tag, $content) + { + $name = '\'' . $tag['name'] . '\''; + $flag = substr($tag['value'], 0, 1); + if ('$' == $flag || ':' == $flag) { + $value = $this->autoBuildVar($tag['value']); + } else { + $value = '\'' . $tag['value'] . '\''; + } + $parseStr = ''; + return $parseStr; + } + + /** + * for鏍囩瑙f瀽 + * 鏍煎紡锛 + * {for start="" end="" comparison="" step="" name=""} + * content + * {/for} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagFor($tag, $content) + { + //璁剧疆榛樿鍊 + $start = 0; + $end = 0; + $step = 1; + $comparison = 'lt'; + $name = 'i'; + $rand = rand(); //娣诲姞闅忔満鏁帮紝闃叉宓屽鍙橀噺鍐茬獊 + //鑾峰彇灞炴 + foreach ($tag as $key => $value) { + $value = trim($value); + $flag = substr($value, 0, 1); + if ('$' == $flag || ':' == $flag) { + $value = $this->autoBuildVar($value); + } + + switch ($key) { + case 'start': + $start = $value; + break; + case 'end': + $end = $value; + break; + case 'step': + $step = $value; + break; + case 'comparison': + $comparison = $value; + break; + case 'name': + $name = $value; + break; + } + } + + $parseStr = 'parseCondition('$' . $name . ' ' . $comparison . ' $__FOR_END_' . $rand . '__') . ';$' . $name . '+=' . $step . '){ ?>'; + $parseStr .= $content; + $parseStr .= ''; + return $parseStr; + } + + /** + * url鍑芥暟鐨則ag鏍囩 + * 鏍煎紡锛歿url link="妯″潡/鎺у埗鍣/鏂规硶" vars="鍙傛暟" suffix="true鎴栬協alse 鏄惁甯︽湁鍚庣紑" domain="true鎴栬協alse 鏄惁鎼哄甫鍩熷悕" /} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagUrl($tag, $content) + { + $url = isset($tag['link']) ? $tag['link'] : ''; + $vars = isset($tag['vars']) ? $tag['vars'] : ''; + $suffix = isset($tag['suffix']) ? $tag['suffix'] : 'true'; + $domain = isset($tag['domain']) ? $tag['domain'] : 'false'; + return ''; + } + + /** + * function鏍囩瑙f瀽 鍖垮悕鍑芥暟锛屽彲瀹炵幇閫掑綊 + * 浣跨敤锛 + * {function name="func" vars="$data" call="$list" use="&$a,&$b"} + * {if is_array($data)} + * {foreach $data as $val} + * {~func($val) /} + * {/foreach} + * {else /} + * {$data} + * {/if} + * {/function} + * @access public + * @param array $tag 鏍囩灞炴 + * @param string $content 鏍囩鍐呭 + * @return string + */ + public function tagFunction($tag, $content) + { + $name = !empty($tag['name']) ? $tag['name'] : 'func'; + $vars = !empty($tag['vars']) ? $tag['vars'] : ''; + $call = !empty($tag['call']) ? $tag['call'] : ''; + $use = ['&$' . $name]; + if (!empty($tag['use'])) { + foreach (explode(',', $tag['use']) as $val) { + $use[] = '&' . ltrim(trim($val), '&'); + } + } + $parseStr = '' . $content . '' : '?>'; + return $parseStr; + } +} diff --git a/thinkphp/library/think/view/driver/Php.php b/thinkphp/library/think/view/driver/Php.php new file mode 100644 index 000000000..468d3611d --- /dev/null +++ b/thinkphp/library/think/view/driver/Php.php @@ -0,0 +1,164 @@ + +// +---------------------------------------------------------------------- + +namespace think\view\driver; + +use think\App; +use think\exception\TemplateNotFoundException; +use think\Loader; +use think\Log; +use think\Request; + +class Php +{ + // 妯℃澘寮曟搸鍙傛暟 + protected $config = [ + // 瑙嗗浘鍩虹鐩綍锛堥泦涓紡锛 + 'view_base' => '', + // 妯℃澘璧峰璺緞 + 'view_path' => '', + // 妯℃澘鏂囦欢鍚庣紑 + 'view_suffix' => 'php', + // 妯℃澘鏂囦欢鍚嶅垎闅旂 + 'view_depr' => DS, + ]; + + public function __construct($config = []) + { + $this->config = array_merge($this->config, $config); + } + + /** + * 妫娴嬫槸鍚﹀瓨鍦ㄦā鏉挎枃浠 + * @access public + * @param string $template 妯℃澘鏂囦欢鎴栬呮ā鏉胯鍒 + * @return bool + */ + public function exists($template) + { + if ('' == pathinfo($template, PATHINFO_EXTENSION)) { + // 鑾峰彇妯℃澘鏂囦欢鍚 + $template = $this->parseTemplate($template); + } + return is_file($template); + } + + /** + * 娓叉煋妯℃澘鏂囦欢 + * @access public + * @param string $template 妯℃澘鏂囦欢 + * @param array $data 妯℃澘鍙橀噺 + * @return void + */ + public function fetch($template, $data = []) + { + if ('' == pathinfo($template, PATHINFO_EXTENSION)) { + // 鑾峰彇妯℃澘鏂囦欢鍚 + $template = $this->parseTemplate($template); + } + // 妯℃澘涓嶅瓨鍦 鎶涘嚭寮傚父 + if (!is_file($template)) { + throw new TemplateNotFoundException('template not exists:' . $template, $template); + } + // 璁板綍瑙嗗浘淇℃伅 + App::$debug && Log::record('[ VIEW ] ' . $template . ' [ ' . var_export(array_keys($data), true) . ' ]', 'info'); + if (isset($data['template'])) { + $__template__ = $template; + extract($data, EXTR_OVERWRITE); + include $__template__; + } else { + extract($data, EXTR_OVERWRITE); + include $template; + } + } + + /** + * 娓叉煋妯℃澘鍐呭 + * @access public + * @param string $content 妯℃澘鍐呭 + * @param array $data 妯℃澘鍙橀噺 + * @return void + */ + public function display($content, $data = []) + { + if (isset($data['content'])) { + $__content__ = $content; + extract($data, EXTR_OVERWRITE); + eval('?>' . $__content__); + } else { + extract($data, EXTR_OVERWRITE); + eval('?>' . $content); + } + } + + /** + * 鑷姩瀹氫綅妯℃澘鏂囦欢 + * @access private + * @param string $template 妯℃澘鏂囦欢瑙勫垯 + * @return string + */ + private function parseTemplate($template) + { + if (empty($this->config['view_path'])) { + $this->config['view_path'] = App::$modulePath . 'view' . DS; + } + + $request = Request::instance(); + // 鑾峰彇瑙嗗浘鏍圭洰褰 + if (strpos($template, '@')) { + // 璺ㄦā鍧楄皟鐢 + list($module, $template) = explode('@', $template); + } + if ($this->config['view_base']) { + // 鍩虹瑙嗗浘鐩綍 + $module = isset($module) ? $module : $request->module(); + $path = $this->config['view_base'] . ($module ? $module . DS : ''); + } else { + $path = isset($module) ? APP_PATH . $module . DS . 'view' . DS : $this->config['view_path']; + } + + $depr = $this->config['view_depr']; + if (0 !== strpos($template, '/')) { + $template = str_replace(['/', ':'], $depr, $template); + $controller = Loader::parseName($request->controller()); + if ($controller) { + if ('' == $template) { + // 濡傛灉妯℃澘鏂囦欢鍚嶄负绌 鎸夌収榛樿瑙勫垯瀹氫綅 + $template = str_replace('.', DS, $controller) . $depr . $request->action(); + } elseif (false === strpos($template, $depr)) { + $template = str_replace('.', DS, $controller) . $depr . $template; + } + } + } else { + $template = str_replace(['/', ':'], $depr, substr($template, 1)); + } + return $path . ltrim($template, '/') . '.' . ltrim($this->config['view_suffix'], '.'); + } + + /** + * 閰嶇疆妯℃澘寮曟搸 + * @access private + * @param string|array $name 鍙傛暟鍚 + * @param mixed $value 鍙傛暟鍊 + * @return void + */ + public function config($name, $value = null) + { + if (is_array($name)) { + $this->config = array_merge($this->config, $name); + } elseif (is_null($value)) { + return isset($this->config[$name]) ? $this->config[$name] : null; + } else { + $this->config[$name] = $value; + } + } + +} diff --git a/thinkphp/library/think/view/driver/Think.php b/thinkphp/library/think/view/driver/Think.php new file mode 100644 index 000000000..39a14ca33 --- /dev/null +++ b/thinkphp/library/think/view/driver/Think.php @@ -0,0 +1,165 @@ + +// +---------------------------------------------------------------------- + +namespace think\view\driver; + +use think\App; +use think\exception\TemplateNotFoundException; +use think\Loader; +use think\Log; +use think\Request; +use think\Template; + +class Think +{ + // 妯℃澘寮曟搸瀹炰緥 + private $template; + // 妯℃澘寮曟搸鍙傛暟 + protected $config = [ + // 瑙嗗浘鍩虹鐩綍锛堥泦涓紡锛 + 'view_base' => '', + // 妯℃澘璧峰璺緞 + 'view_path' => '', + // 妯℃澘鏂囦欢鍚庣紑 + 'view_suffix' => 'html', + // 妯℃澘鏂囦欢鍚嶅垎闅旂 + 'view_depr' => DS, + // 鏄惁寮鍚ā鏉跨紪璇戠紦瀛,璁句负false鍒欐瘡娆¢兘浼氶噸鏂扮紪璇 + 'tpl_cache' => true, + ]; + + public function __construct($config = []) + { + $this->config = array_merge($this->config, $config); + if (empty($this->config['view_path'])) { + $this->config['view_path'] = App::$modulePath . 'view' . DS; + } + + $this->template = new Template($this->config); + } + + /** + * 妫娴嬫槸鍚﹀瓨鍦ㄦā鏉挎枃浠 + * @access public + * @param string $template 妯℃澘鏂囦欢鎴栬呮ā鏉胯鍒 + * @return bool + */ + public function exists($template) + { + if ('' == pathinfo($template, PATHINFO_EXTENSION)) { + // 鑾峰彇妯℃澘鏂囦欢鍚 + $template = $this->parseTemplate($template); + } + return is_file($template); + } + + /** + * 娓叉煋妯℃澘鏂囦欢 + * @access public + * @param string $template 妯℃澘鏂囦欢 + * @param array $data 妯℃澘鍙橀噺 + * @param array $config 妯℃澘鍙傛暟 + * @return void + */ + public function fetch($template, $data = [], $config = []) + { + if ('' == pathinfo($template, PATHINFO_EXTENSION)) { + // 鑾峰彇妯℃澘鏂囦欢鍚 + $template = $this->parseTemplate($template); + } + // 妯℃澘涓嶅瓨鍦 鎶涘嚭寮傚父 + if (!is_file($template)) { + throw new TemplateNotFoundException('template not exists:' . $template, $template); + } + // 璁板綍瑙嗗浘淇℃伅 + App::$debug && Log::record('[ VIEW ] ' . $template . ' [ ' . var_export(array_keys($data), true) . ' ]', 'info'); + $this->template->fetch($template, $data, $config); + } + + /** + * 娓叉煋妯℃澘鍐呭 + * @access public + * @param string $template 妯℃澘鍐呭 + * @param array $data 妯℃澘鍙橀噺 + * @param array $config 妯℃澘鍙傛暟 + * @return void + */ + public function display($template, $data = [], $config = []) + { + $this->template->display($template, $data, $config); + } + + /** + * 鑷姩瀹氫綅妯℃澘鏂囦欢 + * @access private + * @param string $template 妯℃澘鏂囦欢瑙勫垯 + * @return string + */ + private function parseTemplate($template) + { + // 鍒嗘瀽妯℃澘鏂囦欢瑙勫垯 + $request = Request::instance(); + // 鑾峰彇瑙嗗浘鏍圭洰褰 + if (strpos($template, '@')) { + // 璺ㄦā鍧楄皟鐢 + list($module, $template) = explode('@', $template); + } + if ($this->config['view_base']) { + // 鍩虹瑙嗗浘鐩綍 + $module = isset($module) ? $module : $request->module(); + $path = $this->config['view_base'] . ($module ? $module . DS : ''); + } else { + $path = isset($module) ? APP_PATH . $module . DS . 'view' . DS : $this->config['view_path']; + } + + $depr = $this->config['view_depr']; + if (0 !== strpos($template, '/')) { + $template = str_replace(['/', ':'], $depr, $template); + $controller = Loader::parseName($request->controller()); + if ($controller) { + if ('' == $template) { + // 濡傛灉妯℃澘鏂囦欢鍚嶄负绌 鎸夌収榛樿瑙勫垯瀹氫綅 + $template = str_replace('.', DS, $controller) . $depr . $request->action(); + } elseif (false === strpos($template, $depr)) { + $template = str_replace('.', DS, $controller) . $depr . $template; + } + } + } else { + $template = str_replace(['/', ':'], $depr, substr($template, 1)); + } + return $path . ltrim($template, '/') . '.' . ltrim($this->config['view_suffix'], '.'); + } + + /** + * 閰嶇疆鎴栬呰幏鍙栨ā鏉垮紩鎿庡弬鏁 + * @access private + * @param string|array $name 鍙傛暟鍚 + * @param mixed $value 鍙傛暟鍊 + * @return mixed + */ + public function config($name, $value = null) + { + if (is_array($name)) { + $this->template->config($name); + $this->config = array_merge($this->config, $name); + } elseif (is_null($value)) { + return $this->template->config($name); + } else { + $this->template->$name = $value; + $this->config[$name] = $value; + } + } + + public function __call($method, $params) + { + return call_user_func_array([$this->template, $method], $params); + } +} diff --git a/thinkphp/library/traits/controller/Jump.php b/thinkphp/library/traits/controller/Jump.php new file mode 100644 index 000000000..51f4281fe --- /dev/null +++ b/thinkphp/library/traits/controller/Jump.php @@ -0,0 +1,157 @@ +error(); + * $this->redirect(); + * } + * } + */ +namespace traits\controller; + +use think\Config; +use think\exception\HttpResponseException; +use think\Request; +use think\Response; +use think\response\Redirect; +use think\Url; +use think\View as ViewTemplate; + +trait Jump +{ + /** + * 鎿嶄綔鎴愬姛璺宠浆鐨勫揩鎹锋柟娉 + * @access protected + * @param mixed $msg 鎻愮ず淇℃伅 + * @param string $url 璺宠浆鐨刄RL鍦板潃 + * @param mixed $data 杩斿洖鐨勬暟鎹 + * @param integer $wait 璺宠浆绛夊緟鏃堕棿 + * @param array $header 鍙戦佺殑Header淇℃伅 + * @return void + */ + protected function success($msg = '', $url = null, $data = '', $wait = 3, array $header = []) + { + $code = 1; + if (is_numeric($msg)) { + $code = $msg; + $msg = ''; + } + if (is_null($url) && isset($_SERVER["HTTP_REFERER"])) { + $url = $_SERVER["HTTP_REFERER"]; + } elseif ('' !== $url) { + $url = (strpos($url, '://') || 0 === strpos($url, '/')) ? $url : Url::build($url); + } + $result = [ + 'code' => $code, + 'msg' => $msg, + 'data' => $data, + 'url' => $url, + 'wait' => $wait, + ]; + + $type = $this->getResponseType(); + if ('html' == strtolower($type)) { + $result = ViewTemplate::instance(Config::get('template'), Config::get('view_replace_str')) + ->fetch(Config::get('dispatch_success_tmpl'), $result); + } + $response = Response::create($result, $type)->header($header); + throw new HttpResponseException($response); + } + + /** + * 鎿嶄綔閿欒璺宠浆鐨勫揩鎹锋柟娉 + * @access protected + * @param mixed $msg 鎻愮ず淇℃伅 + * @param string $url 璺宠浆鐨刄RL鍦板潃 + * @param mixed $data 杩斿洖鐨勬暟鎹 + * @param integer $wait 璺宠浆绛夊緟鏃堕棿 + * @param array $header 鍙戦佺殑Header淇℃伅 + * @return void + */ + protected function error($msg = '', $url = null, $data = '', $wait = 3, array $header = []) + { + $code = 0; + if (is_numeric($msg)) { + $code = $msg; + $msg = ''; + } + if (is_null($url)) { + $url = Request::instance()->isAjax() ? '' : 'javascript:history.back(-1);'; + } elseif ('' !== $url) { + $url = (strpos($url, '://') || 0 === strpos($url, '/')) ? $url : Url::build($url); + } + $result = [ + 'code' => $code, + 'msg' => $msg, + 'data' => $data, + 'url' => $url, + 'wait' => $wait, + ]; + + $type = $this->getResponseType(); + if ('html' == strtolower($type)) { + $result = ViewTemplate::instance(Config::get('template'), Config::get('view_replace_str')) + ->fetch(Config::get('dispatch_error_tmpl'), $result); + } + $response = Response::create($result, $type)->header($header); + throw new HttpResponseException($response); + } + + /** + * 杩斿洖灏佽鍚庣殑API鏁版嵁鍒板鎴风 + * @access protected + * @param mixed $data 瑕佽繑鍥炵殑鏁版嵁 + * @param integer $code 杩斿洖鐨刢ode + * @param mixed $msg 鎻愮ず淇℃伅 + * @param string $type 杩斿洖鏁版嵁鏍煎紡 + * @param array $header 鍙戦佺殑Header淇℃伅 + * @return void + */ + protected function result($data, $code = 0, $msg = '', $type = '', array $header = []) + { + $result = [ + 'code' => $code, + 'msg' => $msg, + 'time' => $_SERVER['REQUEST_TIME'], + 'data' => $data, + ]; + $type = $type ?: $this->getResponseType(); + $response = Response::create($result, $type)->header($header); + throw new HttpResponseException($response); + } + + /** + * URL閲嶅畾鍚 + * @access protected + * @param string $url 璺宠浆鐨刄RL琛ㄨ揪寮 + * @param array|integer $params 鍏跺畠URL鍙傛暟 + * @param integer $code http code + * @return void + */ + protected function redirect($url, $params = [], $code = 302) + { + $response = new Redirect($url); + if (is_integer($params)) { + $code = $params; + $params = []; + } + $response->code($code)->params($params); + throw new HttpResponseException($response); + } + + /** + * 鑾峰彇褰撳墠鐨剅esponse 杈撳嚭绫诲瀷 + * @access protected + * @return string + */ + protected function getResponseType() + { + $isAjax = Request::instance()->isAjax(); + return $isAjax ? Config::get('default_ajax_return') : Config::get('default_return_type'); + } +} diff --git a/thinkphp/library/traits/model/SoftDelete.php b/thinkphp/library/traits/model/SoftDelete.php new file mode 100644 index 000000000..8781544f6 --- /dev/null +++ b/thinkphp/library/traits/model/SoftDelete.php @@ -0,0 +1,153 @@ +getDeleteTimeField(); + if (!empty($this->data[$field])) { + return true; + } + return false; + } + + /** + * 鏌ヨ杞垹闄ゆ暟鎹 + * @access public + * @return Query + */ + public static function withTrashed() + { + $model = new static(); + $field = $model->getDeleteTimeField(true); + return $model->db(false)->removeWhereField($field); + } + + /** + * 鍙煡璇㈣蒋鍒犻櫎鏁版嵁 + * @access public + * @return Query + */ + public static function onlyTrashed() + { + $model = new static(); + $field = $model->getDeleteTimeField(true); + return $model->db(false)->where($field, 'exp', 'is not null'); + } + + /** + * 鍒犻櫎褰撳墠鐨勮褰 + * @access public + * @param bool $force 鏄惁寮哄埗鍒犻櫎 + * @return integer + */ + public function delete($force = false) + { + if (false === $this->trigger('before_delete', $this)) { + return false; + } + $name = $this->getDeleteTimeField(); + if (!$force) { + // 杞垹闄 + $this->change[] = $name; + $this->data[$name] = $this->autoWriteTimestamp($name); + $result = $this->isUpdate()->save(); + } else { + $result = $this->db(false)->delete($this->data); + } + + $this->trigger('after_delete', $this); + return $result; + } + + /** + * 鍒犻櫎璁板綍 + * @access public + * @param mixed $data 涓婚敭鍒楄〃 鏀寔闂寘鏌ヨ鏉′欢 + * @param bool $force 鏄惁寮哄埗鍒犻櫎 + * @return integer 鎴愬姛鍒犻櫎鐨勮褰曟暟 + */ + public static function destroy($data, $force = false) + { + // 鍖呭惈杞垹闄ゆ暟鎹 + $query = self::withTrashed(); + if (is_array($data) && key($data) !== 0) { + $query->where($data); + $data = null; + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [ & $query]); + $data = null; + } elseif (is_null($data)) { + return 0; + } + + $resultSet = $query->select($data); + $count = 0; + if ($resultSet) { + foreach ($resultSet as $data) { + $result = $data->delete($force); + $count += $result; + } + } + return $count; + } + + /** + * 鎭㈠琚蒋鍒犻櫎鐨勮褰 + * @access public + * @param array $where 鏇存柊鏉′欢 + * @return integer + */ + public function restore($where = []) + { + $name = $this->getDeleteTimeField(); + if (empty($where)) { + $pk = $this->getPk(); + $where[$pk] = $this->getData($pk); + $where[$name] = ['not null', '']; + } + // 鎭㈠鍒犻櫎 + return $this->db(false)->removeWhereField($this->getDeleteTimeField(true))->where($where)->update([$name => null]); + } + + /** + * 鏌ヨ榛樿涓嶅寘鍚蒋鍒犻櫎鏁版嵁 + * @access protected + * @param Query $query 鏌ヨ瀵硅薄 + * @return void + */ + protected function base($query) + { + $field = $this->getDeleteTimeField(true); + $query->where($field, 'null'); + } + + /** + * 鑾峰彇杞垹闄ゅ瓧娈 + * @access public + * @param bool $read 鏄惁鏌ヨ鎿嶄綔 鍐欐搷浣滅殑鏃跺欎細鑷姩鍘绘帀琛ㄥ埆鍚 + * @return string + */ + protected function getDeleteTimeField($read = false) + { + $field = isset($this->deleteTime) ? $this->deleteTime : 'delete_time'; + if (!strpos($field, '.')) { + $field = $this->db(false)->getTable() . '.' . $field; + } + if (!$read && strpos($field, '.')) { + $array = explode('.', $field); + $field = array_pop($array); + } + return $field; + } +} diff --git a/thinkphp/library/traits/think/Instance.php b/thinkphp/library/traits/think/Instance.php new file mode 100644 index 000000000..ba45ddd83 --- /dev/null +++ b/thinkphp/library/traits/think/Instance.php @@ -0,0 +1,45 @@ + +// +---------------------------------------------------------------------- + +namespace traits\think; + +use think\Exception; + +trait Instance +{ + protected static $instance = null; + + /** + * @param array $options + * @return static + */ + public static function instance($options = []) + { + if (is_null(self::$instance)) { + self::$instance = new self($options); + } + return self::$instance; + } + + // 闈欐佽皟鐢 + public static function __callStatic($method, $params) + { + if (is_null(self::$instance)) { + self::$instance = new self(); + } + $call = substr($method, 1); + if (0 === strpos($method, '_') && is_callable([self::$instance, $call])) { + return call_user_func_array([self::$instance, $call], $params); + } else { + throw new Exception("method not exists:" . $method); + } + } +} diff --git a/thinkphp/logo.png b/thinkphp/logo.png new file mode 100644 index 000000000..25fd05936 Binary files /dev/null and b/thinkphp/logo.png differ diff --git a/thinkphp/phpunit.xml b/thinkphp/phpunit.xml new file mode 100644 index 000000000..7c6ef03ca --- /dev/null +++ b/thinkphp/phpunit.xml @@ -0,0 +1,35 @@ + + + + + ./tests/thinkphp/ + + + + + + + + ./ + + tests + vendor + + + + + + + + + + diff --git a/thinkphp/start.php b/thinkphp/start.php new file mode 100644 index 000000000..8af62ef3a --- /dev/null +++ b/thinkphp/start.php @@ -0,0 +1,18 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +// ThinkPHP 寮曞鏂囦欢 +// 鍔犺浇鍩虹鏂囦欢 +require __DIR__ . '/base.php'; +// 鎵ц搴旂敤 +App::run()->send(); diff --git a/thinkphp/tpl/default_index.tpl b/thinkphp/tpl/default_index.tpl new file mode 100644 index 000000000..8538b4dfa --- /dev/null +++ b/thinkphp/tpl/default_index.tpl @@ -0,0 +1,10 @@ +*{ padding: 0; margin: 0; } div{ padding: 4px 48px;} a{color:#2E5CD5;cursor: pointer;text-decoration: none} a:hover{text-decoration:underline; } body{ background: #fff; font-family: "Century Gothic","Microsoft yahei"; color: #333;font-size:18px;} h1{ font-size: 100px; font-weight: normal; margin-bottom: 12px; } p{ line-height: 1.6em; font-size: 42px }

    :)

    ThinkPHP V5
    鍗佸勾纾ㄤ竴鍓 - 涓篈PI寮鍙戣璁$殑楂樻ц兘妗嗘灦

    [ V5.0 鐗堟湰鐢 涓冪墰浜 鐙璧炲姪鍙戝竷 ]
    '; + } +} diff --git a/thinkphp/tpl/dispatch_jump.tpl b/thinkphp/tpl/dispatch_jump.tpl new file mode 100644 index 000000000..18ee01bd5 --- /dev/null +++ b/thinkphp/tpl/dispatch_jump.tpl @@ -0,0 +1,48 @@ +{__NOLAYOUT__} + + + + 璺宠浆鎻愮ず + + + +
    + + +

    :)

    +

    + + +

    :(

    +

    + + +

    +

    + 椤甸潰鑷姩 璺宠浆 绛夊緟鏃堕棿锛 +

    +
    + + + diff --git a/thinkphp/tpl/page_trace.tpl b/thinkphp/tpl/page_trace.tpl new file mode 100644 index 000000000..7c5df6fb5 --- /dev/null +++ b/thinkphp/tpl/page_trace.tpl @@ -0,0 +1,71 @@ +
    + + +
    +
    +
    + +
    + + diff --git a/thinkphp/tpl/think_exception.tpl b/thinkphp/tpl/think_exception.tpl new file mode 100644 index 000000000..e80c57b25 --- /dev/null +++ b/thinkphp/tpl/think_exception.tpl @@ -0,0 +1,507 @@ +'.end($names).''; + } + } + + if(!function_exists('parse_file')){ + function parse_file($file, $line) + { + return ''.basename($file)." line {$line}".''; + } + } + + if(!function_exists('parse_args')){ + function parse_args($args) + { + $result = []; + + foreach ($args as $key => $item) { + switch (true) { + case is_object($item): + $value = sprintf('object(%s)', parse_class(get_class($item))); + break; + case is_array($item): + if(count($item) > 3){ + $value = sprintf('[%s, ...]', parse_args(array_slice($item, 0, 3))); + } else { + $value = sprintf('[%s]', parse_args($item)); + } + break; + case is_string($item): + if(strlen($item) > 20){ + $value = sprintf( + '\'%s...\'', + htmlentities($item), + htmlentities(substr($item, 0, 20)) + ); + } else { + $value = sprintf("'%s'", htmlentities($item)); + } + break; + case is_int($item): + case is_float($item): + $value = $item; + break; + case is_null($item): + $value = 'null'; + break; + case is_bool($item): + $value = '' . ($item ? 'true' : 'false') . ''; + break; + case is_resource($item): + $value = 'resource'; + break; + default: + $value = htmlentities(str_replace("\n", '', var_export(strval($item), true))); + break; + } + + $result[] = is_int($key) ? $value : "'{$key}' => {$value}"; + } + + return implode(', ', $result); + } + } +?> + + + + + 绯荤粺鍙戠敓閿欒 + + + + +
    + +
    + +
    +
    + +
    +
    +

    []

    +
    +

    +
    + +
    + +
    +
      $value) { ?>
    +
    + +
    +

    Call Stack

    +
      +
    1. + +
    2. + +
    3. + +
    +
    +
    + +
    + +

    + +
    + + + +
    +

    Exception Datas

    + $value) { ?> + + + + + + + $val) { ?> + + + + + + + +
    empty
    + +
    + +
    + + + +
    +

    Environment Variables

    + $value) { ?> + + + + + + + $val) { ?> + + + + + + + +
    empty
    + +
    + +
    + + + + + + + + diff --git a/vendor/aferrandini/phpqrcode/.gitignore b/vendor/aferrandini/phpqrcode/.gitignore new file mode 100644 index 000000000..485dee64b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/.gitignore @@ -0,0 +1 @@ +.idea diff --git a/vendor/aferrandini/phpqrcode/LICENSE b/vendor/aferrandini/phpqrcode/LICENSE new file mode 100644 index 000000000..188330326 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/LICENSE @@ -0,0 +1,165 @@ + GNU LESSER GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + + This version of the GNU Lesser General Public License incorporates +the terms and conditions of version 3 of the GNU General Public +License, supplemented by the additional permissions listed below. + + 0. Additional Definitions. + + As used herein, "this License" refers to version 3 of the GNU Lesser +General Public License, and the "GNU GPL" refers to version 3 of the GNU +General Public License. + + "The Library" refers to a covered work governed by this License, +other than an Application or a Combined Work as defined below. + + An "Application" is any work that makes use of an interface provided +by the Library, but which is not otherwise based on the Library. +Defining a subclass of a class defined by the Library is deemed a mode +of using an interface provided by the Library. + + A "Combined Work" is a work produced by combining or linking an +Application with the Library. The particular version of the Library +with which the Combined Work was made is also called the "Linked +Version". + + The "Minimal Corresponding Source" for a Combined Work means the +Corresponding Source for the Combined Work, excluding any source code +for portions of the Combined Work that, considered in isolation, are +based on the Application, and not on the Linked Version. + + The "Corresponding Application Code" for a Combined Work means the +object code and/or source code for the Application, including any data +and utility programs needed for reproducing the Combined Work from the +Application, but excluding the System Libraries of the Combined Work. + + 1. Exception to Section 3 of the GNU GPL. + + You may convey a covered work under sections 3 and 4 of this License +without being bound by section 3 of the GNU GPL. + + 2. Conveying Modified Versions. + + If you modify a copy of the Library, and, in your modifications, a +facility refers to a function or data to be supplied by an Application +that uses the facility (other than as an argument passed when the +facility is invoked), then you may convey a copy of the modified +version: + + a) under this License, provided that you make a good faith effort to + ensure that, in the event an Application does not supply the + function or data, the facility still operates, and performs + whatever part of its purpose remains meaningful, or + + b) under the GNU GPL, with none of the additional permissions of + this License applicable to that copy. + + 3. Object Code Incorporating Material from Library Header Files. + + The object code form of an Application may incorporate material from +a header file that is part of the Library. You may convey such object +code under terms of your choice, provided that, if the incorporated +material is not limited to numerical parameters, data structure +layouts and accessors, or small macros, inline functions and templates +(ten or fewer lines in length), you do both of the following: + + a) Give prominent notice with each copy of the object code that the + Library is used in it and that the Library and its use are + covered by this License. + + b) Accompany the object code with a copy of the GNU GPL and this license + document. + + 4. Combined Works. + + You may convey a Combined Work under terms of your choice that, +taken together, effectively do not restrict modification of the +portions of the Library contained in the Combined Work and reverse +engineering for debugging such modifications, if you also do each of +the following: + + a) Give prominent notice with each copy of the Combined Work that + the Library is used in it and that the Library and its use are + covered by this License. + + b) Accompany the Combined Work with a copy of the GNU GPL and this license + document. + + c) For a Combined Work that displays copyright notices during + execution, include the copyright notice for the Library among + these notices, as well as a reference directing the user to the + copies of the GNU GPL and this license document. + + d) Do one of the following: + + 0) Convey the Minimal Corresponding Source under the terms of this + License, and the Corresponding Application Code in a form + suitable for, and under terms that permit, the user to + recombine or relink the Application with a modified version of + the Linked Version to produce a modified Combined Work, in the + manner specified by section 6 of the GNU GPL for conveying + Corresponding Source. + + 1) Use a suitable shared library mechanism for linking with the + Library. A suitable mechanism is one that (a) uses at run time + a copy of the Library already present on the user's computer + system, and (b) will operate properly with a modified version + of the Library that is interface-compatible with the Linked + Version. + + e) Provide Installation Information, but only if you would otherwise + be required to provide such information under section 6 of the + GNU GPL, and only to the extent that such information is + necessary to install and execute a modified version of the + Combined Work produced by recombining or relinking the + Application with a modified version of the Linked Version. (If + you use option 4d0, the Installation Information must accompany + the Minimal Corresponding Source and Corresponding Application + Code. If you use option 4d1, you must provide the Installation + Information in the manner specified by section 6 of the GNU GPL + for conveying Corresponding Source.) + + 5. Combined Libraries. + + You may place library facilities that are a work based on the +Library side by side in a single library together with other library +facilities that are not Applications and are not covered by this +License, and convey such a combined library under terms of your +choice, if you do both of the following: + + a) Accompany the combined library with a copy of the same work based + on the Library, uncombined with any other library facilities, + conveyed under the terms of this License. + + b) Give prominent notice with the combined library that part of it + is a work based on the Library, and explaining where to find the + accompanying uncombined form of the same work. + + 6. Revised Versions of the GNU Lesser General Public License. + + The Free Software Foundation may publish revised and/or new versions +of the GNU Lesser General Public License from time to time. Such new +versions will be similar in spirit to the present version, but may +differ in detail to address new problems or concerns. + + Each version is given a distinguishing version number. If the +Library as you received it specifies that a certain numbered version +of the GNU Lesser General Public License "or any later version" +applies to it, you have the option of following the terms and +conditions either of that published version or of any later version +published by the Free Software Foundation. If the Library as you +received it does not specify a version number of the GNU Lesser +General Public License, you may choose any version of the GNU Lesser +General Public License ever published by the Free Software Foundation. + + If the Library as you received it specifies that a proxy can decide +whether future versions of the GNU Lesser General Public License shall +apply, that proxy's public statement of acceptance of any version is +permanent authorization for you to choose that version for the +Library. diff --git a/vendor/aferrandini/phpqrcode/VERSION b/vendor/aferrandini/phpqrcode/VERSION new file mode 100644 index 000000000..e0d3a2ee7 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/VERSION @@ -0,0 +1,2 @@ +1.1.5 +2012021604 diff --git a/vendor/aferrandini/phpqrcode/cache/frame_1.dat b/vendor/aferrandini/phpqrcode/cache/frame_1.dat new file mode 100644 index 000000000..be28feac4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_1.dat @@ -0,0 +1,2 @@ +x跐惲 E9硊悕`"P艅C坨墬T!0$ +E暽睶櫇蒻晋h劬9{kI" 9Ln)Apゅ志>過囌z砿雗艝;m遪λ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_1.png b/vendor/aferrandini/phpqrcode/cache/frame_1.png new file mode 100644 index 000000000..86ce6e98d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_1.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_10.dat b/vendor/aferrandini/phpqrcode/cache/frame_10.dat new file mode 100644 index 000000000..aff163f68 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_10.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_10.png b/vendor/aferrandini/phpqrcode/cache/frame_10.png new file mode 100644 index 000000000..dbfcd70b5 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_10.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_11.dat b/vendor/aferrandini/phpqrcode/cache/frame_11.dat new file mode 100644 index 000000000..95af68a47 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_11.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_11.png b/vendor/aferrandini/phpqrcode/cache/frame_11.png new file mode 100644 index 000000000..c07c761f1 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_11.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_12.dat b/vendor/aferrandini/phpqrcode/cache/frame_12.dat new file mode 100644 index 000000000..73228b362 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_12.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_12.png b/vendor/aferrandini/phpqrcode/cache/frame_12.png new file mode 100644 index 000000000..8ba67822c Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_12.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_13.dat b/vendor/aferrandini/phpqrcode/cache/frame_13.dat new file mode 100644 index 000000000..2256f0e34 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_13.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_13.png b/vendor/aferrandini/phpqrcode/cache/frame_13.png new file mode 100644 index 000000000..6e49d35a0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_13.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_14.dat b/vendor/aferrandini/phpqrcode/cache/frame_14.dat new file mode 100644 index 000000000..e9ae09329 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_14.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_14.png b/vendor/aferrandini/phpqrcode/cache/frame_14.png new file mode 100644 index 000000000..efc36c034 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_14.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_15.dat b/vendor/aferrandini/phpqrcode/cache/frame_15.dat new file mode 100644 index 000000000..18727818d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_15.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_15.png b/vendor/aferrandini/phpqrcode/cache/frame_15.png new file mode 100644 index 000000000..a9f416c7c Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_15.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_16.dat b/vendor/aferrandini/phpqrcode/cache/frame_16.dat new file mode 100644 index 000000000..60af67845 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_16.dat @@ -0,0 +1 @@ +x陧橝 E]s隝X;n6萡俼旉W6皴歚%A/3!偄!g柸獭1N) 镋⑾|;畻>6飧忁97$肽綦c]kk鰪w1贮[穖瑿蜏cR耗旯>﹁稻欵,昲艍p#醲F漼W锨VWG珧3颊+堽嗨擔S庘}臑#G8b^c^c侠巔刢&3YQ"庽缣v爹泤襦牺柟k9娷圇}敼筹 ⒛繌QL/菰老 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_16.png b/vendor/aferrandini/phpqrcode/cache/frame_16.png new file mode 100644 index 000000000..6ac8fe890 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_16.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_17.dat b/vendor/aferrandini/phpqrcode/cache/frame_17.dat new file mode 100644 index 000000000..87f0cf593 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_17.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_17.png b/vendor/aferrandini/phpqrcode/cache/frame_17.png new file mode 100644 index 000000000..5b929ac73 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_17.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_18.dat b/vendor/aferrandini/phpqrcode/cache/frame_18.dat new file mode 100644 index 000000000..bb7138c1d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_18.dat @@ -0,0 +1,2 @@ +x陧橝 +0E]缰,2;s冧&赏歨坳O◆萑嚆1&09OIv@DD &賶K峏日Fv<羋q9<%h暪 Y飐 !(dゲ雜;~||b(哮Y暖g#礰淜牨S寂艄亩橓s鄆d邖Lg:訖蝨/gm獫檭kM3硔4rT萉eシs><訖蝨3濋;颒#褭蝨3蚘+og﹉蟮俳琹n瘥F>豬^#awm;g鑯p踘霳s{6z拺汇猴掬p臼' \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_18.png b/vendor/aferrandini/phpqrcode/cache/frame_18.png new file mode 100644 index 000000000..ee0d6a35e Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_18.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_19.dat b/vendor/aferrandini/phpqrcode/cache/frame_19.dat new file mode 100644 index 000000000..95e26adc1 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_19.dat @@ -0,0 +1,3 @@ +x陧欰 + E祸.蘊o7褯嵍iiR贜2嬦W%饃罖谮滉' +u6钻.*S;}槴颐犗T zrt癸%,遗阝蝳;撯)篃樷軿陬L彖棍琍珑$蟩薵淟卖dJ;茚w该.]z#熅玔蜐斤Og偔启"兴 酈瞀}莭;茚w该顦#1Gb巹;茚w该齙軨+wDf黠鐧u濜2櫯谏N9R7|pW遦稃撨k忽胡患斡 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_19.png b/vendor/aferrandini/phpqrcode/cache/frame_19.png new file mode 100644 index 000000000..20fddd84c Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_19.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_2.dat b/vendor/aferrandini/phpqrcode/cache/frame_2.dat new file mode 100644 index 000000000..7e42f31ca --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_2.dat @@ -0,0 +1 @@ +x谕捦 F{v& &癥+?Z1鲂S'y!羇815&鄞庂H澹贋c硶lF畦1#6 f手6镍昈7埁咰珱鹬彏愊8gI鰂Bγ鎈D曰( \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_2.png b/vendor/aferrandini/phpqrcode/cache/frame_2.png new file mode 100644 index 000000000..9c150ebe3 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_2.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_20.dat b/vendor/aferrandini/phpqrcode/cache/frame_20.dat new file mode 100644 index 000000000..d5ecc1d8a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_20.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_20.png b/vendor/aferrandini/phpqrcode/cache/frame_20.png new file mode 100644 index 000000000..23a061d54 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_20.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_21.dat b/vendor/aferrandini/phpqrcode/cache/frame_21.dat new file mode 100644 index 000000000..1974dd9d4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_21.dat @@ -0,0 +1 @@ +x陧欰 E]s隝X;n6Up崜胁]贅< i-eW鰦稑)卒艜忌聟H\jvq貶L\6枤菪rI芄苷%覢逮盫梫坡(螾4|蝀n襣蓾紐]D旧兆u1Us S\皜,2?D汯狐F-:揺J]p_皜,榓0胉喠 X癭喠 w,` X碷槇檪箣槹5 壆Y4{灞骜2睇鍁鏙s啽垌9睒)鮱臂龟县,玗笓嬞^_7$僟 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_21.png b/vendor/aferrandini/phpqrcode/cache/frame_21.png new file mode 100644 index 000000000..291598c72 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_21.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_22.dat b/vendor/aferrandini/phpqrcode/cache/frame_22.dat new file mode 100644 index 000000000..0f01802d2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_22.dat @@ -0,0 +1,3 @@ +x陧欰 +0 E]{雭.抅{{{畅Zep辸e@廣汦RZ3涣"*2o4)i#d襜dF覅孖"鷳4灲W璉韚娪45選.Z璖賩翢8逅k={o.眖寿溚:甯抭货儁 +)t#釀N8醖Cj-O漁G}:/:s弞!)^<鵨椒S穟鈡 '減 '==='減 '減⑦__N8釀9┆堲pQQ鮙H詐z緩谿淾娼Q簶I|具碀u;9櫸镎雂;揦锝$速蓆妒坶dy \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_22.png b/vendor/aferrandini/phpqrcode/cache/frame_22.png new file mode 100644 index 000000000..bc97bd01b Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_22.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_23.dat b/vendor/aferrandini/phpqrcode/cache/frame_23.dat new file mode 100644 index 000000000..ee3b37073 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_23.dat @@ -0,0 +1,3 @@ +x陧欰 + E出fo7褯U) %M!螖蔓Yu(<氿搒K矒T湜 +&I赲i+バ獧(mFQ」痟辨鳇鑦~n1o蟏s腌诛逕3`頮w2跞箷lc[紩;踓譄摔扤螵4躳 7躳庙mT笢洃蓐r辤皲_冪縫S=77躳 7躳脥>I煠O- 7躳 7$}>煞7躳 tss売rs уV臀芷鲯m∫跪嚌揆}R~7羿&诀?7鶟摭澰颾h鈡婊<繫i- \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_23.png b/vendor/aferrandini/phpqrcode/cache/frame_23.png new file mode 100644 index 000000000..b8f16ae23 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_23.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_24.dat b/vendor/aferrandini/phpqrcode/cache/frame_24.dat new file mode 100644 index 000000000..7b92e29c4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_24.dat @@ -0,0 +1 @@ +x陧汚 E祸諱X0;竵nVP4贖SS粁遀3/O待 LiJ4彏盫 JC%龎6VR&棉D態淗jD鶄J??櫙闎l璫潜窠'骍X颱镛0婷yw湍j霕硛3艣倦榗j嗼ē:GqG蓠鵁癗唙;苟绗揓 嚹<麌蒥庩肴髨#8鈭#8釮'由GqG殇tr:9#8鈭#8庳揾瓐疦詔攲粗_葚>t筫腱S︽烓^漒g蜵e?鵹u鎏o镎;>靾*飛lm \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_24.png b/vendor/aferrandini/phpqrcode/cache/frame_24.png new file mode 100644 index 000000000..397c64f85 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_24.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_25.dat b/vendor/aferrandini/phpqrcode/cache/frame_25.dat new file mode 100644 index 000000000..ba125182e --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_25.dat @@ -0,0 +1,3 @@ +x陧跘 + 呩瑂雮嫏]r儀結51mM菳G +*Sx|Ua5频Z棅-,1洳H裀襌j枤X5Мi啯酖>W濺秭/蒜+uT寤 嫌棿猽嫦ペ[S韆kv眵5+5nЯ碕贶%+V琗眀艎踹瑄'澅麪SR鳉tzZ混++V琗眀艎曎熧熧燐+V琗眀攀煜煜煜}艎+V琗宾豹ぅ諺I种+kq[t幏oVZ威voNV硍莭祘硆<媒R"漅娃] W玶} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_25.png b/vendor/aferrandini/phpqrcode/cache/frame_25.png new file mode 100644 index 000000000..25bc4454a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_25.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_26.dat b/vendor/aferrandini/phpqrcode/cache/frame_26.dat new file mode 100644 index 000000000..d34a73f15 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_26.dat @@ -0,0 +1,2 @@ +x陧汚 + E出謪,t񘻉褯U E)i7锘*~c命拍X諩B畦FC枠6:&鏛,瀣Mv.幝捂Kg熣搞YM>熚>鹠蹥?獣v艄緈g?忂冶牺虫畏猟槗C筓οIk曏贓\召Msf榓f榓>淸s訄9飕┺8b祟擼喏Lg奢翛宠句Eo w1 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_26.png b/vendor/aferrandini/phpqrcode/cache/frame_26.png new file mode 100644 index 000000000..f4a6b3938 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_26.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_27.dat b/vendor/aferrandini/phpqrcode/cache/frame_27.dat new file mode 100644 index 000000000..b4d9ffd4f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_27.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_27.png b/vendor/aferrandini/phpqrcode/cache/frame_27.png new file mode 100644 index 000000000..8419ec230 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_27.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_28.dat b/vendor/aferrandini/phpqrcode/cache/frame_28.dat new file mode 100644 index 000000000..8cbaa1961 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_28.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_28.png b/vendor/aferrandini/phpqrcode/cache/frame_28.png new file mode 100644 index 000000000..7609d8e1f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_28.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_29.dat b/vendor/aferrandini/phpqrcode/cache/frame_29.dat new file mode 100644 index 000000000..5e4a71103 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_29.dat @@ -0,0 +1,2 @@ +x陧蹵 卆邹 嫏澻@n7+*稓众4!?甁氿 硱旀姰玗椛S熲Tf)栙s奍"吶攂炤0厞|"Lu俑,幾E1\6*蟯Q?>a滔呫幠R-r瓝鱪.镪瘚\甌奎:*)|)袄 ,袄 ,怅箦閤_悻}:^R剝Uo散塽羱赁X`X袕袕袕袕癬`X`X袕袕袕癬`X`X袕袕袕袕皐bX`骏PU貊)D赞"c葅媧缥3隂<}阁?b鱩温鞛冺喊惑a麕磼玛.怾 +硔Q6u酺,9 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_29.png b/vendor/aferrandini/phpqrcode/cache/frame_29.png new file mode 100644 index 000000000..ffe072c86 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_29.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_3.dat b/vendor/aferrandini/phpqrcode/cache/frame_3.dat new file mode 100644 index 000000000..188d531c2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_3.dat @@ -0,0 +1 @@ +x陧摿 E{v& &癥+衚'y鶓た羇:淅TXl澽$W+訌v顪9}gR娆@H0YPB狡肊m谮?麥表s溨"b奠t2cn珠喩:锖;縔脄Q7吭 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_3.png b/vendor/aferrandini/phpqrcode/cache/frame_3.png new file mode 100644 index 000000000..945ee7cb9 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_3.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_30.dat b/vendor/aferrandini/phpqrcode/cache/frame_30.dat new file mode 100644 index 000000000..44cf3d317 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_30.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_30.png b/vendor/aferrandini/phpqrcode/cache/frame_30.png new file mode 100644 index 000000000..75dbddd24 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_30.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_31.dat b/vendor/aferrandini/phpqrcode/cache/frame_31.dat new file mode 100644 index 000000000..ce429d0a6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_31.dat @@ -0,0 +1 @@ +x陧蹵 卆邹 厣 &r硞犃锤4珕磞鶄姆!V3I嵉v!覝2\NSS4EF2+65e峻/W渟]汃緣!劻?p捧=S~膼?她+䎬χ6r6y迟箎撉礄胱eR1-峎l傲l傲淃覍轝焭/>V娅幅:帽乙腁8-+mT蝏l傲l傲l铓tM&]摮l傲l傲&]摦I卒剂l傲l傲&]摦蓎 6豟 6豟兺輎瑄y簇XW蜩鄙i\t嘄z晽>.顢z緆蔬 t7┻7騱J跸敹4襴憹覉哂滞85 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_31.png b/vendor/aferrandini/phpqrcode/cache/frame_31.png new file mode 100644 index 000000000..b14d1fa26 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_31.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_32.dat b/vendor/aferrandini/phpqrcode/cache/frame_32.dat new file mode 100644 index 000000000..aaa0808e2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_32.dat @@ -0,0 +1,2 @@ +x陧芰 + 呩志鮻嬰. 逥遧, Mz6喢 gcJ薉;'.瓵扞q炥壞I,Irɑ懰Fk%塂﨩鎦|ED狣(L_Y嵤>*邭?a蔕k盠_<[c楍讹>蔯藰鮱訪I湔%#0#0#屴ot癣汋祡跑4蚮漹_)壜E彫h5R窂881#0#0⒂襥榇tZ#0#0#0⒂襥榇tZ#0#0#0⒂襥榇tZ蝜0#0w9q"⑸H軠H橯䴘碉"壅L5}-蒈Y拙雨竗靈も>秡楦钞4&襭麽!憡!`:5 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_32.png b/vendor/aferrandini/phpqrcode/cache/frame_32.png new file mode 100644 index 000000000..58d42db3c Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_32.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_33.dat b/vendor/aferrandini/phpqrcode/cache/frame_33.dat new file mode 100644 index 000000000..a2613755a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_33.dat @@ -0,0 +1,14 @@ +x陧蹵 卆邹簤嫏澻@n7+*L++渔煯篁壧bb*LC懓噳c k橦r殧j暡怞5Y韎~0昣岥譚蔜鮹鍡e>5慴_鍂型?郡哽妲周哱Ra苅+7踹W│\泾wLUN錖β ++ ++玛璲咭O窡kc朕耵鏫拴|%昽<釈k柇L++蝪 ++ ++聤>} 8 ++ ++ ++ 3術嗉 ++ ++ ++3術犗@焷 ++ ++ ++:R墾猉弛B9珨I=鏺彵o/Sw缲槞侬蟕g夺攀萺_贆橸緝VSY櫯zIefnmQoz > \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_33.png b/vendor/aferrandini/phpqrcode/cache/frame_33.png new file mode 100644 index 000000000..924c728e0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_33.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_34.dat b/vendor/aferrandini/phpqrcode/cache/frame_34.dat new file mode 100644 index 000000000..7ceb0259d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_34.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_34.png b/vendor/aferrandini/phpqrcode/cache/frame_34.png new file mode 100644 index 000000000..a477042d8 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_34.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_35.dat b/vendor/aferrandini/phpqrcode/cache/frame_35.dat new file mode 100644 index 000000000..56bc3e28e Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_35.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_35.png b/vendor/aferrandini/phpqrcode/cache/frame_35.png new file mode 100644 index 000000000..d29806c60 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_35.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_36.dat b/vendor/aferrandini/phpqrcode/cache/frame_36.dat new file mode 100644 index 000000000..282c60d23 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_36.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_36.png b/vendor/aferrandini/phpqrcode/cache/frame_36.png new file mode 100644 index 000000000..96ecb4213 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_36.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_37.dat b/vendor/aferrandini/phpqrcode/cache/frame_37.dat new file mode 100644 index 000000000..015c0f24b Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_37.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_37.png b/vendor/aferrandini/phpqrcode/cache/frame_37.png new file mode 100644 index 000000000..fcc51627d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_37.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_38.dat b/vendor/aferrandini/phpqrcode/cache/frame_38.dat new file mode 100644 index 000000000..71cf53eb5 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_38.dat @@ -0,0 +1 @@ +x陧軦獌0袔輚罙2;袧栉k(g纠y晅p9$蓑櫣D湐蚣\篹^'t-aI簥FM歋歬吗I笈び:7|L鷎烴8N7疁鰅}鰢焛,焄W喛g哟半?31鱥櫨N穧}=翺M:4摂)S2e蕯)S#$ 袈J悱JM:}齗槙諰涃SQL2e蕯)S2铡鑀t(:婓)S2e蕯)S:E嚔Cq2e蕯)S2e蕯E嚔C选8O2e蕯)S2e蔜嚔C选鑀湩L2e蕯)S2輷sJCIK衷俰93嶔n篲羊 究+Ri棼4鰸\莋骏飵;% }遖辬 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_38.png b/vendor/aferrandini/phpqrcode/cache/frame_38.png new file mode 100644 index 000000000..89238f3c5 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_38.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_39.dat b/vendor/aferrandini/phpqrcode/cache/frame_39.dat new file mode 100644 index 000000000..53511f736 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_39.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_39.png b/vendor/aferrandini/phpqrcode/cache/frame_39.png new file mode 100644 index 000000000..1dc9cd1bc Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_39.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_4.dat b/vendor/aferrandini/phpqrcode/cache/frame_4.dat new file mode 100644 index 000000000..67b30e82a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_4.dat @@ -0,0 +1 @@ +x陧斄 E=籾 p噩Q曍COM'盟$ 矦3e朏‐FNXRy韶綜{塧8R 艃a2@皴湁qk呱H1(埮`c硚0謦棵塄炙秘筺XG嵞 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_4.png b/vendor/aferrandini/phpqrcode/cache/frame_4.png new file mode 100644 index 000000000..b72f9e70d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_4.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_40.dat b/vendor/aferrandini/phpqrcode/cache/frame_40.dat new file mode 100644 index 000000000..90d36dd13 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_40.dat @@ -0,0 +1,2 @@ +x陧軦妰@鞋锦嬵澽@o7摌`換fe轰暙PA>彟磊惆<]哒须叱bZ玭悭^A涼Q}[9^猐珁najM車K虡1c茖3f虡1愀苳遅5}缃{脱7lM咭镛歺躀<坚K建漆伪yl3f虡1c茖3f虡1惬刍倩={窊伪yl3f虡1c茖3f虡1惬刍倩={窊伪yl3f虡1c茖3f虡1惬刍倩={窊伪yl3f虡1c茖3f虡1惬刍倩={窊伪yl3f虡1c茖3f虡聍鐪S蕬層7芀藜g\缇鈛踹蟔猺'4[甾-芣泤q堺L风8茲臂Y1q劵弸!顥拊/(% \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_40.png b/vendor/aferrandini/phpqrcode/cache/frame_40.png new file mode 100644 index 000000000..8034d862d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_40.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_5.dat b/vendor/aferrandini/phpqrcode/cache/frame_5.dat new file mode 100644 index 000000000..d5dafe186 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/frame_5.dat @@ -0,0 +1 @@ +x陧1 E澖u7袥劳Z|N啑麯 B0@R$l,->VKZ[<趜楊槗q茙ㄘYJ&僫鍤倠Zy:Y'氙Y盗V&梕昍"j┹r+碎夰.稭茙粬9域z祍帞, \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/frame_5.png b/vendor/aferrandini/phpqrcode/cache/frame_5.png new file mode 100644 index 000000000..96b6494f3 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_5.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_6.dat b/vendor/aferrandini/phpqrcode/cache/frame_6.dat new file mode 100644 index 000000000..0fc3d039e Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_6.png b/vendor/aferrandini/phpqrcode/cache/frame_6.png new file mode 100644 index 000000000..05ca358b0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_6.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_7.dat b/vendor/aferrandini/phpqrcode/cache/frame_7.dat new file mode 100644 index 000000000..43375960f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_7.png b/vendor/aferrandini/phpqrcode/cache/frame_7.png new file mode 100644 index 000000000..7d2ff4f3e Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_7.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_8.dat b/vendor/aferrandini/phpqrcode/cache/frame_8.dat new file mode 100644 index 000000000..669b325f3 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_8.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_8.png b/vendor/aferrandini/phpqrcode/cache/frame_8.png new file mode 100644 index 000000000..db1f1877a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_8.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_9.dat b/vendor/aferrandini/phpqrcode/cache/frame_9.dat new file mode 100644 index 000000000..d79295ee0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_9.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/frame_9.png b/vendor/aferrandini/phpqrcode/cache/frame_9.png new file mode 100644 index 000000000..74ddf08da Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/frame_9.png differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_101_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_101_0.dat new file mode 100644 index 000000000..51deabaef Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_101_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_105_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_105_0.dat new file mode 100644 index 000000000..97e9e5dff Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_105_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_109_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_109_0.dat new file mode 100644 index 000000000..eadf83a2c --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_109_0.dat @@ -0,0 +1,2 @@ +x陧= +0 薪9'舃$牼t邴^#i潽薸?撑踒鶮[AU谾寰澠的硏]m焆2帊崓崓-臇滽巭 蟅w}禭淃破破破&O采撧666666yR灁'%lllll/村h渓吤頼 工陱醖雔櫠3+镗m瞳 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_113_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_113_0.dat new file mode 100644 index 000000000..5eb7f5de2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_113_0.dat @@ -0,0 +1,2 @@ +x陧; +0>獻隹9+E駜俿=悉L1虅[笷醃U4?i<愮;7嶇蜱;嚻P#W-[莜6鲝礵dddddc",;"澕焥k嶆憫憫憫憫Q&椛erw######.滏憫憫憫懕袠y1^瞬\驆钇3柒弛芋 懷v \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_117_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_117_0.dat new file mode 100644 index 000000000..781c7f875 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_117_0.dat @@ -0,0 +1,2 @@ +x陧贏 +0 衹O愚r砇,#3癀,饷給5烠地衠:貂;;瑆vNJZG=宮伝鹽邛 驯郸2廪i婻镲k苆_YYYYYYYYe賍龇快/WVVVVVVk钺齞-,#繭治Zc]|噞啪玖埈$ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_121_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_121_0.dat new file mode 100644 index 000000000..68810c347 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_121_0.dat @@ -0,0 +1 @@ +x陧1 袨永/窌w YM黃鵂8>2S宣諪燨璄徲忍觕玏\細媨cфpK汫濨沸汘韒麕xhfffffff/s2矍2W|*鱠1巯烫烫烫烫剃*5澈铺烫烫烫态甊W┇渁烫烫烫烫s鲵肆\xm~8弋竷r0w鎗sdm鳘&鈟 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_125_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_125_0.dat new file mode 100644 index 000000000..2c73ef1a8 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_125_0.dat @@ -0,0 +1,2 @@ +x陧跘 + 薪а鸰畷慣H`3AOL4 k恚兔(嵬踖w鰶GW僮. #棉2葵 \腌Yごgggggggggg_d>巷凋j^櫿豷眦;;;;;;;;;;'巧q;;;;;;;;;'税藂⺷祆祆祆祆登_P靿顽鰲象Yw{e=d妥听礼/龘 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_129_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_129_0.dat new file mode 100644 index 000000000..812ee8a66 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_129_0.dat @@ -0,0 +1,2 @@ +x陧1 + 擂予/*焺D鳨'蒱gt-靰_V \"蘠=s䲢撂鍣骩冃J岕=8Dh舘嗪蹫' 0X 鄞彥钁琫0`纮 羰 j" 0`览紶Wf`^P0`纮2髠毴 家 笊鬱07(Y/XLGb厃"pT \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_137_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_137_0.dat new file mode 100644 index 000000000..f6d993b03 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_137_0.dat @@ -0,0 +1 @@ +x陧1 袨永/窌+F蒢?J毶 L7院剌*袯赼%L~殞嘶蜾斏亻蕮C擆J洿Y領WJ 綑.K]馬0a聞 $鐡颕NTw蝜L瀉璞L0a聞 &Ld@P設0a聞 &L0e@P?a聞 &L0a翫e@齽 &L0a落螹矡IlL&)dl橅gacR<$鰒,齽珊?U2鏬 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_141_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_141_0.dat new file mode 100644 index 000000000..8c685c8ea --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_141_0.dat @@ -0,0 +1,2 @@ +x陧= +0 薪9'E漾諨x蜆%獻隹9+E袂{$錷唭^&祹汲S"低D岨6跓]98霼揗揵砸荆Yひ[2鎷壞#F1b膱O%舻iRN氭綕栄硩#;#F1b膱慛1襥#F1b膱#F宼Z}崙Nk1b膱#F1bび阫宼Z;#F1b蔫鮂廣-屄u"IoD-*7uj>bMV+ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_149_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_149_0.dat new file mode 100644 index 000000000..d25835029 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_149_0.dat @@ -0,0 +1,3 @@ +x陧蹵 + 薪а鸰甅EQXP唰.輡94嘲謊盗{矹L╲殜#玘荀蟦[ ?; +ZIV-窕谀*w锔藪1*+V琗眀艎X矾gホw岃qX}荍絉踄眀艎+V琗眀e螤wfe螤^眀艎+V琗眀攀淎锾蕼A絙艎+V琗眀艎9冝贉翜A絙艎+V琗眀艎誰曮0尬*0T遤`?橏菸震;X=zZr* \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_153_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_153_0.dat new file mode 100644 index 000000000..fc79e9ede --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_153_0.dat @@ -0,0 +1 @@ +x陧1 袧永/绀慼&F菌`煲粿I;P爰Z繼昘弁屬mf掀.=5 [if-鯗璕+!wr凰済\j虡1c茖3f虡1cf栐o.2?1麨檢 `茖3f虡1c茖櫣潤箚z茖3f虡1c茖3fztf3f虡1c茖3f虡檏榢03尊0c茖3f虡1c超9;承膸鰶`vf曎虇咄歓蠘澾禬9 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_157_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_157_0.dat new file mode 100644 index 000000000..ad749f305 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_157_0.dat @@ -0,0 +1,2 @@ +x陧蹵 + 薪а鸰畷QRЩY k慝呾*q偷宇=j7賜N.p%诘鋝穒菡.阶撒乇c菐;v熵眥.-W2={嶊炀mg辻+乇c菐;v熵沙2;y旨c菐;v熵眂荖灂韶沙;v熵眂菐;v2<薔烏V癱菐;v熵睋ge2v颥y菐;v熵眂穷;秽嫫v"鯊逎]e'粯;[撶畸{緗A \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_161_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_161_0.dat new file mode 100644 index 000000000..4bdc5fdd3 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_161_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_165_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_165_0.dat new file mode 100644 index 000000000..3a17a0510 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_165_0.dat @@ -0,0 +1,2 @@ +x陧; +0>9+培Dy罥4藸5:禬賤玠錻襁溰<d2x%[炈U%疠揞2]&K,Y瞕蓲%恕,S空棑r2yd=,k簕X瞕蓲%K,Y瞕)0彸攎钘,Y瞕蓲%K,Y瞕)0彸攎钘,Y瞕蓲%K,Y矓m樓e,e%K,Y瞕蓲%K,e鎞柌 菝%K,Y瞕蓲鍏e顝:<硨!⌒YV,唏:B縟鵿O$*# \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_169_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_169_0.dat new file mode 100644 index 000000000..c4787d9db --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_169_0.dat @@ -0,0 +1,2 @@ +x陧1 +0>疘9EQ=槫Ls 驿禝锼亄砕t°碦欮}蚶麸揝羽昻:獆R[铦?疃_*S2e蕯)S&夕隝 氷疧2誒2e蕯)S2e*C1锼P樖P鬝2e蕯)S櫴P帖22龜)S2e蕯)Se(鎪2龜)S2e蕯)Se(鎪2龜)S2e蕯)SΤi鯿7;"呣橣挺迱贆v 荦^2}oO溥'r \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_173_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_173_0.dat new file mode 100644 index 000000000..5ef85e7ad --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_173_0.dat @@ -0,0 +1 @@ +x陧10擂筎 琜4v殒2仄給k輫帐;τ瞉f譃齞仅ljlG0n+呋m矴屗-[秎俨e"坳锓鸜粆o砝栱誚奘[秎俨e藮-[秎俨e诱罢兩[2秎俨e藮-[秎俨e诱罢兩[2秎俨e藮-[秎俨e诱罢兩[秎俨e藮-[秎俨e薞Λa摲l俨e藮-[秎俨ef[B皳m邪鞫E麆;禢稘-蹨T/r挿l?卦* \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_177_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_177_0.dat new file mode 100644 index 000000000..78a26a77b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_177_0.dat @@ -0,0 +1,2 @@ +x陧1 +0>9+?姴罔邅棓i謬 d毺汞誼謭x軳跳/普{鲐覆鲮8d0騢=鳄cFf虡1c茖3f虡q紂=飛6;l坊4c茣<柷3f虡1c茖3f蘕铴一1只蒫茖3f虡1c茖3f蘕铴b瑆撉3f虡1c茖3f虡鞭M'腦&忓1c茖3f虡1c茖3只閯鬾岝n驑1c茖3f虡1c茖脥3U< \7+掁(<O茖肺姍柷屵珍4@ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_21_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_21_0.dat new file mode 100644 index 000000000..368c9941f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_21_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_25_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_25_0.dat new file mode 100644 index 000000000..e4a5b6d8a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_25_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_29_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_29_0.dat new file mode 100644 index 000000000..74a216b4f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_29_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_33_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_33_0.dat new file mode 100644 index 000000000..2ec712a7a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_33_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_37_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_37_0.dat new file mode 100644 index 000000000..1588cfce1 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_37_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_41_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_41_0.dat new file mode 100644 index 000000000..e369027e3 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_41_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_45_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_45_0.dat new file mode 100644 index 000000000..452f126c8 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_45_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_49_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_49_0.dat new file mode 100644 index 000000000..fdd2aac15 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_49_0.dat @@ -0,0 +1,2 @@ +x陧誎 E9珌齩#?H/6g$-崻,X] +槢x輼;鲤窐 X园9簣<苎氦錻2燗f黝H7鴿/5We{#犰f蕤列?喈4=N >擎 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_53_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_53_0.dat new file mode 100644 index 000000000..572d279ef --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_53_0.dat @@ -0,0 +1,2 @@ +x陧諯 +@!泄齩甉严:(m&癁s-宫6跹Z{惀m4Y啿X怡.F蕾瓔Zi沸鵭=:苇謰b蹇慥H 8 #楖簿嬨Y茬綹X憎e \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_57_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_57_0.dat new file mode 100644 index 000000000..ea81e6dc6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_57_0.dat @@ -0,0 +1,4 @@ +x陧諥 + 纚^擖s=YL諝 ( +o乽爅)  +Z7y勝婉v,源靪VQ 瀒护Gi窑古Df论暽wo4绚剐o蜭涕L燃翜┓浟痴靰4 h \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_61_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_61_0.dat new file mode 100644 index 000000000..93d2444d8 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_61_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_65_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_65_0.dat new file mode 100644 index 000000000..df29d7bfb Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_65_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_69_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_69_0.dat new file mode 100644 index 000000000..8a2cfbd7c --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_69_0.dat @@ -0,0 +1 @@ +x陧譑 =鸰+璵愊B爨抎|Q"s+1"),=揈坂a T"缫艕鉵鍱-3 ,袄KY媤=ZZ颰 .,袄K1#譃!嵟+V锦鲰.蒗2疿眀艎+V琗睎碴.kB柁z狈詗饍毡μg蘫Z \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_89_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_89_0.dat new file mode 100644 index 000000000..aaa4c5267 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_89_0.dat @@ -0,0 +1 @@ +x陧1 呩涧i9'寛炎Hl?L氞^"&M棛?b顥q密?烁渖,9猖!祲zV耆Sc茖3鎋殨鬣綾娱!頯n汪3f虡1c铺寗 3f虡1/f>.U妞腸笏; 2;鏅Y炚+7摭 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_93_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_93_0.dat new file mode 100644 index 000000000..e218fa0ef --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_93_0.dat @@ -0,0 +1,3 @@ +x陧貹 + E褃V骺箮,贠琺逘r鉖H0爗崚礆棒溴2鸼雳忑辌{t醿] +{Q辿撧{寮挙菐;v熵_诔哒}下鼾薒}lн眂菐;v焯懱懱懾眂菐鳇形.Legw3龃瀜e丫@廼 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_0/mask_97_0.dat b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_97_0.dat new file mode 100644 index 000000000..74ac719d0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_0/mask_97_0.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_101_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_101_1.dat new file mode 100644 index 000000000..ec939b521 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_101_1.dat @@ -0,0 +1,2 @@ +x陧1 + 薪骺\Q鳘E簋  1ⅵ蘊<#剿諙-7u欮.糜l苍iXXXXXRZV硎VeIo1,,,,,v%?屬gaaaa摈Y K&K=/窚+蹗吮蟛烛嫁庛 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_105_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_105_1.dat new file mode 100644 index 000000000..e1f5c99b4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_105_1.dat @@ -0,0 +1 @@ +x陧1 休訑鸰违嘼 K驜?"*#W淌榯橥黦訋揓铋qUM9笗噳墘墿訲赙LLv摼扦镔LLLLLLz摼gG01111汄y刬堰槚4m=諞nД+2 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_109_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_109_1.dat new file mode 100644 index 000000000..7e0d6d164 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_109_1.dat @@ -0,0 +1 @@ +x陧直 >訕龡K儉皶纝搱:!鑙Y墮'*3f踫凵右頼购b[轏缕破破贫艝軰9泒cccccc'u.6破破破遍撧6666禦[^g鹻/l秦俜 7蛡 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_113_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_113_1.dat new file mode 100644 index 000000000..1dd666d9b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_113_1.dat @@ -0,0 +1 @@ +x陧1 擂 -8fL(pB巐DM欒9";-嶗蝼;?1頿庑{糪弃%-寣寣寣3:@踑d4弻寣寣寣岤*钔adddddd陨########c]75遐恽棋1蟲Yu \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_117_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_117_1.dat new file mode 100644 index 000000000..8921f6437 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_117_1.dat @@ -0,0 +1,2 @@ +x陧只 >訕龡K儉$鴁 8YQSV'z8扊jz胩蕠瑱^]拮〉嚞ekXYYYYYYYj莸# ++珁eeeeeeee#膊歐VVVVVVVV;"+珁eeeeeee漧'雃;b槐&瞊9瘂/蔎$p \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_121_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_121_1.dat new file mode 100644 index 000000000..64bd8ba04 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_121_1.dat @@ -0,0 +1,2 @@ +x陧1 + 薪骺\棤b褩E亮$扗d譬氖昚怀焧脍咑位击险垠0$祆祆祆祆祆涭莆儋儋儋儋儋躞祆隄潩潩潩潩g朼7莥胴儋儋儋侬崇椊Y??刭{調嶟綝搹 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_129_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_129_1.dat new file mode 100644 index 000000000..62cd1c9a1 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_129_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_133_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_133_1.dat new file mode 100644 index 000000000..18d68dce8 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_133_1.dat @@ -0,0 +1 @@ +x陧1 休訑鸰螀摵h縣槡襹"z谯墮-*dN菱遭袤嚝Q蘑誖脗 ,X癭馽嫭9蝁(na_癭羵 ,X铦,X,X癭羵 #:8 粨 ,X癭罛飀谅綻羵 ,X|息Y\X; 7-; ` \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_137_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_137_1.dat new file mode 100644 index 000000000..284d7beab --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_137_1.dat @@ -0,0 +1,3 @@ +x陧1 +0 休婴骺淜h]D,-t #糙寠Q[T┭ 揔搒7_姏瑜?9|旴&賆沕L0a聞陕&3鲹勆M&L0a聞 &2牸肈4c0a聞 &L樔袑e聞 &L0ab窏w樔f,&L0a聞7&y2筧豺n遫L<01O +橃剧 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_141_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_141_1.dat new file mode 100644 index 000000000..83220ddb4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_141_1.dat @@ -0,0 +1,2 @@ +x陧1 + >9򏛸)3$`s 哺u十>Wd毦 )沢硶'氨M聓3\漝6u箳b匕a脝 6l丶n雏]躈匕9腇薨a脝 6l匕a3癮#o匕a脝 6l匕匕5e16l匕a脝 ]S焍6l匕a脝 m陾亭瓦;C鹀胒憤糏ds怗 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_145_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_145_1.dat new file mode 100644 index 000000000..6a9950f7a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_145_1.dat @@ -0,0 +1 @@ +x陧!0@蟢Α 4a)q2i.YCUO{35睻Z艶宯]蟜N>bdpwtzJ丗}F1b膱#F(’F6r1b膱#F1褽1襥輑F1b膱#F宼F屲#F1b膱#F宼Z}#鲌#F1b膱裭 ̄e衾H声G5嵻贊掝 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_149_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_149_1.dat new file mode 100644 index 000000000..02a3cdc6e --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_149_1.dat @@ -0,0 +1 @@ +x陧1 休訑鸰螀垉q罤轤X騝夎i#Gd虡娬斦涐鱣叕LU鑎莓VR>dKV琗眀艎+V踃e鵲课X壁蘆_眀艎+V琗眀攀;冚+}艎+V琗眀艎+V鯝V昻g艎+V琗眀艎晑+ 鷬+V琗眀艎Vj沂>鴋ewf鯖暰*`uTq \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_153_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_153_1.dat new file mode 100644 index 000000000..2abfca20b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_153_1.dat @@ -0,0 +1,2 @@ +x陧1 +0擂譢9浢bt硩E歝'洩蜨H暁9砮f邽mff睲镔愍涻#.虡1c茖3f虡1cf栐73f虡賕虡1c茖3f虡1c2c鎉?3f虡1c茖3f5鬗f天3f虡1c茖3f虡17utf天3f虡1c茖3f虡=铁薼均jΓ3>廴ばV \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_157_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_157_1.dat new file mode 100644 index 000000000..17344b891 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_157_1.dat @@ -0,0 +1,2 @@ +x陧1 + >94Sd/5傉1仁V)Sk嶫灥画v哕7闯掭黹呟蚭G浒c菐;v熵眂鞑]Z钯眂'镓眂菐;v熵庇+乇觛;v熵眂菐;}V痐荖搦+乇c菐;v熵:;v驇;v熵眂菐;絺;}V薇c菐;v熵当'煌vz#;潓]対糼lw痮A` \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_161_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_161_1.dat new file mode 100644 index 000000000..669ade1b8 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_161_1.dat @@ -0,0 +1 @@ +x陧10休有鸰螀X◤拧 yi~厛琎゜誯v萷贬酬7'M u纲=糫嗁([ 2d葠嵯 +\' 2攪 2d葠!C 2磗0/3d() 2d葠!C 241dh 2d葠!C 2dh謈葠S2e蕯)S2瓬M S蚐2e蕯)S2昅鍈蚐S2e蕯)SeSy)S笤艛)S2e蕯)S;贁)S;髷)S2e蕯)S()S()S2e蕯)SL骴和T6}a*3mljmzC' \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_173_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_173_1.dat new file mode 100644 index 000000000..436918c0b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_173_1.dat @@ -0,0 +1 @@ +x陧1 休訑鸰违''覢y]X1?"g:1鐘漟n硕迢嘶霈伃m挤粰m.?秎俨e藮-跢东>gl俨暦2秎俨e藮-[秎俨e玂`藮e藮-[秎俨e藮-[秎 l俨暦l俨e藮-[秎俨e薞`藮e藮-[秎俨e藮-[秡0}[秡0y藮-[秎俨e藮-[禘儿e[h玂VW枚叚=迢t*| \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_177_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_177_1.dat new file mode 100644 index 000000000..12e2e522f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_177_1.dat @@ -0,0 +1 @@ +x陧1 擂溚Eì宲0譞朽,a#r谑}6}菍抠欢壉~巼8爼茖3f虡1c茖7蝱3f,忀y3f虡1c茖3f蘕_’`蘕&3f虡1c茖3f虡鞭M_翗1cy虡1c茖3f虡1c茖+3f,忀y3f虡1c茖3f蘕_b蘕&3f虡1c茖3f蘹泓2瀌X'膞[cy| 笗3 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_21_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_21_1.dat new file mode 100644 index 000000000..f87e0a112 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_21_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_25_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_25_1.dat new file mode 100644 index 000000000..3a225e30f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_25_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_29_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_29_1.dat new file mode 100644 index 000000000..0a1cb3b5d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_29_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_33_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_33_1.dat new file mode 100644 index 000000000..318949df0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_33_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_37_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_37_1.dat new file mode 100644 index 000000000..5bd9e3aa0 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_37_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_41_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_41_1.dat new file mode 100644 index 000000000..52e9e58f4 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_41_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_45_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_45_1.dat new file mode 100644 index 000000000..b35c567dc Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_45_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_49_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_49_1.dat new file mode 100644 index 000000000..d20d71710 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_49_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_53_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_53_1.dat new file mode 100644 index 000000000..a676d7dfa Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_53_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_57_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_57_1.dat new file mode 100644 index 000000000..896ed4357 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_57_1.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_61_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_61_1.dat new file mode 100644 index 000000000..4165a4bd2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_61_1.dat @@ -0,0 +1 @@ +x30CbpP欇嘏i`鰼&谶H^n╝哾丵籊祘詎а鸰.嵞4I戩鈬y韨帰竊)-5*ョ(欓f[觭m}逖闽跎6YM ;;;;;鸊{艋z肇返猾祆祆祆靭1vw暆潩潩}=鷚u齃%?"=莮e梚 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_1/mask_97_1.dat b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_97_1.dat new file mode 100644 index 000000000..24fa60fc3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_1/mask_97_1.dat @@ -0,0 +1,2 @@ +x陧1 +0薪1骺\椥tncK帖珒塂"慔$输D櫆H$塂"Q&梂er鸋$塂"辩*x[(?/谇鲌'煀nd \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_117_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_117_2.dat new file mode 100644 index 000000000..b4dcce46f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_117_2.dat @@ -0,0 +1,2 @@ +x陧1 + >94!m dO僺\0X,la5#亏E潽>Z[頊址幧赉侥R㏕*旿?Q-満*疶*旿┖?U鳁W*旿㏕轃+旿┰ぺ~m5;鶶&+ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_121_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_121_2.dat new file mode 100644 index 000000000..a2a0097b1 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_121_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_125_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_125_2.dat new file mode 100644 index 000000000..0ea40fdae --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_125_2.dat @@ -0,0 +1 @@ +x陧! P坑旣_傽 U(踜p@^MZ5舄铗-鷫:瘀VF_炋\t:漀в閠跫y濶譹t:漀в閠G椼;漀в閠:.诵8:漀в閠zA}祴粼y濶譹垦;+n& \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_129_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_129_2.dat new file mode 100644 index 000000000..bf048394b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_129_2.dat @@ -0,0 +1,2 @@ +x陧1 +0薪а鸰甂萔凇▉'.婶!秝]A0X鹉~槧  !洑啵爁K# x楩y亐4 vey亐@^+  簙櫈 ㄛ L#陵veI \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_133_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_133_2.dat new file mode 100644 index 000000000..9e78b6de5 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_133_2.dat @@ -0,0 +1,10 @@ +x陧1 + 薪骺湅&碻Ⅳ匧怵旤翀Q-絞=Aq昩膪箠湿<媣歜+)((((((((In*鳝l 唵f幤動行行行行腥44& 崿)O讶氜Y揊󨳄44444444c4~9S:3行屪榩毲> \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_145_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_145_2.dat new file mode 100644 index 000000000..9ff2bbf3f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_145_2.dat @@ -0,0 +1,4 @@ +x陧1 + 薪r] +,t圦^&蔆堭摟埔~ +褖⑸j~m删.斟嶧g腰MDDDDDDDDDDD許T欁垐闐垐垐垐垐圚巿dZ慙+纱DDDDDDDDDDD2-戦'"""""""":B凿转紼錣剤―d鷫钧 唺 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_149_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_149_2.dat new file mode 100644 index 000000000..d52e0484b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_149_2.dat @@ -0,0 +1 @@ +x陧;@擂永/gcaG諦XB'-埣聢籵u乍UQ絛鮎VOmT兾*谦;;殜妸妸妸妸jí寤潑狤e2PQQQQQQQQQQQè TTTTTTTTTTTTr3暅良ⅱⅱⅱⅱⅱ3豏╀ &者Ts䦟坊猭あz_e2P=狅d \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_153_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_153_2.dat new file mode 100644 index 000000000..3b0604101 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_153_2.dat @@ -0,0 +1,2 @@ +x陧1 +0 休婴骺湅8ZP!窥B兑蜩Zu匀璩"麚靊u厚鰴嬋*钗查)]MFFFFFFFFFFFF鰡%= #嫅ddddddddddddr ot2箚yFFFFFFFFFFFF& #搆悜憫憫憫憫憫懮5热L 2222222222(Y陲湡惖7扇"d袤@H \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_157_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_157_2.dat new file mode 100644 index 000000000..2baf535e2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_157_2.dat @@ -0,0 +1,3 @@ +x陧1 +0>s6侻箥蝢UH1&U缣樈f/u-琨'耥.蝴[蚄GGGGGGGGGG鱄讄NG〒(ttttttttttNF;::::::::::}枎N煡#####觶梃$ +>珦验厧帋帋帋巒 麬#侯^AG棫(t =3{ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_161_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_161_2.dat new file mode 100644 index 000000000..d2df7594e Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_161_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_165_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_165_2.dat new file mode 100644 index 000000000..2e6cd7c6f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_165_2.dat @@ -0,0 +1,2 @@ +x陧1 +0 休婴骺湅婅?駼芕侟G嵓灥%妁嘘遒*+鷂巉仕s MIIIIIIIII鶘2d;¥l4()))))))))))e鰍J賳IDIIIIIIIIIII)蹱mPRRRRRRRRRRR6l簲敳 JJJJJJJJJJJJ賳}淩禷QRRRRRRRRRRNe騅?R韾敂蛿&錡3U \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_169_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_169_2.dat new file mode 100644 index 000000000..4052062b8 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_169_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_173_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_173_2.dat new file mode 100644 index 000000000..0a30ba530 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_173_2.dat @@ -0,0 +1 @@ +x陧+@ P蟟瑚佢f輙髿:>y &d U攥札蚐创[睨≌]5Z;a价5V蹫櫞创创创创碅诶颷Z谒寲枛枛枛枛枛枛VΛ≌僆0ZZZZZZZZZZZZZZ=-L傃乙乙乙乙乙乙谊羑i鮜寲枛枛枛枛枛枛VFK摲创创创创创创哨?傒hio襃0卩}尝o \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_177_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_177_2.dat new file mode 100644 index 000000000..d2c52f990 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_177_2.dat @@ -0,0 +1,2 @@ +x陧1 + E>4扸$蛜 ,C妳&U髮搠;Oo妅5b餮蠙徕髱G皒9ホ%&&&&&&&&&&&n$铞鮋L|囏v#&&&&&&&&&&&&&州bb輱槝槝槝槝槝槝槝Xw#&州l7bbbbbbbbbbbbbb軲"州l7bbbbbbbbbbbbbb軲a!&州垑墘墘墘墘墘3)U*F> \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_45_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_45_2.dat new file mode 100644 index 000000000..ad44ff188 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_45_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_49_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_49_2.dat new file mode 100644 index 000000000..6e8edff24 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_49_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_53_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_53_2.dat new file mode 100644 index 000000000..682cae2aa Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_53_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_57_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_57_2.dat new file mode 100644 index 000000000..66a5c056b Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_57_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_61_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_61_2.dat new file mode 100644 index 000000000..77d3815eb Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_61_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_65_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_65_2.dat new file mode 100644 index 000000000..caf184ada Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_65_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_69_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_69_2.dat new file mode 100644 index 000000000..6a3801bf5 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_69_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_73_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_73_2.dat new file mode 100644 index 000000000..74945b718 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_73_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_77_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_77_2.dat new file mode 100644 index 000000000..903cba4a0 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_77_2.dat @@ -0,0 +1 @@ +x陧1 薪Я鸰畫CM娯>騁髏 勋e蟥+幾FWZE辑m&g迖FQ秇夰F+譼/岶YYv岶歹筹弟欹誮玺[*7蛘a \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_81_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_81_2.dat new file mode 100644 index 000000000..17a9ac2a1 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_81_2.dat @@ -0,0 +1,2 @@ +x陧1 +0薪а鸰甂梢¬i!O\"釕A鴲埆騗:倄粦缑bW到1舥圈&壞_壍T 螊6淗$壱U^塂鈤b庄b=g埑鉃鸛 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_85_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_85_2.dat new file mode 100644 index 000000000..72c74ff9a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_85_2.dat @@ -0,0 +1,2 @@ +x陧1 +0=1骺\桞7≈O$釕A0$猬8W瓲﹃wjgu辵凯妲姟R㏕*u曶S支帹鏁J┺T驤R鰢贞蔻Nㄦ禈V \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_89_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_89_2.dat new file mode 100644 index 000000000..06c9a4fe3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_89_2.dat @@ -0,0 +1 @@ +x陧俦 0 >尤/&E*c馇Qq怆门 獄f$驓藃M呥弭勥<幦sa#d2橪&?顰rY熒d2橪鯣"3橪&搟寿隦ygw;鳃 凂贗 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_93_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_93_2.dat new file mode 100644 index 000000000..f5202963d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_93_2.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_2/mask_97_2.dat b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_97_2.dat new file mode 100644 index 000000000..38842b989 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_2/mask_97_2.dat @@ -0,0 +1,2 @@ +x陧1 +0薪а鸰甂i穠H4 )隷%s譥秸僤3K嶰1^歛啩L熱,$扝"$KzRP梩[I&X捿9$扝"$掍I$y抯I$慏I$蓳銲$I$慏蓫%e胩s!辜=棛備LAZ5棞'逄IV龘r頽富/2o茀 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_113_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_113_3.dat new file mode 100644 index 000000000..023b27304 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_113_3.dat @@ -0,0 +1,2 @@ +x陧贏 +0 D複N撧rnD肼Fj2鶮偮菴t?W莶頩搃昀.q樟媽o齈 舣%Smj致7宿:珐*N┦:@:嚪忑翃竞*箠**f┮W9d2槴á*j*}昐@犑`*j姜姣唰6餔l壕暄棵}照}照昑嬁U遖24hnt \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_117_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_117_3.dat new file mode 100644 index 000000000..79cc04d14 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_117_3.dat @@ -0,0 +1,4 @@ +x陧1 +0 衇.I贐$鷮噣?~!<荙?#栫缎5/tI軯8ow鴌閧哏猬}鴺^愩~嚔絨鐚塒聞 &L0酛幄1竖匣茛餌譑'琂 &L0釢咍胣 +0a聞 &LxgaG儼*&L0a聓 +g{2噑螇~橉7\]%r9nZ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_121_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_121_3.dat new file mode 100644 index 000000000..aff5a7be5 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_121_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_125_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_125_3.dat new file mode 100644 index 000000000..e2febdbdd --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_125_3.dat @@ -0,0 +1,2 @@ +x陧1 + 衺N3r蹌]穳B袘'?<锣/碳V鄣蒹 镆彅<0-_盝鋄? +w阅^;*y蜾蓳'O</9舷莎9<犷y悠'Oz喼#E)R慫@猙耹-)梯狂<擐*㎡UR"5&曵洑5*i鷈 J]+ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_145_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_145_3.dat new file mode 100644 index 000000000..338b7e7a8 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_145_3.dat @@ -0,0 +1,3 @@ +x陧芰 +侤旋|腕畭:宆 Jy獭おyMь譲嬜-' +9桽埤枝K订9肭e)P韞U梗we-m j渊Q5j渊峈薸佾訤4譥wk}0+档jRBR5j渊Q粌毦逴MBJHj渊Q5jwP幼挫霜IH I5j渊Q统曲殩碣c 閣5j渊Q祳:褄S珟*2くUZ_C易*璭_O泶颶%dI券鈫b \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_149_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_149_3.dat new file mode 100644 index 000000000..30bc5fabc --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_149_3.dat @@ -0,0 +1 @@ +x陧蹵0衹O蠊遘犫H闋聧魮'琙2{oV踻螜%禉>倽yR{!8钿脗I+JpI|#遁f鞏5魏[釶 A $H 俀})&<棍E偋 JQ)J A $H 羢5z嶠エ%H A $H愢4A峖5A)*E $H A $杞 嵽3颩HQ $H A 纥3蕨`轕畂⑤>X{讟澡+毖Wb?∴爝`I雁)5%d \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_153_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_153_3.dat new file mode 100644 index 000000000..89cdec031 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_153_3.dat @@ -0,0 +1,2 @@ +x陧蹵 +0衹N髎斯)7mJ︴,}8店X=c縒^G揺捿N}橛o隣%uJV/蛋搟縿䲢%夸>}橛燓陖闗~蒓>}橛>齉跽=/褩鼟>}橛>}戢鱱{栾讞鼟>}橛>}戢鱱{锜/%?}橛>}庄.N4醿廣Mm枹R暁憪鴻裔竧(1參| \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_177_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_177_3.dat new file mode 100644 index 000000000..9586979a1 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_177_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_21_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_21_3.dat new file mode 100644 index 000000000..bcb4eec49 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_21_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_25_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_25_3.dat new file mode 100644 index 000000000..0ffc375fc Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_25_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_29_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_29_3.dat new file mode 100644 index 000000000..6150ac128 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_29_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_33_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_33_3.dat new file mode 100644 index 000000000..6053b5e39 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_33_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_37_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_37_3.dat new file mode 100644 index 000000000..5dea5b9cb Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_37_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_41_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_41_3.dat new file mode 100644 index 000000000..ca9ddc2aa Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_41_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_45_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_45_3.dat new file mode 100644 index 000000000..3daad97f8 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_45_3.dat @@ -0,0 +1,2 @@ +x陧擪 + D鱯汃䲢隖J(&)舫0d▲袊挸F辡!孾8=枸&薸a挛D)興8&A苏a翅累 1'橧朓敨壮𽛽 ex纴撨救 澞襂禧&譂疮C鐄赐Jy \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_49_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_49_3.dat new file mode 100644 index 000000000..7f6508ddd Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_49_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_53_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_53_3.dat new file mode 100644 index 000000000..8800beab1 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_53_3.dat @@ -0,0 +1,2 @@ +x陧朘 +0D鱯搴 壯! -(.Bp&弢"-t&`苢慟-"嗮9賍+)Be/H8俱D%a~毿}spKFN橔鰷=,;;a^t4鳀壪\欈F櫸吕饰SNч \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_57_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_57_3.dat new file mode 100644 index 000000000..4e1e5da38 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_57_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_61_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_61_3.dat new file mode 100644 index 000000000..bf1a3cc7f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_61_3.dat @@ -0,0 +1,2 @@ +x陧朅 +0锞f鐉4-%殨愐*免d靶p!輞Z塬夋倁鋸(~=&邵摯)溙R2"/"<9鐘F螉=rb驓"/搩钐礴rw"2沌萨鵅蜩#3-0-K徖W \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_65_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_65_3.dat new file mode 100644 index 000000000..85892089b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_65_3.dat @@ -0,0 +1,2 @@ +x陧桻 + D4击縗?R曐 吪,!∣姴-騈v伄1:耤渦飦 "U婱脮F ~jK█磪-la[^q^砆\闇=崊o-laZpU肄B瑒 盄I抵K礘z散|1脥 鲪 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_69_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_69_3.dat new file mode 100644 index 000000000..55318a87b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_69_3.dat @@ -0,0 +1,2 @@ +x陧琢 +0 衶&燇2'棶d l=,伖F筋閥;$ъ嚖擶E-R剨ǔ:ⅰ%T,O2阶g"",踩/D蛓膱樄羯﨩洚橙",:N剉E馱N#(&,,戜暫姰隴x貐 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_73_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_73_3.dat new file mode 100644 index 000000000..15be77f64 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_73_3.dat @@ -0,0 +1,2 @@ +x陧楺 +0 C{氺䲢骻;J?d袁K=赗a胹JhTJ6ex蝛a簮$憂IE,-/延XB濊*褏=蒜烅谝齱ee4櫌濊竧鎏抰歀褖瞰t爐斶珳栲b 杇F協 醧乎o謉⑿dn-? \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_77_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_77_3.dat new file mode 100644 index 000000000..ec7828044 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_77_3.dat @@ -0,0 +1,2 @@ +x陧谹 +0 利&缂瑽奿諶aK"t閌I@扯|逗咁篂fX楒痹猸yi鲥髄鮁:Sza18G纠塱f楰*─?:Y蹸1鞂烄(釅窏牿J*─捑抝蔿*枕昑RI譑填R^貦鏈辦s)c)c)矹Z扼恁鉧 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_81_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_81_3.dat new file mode 100644 index 000000000..47bc0f793 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_81_3.dat @@ -0,0 +1,2 @@ +x陧1 + F=骺\,J袵A齢j捞>谫#3X櫪掵:鏺髟筡鸉M J輍u伹3辅>T璟鯚詛倾PSg鳳'股k訴竕摘U_郫U玍暬=P 蝟哋:W覞嬣漭j黐焀x蝝 5 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_85_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_85_3.dat new file mode 100644 index 000000000..02c4f8cdb Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_85_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_89_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_89_3.dat new file mode 100644 index 000000000..2b4cb59fc --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_89_3.dat @@ -0,0 +1,2 @@ +x陧1 呩涧)骺 *.@郩咙 冇觸倍夘箳e诺6邰倠鋡5*黥) o鬷↘畱4nk>凌1}d>溞@ 4蠿YC緊 邸1<〢h爜Ft + 4蠤嵉5棡1罻r绎>7麊G祡}ぁx7|荖慻祁贜 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_93_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_93_3.dat new file mode 100644 index 000000000..b4cc8a971 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_93_3.dat @@ -0,0 +1,2 @@ +x陧貯 +0 D複N撧rnJ瑟Q~锽▽06狖產岣<<ψ噰e6譓桼欭C臥 辣籼搃氊9M 2 僉k女絛D籿*"a吩X唈Bd怉d猟梏Z隼Td怉d異qY0ex鲼黾焣eх蔔岤潻&WV撴Qvc \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_3/mask_97_3.dat b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_97_3.dat new file mode 100644 index 000000000..7adc9ebaa Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_3/mask_97_3.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_101_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_101_4.dat new file mode 100644 index 000000000..1c97dc048 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_101_4.dat @@ -0,0 +1,2 @@ +x陧欰 锛s^1燽漆袠猐秘4歮+8+榁e^H璕]朶c畩 +o鮓N#割X樨+l 灜HE砪p \梌.9q馱9镧"騾:咐.熸B篦鵅 \0蘟樏P迁cp \尥棚O晵婲jpG拄}擊}ナ$.渌厞 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_105_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_105_4.dat new file mode 100644 index 000000000..0211cdb3a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_105_4.dat @@ -0,0 +1,2 @@ +x陧欿 +0 D=Mr斯戻A 呂T醗慐F驦2 #幦涕耿卂I!搵啁g霐ぱ簭摀-斋km縊襻]sS T6*'8 N熲$揞'N嬍韃箎r諹*G9r|篶蘙粒cN猬轠_=谱5糬蚃紒y 1*q泇 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_117_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_117_4.dat new file mode 100644 index 000000000..386725918 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_117_4.dat @@ -0,0 +1,2 @@ +x陧谒 + 薪_s]4ォDgn蚋鄩2Jj}粗谝痉RsSWG秿R繉骚啚)5瑵晦E嗎鋗#墀曘菊餶_"脄煍3耚箁迨+r 璍籯|/{劰;'奏/#\p\p羺># \p\p#鷪>q巔\p僚.壕$亢菼q饯 d魬袵扲话_4﹁  \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_137_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_137_4.dat new file mode 100644 index 000000000..0c09c487c Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_137_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_141_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_141_4.dat new file mode 100644 index 000000000..62b03f243 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_141_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_145_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_145_4.dat new file mode 100644 index 000000000..33fb21124 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_145_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_149_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_149_4.dat new file mode 100644 index 000000000..de99310f3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_149_4.dat @@ -0,0 +1,2 @@ +x陧芰 +!薪_sm溕 寗+X計9=.=Zk鏰敋]辙> K痡虼o 惊齶疭獖SWK鵝m俗尯兠j閺線肻拪圭歇2 W\qW\q耪癞"~愤 璲v趖蜿獖v韮\qW\qW\q%g3 鷷}+鈯+鈯+ r9儨A坎rW\qW\q艜淎 g3鑇鯝 W\qW\qW棲]錠祣v{闐煓3匀!躬僜閃囚耝<韗/ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_153_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_153_4.dat new file mode 100644 index 000000000..e827dd16f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_153_4.dat @@ -0,0 +1,2 @@ +x陧10擂si,')p!涞4.;苯縒Um哏j=坎劜臧.NO峻>氱T第峓骶S7v苒済秖晞?骥抟 +歿pe筼𚕶銓38銓38{豗Xz疳,安踎鶲覻fe裎3s榆鋵38銓38銓3箚\C!仔咸Ms𚕶銓38銓3武r 箚\C?379銓38銓38銓3箚\C!07蚆8銓38銓3挝q二陂,徊mMrs沰v3纞朩 WB \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_157_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_157_4.dat new file mode 100644 index 000000000..ad5fcf69d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_157_4.dat @@ -0,0 +1 @@ +x陧10擂s4"FP=iRX垄X0珉┆u 4f弔阎l}襪绝啐為鉃S|菜褞隨 幠P5<]錾rw躴w躴鳎蝗^Q衢N6脧閆竤邫虫,w躴w躴菨>瓿5g广;罡銕;铘Y}V熣g輜,w躴w躴w>瓿娆9w躴w躴瓿>刖3g蚘罡銕;罡泐[ww?P3苠й蜓茦輌gt燀惍;鱙.3w4A \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_161_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_161_4.dat new file mode 100644 index 000000000..7604c4540 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_161_4.dat @@ -0,0 +1 @@ +x陧贏侤 旋渇纩梥!AL澞_|,4騦)i锆m策l0'槛 +E耕叄 鳀]N\宏黿#嚙2熺/_{7g垓9鞆假吩乐}箫2r!嚂?}-#T椪e9鋹C9鋹C96S蕠瓴禾!r!r!囨羲捋榮誩9鋹C9鋹C94鐮_鲛茰億.r!r!歴/泂0 2r!r柔8歆}D偯戊鐆y齬忥棲DX巍|x脇!2 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_165_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_165_4.dat new file mode 100644 index 000000000..d83d63165 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_165_4.dat @@ -0,0 +1,3 @@ +x陧贏 +1 衹O愚r."*?鷉雅蘌傮LHI飾k滴ZQ8 +Gy麝qk-灴蝞5节+?桃聩葩盔|謳饕穔Kn瓻衽逛扠.逛扠.?2海.螅|糆J2<:.錕u\鐠K.逛扠.逛捤楘窨昵mu)_8梊r%梊r%梊氎袕;'2!_8梊r%梊r%梊氎袕;'2!_\r%梊r%梊rinC?nn9懝 鵕鐠K.逛扠.逛;.驢qY'瘦介﨨稦?轐翥暍,闞军閨 My*3 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_169_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_169_4.dat new file mode 100644 index 000000000..4aac95c1e Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_169_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_173_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_173_4.dat new file mode 100644 index 000000000..9df4d865d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_173_4.dat @@ -0,0 +1,2 @@ +x陧豄 +1纝N撥rna ~ZY!涡J裻^镒5(/蘪k杬[pj掸_?~v:|jw諙_m頧z騩鯘栍6?n宿<霭j$p-奋r-穇p[z=V礞朡恙3萱T閂空o 躵-奋r-奋r-穜09L 躵-奋r-奋r-穜09L 躵-奋r-奋r-穜09L 躵-奋r-奋r-穜09L 躵-奋r-奋r涙v6鬟?,e錪=*K6诃~6` 彺* \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_177_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_177_4.dat new file mode 100644 index 000000000..6437d2511 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_177_4.dat @@ -0,0 +1,2 @@ +x陧谹 +!阑.1B勯My![ c齄\c尣鸝V\管q搳7哜軪髧O齴g~k滕丝j;pg攰u餐[*7娮渐忽缞骉p1s1s恬q!嬨爦w/騬8X曲眫l榗9鏄c9鏄c9柣奢鋘r7箾~l0Wp1s1s1溯鋘r7箾~l0Wp1s1s1溯鋘r7箾~+8鏄c9鏄c9鏄c庡nr7箾躆瞀+8鏄c9鏄c9鏄鉻躯E菣熒Ny筟徦軷+緣cqN\,4J \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_21_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_21_4.dat new file mode 100644 index 000000000..e006b67e4 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_21_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_25_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_25_4.dat new file mode 100644 index 000000000..0c7c44bbb Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_25_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_29_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_29_4.dat new file mode 100644 index 000000000..c28dc20e8 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_29_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_33_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_33_4.dat new file mode 100644 index 000000000..5834b6fb5 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_33_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_37_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_37_4.dat new file mode 100644 index 000000000..4bf2e26ed Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_37_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_41_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_41_4.dat new file mode 100644 index 000000000..b75b7d052 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_41_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_45_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_45_4.dat new file mode 100644 index 000000000..1b921f300 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_45_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_49_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_49_4.dat new file mode 100644 index 000000000..e417f9476 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_49_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_53_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_53_4.dat new file mode 100644 index 000000000..7e88826dd Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_53_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_57_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_57_4.dat new file mode 100644 index 000000000..84669c7d6 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_57_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_61_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_61_4.dat new file mode 100644 index 000000000..d127c3be2 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_61_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_65_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_65_4.dat new file mode 100644 index 000000000..c24343d94 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_65_4.dat @@ -0,0 +1,2 @@ +x陧桲 D鳒錿乆P.4覧1^裸觝v喕O蛒匇閃擆0 癜JH呕嚱z[^軋怺v馊 +yy悋Z纊=`剻陃穋莾顓<葍<C/)z軕 ∪们蜟寶"" \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_69_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_69_4.dat new file mode 100644 index 000000000..a73b1144d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_69_4.dat @@ -0,0 +1 @@ +x陧;0CwN骺\棃~え;C$3$<)/me违T睉2 牀:^VV%瞶A飊萷v卻"yyy)z鲠膗6洁紾葖既薸^/<蕋觬聄r)9亭l鮮祗倎.秶 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_73_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_73_4.dat new file mode 100644 index 000000000..72f89227e --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_73_4.dat @@ -0,0 +1,3 @@ +x陧楢 +0 飝M祥〈)1-俵E3=詝瑘(拄9烼莇因朎/eO ZO烆阚呕K鞾;埴pS5+懩隢炰I瀶炧| +"〆靨+D輷鍺s'O颏W巏钿I灳饎靻虨袱Fr躨9&篝 嘳?皿A \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_77_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_77_4.dat new file mode 100644 index 000000000..993c48608 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_77_4.dat @@ -0,0 +1,2 @@ +x陧樠 +0E啐?姿jFe80犏 u豊0塴B磫6:玥<鯑ДC躤\, 嗢攒梸濜& rd0 籗z7z+魱阰夯榆脘嶸鈲u:7z7Wo繗 宋S)霐掼mo3 O嫕y;:*hマ>a顲e \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_81_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_81_4.dat new file mode 100644 index 000000000..dd6521613 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_81_4.dat @@ -0,0 +1,3 @@ +x陧楢 +0 飝M蟳慩E m7"89錉2捬笭鸔1莩屉+竫墄;庞t35DIY鞫1x\:u灣}瞀e/ 戋#聒彚Th< +夰鵘z<沧5駡G<捵5{岹<彚慨5鸟<楗>痌炞U膔徔xu 茺 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_85_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_85_4.dat new file mode 100644 index 000000000..c8d5123e2 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_85_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_89_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_89_4.dat new file mode 100644 index 000000000..5b9bd7ec3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_89_4.dat @@ -0,0 +1,2 @@ +x陧1 +0 呩涧I9%  V摒趢f譺髋0}騴=#9辂武視:麀泂蠊1B諄g&4pg渜.p.戆&g耇05蓃g鷻sg渜鐁g鎹u38銓箅漦.Egm不b*型&7陨臀?礴 :撄 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_93_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_93_4.dat new file mode 100644 index 000000000..be7f5e525 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_93_4.dat @@ -0,0 +1,2 @@ +x陧貹 + 勧涧I筺$}P偱孊鵠笐N椐菮臀犷%s宖k耀Cz驆oA}a式2絴ゎ閪乩狣&瀕=リ儡绂峐湲暳w躴莭郴爍\蹺閅硺j岙K_馕y鐬w躴wz$==;罡銕鸰輯>荃闊+裵佑鯤擌9铫D辤 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_4/mask_97_4.dat b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_97_4.dat new file mode 100644 index 000000000..5d848caad Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_4/mask_97_4.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_101_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_101_5.dat new file mode 100644 index 000000000..c21869e83 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_101_5.dat @@ -0,0 +1,2 @@ +x陧氀 + E啐氻+%=躆3Cbv蟾 熏洬蚇k没勾gqkqq{%鬙恳螯o鐬,i枿臟謊渆缌3[泑夘i灦鲊觉樸¬ゥh]` 芵 0κ暆备z舜T0Gu嘴/q8F畡1儽3:W黍>鯓#侨0c0鯭8圾E=F#帒+a 掐腦滞+cV%9W>唔Q瀅Tk┕揧-gL各箢弽q珼鑹嬒 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_105_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_105_5.dat new file mode 100644 index 000000000..bc8798c64 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_105_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_109_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_109_5.dat new file mode 100644 index 000000000..25a394402 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_109_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_113_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_113_5.dat new file mode 100644 index 000000000..25f42b8b2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_113_5.dat @@ -0,0 +1,9 @@ +x陧浟 +0D稆気筤663[鑋EDq蘡樑+蟡脃れ81曽\c +穻7赾觖?u齷DK4,k鹓--3[鹘碪吰倅刄X匲X齋V牫:汐嶎,麒|纨泻S緤 + + +V色⒓*裍襕岖* + + +z奤*N鱒*綣UX匲X呎S箬XijTi4f辊ZkU^鑏囲戥~奤戠x }债Z/r \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_117_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_117_5.dat new file mode 100644 index 000000000..f236940de --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_117_5.dat @@ -0,0 +1 @@ +x陧浟 D飢Mn/*{罬φ+剺p蠭_&m悎-鋷柩緸C骢洵蛳32弖?o-kgB7wc=瘷應U%y韔甍R鎕盈s苔Do:侄ykQ^a哸哸濁qOg趇J ;鷾濁qOg)訆忝0 0 0 灭[v酌>鐏=>忝0 0 0 肙f榸3幓=>忝0 0 0.3黠Z$8\鮬斣鸣w4角:悒諾纏:q怪涅 7 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_121_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_121_5.dat new file mode 100644 index 000000000..9bb5c4158 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_121_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_125_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_125_5.dat new file mode 100644 index 000000000..2161c50a4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_125_5.dat @@ -0,0 +1,2 @@ +x陧汚 + E鳛&錰c;S炊$?嫈娯弿廞4脶Ya遠没yJ}9篻=宗=﹍i煷.;nh_驮wz龔.濛q魞CW敹颧y鹤 u朕霋Pk;堂<堂<堂<堂|*髊膣吸, m搆W鎞鏝蔿% 熐珩y槆y槆y槆y槆套^2浒卻X|熐鏰鎍鎍鎍3蠙9lH嬒泱<堂<堂<堂<炭涽鞗伩蘙恝觞阯`鞕T碲鹮8蟆^v魕帿 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_129_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_129_5.dat new file mode 100644 index 000000000..f0c1d6502 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_129_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_133_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_133_5.dat new file mode 100644 index 000000000..46be8b094 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_133_5.dat @@ -0,0 +1,2 @@ +x陧蹵 +0 D呀Ocr]4%1mCT迳瘎绪x螠[Dv=漆{F朎は彽堜q?菘9鹝e循鶀'2^昪滅4G敒澺:3=J-汈F0宍#虷隐w'#<{趡梈4 :BG鑸F0宍酖~剮:宍#?聫#t剮w-宍#?聫##t#F0r咃}Q鯕}e蒡R嬟;<C閂}褩戩螈速Y-H \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_137_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_137_5.dat new file mode 100644 index 000000000..064e7f2f3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_137_5.dat @@ -0,0 +1,3 @@ +x陧苎 +0 呩>M簋/k1m馒X撼=墫鼍等h陬s忓揍H"k M:铤陽3q襉W}9捲柆I攭擙H1貵;- ﹕祛诙?襕%M + v皟禳#;g^菰3誨}6嫇9燮唯迿譆7鯏覄镳#gv皟靈;壮^‐$耘鍩久w鋖亓v皟靈v;迧謢7蕎邞砤;亓v皟kA旓#gv皟靈=N2w鷛g婴@n?垭c}絊Q:鸝d?+ㄏ晰9v莆z)I孑酨 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_141_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_141_5.dat new file mode 100644 index 000000000..60c1a8e81 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_141_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_145_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_145_5.dat new file mode 100644 index 000000000..9303c07f9 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_145_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_149_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_149_5.dat new file mode 100644 index 000000000..4256cefd0 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_149_5.dat @@ -0,0 +1,3 @@ +x陧躘 +0&往c禒}-s+'嚋^;A莤庩=ε隥_gU輳锏鉿轌T斫肴V鹿嵇淯猆E瀣骢_I谴;幥T1做朋籼爦┑ +]W 2 2揉 秥o5豼忬陠瑄I:(W滽璘韬拑rP蔄d怉d怉; v裲磔_zNO嫦{2rP蔄9 2 2揉9硌遾廭O鎸瀸攦r衱Qd怉d怉齨綖屵M華9(d怉d怉痎W髣 庝Z呡.+暢G焇隟晟苎`椎}薫_F株鳚k \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_153_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_153_5.dat new file mode 100644 index 000000000..deea09d77 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_153_5.dat @@ -0,0 +1,2 @@ +x陧苎 +侤衱縡畻鞋`镏貲"I读胑<:a評,7铽鯫f鄢歶离P6庮~耥s緕澱料s尴,j恼琧胩V濚Zv擗m榘 胳梥^u蹾甕饀&沴采&沴采&沖9く壁荜 ;]衬^衘s玂;鞓軘況Sn采&沴采&沴9酵y讻冎鑪A 挍rSn蔒6賒揗6賒揗6У檗击@/璞$7濡軘沴采&沴采&沴z沤zA濩憶rSn蔒6賒揗6賒撏Ol铪7岚歎靧禢譀籉c敷剃蚉砅棞S,l;H鸒 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_157_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_157_5.dat new file mode 100644 index 000000000..176e2a69f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_157_5.dat @@ -0,0 +1 @@ +x陧10D艳яri湗( r*彞 刓櫿~>C铮*瑅癸s]艤_{W!莦锒/陷搋鬏耪)剿櫥沽鲣弙 浵V懏6躒憰藁鸸,f1媃蘠硺嵤 蘮^o>傀韁O颹,向,蟗硺,f1媃yV灂g錣炲Y炲Y蘠硺,f1+鲜瞅<顺<顺緧1媃蘠硺努<+鲜瞅,向,蟗硺,f1媃yV灂g錣炲Y炲Y蘠硺,f镙ukys77椚靰飗鐘澤鎚b=賡w)锘tW営: \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_161_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_161_5.dat new file mode 100644 index 000000000..70d5fb008 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_161_5.dat @@ -0,0 +1,2 @@ +x陧躀 +傽薪ч宀袲p@ 鑟|m 槩遰Hk悒ぱ~c樈o砠菝Jz皞#5l豢盥_92垲 ⿹[瘆拒萌Z脢焽=滌T2苺譖憴2[cV扈藽朰f欏'-垤X9>黺~u粨絪K5靈蔱,楀2,程2,程2绥-霔咝///o_q緘K晴句瞈査r檈朰f檈朰f欏硸跛o钘+/,楀瞈f檈朰f檈朰f俪鷈鲓x禕.薳固2,程2,程瞘+羲駆匼査r16,程2,滁<渧.譺毳喠睫紓欭rO5燊詓_齔紊攗薎_燹* \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_165_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_165_5.dat new file mode 100644 index 000000000..94af813d9 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_165_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_169_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_169_5.dat new file mode 100644 index 000000000..921a77076 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_169_5.dat @@ -0,0 +1 @@ +x陧垩j0绪|弯/秐i耽K梗'娙.r:罷-m&禯踴9y黴3r礝g鄓糘O_z庄`]肪遡t胴鉻劫駃lc傱13j9%_g|朐)诚>诚>翱=te&觜_4ま部=t熓}/麇>诚>诚>鸒疮r5/u摴>/麇拒g焳鲑g焳鲑g整O桔sv甇钏}/髻g焳鲑g焳鲑窎G給-弚{扄r_顊鎔焳鲑g焳鲑g_蒋搠n彽诚=熳n]4Nkе尾_署玀娇灕8韒?碨F叿< \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_173_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_173_5.dat new file mode 100644 index 000000000..f9a674136 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_173_5.dat @@ -0,0 +1,4 @@ +x陧踇 +0&王G1续g銬)[癈z摈eD盅费嗺=R綨6差筅F翼Jm 尴Jq鱌渳離搛 s轭裚觉}堕G佢F﹜;烅 +[;齗欥榀e胟[QbT玬眣&0 L`绿刌?屺丘乇葳婕w貨 f齽胏鞘誚N9两樌&0 L`槓讋Z齽0=F=F9ANL`樌&0 L`Bz琘瞗I忨=F9AN&0 L`樌&0 瀷磃I徰s憆倻 ' L`樌&0 L`<i蛼g"9AN樌&0 L`橉V璦B疿"齽辈,U璵>伥 +=苭槓骦秅B螠P !8 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_177_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_177_5.dat new file mode 100644 index 000000000..b07c636b6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_177_5.dat @@ -0,0 +1,11 @@ +x陧菅娐0绪~吞芫丨垰 蝩O,"% 耵:$Xui=拐讯諆燂舾g髌?螐q.So~鎧讐辊W:髾=巋1c躴篪桉⺌鞌嵌碱i侵!骷哏韗哳槚8苼髽琡+X +V皞 鯅雨廽浵j;8苼骕異+ +孤oV皞琡+X +V扳?[1礮駂-殖5Z;Ф┒鵵mS異+ +孤oV皞琡+X +V扳;薢,低Y礛笲異+ +V皞琡+X +V皞鰴Z媏煩讦}匼!W琡+X +V皞琡+3Km>S笲異+ +V皞琡+X +V阿c銑浨{g;Qq5U黄蛰烸跮0+*&YD弎锐轸*6 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_21_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_21_5.dat new file mode 100644 index 000000000..04f97ea69 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_21_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_25_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_25_5.dat new file mode 100644 index 000000000..c20b59b1f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_25_5.dat @@ -0,0 +1,2 @@ +x跐慳 +@!4铪 蕹盛 ?,"旁"j溅?n<禣垆a 瑆, l}rG婱;夕9[烅 あ_漻|熓=磍4lKv \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_29_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_29_5.dat new file mode 100644 index 000000000..217ec1b8a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_29_5.dat @@ -0,0 +1,2 @@ +x谡抅 + 冞=M葵_n0W .珟=-翢4y譈煊+ 壏躌揍撶魣媶$辑苾.=s/,+頑7烍碀峲畓膥>=G橘陰Z鵼:椳DR煫 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_33_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_33_5.dat new file mode 100644 index 000000000..726d7fd75 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_33_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_37_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_37_5.dat new file mode 100644 index 000000000..6d32ca6fa Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_37_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_41_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_41_5.dat new file mode 100644 index 000000000..e07c6172a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_41_5.dat @@ -0,0 +1,2 @@ +x陧TA + 击5?7槴萖Mtx掖x─ 吀?@倶7@蚓~"N$鯂S稚澳{+C敞A'述r\P宲<迯- 秃:S3s乖夰刍宿┪z#烍qw欝 > \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_45_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_45_5.dat new file mode 100644 index 000000000..5168a17f9 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_45_5.dat @@ -0,0 +1 @@ +x陧UA 伙5?剱U:N&Z"啍:;4P1=僢NvSG芃1盾藳絥<雟`q长{祛進gс4=G-T箣?洆赃='k砣u欅瓭灀>戨'琬(槲簬 J魗E哐祍蔧 ,sq \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_49_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_49_5.dat new file mode 100644 index 000000000..9f3f3cd7d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_49_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_53_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_53_5.dat new file mode 100644 index 000000000..449807bae --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_53_5.dat @@ -0,0 +1 @@ +x陧VA 惑享" zYf5茞楯C 楢;能衢韭l鸤,嗠d騌. \(錯_蔸 械aNi5猏偶巻行雵aLP蝌(嘁;澴2砖勉瀵j萅6O u+榀裭{y暙6od^ 阙噻C[%  \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_57_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_57_5.dat new file mode 100644 index 000000000..c7dd81f39 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_57_5.dat @@ -0,0 +1,2 @@ +x陧VA + 击5?穻N輑獌Z婬AbBZ0a 煳M埔胐`1輟'"<諓1欐9nv通.泗 )峛莼祣;9語#t魾羱 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_65_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_65_5.dat new file mode 100644 index 000000000..ecd938068 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_65_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_69_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_69_5.dat new file mode 100644 index 000000000..ead4edc1f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_69_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_73_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_73_5.dat new file mode 100644 index 000000000..00001176d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_73_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_77_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_77_5.dat new file mode 100644 index 000000000..1652cdc2d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_77_5.dat @@ -0,0 +1 @@ +x陧楺 Cw汋䲢#&C`T16紨乒桞饵(9 '謫阎靸冖永纙k峰"h踯贻v.` 圾恻cX鯞5[(胖F>71/3嶒4呜为亃鱚'言輀楩yglg暜M>昈臫L4蠑{&3隬y*熓*︼`黑<3;磻钷Vo0/s6n崙0蛓a竭[麛猰cE \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_81_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_81_5.dat new file mode 100644 index 000000000..71215e952 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_81_5.dat @@ -0,0 +1,3 @@ +x陧樍 + C飤M?访tz癠4綄" }t仛绽MX2|.钌嫺藱F\崬鸶跹聗m鯌4崰鞅控X乽 +贁, 吞w:棤E喥>姷X濙摊=_謁峠>>舐苲孀瀲/)5讱kkk黌s竂嫷X楷Y{墚谚}舳徵~mt挎顨烀淆:S#櫟&;U#)褛 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_85_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_85_5.dat new file mode 100644 index 000000000..09cf0e281 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_85_5.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_89_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_89_5.dat new file mode 100644 index 000000000..5fff53069 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_89_5.dat @@ -0,0 +1,2 @@ +x陧欋 + 4甥寇5e瀑瀹npQご G鷻c滞f嚨餷^^;;b5;`琸订禪凸弋j`蚇髎荠鵒=赲[a6嚍涟秪nL虳? !6钠榜uF%w*痊蔡k矽77擲膯b兠嗞峏odw_鶟芈梞胆鈫鈫bCl萅蹤 ck&漕YVo堋欁稡注回Al6 氭Jj躼 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_93_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_93_5.dat new file mode 100644 index 000000000..ec4240bd3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_93_5.dat @@ -0,0 +1,2 @@ +x陧橩 +0 D>嶆䲢&┗鸠& 嵘 fP^斄8BY5是s(im湲假耶け=f椀3/w郄E滑y燂Y嶲wf戓[} 矂[򗔀303蟚f窍虣亳3'3=<蚱K{n礞糞g3鼪`f`f`婵蘢鴻逎掼畂iξPg7 汤 誊G礿k轌]韢?V/骐 k易搠繵i72朖s檝脇*"f^ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_5/mask_97_5.dat b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_97_5.dat new file mode 100644 index 000000000..509d1174f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_5/mask_97_5.dat @@ -0,0 +1 @@ +x陧欰 E鳒錰!3bf5懑萍B垜#f=<3l6<洤巁+xj┝) 撡荸脃皳 塉 箉Xi镘华5zs贋頲雞膃Q  KKK穌)鈼>,图wx瓺冎.,途懵s%g,,,=ル齊畜峤摷7u壓KK棒T朌<(n lY揀熮h轩V酃s鬓魕笇ゴ齈抱E<个q \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_101_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_101_6.dat new file mode 100644 index 000000000..13f97a0fd --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_101_6.dat @@ -0,0 +1,2 @@ +x陧歬 +0 4升/匪n_.壥2#圫婬6盻揚廪Z牄蟜顺_☆,7貫+%族W锹頳劼栤蟦鍈枔_&玂k;対P厩f>硉倅Y綸謓s ;s锋s傏,!Lk詤cbL寜12c訶9梭1犜繸#X序盓n#;蟬v孴~L~L盧寜11vs.嫊敵1傭笨111臞1&颇貚淤艃111臞1&茷g素 KL篇jlk{恎稼5驥1/肭慁瘇,a尰鶋鮟$撧 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_105_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_105_6.dat new file mode 100644 index 000000000..a58fec749 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_105_6.dat @@ -0,0 +1,3 @@ +x陧歈 +傽 Ds氫䲢k‘貰襔#o)Sd峿澨G芊婂筶謂嫕烔)G]S4軸纂氜揆阁?#B篫:浙+{譻ЯHK﨨骾I幝!Λm騟1 +RWe9!鯼锒霼y:妾灮胫鍤U=w-鹢曉wB穧cM轐呙麒拱{{=y呙麒拱{{=y呙麒拱{皐S訾cao靑'輋龊榻鮵O=C晊O=C晊瘺遊燋捏{S藁=;焲v4}席2ǜ0 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_109_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_109_6.dat new file mode 100644 index 000000000..be7b4749e --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_109_6.dat @@ -0,0 +1 @@ +x陧欰0 飤嶟蟻DBH覥H鉜2嚻0賜o帛嘶=邰s9[垘鮨'?欗7享R" &摟2櫿呦:7QqX_昻 鸧$诱殩EIY袅*Τ芁旤q0 0 0鵾LJ蟹(s\澤硍滛緣-7^鲅It潉I槃I槃I$~?塏0 0 0墴腛'袸t抔7L$L-摏劇Iuzrfr M鲹^'}撻枉摃(蒓~R熼]1Y层L澞瀠义9蟆Q陀 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_113_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_113_6.dat new file mode 100644 index 000000000..397f52741 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_113_6.dat @@ -0,0 +1,3 @@ +x陧浹 E啕5N挨諌膈撩萣F6on炊,m>穏S9祚轗W金坈屒9&%1蟔鬓cx= GR^皻峸-z?藐浅鮲=,龟皚豫?枪跇:9渕==@U敹佷矇刄X匲XVe牫芘~誦y4W閕:輊=髩蟆鶔觞杉椕嗿$<>疧阳v蚴'披錍缪yt鎍鎍鎍鎍~淁蝓|隇再'9li嬑s<堂<堂<堂<堂箪I懨︾拌<:徫<堂<堂<堂筚晛W媲哺倅:.z=开 屳撌 渟豓毭禘=ll沖k0_#v醢澄瘶m蓣沯 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_129_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_129_6.dat new file mode 100644 index 000000000..b4695c3ff Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_129_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_133_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_133_6.dat new file mode 100644 index 000000000..40911dc57 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_133_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_137_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_137_6.dat new file mode 100644 index 000000000..43ccb68c6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_137_6.dat @@ -0,0 +1,2 @@ +x陧溠 E啕5?積叆胀蚟4f丠錺[1-e)UQ蔞]濽橶鱊5o*8|泴郓霿梗彋6瀊k?蓣弡媐勳|>緎知囲黵6鲻笓6嚝6蓃覠=v岗猖Wy壁 ⒍-甘' +炖炖螣层;q tQ岴>U栂饿f懐曣抄靫馧]Tc(s鉂7t軦w鹳`v`v`v`鏺v坵f巜t;];簝;;;;缳!迿;轪o翦]簝顮;鴏;;;懊窢| 史犺簝钹沉炖炖鞂(3}岇l.瓪?珔"迚r}淝;赲}糞-A灉黽<9;鶨V'ё潏濘饜 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_141_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_141_6.dat new file mode 100644 index 000000000..0340409a4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_141_6.dat @@ -0,0 +1,10 @@ +x陧渁 F4睚/c犚赈蜇葠叆猍铂=绅[E霟撝荏s罟m,闅踗襫/遼kj\淇騤?g捷萚q(NO榋c5麏㏒GGP[oM譜釖指撮f讥鱲諰<非W硰贸粭C鯽飠6钶蒛~一穥`n龝姼鮼偕莼dvV齳~祌穁"鐀粉k{>g$裴XK鄂萓}m\b鱦a辺樑,f1媃蘠隹檷蛸]絲瑧灲儦^瘣.嶋5覽?皱瑴暢r硺,f1媃舔Y>薵湑硆V蝏硺,f1媃>薵,煏硆V晰m孻蘠硺,f,熷硘V问Y9媃蘠硺,f,熷硘V问Y9+g1媃蘠硺龏賸 苔M7>2{9z合秇焟3社鹟騶9選筚紮#f#焫6 槼-v%匩' \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_161_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_161_6.dat new file mode 100644 index 000000000..ecec68b1a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_161_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_165_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_165_6.dat new file mode 100644 index 000000000..d641dfa3d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_165_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_169_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_169_6.dat new file mode 100644 index 000000000..ae689723d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_169_6.dat @@ -0,0 +1 @@ +x陧葆J0鱥掲9[跩由橌k{1腷!g遪hHkS雺-璙?矾KI雉笞佅1韥忼1蓰]/建妨槑嫔`嶠/z)*=盁3趶炜g6貊齘k6钟5廋媲殛Y<蒏ㄍkM鉡鮵@覩&_=~⒉>诚>稻+觜鯋'{珓乍_鮒藠趚-釰<記軛鹯_畛>诚>i;鰷r璒钏}/髻g焳鲑g焳鲑喀}觜诫}黯紉璒钏}/髻g焳鲑g焳鲑窏G給/弡{戾扄r_铥蜗>诚>硂/弞過黯厮#麇拒鳚焳鲑g焳鲑繂d察'熓车|訯R竭NS锓3箳挲硿擞阆錣助蚕溱塟'莔s縀熐穓5 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_173_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_173_6.dat new file mode 100644 index 000000000..95fa97c7b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_173_6.dat @@ -0,0 +1 @@ +x陧輆娐09Mr隧M S*:a_-矏5hh_)﹗踆汁[辋镉o抯煆mK鷼忢鴽N莧鉎?x`l#f9>捼焄漞榉袆蠋邠 ?⺗祗鰉黧*/K薽織鉮藋%匏v-nKlロ宦钖恧kK窵`樌&&剃 g5啹禃(穏簑x鬥④烆掎a卢瀙Vc鳽[壠#O=‰SN9两樌&0 L`槓莿リ 圭't璲宩宂峇N9两樌&0 L`槓s柼YRc麫峇N9 L`樌&0 L`傜"蚘Rc鬨 ' r樌&0 L`樌螮毘て韫H9AN&0 L`樌&帛 7歱6痐聕醪刪m笢s栻 鬃隦5櫝茦饺 k\X/蚕 )g9 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_177_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_177_6.dat new file mode 100644 index 000000000..e9f0476f0 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_177_6.dat @@ -0,0 +1,14 @@ +x陧菅n 呩{遐*4v鄞u1墸缠{<丩璆i慘恁 侩GU添鋳喵荾鏼/6炣踛?mZ鼁蘺/緂蚸\陶摋<`臺躲>f{啃砡聩,,K9o 4幥掉鶏7l笋辬iJiggi錼<-MG + + + +瑇uV镊+z桃RC蘲9+Gq6轖W烋b"譗e罪"WL簜疇+ + XXXXX/翃|~舑,謓屩mu糓捩谯+ +簜 XXXXX癔Ya,柋X纪;糓簜疇+ + XXXXX鏅媏)o訕辠抾窓5嵅H擕鸽姎恁7/D \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_21_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_21_6.dat new file mode 100644 index 000000000..6bd505b4a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_21_6.dat @@ -0,0 +1 @@ +x跐怮 C9M{薸]枰齒1だ犈- C!粷矰禧7 W 熨湩&rD)~蝅<M 3(>{冿A ~韼a布S \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_25_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_25_6.dat new file mode 100644 index 000000000..d45083aab --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_25_6.dat @@ -0,0 +1 @@ +x跐QA 击5熪2)阤+(Xm蔤t*(踮公;澝鐃J<宄倖_冥3皁节"挞azh}&qv礢G手,-鲊橨浲4}硻S[鼄w \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_29_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_29_6.dat new file mode 100644 index 000000000..0408e2240 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_29_6.dat @@ -0,0 +1,3 @@ +x谡RA +0 蝴氫熪Xc愒醪覮(4E斣圔 +8C周巨硁肕+l菨諉O為1]&曏崶4U隓-6-$:6蔰Z?y踠糨f?順 鍘8?炦脖<踠魹}灩g沢頎* \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_33_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_33_6.dat new file mode 100644 index 000000000..8de4ba5c9 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_33_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_37_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_37_6.dat new file mode 100644 index 000000000..b37ff0ab6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_37_6.dat @@ -0,0 +1 @@ +x陧擜 锛嘄缵T `紧鑚_輳`迕8幑G1摐`B斴`;基+芈靰迥特&sR鐚]<匭F=mb3 U孎'谪6茤釀E 恻5鮆a鱳Z刂60 +遊狗攵9”9稊靮翓S踈轂8佌`k&渙s}{鈁R君[+楟w鸷誒嬭d^jW \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_61_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_61_6.dat new file mode 100644 index 000000000..f2d3f10df --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_61_6.dat @@ -0,0 +1,2 @@ +x陧桲 D鱯遐礒~輦峣h棠' +屠騈'oVW唛啵jsp柃Cz誉煅慰━胊瞑浱蟝!珔#WD%~芩僘?盝踗屭錜E涪幜怟o嫱紮菢|蓛藥*蘋P7彔姙囧踤倦冀7s銌`灲濓掏掷鮯鏳鬌緹篃s閪><i蘇'巐'缸9%.祁7 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_65_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_65_6.dat new file mode 100644 index 000000000..550fc8fe7 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_65_6.dat @@ -0,0 +1 @@ +x陧WQ i濑梴K濶Lk?悩e$孮磇k蒽期4哥1{樶夝譧+!M ? 猁爴叭1b8啚 7.屴^崈w剔轨鑣n擦Fj鑽5匛aQX|=w浼@2v<蚺嬇灮東4w魂闬UX荁Qz+笫T比T怪c鳥觶/48针,懟5`エ积谷 OV$ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_69_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_69_6.dat new file mode 100644 index 000000000..a3e4fa0f2 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_69_6.dat @@ -0,0 +1 @@ +x陧楰 @鳒箼dTh灭恙 hLSSEq eY@<惘+*哏 %嫯>ァó嫻z*7霒舉6鉗督㏒`氜.>籹E '%@[憥6癅P0胔 F誼tpl2麦 崌Q-g1N唂eo^啉腰雴0F痙T>N_誒鰭憌婫3禊猽嗗溋籫堝 峽3<[炘 b?'6^簥 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_73_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_73_6.dat new file mode 100644 index 000000000..ab71b70ae Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_73_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_77_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_77_6.dat new file mode 100644 index 000000000..ad5a660e3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_77_6.dat @@ -0,0 +1 @@ +x陧樠 E啭熪瞚嬞臯鋋.殾盕┃pST沋拄4q鼹~磟叓=熶:鐾 6忘鬽8懢:#0P囟恑Dy:2艩'Z璼眾&}萧珂粶\r0\ヅ殨榅w;iP嗳擫)襍e詴躿h圖u9L憺b嵔暕J抟S孲)檼)欈gZ歿齄e茗熼)銛qJLw+#3-V0耪穕顛j児欭跔譙-誗 杲9=莜5罰P秕q1M?錱 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_81_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_81_6.dat new file mode 100644 index 000000000..28a6d0752 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_81_6.dat @@ -0,0 +1,3 @@ +x陧橯0D9 r簧赗矻誺鰇嚉芵搐0 ;鎖6茺缪赲|耥_筷cc1h君u導絠o摅#2儸}x*.Y鹴&腑 +职独謖/K;3蘶鋏刑葕侫跦?`]5茵K俊庶磫w!}{Z没卟濿 +寯y夥纠瀆ykk睎^K瘯转醔-謆軾S诟'譁N笕u米鯕#M絝鮄S疩?|]疘A期i橫憏盱uW \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_85_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_85_6.dat new file mode 100644 index 000000000..d5403e49a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_85_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_89_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_89_6.dat new file mode 100644 index 000000000..eeeb5d197 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_89_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_93_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_93_6.dat new file mode 100644 index 000000000..6ff38db68 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_93_6.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_6/mask_97_6.dat b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_97_6.dat new file mode 100644 index 000000000..3a2072efd --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_6/mask_97_6.dat @@ -0,0 +1,2 @@ +x陧歛0s宥d抨乙Fx[=4艸鶌o妀34&}*幟灄 鱝 V笉c&3炾5塧rW鏭a鱈Cl梢z瀜,萏1唜 祲SQN霍籡发/Giu腵&牏悥w%,%睤柸Y"K絫袱+錒E'|R2(v1v枹qi焣d,%睤杶%鼗Jj飣硗耗耗gY"Kd,+K +]W杢+sF/)黓.zN蜈'`>伄1='#朻+b儱l]Z睌 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_101_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_101_7.dat new file mode 100644 index 000000000..1f6bc5129 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_101_7.dat @@ -0,0 +1 @@ +x陧歈 C}r薽撌`橼蔲猨愹乀#ピ滢54't咁fa刃噊皷$眂<偎赉萦振H轄蘦8刲折雘=侥剮脓郒v憨;媧!億/婩幮3cf虒S资叡銔偧犟 8㎏)49爺Q11譐3f铺嘏汼+舧謈珲橌橌榢3cf旌岡钼緕蘺蘺痰覍櫛1úbL鞫缤X橲+髳.to{>魳頼OJ23c<6鉞n0F ) \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_105_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_105_7.dat new file mode 100644 index 000000000..6b0cacfe9 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_105_7.dat @@ -0,0 +1,2 @@ +x陧欰 +0E鳛F筧燤覚楴綪x)竝Q<鑏鑯|帽(b繚F拕氱$.煃a縪撦騑GN赑鎁菛穳M%{囯o鬑Q糍U駆纯瑵L^>+社m瓝#{{廵o&Y2s踥M)gncO9s涍Z姵3w莖檮+妈{=f.炝炝z絿蓿镅鳂拱{{=z従G遚媛炝辳賙CQ蠟p禹蔨秬&褅^w糔锃史U镎祜 靍5扮}E墂G遪+o \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_109_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_109_7.dat new file mode 100644 index 000000000..9875cbe8c --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_109_7.dat @@ -0,0 +1,2 @@ +x陧欰 +0 D9Mr藑* _x-d悩:"钋姠-k"娾d寋浛栈否惘蟪奘讙涩|'儛亿r郠5+ s嶜尅)踓曊7-颜蠡1nn頄簈蓴栐Jtg琟钍墂虡艊-鶅?*&Mm鼲ee5轼^ +c + +廓,颾颸1孔3齤4豑縕瀎泞穽籵癄* + + +:沀t*簥 + + +珦X暖0<稡W裊tVaVaV僘誰 +薨k齠5辇猒 开賙W5{祤鋿5pn魿楠Z: bv \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_117_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_117_7.dat new file mode 100644 index 000000000..cde78c10d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_117_7.dat @@ -0,0 +1,2 @@ +x陧沋 D4p薝暡汝I>"%6鎒d 懸8r餝 蚇s幍Un雓5ej晦u椲瞀艨霰V踃g,l`顄菔李鳎!俬塜Z鵟VlM膘雦[同萋0 0 0 庙#hF'滕c]伈灳轫>禚潕鸋衋t哸哸哸~<迷脝z貀苢kOC0 0 0 灭鎸悛F囇a哸哸)啩2撬伴f脪狺%兝z8醫O=紴鈥3=巶嶂3溰:c峸 +怴$ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_121_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_121_7.dat new file mode 100644 index 000000000..d5d577f7e --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_121_7.dat @@ -0,0 +1,2 @@ +x陧沎 + E硽d屍i;^az,#6茷^r葑偋h舻汝&蒽ㄧ躛am罆Y9譥邑呼5C穜6氘餿^鵡臰lE錁糌瞺煽|鐼m遨mS騷鞫( 郯 郯 郯}巑Q絔孼黴絔加v搜"橫1f羸G咱, q浇虰费mt禷禷禷脉膠籩F3澔cxI箺D费mt禷禷禷焟蝞饞媒$簫n[ 郯 郯 蹚a[}铴`[y腈丽柊巨栟)蟦<规4躃/轶激O⒒s╪鏻詍煹/G胭 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_125_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_125_7.dat new file mode 100644 index 000000000..f9ec0887b Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_125_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_129_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_129_7.dat new file mode 100644 index 000000000..9bf51d529 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_129_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_133_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_133_7.dat new file mode 100644 index 000000000..b643ffeda Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_133_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_137_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_137_7.dat new file mode 100644 index 000000000..11d212bf4 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_137_7.dat @@ -0,0 +1,5 @@ +x陧溳 + F矬4/ 鷆嫬_莻+{S贙<熩蝟[l +炖炖螣灿 +07謮礦苐;b秿潠朦铏7骢fM維磫顮;1炖炖炖霯矯<逹盹湱[+@wt嚇 v`v`v`v&! 涳7赬+@wt嚇 v`v`v`v8 蔣P蝹;篊;;;皊aG{櫮;诨>vR渱炲鶐霎廗荆诫XM勿H +偝#袧(S浰悟q嚏d \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_141_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_141_7.dat new file mode 100644 index 000000000..98dffab00 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_141_7.dat @@ -0,0 +1 @@ +x陧淎 E= r揑雽TY@可脛0!跌鴟1敳棜t筨G0垫詶绚卻仭2Z/铉縪峚傡\qz苒On覌M合褶n櫄駎勧X"K梞慹M}缵駽p淀惸P橪^镜S0S0S0S )菙偀跇闥呔%炘b妖,鮏崺l?誾C)t姷LLLLI2厽z崯奟X秇荆@)t姷LLLLI1厽z憻b戏髱滲)謣0S0S0S07带)鰘鶈滲)槀)槀)槀3諙L% tfwM睙*:~猦弧Z虹玸n抍$1UTtJg8嬏OY篍 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_145_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_145_7.dat new file mode 100644 index 000000000..4aa2bac11 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_145_7.dat @@ -0,0 +1,2 @@ +x陧溠 + E唧5?W6Z禒-沕楸鄡2塹ё腷G煝X6(茐u"揕b裝軬u幬籍贩Gk:HwA[縥氃mHⅧ烇3锱钸釵餶Q{l穦昊鑄泦Em Jf貺?熩2"&)膋┹Rf賑濓撗虊錐娧,z│=簡噭5X5X5X兊7盕级\鸡p枬赨s#簡噭5X5X5X兊癋级d计Y鰰簡k!a 謄 謄 謄峯弶8随踓t ]C棕C gM瓊5匸碞%槟k鱤曬Zp?疘埽曄矁^n$堁孓Y7A制隯P[ f葥0 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_149_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_149_7.dat new file mode 100644 index 000000000..809f00554 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_149_7.dat @@ -0,0 +1 @@ +x陧苎n 呩{? 藆撢啳叴2徽G$n垐c珑菷〉K肷仔b3酤知t愁Pc 缣ビ7[?9:裾齕'榱9'*訓 鄰G孳a敖h謃抗/珃+6XB>陿2魆Y繨0戍榛弽欱fa 1埩 9c7G 鵒l,臹ミ嵟杲擈3冋駻:H  1圓 b泥0X4%佾禁#窃d>&C  1圓 b<欰{鬿縂&驦M褹:H1圓 b 1杩 {鬸2䴖爟tb 1圓 &1X鏯糭 9垧蔪珯瓸樁G_鏻5:g5櫠ajM嫱L粤娝僆 <|r. \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_153_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_153_7.dat new file mode 100644 index 000000000..c1ab27664 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_153_7.dat @@ -0,0 +1,2 @@ +x陧躋0勧= 鍖塀5 +褞鎯/嵧插镓I賗篰-囝甖km|YDh9 浳昝\{扠瞧斗{煆y棾3遶喍袒淓]u`R谇r"涍}Y 诫e2{踁V8][.n:+2_缍事B&6眽Mlb涁&6違3i9廦冲螌璜羸U}詿82f-v跂6爮n襇篒7眽Mlb涁&6冰髄骖7阄饌 t搉襇lb涁&6眽Ml2浳^/坣襇篒7眽Mlb涁&6遍;gxA綜t搉鶲婱lb涁&6眣蛓暻fPフ[64哼太佡k`v^+9 尸O郾}湘敤洐+ZX龛=:iWE \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_157_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_157_7.dat new file mode 100644 index 000000000..2db27f685 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_157_7.dat @@ -0,0 +1,2 @@ +x陧躆 +0E狴[M蜁偔s 瀴芹 瓆F箚Qks絿蘽%禬q埙最.灜璢8evV竭bvV}璁_弟E袒n3W^K肻`渲V旌界Zl硺,f1媃烃;砤f痹k灧蕤i噼摿膠汪炙z?urV问Y9媃蘠硺,f,熷硘V问Y9+g1媃蘠硺,熷硘栂蔣9+g7,f1媃蘠硘栂験>+g瀣溑,f1媃蘠硘栂験>+g瀣湑硺,f1媃烃萳怡砳o橗.\ο茲o<鷏;9:耮[秎焟猎^淚>=g蠙潨-yk_TA \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_161_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_161_7.dat new file mode 100644 index 000000000..35ba8ff48 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_161_7.dat @@ -0,0 +1 @@ +x陧苎幝 呩鹹汄/鐨‥)飘涱1~4崧93,说礌+豘T=Ze舛C还.~i邚掸拗&>,锎6e鶚讽曋~,l醀] 2完旰惐輁瑶;椎2j"柋宔,藃X屳礦(c斓礪訓1憴8蕞恣弶/,'秦t.觘簩e,c薠2柋屽弑,_䦅|yt觞纾飢诬]λt.c薠2柋宔,cy蓑/蓷ys顙.觘篖棻宔,c薠2柋lo厊賈{+2]λbl,c薠2柋宔{+渌謯豙A楅2]λX2柋宔,c9忓藽X<膻乞挓/owsP緶y铸]緶y謢稸d凫姘汞廳9J椨囲)1 jI2 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_165_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_165_7.dat new file mode 100644 index 000000000..e27fb8edf --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_165_7.dat @@ -0,0 +1 @@ +x陧軶巶0窔蚆&A,N颂!梮0閲&XZ讎Z-K孬[贽蒭蒵嚱岆鎥,儳贱.锓g}隸;+k{搜~1煄}+k{素櫨SG鵚鉭題囕珸?_钓8韾緊+穔][h魯W8~穟躆g渜g|j<醴囤*遯=<瘮宵ゆ蹞R弚茡腭狰厅奶闊萹9.清8銓38銓38鉏普U廏?~祍nr\幩q98銓38銓3蝬q貘钦惝亂979.清g渜g渜g=)阸桌'E幩q9.g渜g渜wO妟50領戙r\廂8銓38銓_气Vvd}睬搌a鮴睡芹5犟)&9)濕俎f讥鐇3鳼H]S^*/ \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_169_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_169_7.dat new file mode 100644 index 000000000..ef1a181f2 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_169_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_173_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_173_7.dat new file mode 100644 index 000000000..3b5137125 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_173_7.dat @@ -0,0 +1 @@ +x陧葺j0 衶=M蟒/ Rb諬rj\=,簃穋o鶽8;伙m削絤圯氡0b鲀_邭t$~Wu5矍歂n'?d說2俄(o溜嵾G,L6_{z叛髌? 9鷚q9 S!茖脎9蝟雬2 L`槓榩祵瑬朞En锒v8L槻v8L负 U嶀Y"鹍} K況倻 ' L`樌&0 L`B'f:颫8跿cTc\k宺倻 ' L`樌&0 L`B靁瞘I嶑宺倻 ' L`樌&0 L`傜"鞾Rc鬨 ' r樌&0 L`樌螮诔て韫H9AN&0 L`樌&'!;昱q+禮鮿l*窜炒=貈9j孾2谥 Ms陇焽詣栫* \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_177_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_177_7.dat new file mode 100644 index 000000000..068477c92 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_177_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_21_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_21_7.dat new file mode 100644 index 000000000..4f9f1386d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_21_7.dat @@ -0,0 +1,4 @@ +x跐怮 +0 B=嵽r]殬-?]Rl悋2暒筺c +餥幑鲚n鶑A".杍+阨 +~媥3<鷄X趝H喼C1x衫)鈩S \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_25_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_25_7.dat new file mode 100644 index 000000000..cefe1b972 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_25_7.dat @@ -0,0 +1 @@ +x跐QA 击5焄F@┼a儎脑諾滩 >;捫繯笔I掷QH8R捡覉G"z,&囻;杯'o窕97%P8%6o墙;]搡NWn輀鋐7陶v \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_29_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_29_7.dat new file mode 100644 index 000000000..e3d7391b7 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_29_7.dat @@ -0,0 +1,2 @@ +x谡R9 埤汎煫娙QpX$l媾瞗!I2pg峉悛戟MZj泛筇寐"鷗婐铩e0究#詻`恄1-鞭髱榇c廨枉騢a~/h4ε"榽 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_37_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_37_7.dat new file mode 100644 index 000000000..87d9a1a9d Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_37_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_41_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_41_7.dat new file mode 100644 index 000000000..8acec04f0 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_41_7.dat @@ -0,0 +1 @@ +x陧TA 击5焄fDY(O抆bR3/~t/L"鼓7SQQ5卝診S棜i眈蝏_#諉葟伦+牵#鎧鮴?憬-霘w晆鏇慪7$b.%A饲;晈玆调耧暅oxG}? \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_45_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_45_7.dat new file mode 100644 index 000000000..dbba31d0b Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_45_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_49_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_49_7.dat new file mode 100644 index 000000000..be5dce8b7 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_49_7.dat @@ -0,0 +1 @@ +x陧V0击k蝖淐-X.炂<歙h6虌 3澜,B 佋旙,鏎蠍d5$K"T|p%9"紋p,镡妗=<蒻鳂詨!嫬?DW"槻(A-謀- J\﹞0pWW]円4簽?4h= Z_冎d+齡(**+攵(锨 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_53_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_53_7.dat new file mode 100644 index 000000000..7028ef6d7 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_53_7.dat @@ -0,0 +1 @@ +x陧朘 D鱯鍤&碦錵篿&,H)鴗D譲 =ǖ形Qa槥玃"d箂1ZyC畖hey Cv閍H皼7+鵒吪u仴Q蝧Ztb叞宛7 祐泮缞環V堤 祣棂喾1#~嘯f嗼冁fHkZ禞t禍焟Z蜗6綏蓅怔釈 僿竢&C叨f \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_57_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_57_7.dat new file mode 100644 index 000000000..ee3107a3c --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_57_7.dat @@ -0,0 +1 @@ +x陧桝0稃汌3&曇H)Kb夹&q胖炚[-汤尴謱 呞9蓃Ys .i萦綻挫蜇鷡棴隚愰{ci冋偦\Y<.﹟放2簇蹮2 QQ[[鱚N冯88l硐昧[}[歅l?g+o*蠔4W尔1 33樢[縗Gos蟓bk鑝攎唆m顗蔨h \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_61_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_61_7.dat new file mode 100644 index 000000000..76f8d7271 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_61_7.dat @@ -0,0 +1,2 @@ +x陧桲 + D鱯氫䲢+-*5m焌T񇹋梺Y天導闘9 绯冬茬韘//i噰蔦勯b雯媛墨廪-"^睤H-i;bnA⒓7骹頴t7俜y蹋E:r<46(黸鱵{K3o"筇齶a綕o?釡焽镧珶擈磒9妪渏~N昶浫靤澒宴TY 聈 \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_65_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_65_7.dat new file mode 100644 index 000000000..d8b920626 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_65_7.dat @@ -0,0 +1 @@ +x陧桝 稃鐚I*6^L!閰岸 竔)mT荰]燰W\9恊@4Ku^#N%:,仲mY嚷JN銊9瘬娻订坴A2严湐H"Y鼩跑 +?`鼴XD麭X探{嬃.乘w╠,~蝹k,竈颾Q鈤.绵\痺 隵侘頞Y樲E鬔mX严"% \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_69_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_69_7.dat new file mode 100644 index 000000000..c2db0204a Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_69_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_73_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_73_7.dat new file mode 100644 index 000000000..f414e4a58 Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_73_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_77_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_77_7.dat new file mode 100644 index 000000000..3e52bfd3f Binary files /dev/null and b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_77_7.dat differ diff --git a/vendor/aferrandini/phpqrcode/cache/mask_7/mask_81_7.dat b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_81_7.dat new file mode 100644 index 000000000..78e08dfc6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/cache/mask_7/mask_81_7.dat @@ -0,0 +1 @@ +x陧櫫 D矬5?譴"蛫輹箄4崂a>ukv o40蓣鉚%9瘧6颤鳘U5*sI{`躬檁>S?鐌(:y釺飈飡G&E\斨6}"A挙X嬽 X犀城<耺wx癴茚慓width = $width; + $this->frame = $frame; + $this->x = $width - 1; + $this->y = $width - 1; + $this->dir = -1; + $this->bit = -1; + } + + //---------------------------------------------------------------------- + public function setFrameAt($at, $val) + { + $this->frame[$at['y']][$at['x']] = chr($val); + } + + //---------------------------------------------------------------------- + public function getFrameAt($at) + { + return ord($this->frame[$at['y']][$at['x']]); + } + + //---------------------------------------------------------------------- + public function next() + { + do { + + if($this->bit == -1) { + $this->bit = 0; + return array('x'=>$this->x, 'y'=>$this->y); + } + + $x = $this->x; + $y = $this->y; + $w = $this->width; + + if($this->bit == 0) { + $x--; + $this->bit++; + } else { + $x++; + $y += $this->dir; + $this->bit--; + } + + if($this->dir < 0) { + if($y < 0) { + $y = 0; + $x -= 2; + $this->dir = 1; + if($x == 6) { + $x--; + $y = 9; + } + } + } else { + if($y == $w) { + $y = $w - 1; + $x -= 2; + $this->dir = -1; + if($x == 6) { + $x--; + $y -= 8; + } + } + } + if($x < 0 || $y < 0) return null; + + $this->x = $x; + $this->y = $y; + + } while(ord($this->frame[$y][$x]) & 0x80); + + return array('x'=>$x, 'y'=>$y); + } + +} ; \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRbitstream.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRbitstream.php new file mode 100644 index 000000000..93606f13f --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRbitstream.php @@ -0,0 +1,182 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRbitstream { + + public $data = array(); + + //---------------------------------------------------------------------- + public function size() + { + return count($this->data); + } + + //---------------------------------------------------------------------- + public function allocate($setLength) + { + $this->data = array_fill(0, $setLength, 0); + return 0; + } + + //---------------------------------------------------------------------- + public static function newFromNum($bits, $num) + { + $bstream = new QRbitstream(); + $bstream->allocate($bits); + + $mask = 1 << ($bits - 1); + for($i=0; $i<$bits; $i++) { + if($num & $mask) { + $bstream->data[$i] = 1; + } else { + $bstream->data[$i] = 0; + } + $mask = $mask >> 1; + } + + return $bstream; + } + + //---------------------------------------------------------------------- + public static function newFromBytes($size, $data) + { + $bstream = new QRbitstream(); + $bstream->allocate($size * 8); + $p=0; + + for($i=0; $i<$size; $i++) { + $mask = 0x80; + for($j=0; $j<8; $j++) { + if($data[$i] & $mask) { + $bstream->data[$p] = 1; + } else { + $bstream->data[$p] = 0; + } + $p++; + $mask = $mask >> 1; + } + } + + return $bstream; + } + + //---------------------------------------------------------------------- + public function append(QRbitstream $arg) + { + if (is_null($arg)) { + return -1; + } + + if($arg->size() == 0) { + return 0; + } + + if($this->size() == 0) { + $this->data = $arg->data; + return 0; + } + + $this->data = array_values(array_merge($this->data, $arg->data)); + + return 0; + } + + //---------------------------------------------------------------------- + public function appendNum($bits, $num) + { + if ($bits == 0) + return 0; + + $b = QRbitstream::newFromNum($bits, $num); + + if(is_null($b)) + return -1; + + $ret = $this->append($b); + unset($b); + + return $ret; + } + + //---------------------------------------------------------------------- + public function appendBytes($size, $data) + { + if ($size == 0) + return 0; + + $b = QRbitstream::newFromBytes($size, $data); + + if(is_null($b)) + return -1; + + $ret = $this->append($b); + unset($b); + + return $ret; + } + + //---------------------------------------------------------------------- + public function toByte() + { + + $size = $this->size(); + + if($size == 0) { + return array(); + } + + $data = array_fill(0, (int)(($size + 7) / 8), 0); + $bytes = (int)($size / 8); + + $p = 0; + + for($i=0; $i<$bytes; $i++) { + $v = 0; + for($j=0; $j<8; $j++) { + $v = $v << 1; + $v |= $this->data[$p]; + $p++; + } + $data[$i] = $v; + } + + if($size & 7) { + $v = 0; + for($j=0; $j<($size & 7); $j++) { + $v = $v << 1; + $v |= $this->data[$p]; + $p++; + } + $data[$bytes] = $v; + } + + return $data; + } + +} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRcode.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRcode.php new file mode 100644 index 000000000..08b602435 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRcode.php @@ -0,0 +1,158 @@ +getVersion() < 0 || $input->getVersion() > Constants::QRSPEC_VERSION_MAX) { + throw new Exception('wrong version'); + } + if($input->getErrorCorrectionLevel() > Constants::QR_ECLEVEL_H) { + throw new Exception('wrong level'); + } + + $raw = new QRrawcode($input); + + QRtools::markTime('after_raw'); + + $version = $raw->version; + $width = QRspec::getWidth($version); + $frame = QRspec::newFrame($version); + + $filler = new FrameFiller($width, $frame); + if(is_null($filler)) { + return NULL; + } + + // inteleaved data and ecc codes + for($i=0; $i<$raw->dataLength + $raw->eccLength; $i++) { + $code = $raw->getCode(); + $bit = 0x80; + for($j=0; $j<8; $j++) { + $addr = $filler->next(); + $filler->setFrameAt($addr, 0x02 | (($bit & $code) != 0)); + $bit = $bit >> 1; + } + } + + QRtools::markTime('after_filler'); + + unset($raw); + + // remainder bits + $j = QRspec::getRemainder($version); + for($i=0; $i<$j; $i++) { + $addr = $filler->next(); + $filler->setFrameAt($addr, 0x02); + } + + $frame = $filler->frame; + unset($filler); + + + // masking + $maskObj = new QRmask(); + if($mask < 0) { + + if (Constants::QR_FIND_BEST_MASK) { + $masked = $maskObj->mask($width, $frame, $input->getErrorCorrectionLevel()); + } else { + $masked = $maskObj->makeMask($width, $frame, (intval(Constants::QR_DEFAULT_MASK) % 8), $input->getErrorCorrectionLevel()); + } + } else { + $masked = $maskObj->makeMask($width, $frame, $mask, $input->getErrorCorrectionLevel()); + } + + if($masked == NULL) { + return NULL; + } + + QRtools::markTime('after_mask'); + + $this->version = $version; + $this->width = $width; + $this->data = $masked; + + return $this; + } + + //---------------------------------------------------------------------- + public function encodeInput(QRinput $input) + { + return $this->encodeMask($input, -1); + } + + //---------------------------------------------------------------------- + public function encodeString8bit($string, $version, $level) + { + if(string == NULL) { + throw new Exception('empty string!'); + return NULL; + } + + $input = new QRinput($version, $level); + if($input == NULL) return NULL; + + $ret = $input->append($input, Constants::QR_MODE_8, strlen($string), str_split($string)); + if($ret < 0) { + unset($input); + return NULL; + } + return $this->encodeInput($input); + } + + //---------------------------------------------------------------------- + public function encodeString($string, $version, $level, $hint, $casesensitive) + { + + if($hint != Constants::QR_MODE_8 && $hint != Constants::QR_MODE_KANJI) { + throw new Exception('bad hint'); + return NULL; + } + + $input = new QRinput($version, $level); + if($input == NULL) return NULL; + + $ret = QRsplit::splitStringToQRinput($string, $input, $hint, $casesensitive); + if($ret < 0) { + return NULL; + } + + return $this->encodeInput($input); + } + + //---------------------------------------------------------------------- + public static function png($text, $outfile = false, $level = Constants::QR_ECLEVEL_L, $size = 3, $margin = 4, $saveandprint=false) + { + $enc = QRencode::factory($level, $size, $margin); + return $enc->encodePNG($text, $outfile, $saveandprint=false); + } + + //---------------------------------------------------------------------- + public static function text($text, $outfile = false, $level = Constants::QR_ECLEVEL_L, $size = 3, $margin = 4) + { + $enc = QRencode::factory($level, $size, $margin); + return $enc->encode($text, $outfile); + } + + //---------------------------------------------------------------------- + public static function raw($text, $outfile = false, $level = Constants::QR_ECLEVEL_L, $size = 3, $margin = 4) + { + $enc = QRencode::factory($level, $size, $margin); + return $enc->encodeRAW($text, $outfile); + } +} diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRencode.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRencode.php new file mode 100644 index 000000000..d05ab6b7b --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRencode.php @@ -0,0 +1,137 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +use Exception; + +class QRencode { + + public $casesensitive = true; + public $eightbit = false; + + public $version = 0; + public $size = 3; + public $margin = 4; + + public $structured = 0; // not supported yet + + public $level = Constants::QR_ECLEVEL_L; + public $hint = Constants::QR_MODE_8; + + //---------------------------------------------------------------------- + public static function factory($level = Constants::QR_ECLEVEL_L, $size = 3, $margin = 4) + { + $enc = new QRencode(); + $enc->size = $size; + $enc->margin = $margin; + + switch ($level.'') { + case '0': + case '1': + case '2': + case '3': + $enc->level = $level; + break; + case 'l': + case 'L': + $enc->level = Constants::QR_ECLEVEL_L; + break; + case 'm': + case 'M': + $enc->level = Constants::QR_ECLEVEL_M; + break; + case 'q': + case 'Q': + $enc->level = Constants::QR_ECLEVEL_Q; + break; + case 'h': + case 'H': + $enc->level = Constants::QR_ECLEVEL_H; + break; + } + + return $enc; + } + + //---------------------------------------------------------------------- + public function encodeRAW($intext, $outfile = false) + { + $code = new QRcode(); + + if($this->eightbit) { + $code->encodeString8bit($intext, $this->version, $this->level); + } else { + $code->encodeString($intext, $this->version, $this->level, $this->hint, $this->casesensitive); + } + + return $code->data; + } + + //---------------------------------------------------------------------- + public function encode($intext, $outfile = false) + { + $code = new QRcode(); + + if($this->eightbit) { + $code->encodeString8bit($intext, $this->version, $this->level); + } else { + $code->encodeString($intext, $this->version, $this->level, $this->hint, $this->casesensitive); + } + + QRtools::markTime('after_encode'); + + if ($outfile!== false) { + file_put_contents($outfile, join("\n", QRtools::binarize($code->data))); + } else { + return QRtools::binarize($code->data); + } + } + + //---------------------------------------------------------------------- + public function encodePNG($intext, $outfile = false,$saveandprint=false) + { + try { + ob_start(); + $tab = $this->encode($intext); + $err = ob_get_contents(); + ob_end_clean(); + + if ($err != '') + QRtools::log($outfile, "ERROR: " . $err); + + $maxSize = (int)(Constants::QR_PNG_MAXIMUM_SIZE / (count($tab)+2*$this->margin)); + + QRimage::png($tab, $outfile, min(max(1, $this->size), $maxSize), $this->margin,$saveandprint); + } catch (Exception $e) { + echo $e->getMessage(); + die(); + + QRtools::log($outfile, $e->getMessage()); + } + } +} diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRimage.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRimage.php new file mode 100644 index 000000000..430a16f84 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRimage.php @@ -0,0 +1,95 @@ + + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRimage { + + //---------------------------------------------------------------------- + public static function png($frame, $filename = false, $pixelPerPoint = 4, $outerFrame = 4,$saveandprint=FALSE) + { + $image = self::image($frame, $pixelPerPoint, $outerFrame); + + if ($filename === false) { + Header("Content-type: image/png"); + ImagePng($image); + } else { + if($saveandprint===TRUE){ + ImagePng($image, $filename); + header("Content-type: image/png"); + ImagePng($image); + }else{ + ImagePng($image, $filename); + } + } + + ImageDestroy($image); + } + + //---------------------------------------------------------------------- + public static function jpg($frame, $filename = false, $pixelPerPoint = 8, $outerFrame = 4, $q = 85) + { + $image = self::image($frame, $pixelPerPoint, $outerFrame); + + if ($filename === false) { + Header("Content-type: image/jpeg"); + ImageJpeg($image, null, $q); + } else { + ImageJpeg($image, $filename, $q); + } + + ImageDestroy($image); + } + + //---------------------------------------------------------------------- + private static function image($frame, $pixelPerPoint = 4, $outerFrame = 4) + { + $h = count($frame); + $w = strlen($frame[0]); + + $imgW = $w + 2*$outerFrame; + $imgH = $h + 2*$outerFrame; + + $base_image =ImageCreate($imgW, $imgH); + + $col[0] = ImageColorAllocate($base_image,255,255,255); + $col[1] = ImageColorAllocate($base_image,0,0,0); + + imagefill($base_image, 0, 0, $col[0]); + + for($y=0; $y<$h; $y++) { + for($x=0; $x<$w; $x++) { + if ($frame[$y][$x] == '1') { + ImageSetPixel($base_image,$x+$outerFrame,$y+$outerFrame,$col[1]); + } + } + } + + $target_image =ImageCreate($imgW * $pixelPerPoint, $imgH * $pixelPerPoint); + ImageCopyResized($target_image, $base_image, 0, 0, 0, 0, $imgW * $pixelPerPoint, $imgH * $pixelPerPoint, $imgW, $imgH); + ImageDestroy($base_image); + + return $target_image; + } +} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRinput.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRinput.php new file mode 100644 index 000000000..8bdd21e71 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRinput.php @@ -0,0 +1,486 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +use Exception; + +class QRinput { + + public $items; + + private $version; + private $level; + + //---------------------------------------------------------------------- + public function __construct($version = 0, $level = Constants::QR_ECLEVEL_L) + { + if ($version < 0 || $version > Constants::QRSPEC_VERSION_MAX || $level > Constants::QR_ECLEVEL_H) { + throw new Exception('Invalid version no'); + return NULL; + } + + $this->version = $version; + $this->level = $level; + } + + //---------------------------------------------------------------------- + public function getVersion() + { + return $this->version; + } + + //---------------------------------------------------------------------- + public function setVersion($version) + { + if($version < 0 || $version > Constants::QRSPEC_VERSION_MAX) { + throw new Exception('Invalid version no'); + return -1; + } + + $this->version = $version; + + return 0; + } + + //---------------------------------------------------------------------- + public function getErrorCorrectionLevel() + { + return $this->level; + } + + //---------------------------------------------------------------------- + public function setErrorCorrectionLevel($level) + { + if($level > Constants::QR_ECLEVEL_H) { + throw new Exception('Invalid ECLEVEL'); + return -1; + } + + $this->level = $level; + + return 0; + } + + //---------------------------------------------------------------------- + public function appendEntry(QRinputItem $entry) + { + $this->items[] = $entry; + } + + //---------------------------------------------------------------------- + public function append($mode, $size, $data) + { + try { + $entry = new QRinputItem($mode, $size, $data); + $this->items[] = $entry; + return 0; + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + + public function insertStructuredAppendHeader($size, $index, $parity) + { + if( $size > Constants::MAX_STRUCTURED_SYMBOLS ) { + throw new Exception('insertStructuredAppendHeader wrong size'); + } + + if( $index <= 0 || $index > Constants::MAX_STRUCTURED_SYMBOLS ) { + throw new Exception('insertStructuredAppendHeader wrong index'); + } + + $buf = array($size, $index, $parity); + + try { + $entry = new QRinputItem(Constants::QR_MODE_STRUCTURE, 3, buf); + array_unshift($this->items, $entry); + return 0; + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + public function calcParity() + { + $parity = 0; + + foreach($this->items as $item) { + if($item->mode != Constants::QR_MODE_STRUCTURE) { + for($i=$item->size-1; $i>=0; $i--) { + $parity ^= $item->data[$i]; + } + } + } + + return $parity; + } + + //---------------------------------------------------------------------- + public static function checkModeNum($size, $data) + { + for($i=0; $i<$size; $i++) { + if((ord($data[$i]) < ord('0')) || (ord($data[$i]) > ord('9'))){ + return false; + } + } + + return true; + } + + //---------------------------------------------------------------------- + public static function estimateBitsModeNum($size) + { + $w = (int)$size / 3; + $bits = $w * 10; + + switch($size - $w * 3) { + case 1: + $bits += 4; + break; + case 2: + $bits += 7; + break; + default: + break; + } + + return $bits; + } + + //---------------------------------------------------------------------- + public static $anTable = array( + -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, + -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, + 36, -1, -1, -1, 37, 38, -1, -1, -1, -1, 39, 40, -1, 41, 42, 43, + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 44, -1, -1, -1, -1, -1, + -1, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, + 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, -1, -1, -1, -1, -1, + -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, + -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1 + ); + + //---------------------------------------------------------------------- + public static function lookAnTable($c) + { + return (($c > 127)?-1:self::$anTable[$c]); + } + + //---------------------------------------------------------------------- + public static function checkModeAn($size, $data) + { + for($i=0; $i<$size; $i++) { + if (self::lookAnTable(ord($data[$i])) == -1) { + return false; + } + } + + return true; + } + + //---------------------------------------------------------------------- + public static function estimateBitsModeAn($size) + { + $w = (int)($size / 2); + $bits = $w * 11; + + if($size & 1) { + $bits += 6; + } + + return $bits; + } + + //---------------------------------------------------------------------- + public static function estimateBitsMode8($size) + { + return $size * 8; + } + + //---------------------------------------------------------------------- + public function estimateBitsModeKanji($size) + { + return (int)(($size / 2) * 13); + } + + //---------------------------------------------------------------------- + public static function checkModeKanji($size, $data) + { + if($size & 1) + return false; + + for($i=0; $i<$size; $i+=2) { + $val = (ord($data[$i]) << 8) | ord($data[$i+1]); + if( $val < 0x8140 + || ($val > 0x9ffc && $val < 0xe040) + || $val > 0xebbf) { + return false; + } + } + + return true; + } + + /*********************************************************************** + * Validation + **********************************************************************/ + + public static function check($mode, $size, $data) + { + if($size <= 0) + return false; + + switch($mode) { + case Constants::QR_MODE_NUM: return self::checkModeNum($size, $data); break; + case Constants::QR_MODE_AN: return self::checkModeAn($size, $data); break; + case Constants::QR_MODE_KANJI: return self::checkModeKanji($size, $data); break; + case Constants::QR_MODE_8: return true; break; + case Constants::QR_MODE_STRUCTURE: return true; break; + + default: + break; + } + + return false; + } + + + //---------------------------------------------------------------------- + public function estimateBitStreamSize($version) + { + $bits = 0; + + foreach($this->items as $item) { + $bits += $item->estimateBitStreamSizeOfEntry($version); + } + + return $bits; + } + + //---------------------------------------------------------------------- + public function estimateVersion() + { + $version = 0; + $prev = 0; + do { + $prev = $version; + $bits = $this->estimateBitStreamSize($prev); + $version = QRspec::getMinimumVersion((int)(($bits + 7) / 8), $this->level); + if ($version < 0) { + return -1; + } + } while ($version > $prev); + + return $version; + } + + //---------------------------------------------------------------------- + public static function lengthOfCode($mode, $version, $bits) + { + $payload = $bits - 4 - QRspec::lengthIndicator($mode, $version); + switch($mode) { + case Constants::QR_MODE_NUM: + $chunks = (int)($payload / 10); + $remain = $payload - $chunks * 10; + $size = $chunks * 3; + if($remain >= 7) { + $size += 2; + } else if($remain >= 4) { + $size += 1; + } + break; + case Constants::QR_MODE_AN: + $chunks = (int)($payload / 11); + $remain = $payload - $chunks * 11; + $size = $chunks * 2; + if($remain >= 6) + $size++; + break; + case Constants::QR_MODE_8: + $size = (int)($payload / 8); + break; + case Constants::QR_MODE_KANJI: + $size = (int)(($payload / 13) * 2); + break; + case Constants::QR_MODE_STRUCTURE: + $size = (int)($payload / 8); + break; + default: + $size = 0; + break; + } + + $maxsize = QRspec::maximumWords($mode, $version); + if($size < 0) $size = 0; + if($size > $maxsize) $size = $maxsize; + + return $size; + } + + //---------------------------------------------------------------------- + public function createBitStream() + { + $total = 0; + + foreach($this->items as $item) { + $bits = $item->encodeBitStream($this->version); + + if($bits < 0) + return -1; + + $total += $bits; + } + + return $total; + } + + //---------------------------------------------------------------------- + public function convertData() + { + $ver = $this->estimateVersion(); + if($ver > $this->getVersion()) { + $this->setVersion($ver); + } + + for(;;) { + $bits = $this->createBitStream(); + + if($bits < 0) + return -1; + + $ver = QRspec::getMinimumVersion((int)(($bits + 7) / 8), $this->level); + if($ver < 0) { + throw new Exception('WRONG VERSION'); + return -1; + } else if($ver > $this->getVersion()) { + $this->setVersion($ver); + } else { + break; + } + } + + return 0; + } + + //---------------------------------------------------------------------- + public function appendPaddingBit(&$bstream) + { + $bits = $bstream->size(); + $maxwords = QRspec::getDataLength($this->version, $this->level); + $maxbits = $maxwords * 8; + + if ($maxbits == $bits) { + return 0; + } + + if ($maxbits - $bits < 5) { + return $bstream->appendNum($maxbits - $bits, 0); + } + + $bits += 4; + $words = (int)(($bits + 7) / 8); + + $padding = new QRbitstream(); + $ret = $padding->appendNum($words * 8 - $bits + 4, 0); + + if($ret < 0) + return $ret; + + $padlen = $maxwords - $words; + + if($padlen > 0) { + + $padbuf = array(); + for($i=0; $i<$padlen; $i++) { + $padbuf[$i] = ($i&1)?0x11:0xec; + } + + $ret = $padding->appendBytes($padlen, $padbuf); + + if($ret < 0) + return $ret; + + } + + $ret = $bstream->append($padding); + + return $ret; + } + + //---------------------------------------------------------------------- + public function mergeBitStream() + { + if($this->convertData() < 0) { + return null; + } + + $bstream = new QRbitstream(); + + foreach($this->items as $item) { + $ret = $bstream->append($item->bstream); + if($ret < 0) { + return null; + } + } + + return $bstream; + } + + //---------------------------------------------------------------------- + public function getBitStream() + { + + $bstream = $this->mergeBitStream(); + + if($bstream == null) { + return null; + } + + $ret = $this->appendPaddingBit($bstream); + if($ret < 0) { + return null; + } + + return $bstream; + } + + //---------------------------------------------------------------------- + public function getByteStream() + { + $bstream = $this->getBitStream(); + if($bstream == null) { + return null; + } + + return $bstream->toByte(); + } +} + + diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRinputItem.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRinputItem.php new file mode 100644 index 000000000..1e5eb18de --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRinputItem.php @@ -0,0 +1,246 @@ +mode = $mode; + $this->size = $size; + $this->data = $setData; + $this->bstream = $bstream; + } + + //---------------------------------------------------------------------- + public function encodeModeNum($version) + { + try { + + $words = (int)($this->size / 3); + $bs = new QRbitstream(); + + $val = 0x1; + $bs->appendNum(4, $val); + $bs->appendNum(QRspec::lengthIndicator(Constants::QR_MODE_NUM, $version), $this->size); + + for($i=0; $i<$words; $i++) { + $val = (ord($this->data[$i*3 ]) - ord('0')) * 100; + $val += (ord($this->data[$i*3+1]) - ord('0')) * 10; + $val += (ord($this->data[$i*3+2]) - ord('0')); + $bs->appendNum(10, $val); + } + + if($this->size - $words * 3 == 1) { + $val = ord($this->data[$words*3]) - ord('0'); + $bs->appendNum(4, $val); + } else if($this->size - $words * 3 == 2) { + $val = (ord($this->data[$words*3 ]) - ord('0')) * 10; + $val += (ord($this->data[$words*3+1]) - ord('0')); + $bs->appendNum(7, $val); + } + + $this->bstream = $bs; + return 0; + + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + public function encodeModeAn($version) + { + try { + $words = (int)($this->size / 2); + $bs = new QRbitstream(); + + $bs->appendNum(4, 0x02); + $bs->appendNum(QRspec::lengthIndicator(Constants::QR_MODE_AN, $version), $this->size); + + for($i=0; $i<$words; $i++) { + $val = (int)QRinput::lookAnTable(ord($this->data[$i*2 ])) * 45; + $val += (int)QRinput::lookAnTable(ord($this->data[$i*2+1])); + + $bs->appendNum(11, $val); + } + + if($this->size & 1) { + $val = QRinput::lookAnTable(ord($this->data[$words * 2])); + $bs->appendNum(6, $val); + } + + $this->bstream = $bs; + return 0; + + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + public function encodeMode8($version) + { + try { + $bs = new QRbitstream(); + + $bs->appendNum(4, 0x4); + $bs->appendNum(QRspec::lengthIndicator(Constants::QR_MODE_8, $version), $this->size); + + for($i=0; $i<$this->size; $i++) { + $bs->appendNum(8, ord($this->data[$i])); + } + + $this->bstream = $bs; + return 0; + + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + public function encodeModeKanji($version) + { + try { + + $bs = new QRbitstream(); + + $bs->appendNum(4, 0x8); + $bs->appendNum(QRspec::lengthIndicator(Constants::QR_MODE_KANJI, $version), (int)($this->size / 2)); + + for($i=0; $i<$this->size; $i+=2) { + $val = (ord($this->data[$i]) << 8) | ord($this->data[$i+1]); + if($val <= 0x9ffc) { + $val -= 0x8140; + } else { + $val -= 0xc140; + } + + $h = ($val >> 8) * 0xc0; + $val = ($val & 0xff) + $h; + + $bs->appendNum(13, $val); + } + + $this->bstream = $bs; + return 0; + + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + public function encodeModeStructure() + { + try { + $bs = new QRbitstream(); + + $bs->appendNum(4, 0x03); + $bs->appendNum(4, ord($this->data[1]) - 1); + $bs->appendNum(4, ord($this->data[0]) - 1); + $bs->appendNum(8, ord($this->data[2])); + + $this->bstream = $bs; + return 0; + + } catch (Exception $e) { + return -1; + } + } + + //---------------------------------------------------------------------- + public function estimateBitStreamSizeOfEntry($version) + { + $bits = 0; + + if($version == 0) + $version = 1; + + switch($this->mode) { + case Constants::QR_MODE_NUM: $bits = QRinput::estimateBitsModeNum($this->size); break; + case Constants::QR_MODE_AN: $bits = QRinput::estimateBitsModeAn($this->size); break; + case Constants::QR_MODE_8: $bits = QRinput::estimateBitsMode8($this->size); break; + case Constants::QR_MODE_KANJI: $bits = QRinput::estimateBitsModeKanji($this->size);break; + case Constants::QR_MODE_STRUCTURE: return Constants::STRUCTURE_HEADER_BITS; + default: + return 0; + } + + $l = QRspec::lengthIndicator($this->mode, $version); + $m = 1 << $l; + $num = (int)(($this->size + $m - 1) / $m); + + $bits += $num * (4 + $l); + + return $bits; + } + + //---------------------------------------------------------------------- + public function encodeBitStream($version) + { + try { + + unset($this->bstream); + $words = QRspec::maximumWords($this->mode, $version); + + if($this->size > $words) { + + $st1 = new QRinputItem($this->mode, $words, $this->data); + $st2 = new QRinputItem($this->mode, $this->size - $words, array_slice($this->data, $words)); + + $st1->encodeBitStream($version); + $st2->encodeBitStream($version); + + $this->bstream = new QRbitstream(); + $this->bstream->append($st1->bstream); + $this->bstream->append($st2->bstream); + + unset($st1); + unset($st2); + + } else { + + $ret = 0; + + switch($this->mode) { + case Constants::QR_MODE_NUM: $ret = $this->encodeModeNum($version); break; + case Constants::QR_MODE_AN: $ret = $this->encodeModeAn($version); break; + case Constants::QR_MODE_8: $ret = $this->encodeMode8($version); break; + case Constants::QR_MODE_KANJI: $ret = $this->encodeModeKanji($version);break; + case Constants::QR_MODE_STRUCTURE: $ret = $this->encodeModeStructure(); break; + + default: + break; + } + + if($ret < 0) + return -1; + } + + return $this->bstream->size(); + + } catch (Exception $e) { + return -1; + } + } +} diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRmask.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRmask.php new file mode 100644 index 000000000..2be76f471 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRmask.php @@ -0,0 +1,325 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRmask { + + public $runLength = array(); + + //---------------------------------------------------------------------- + public function __construct() + { + $this->runLength = array_fill(0, Constants::QRSPEC_WIDTH_MAX + 1, 0); + } + + //---------------------------------------------------------------------- + public function writeFormatInformation($width, &$frame, $mask, $level) + { + $blacks = 0; + $format = QRspec::getFormatInfo($mask, $level); + + for($i=0; $i<8; $i++) { + if($format & 1) { + $blacks += 2; + $v = 0x85; + } else { + $v = 0x84; + } + + $frame[8][$width - 1 - $i] = chr($v); + if($i < 6) { + $frame[$i][8] = chr($v); + } else { + $frame[$i + 1][8] = chr($v); + } + $format = $format >> 1; + } + + for($i=0; $i<7; $i++) { + if($format & 1) { + $blacks += 2; + $v = 0x85; + } else { + $v = 0x84; + } + + $frame[$width - 7 + $i][8] = chr($v); + if($i == 0) { + $frame[8][7] = chr($v); + } else { + $frame[8][6 - $i] = chr($v); + } + + $format = $format >> 1; + } + + return $blacks; + } + + //---------------------------------------------------------------------- + public function mask0($x, $y) { return ($x+$y)&1; } + public function mask1($x, $y) { return ($y&1); } + public function mask2($x, $y) { return ($x%3); } + public function mask3($x, $y) { return ($x+$y)%3; } + public function mask4($x, $y) { return (((int)($y/2))+((int)($x/3)))&1; } + public function mask5($x, $y) { return (($x*$y)&1)+($x*$y)%3; } + public function mask6($x, $y) { return ((($x*$y)&1)+($x*$y)%3)&1; } + public function mask7($x, $y) { return ((($x*$y)%3)+(($x+$y)&1))&1; } + + //---------------------------------------------------------------------- + private function generateMaskNo($maskNo, $width, $frame) + { + $bitMask = array_fill(0, $width, array_fill(0, $width, 0)); + + for($y=0; $y<$width; $y++) { + for($x=0; $x<$width; $x++) { + if(ord($frame[$y][$x]) & 0x80) { + $bitMask[$y][$x] = 0; + } else { + $maskFunc = call_user_func(array($this, 'mask'.$maskNo), $x, $y); + $bitMask[$y][$x] = ($maskFunc == 0)?1:0; + } + + } + } + + return $bitMask; + } + + //---------------------------------------------------------------------- + public static function serial($bitFrame) + { + $codeArr = array(); + + foreach ($bitFrame as $line) + $codeArr[] = join('', $line); + + return gzcompress(join("\n", $codeArr), 9); + } + + //---------------------------------------------------------------------- + public static function unserial($code) + { + $codeArr = array(); + + $codeLines = explode("\n", gzuncompress($code)); + foreach ($codeLines as $line) + $codeArr[] = str_split($line); + + return $codeArr; + } + + //---------------------------------------------------------------------- + public function makeMaskNo($maskNo, $width, $s, &$d, $maskGenOnly = false) + { + $b = 0; + $bitMask = array(); + + $fileName = Constants::QR_CACHE_DIR.'mask_'.$maskNo.DIRECTORY_SEPARATOR.'mask_'.$width.'_'.$maskNo.'.dat'; + + if (Constants::QR_CACHEABLE) { + if (file_exists($fileName)) { + $bitMask = self::unserial(file_get_contents($fileName)); + } else { + $bitMask = $this->generateMaskNo($maskNo, $width, $s, $d); + if (!file_exists(Constants::QR_CACHE_DIR.'mask_'.$maskNo)) + mkdir(Constants::QR_CACHE_DIR.'mask_'.$maskNo); + file_put_contents($fileName, self::serial($bitMask)); + } + } else { + $bitMask = $this->generateMaskNo($maskNo, $width, $s, $d); + } + + if ($maskGenOnly) + return; + + $d = $s; + + for($y=0; $y<$width; $y++) { + for($x=0; $x<$width; $x++) { + if($bitMask[$y][$x] == 1) { + $d[$y][$x] = chr(ord($s[$y][$x]) ^ (int)$bitMask[$y][$x]); + } + $b += (int)(ord($d[$y][$x]) & 1); + } + } + + return $b; + } + + //---------------------------------------------------------------------- + public function makeMask($width, $frame, $maskNo, $level) + { + $masked = array_fill(0, $width, str_repeat("\0", $width)); + $this->makeMaskNo($maskNo, $width, $frame, $masked); + $this->writeFormatInformation($width, $masked, $maskNo, $level); + + return $masked; + } + + //---------------------------------------------------------------------- + public function calcN1N3($length) + { + $demerit = 0; + + for($i=0; $i<$length; $i++) { + + if($this->runLength[$i] >= 5) { + $demerit += (Constants::N1 + ($this->runLength[$i] - 5)); + } + if($i & 1) { + if(($i >= 3) && ($i < ($length-2)) && ($this->runLength[$i] % 3 == 0)) { + $fact = (int)($this->runLength[$i] / 3); + if(($this->runLength[$i-2] == $fact) && + ($this->runLength[$i-1] == $fact) && + ($this->runLength[$i+1] == $fact) && + ($this->runLength[$i+2] == $fact)) { + if(($this->runLength[$i-3] < 0) || ($this->runLength[$i-3] >= (4 * $fact))) { + $demerit += Constants::N3; + } else if((($i+3) >= $length) || ($this->runLength[$i+3] >= (4 * $fact))) { + $demerit += Constants::N3; + } + } + } + } + } + return $demerit; + } + + //---------------------------------------------------------------------- + public function evaluateSymbol($width, $frame) + { + $head = 0; + $demerit = 0; + + for($y=0; $y<$width; $y++) { + $head = 0; + $this->runLength[0] = 1; + + $frameY = $frame[$y]; + + if ($y>0) + $frameYM = $frame[$y-1]; + + for($x=0; $x<$width; $x++) { + if(($x > 0) && ($y > 0)) { + $b22 = ord($frameY[$x]) & ord($frameY[$x-1]) & ord($frameYM[$x]) & ord($frameYM[$x-1]); + $w22 = ord($frameY[$x]) | ord($frameY[$x-1]) | ord($frameYM[$x]) | ord($frameYM[$x-1]); + + if(($b22 | ($w22 ^ 1))&1) { + $demerit += Constants::N2; + } + } + if(($x == 0) && (ord($frameY[$x]) & 1)) { + $this->runLength[0] = -1; + $head = 1; + $this->runLength[$head] = 1; + } else if($x > 0) { + if((ord($frameY[$x]) ^ ord($frameY[$x-1])) & 1) { + $head++; + $this->runLength[$head] = 1; + } else { + $this->runLength[$head]++; + } + } + } + + $demerit += $this->calcN1N3($head+1); + } + + for($x=0; $x<$width; $x++) { + $head = 0; + $this->runLength[0] = 1; + + for($y=0; $y<$width; $y++) { + if($y == 0 && (ord($frame[$y][$x]) & 1)) { + $this->runLength[0] = -1; + $head = 1; + $this->runLength[$head] = 1; + } else if($y > 0) { + if((ord($frame[$y][$x]) ^ ord($frame[$y-1][$x])) & 1) { + $head++; + $this->runLength[$head] = 1; + } else { + $this->runLength[$head]++; + } + } + } + + $demerit += $this->calcN1N3($head+1); + } + + return $demerit; + } + + + //---------------------------------------------------------------------- + public function mask($width, $frame, $level) + { + $minDemerit = PHP_INT_MAX; + $bestMaskNum = 0; + $bestMask = array(); + + $checked_masks = array(0,1,2,3,4,5,6,7); + + if (Constants::QR_FIND_FROM_RANDOM !== false) { + + $howManuOut = 8-(Constants::QR_FIND_FROM_RANDOM % 9); + for ($i = 0; $i < $howManuOut; $i++) { + $remPos = rand (0, count($checked_masks)-1); + unset($checked_masks[$remPos]); + $checked_masks = array_values($checked_masks); + } + + } + + $bestMask = $frame; + + foreach($checked_masks as $i) { + $mask = array_fill(0, $width, str_repeat("\0", $width)); + + $demerit = 0; + $blacks = 0; + $blacks = $this->makeMaskNo($i, $width, $frame, $mask); + $blacks += $this->writeFormatInformation($width, $mask, $i, $level); + $blacks = (int)(100 * $blacks / ($width * $width)); + $demerit = (int)((int)(abs($blacks - 50) / 5) * Constants::N4); + $demerit += $this->evaluateSymbol($width, $mask); + + if($demerit < $minDemerit) { + $minDemerit = $demerit; + $bestMask = $mask; + $bestMaskNum = $i; + } + } + + return $bestMask; + } + + //---------------------------------------------------------------------- +} diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrawcode.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrawcode.php new file mode 100644 index 000000000..25eae7c8d --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrawcode.php @@ -0,0 +1,117 @@ +datacode = $input->getByteStream(); + if(is_null($this->datacode)) { + throw new Exception('null input string'); + } + + QRspec::getEccSpec($input->getVersion(), $input->getErrorCorrectionLevel(), $spec); + + $this->version = $input->getVersion(); + $this->b1 = QRspec::rsBlockNum1($spec); + $this->dataLength = QRspec::rsDataLength($spec); + $this->eccLength = QRspec::rsEccLength($spec); + $this->ecccode = array_fill(0, $this->eccLength, 0); + $this->blocks = QRspec::rsBlockNum($spec); + + $ret = $this->init($spec); + if($ret < 0) { + throw new Exception('block alloc error'); + return null; + } + + $this->count = 0; + } + + //---------------------------------------------------------------------- + public function init(array $spec) + { + $dl = QRspec::rsDataCodes1($spec); + $el = QRspec::rsEccCodes1($spec); + $rs = QRrs::init_rs(8, 0x11d, 0, 1, $el, 255 - $dl - $el); + + + $blockNo = 0; + $dataPos = 0; + $eccPos = 0; + for($i=0; $iecccode,$eccPos); + $this->rsblocks[$blockNo] = new QRrsblock($dl, array_slice($this->datacode, $dataPos), $el, $ecc, $rs); + $this->ecccode = array_merge(array_slice($this->ecccode,0, $eccPos), $ecc); + + $dataPos += $dl; + $eccPos += $el; + $blockNo++; + } + + if(QRspec::rsBlockNum2($spec) == 0) + return 0; + + $dl = QRspec::rsDataCodes2($spec); + $el = QRspec::rsEccCodes2($spec); + $rs = QRrs::init_rs(8, 0x11d, 0, 1, $el, 255 - $dl - $el); + + if($rs == NULL) return -1; + + for($i=0; $iecccode,$eccPos); + $this->rsblocks[$blockNo] = new QRrsblock($dl, array_slice($this->datacode, $dataPos), $el, $ecc, $rs); + $this->ecccode = array_merge(array_slice($this->ecccode,0, $eccPos), $ecc); + + $dataPos += $dl; + $eccPos += $el; + $blockNo++; + } + + return 0; + } + + //---------------------------------------------------------------------- + public function getCode() + { + $ret = null; + + if($this->count < $this->dataLength) { + $row = $this->count % $this->blocks; + $col = $this->count / $this->blocks; + if($col >= $this->rsblocks[0]->dataLength) { + $row += $this->b1; + } + $ret = $this->rsblocks[$row]->data[$col]; + } else if($this->count < $this->dataLength + $this->eccLength) { + $row = ($this->count - $this->dataLength) % $this->blocks; + $col = ($this->count - $this->dataLength) / $this->blocks; + $ret = $this->rsblocks[$row]->ecc[$col]; + } else { + return 0; + } + $this->count++; + + return $ret; + } +} diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrs.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrs.php new file mode 100644 index 000000000..66f0d5e77 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrs.php @@ -0,0 +1,56 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRrs { + + public static $items = array(); + + //---------------------------------------------------------------------- + public static function init_rs($symsize, $gfpoly, $fcr, $prim, $nroots, $pad) + { + foreach(self::$items as $rs) { + if($rs->pad != $pad) continue; + if($rs->nroots != $nroots) continue; + if($rs->mm != $symsize) continue; + if($rs->gfpoly != $gfpoly) continue; + if($rs->fcr != $fcr) continue; + if($rs->prim != $prim) continue; + + return $rs; + } + + $rs = QRrsItem::init_rs_char($symsize, $gfpoly, $fcr, $prim, $nroots, $pad); + array_unshift(self::$items, $rs); + + return $rs; + } +} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrsItem.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrsItem.php new file mode 100644 index 000000000..ce63a8c35 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrsItem.php @@ -0,0 +1,162 @@ += $this->nn) { + $x -= $this->nn; + $x = ($x >> $this->mm) + ($x & $this->nn); + } + + return $x; + } + + //---------------------------------------------------------------------- + public static function init_rs_char($symsize, $gfpoly, $fcr, $prim, $nroots, $pad) + { + // Common code for intializing a Reed-Solomon control block (char or int symbols) + // Copyright 2004 Phil Karn, KA9Q + // May be used under the terms of the GNU Lesser General Public License (LGPL) + + $rs = null; + + // Check parameter ranges + if($symsize < 0 || $symsize > 8) return $rs; + if($fcr < 0 || $fcr >= (1<<$symsize)) return $rs; + if($prim <= 0 || $prim >= (1<<$symsize)) return $rs; + if($nroots < 0 || $nroots >= (1<<$symsize)) return $rs; // Can't have more roots than symbol values! + if($pad < 0 || $pad >= ((1<<$symsize) -1 - $nroots)) return $rs; // Too much padding + + $rs = new QRrsItem(); + $rs->mm = $symsize; + $rs->nn = (1<<$symsize)-1; + $rs->pad = $pad; + + $rs->alpha_to = array_fill(0, $rs->nn+1, 0); + $rs->index_of = array_fill(0, $rs->nn+1, 0); + + // PHP style macro replacement ;) + $NN =& $rs->nn; + $A0 =& $NN; + + // Generate Galois field lookup tables + $rs->index_of[0] = $A0; // log(zero) = -inf + $rs->alpha_to[$A0] = 0; // alpha**-inf = 0 + $sr = 1; + + for($i=0; $i<$rs->nn; $i++) { + $rs->index_of[$sr] = $i; + $rs->alpha_to[$i] = $sr; + $sr <<= 1; + if($sr & (1<<$symsize)) { + $sr ^= $gfpoly; + } + $sr &= $rs->nn; + } + + if($sr != 1){ + // field generator polynomial is not primitive! + $rs = NULL; + return $rs; + } + + /* Form RS code generator polynomial from its roots */ + $rs->genpoly = array_fill(0, $nroots+1, 0); + + $rs->fcr = $fcr; + $rs->prim = $prim; + $rs->nroots = $nroots; + $rs->gfpoly = $gfpoly; + + /* Find prim-th root of 1, used in decoding */ + for($iprim=1;($iprim % $prim) != 0;$iprim += $rs->nn) + ; // intentional empty-body loop! + + $rs->iprim = (int)($iprim / $prim); + $rs->genpoly[0] = 1; + + for ($i = 0,$root=$fcr*$prim; $i < $nroots; $i++, $root += $prim) { + $rs->genpoly[$i+1] = 1; + + // Multiply rs->genpoly[] by @**(root + x) + for ($j = $i; $j > 0; $j--) { + if ($rs->genpoly[$j] != 0) { + $rs->genpoly[$j] = $rs->genpoly[$j-1] ^ $rs->alpha_to[$rs->modnn($rs->index_of[$rs->genpoly[$j]] + $root)]; + } else { + $rs->genpoly[$j] = $rs->genpoly[$j-1]; + } + } + // rs->genpoly[0] can never be zero + $rs->genpoly[0] = $rs->alpha_to[$rs->modnn($rs->index_of[$rs->genpoly[0]] + $root)]; + } + + // convert rs->genpoly[] to index form for quicker encoding + for ($i = 0; $i <= $nroots; $i++) + $rs->genpoly[$i] = $rs->index_of[$rs->genpoly[$i]]; + + return $rs; + } + + //---------------------------------------------------------------------- + public function encode_rs_char($data, &$parity) + { + $MM =& $this->mm; + $NN =& $this->nn; + $ALPHA_TO =& $this->alpha_to; + $INDEX_OF =& $this->index_of; + $GENPOLY =& $this->genpoly; + $NROOTS =& $this->nroots; + $FCR =& $this->fcr; + $PRIM =& $this->prim; + $IPRIM =& $this->iprim; + $PAD =& $this->pad; + $A0 =& $NN; + + $parity = array_fill(0, $NROOTS, 0); + + for($i=0; $i< ($NN-$NROOTS-$PAD); $i++) { + + $feedback = $INDEX_OF[$data[$i] ^ $parity[0]]; + if($feedback != $A0) { + // feedback term is non-zero + + // This line is unnecessary when GENPOLY[NROOTS] is unity, as it must + // always be for the polynomials constructed by init_rs() + $feedback = $this->modnn($NN - $GENPOLY[$NROOTS] + $feedback); + + for($j=1;$j<$NROOTS;$j++) { + $parity[$j] ^= $ALPHA_TO[$this->modnn($feedback + $GENPOLY[$NROOTS-$j])]; + } + } + + // Shift + array_shift($parity); + if($feedback != $A0) { + array_push($parity, $ALPHA_TO[$this->modnn($feedback + $GENPOLY[0])]); + } else { + array_push($parity, 0); + } + } + } +} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrsblock.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrsblock.php new file mode 100644 index 000000000..c1d01f22e --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRrsblock.php @@ -0,0 +1,25 @@ +encode_rs_char($data, $ecc); + + $this->dataLength = $dl; + $this->data = $data; + $this->eccLength = $el; + $this->ecc = $ecc; + } +}; \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRspec.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRspec.php new file mode 100644 index 000000000..d68432604 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRspec.php @@ -0,0 +1,586 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * The following data / specifications are taken from + * "Two dimensional symbol -- QR-code -- Basic Specification" (JIS X0510:2004) + * or + * "Automatic identification and data capture techniques -- + * QR Code 2005 bar code symbology specification" (ISO/IEC 18004:2006) + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRspec { + + public static $capacity = array( + array( 0, 0, 0, array( 0, 0, 0, 0)), + array( 21, 26, 0, array( 7, 10, 13, 17)), // 1 + array( 25, 44, 7, array( 10, 16, 22, 28)), + array( 29, 70, 7, array( 15, 26, 36, 44)), + array( 33, 100, 7, array( 20, 36, 52, 64)), + array( 37, 134, 7, array( 26, 48, 72, 88)), // 5 + array( 41, 172, 7, array( 36, 64, 96, 112)), + array( 45, 196, 0, array( 40, 72, 108, 130)), + array( 49, 242, 0, array( 48, 88, 132, 156)), + array( 53, 292, 0, array( 60, 110, 160, 192)), + array( 57, 346, 0, array( 72, 130, 192, 224)), //10 + array( 61, 404, 0, array( 80, 150, 224, 264)), + array( 65, 466, 0, array( 96, 176, 260, 308)), + array( 69, 532, 0, array( 104, 198, 288, 352)), + array( 73, 581, 3, array( 120, 216, 320, 384)), + array( 77, 655, 3, array( 132, 240, 360, 432)), //15 + array( 81, 733, 3, array( 144, 280, 408, 480)), + array( 85, 815, 3, array( 168, 308, 448, 532)), + array( 89, 901, 3, array( 180, 338, 504, 588)), + array( 93, 991, 3, array( 196, 364, 546, 650)), + array( 97, 1085, 3, array( 224, 416, 600, 700)), //20 + array(101, 1156, 4, array( 224, 442, 644, 750)), + array(105, 1258, 4, array( 252, 476, 690, 816)), + array(109, 1364, 4, array( 270, 504, 750, 900)), + array(113, 1474, 4, array( 300, 560, 810, 960)), + array(117, 1588, 4, array( 312, 588, 870, 1050)), //25 + array(121, 1706, 4, array( 336, 644, 952, 1110)), + array(125, 1828, 4, array( 360, 700, 1020, 1200)), + array(129, 1921, 3, array( 390, 728, 1050, 1260)), + array(133, 2051, 3, array( 420, 784, 1140, 1350)), + array(137, 2185, 3, array( 450, 812, 1200, 1440)), //30 + array(141, 2323, 3, array( 480, 868, 1290, 1530)), + array(145, 2465, 3, array( 510, 924, 1350, 1620)), + array(149, 2611, 3, array( 540, 980, 1440, 1710)), + array(153, 2761, 3, array( 570, 1036, 1530, 1800)), + array(157, 2876, 0, array( 570, 1064, 1590, 1890)), //35 + array(161, 3034, 0, array( 600, 1120, 1680, 1980)), + array(165, 3196, 0, array( 630, 1204, 1770, 2100)), + array(169, 3362, 0, array( 660, 1260, 1860, 2220)), + array(173, 3532, 0, array( 720, 1316, 1950, 2310)), + array(177, 3706, 0, array( 750, 1372, 2040, 2430)) //40 + ); + + //---------------------------------------------------------------------- + public static function getDataLength($version, $level) + { + return self::$capacity[$version][Constants::QRCAP_WORDS] - self::$capacity[$version][Constants::QRCAP_EC][$level]; + } + + //---------------------------------------------------------------------- + public static function getECCLength($version, $level) + { + return self::$capacity[$version][Constants::QRCAP_EC][$level]; + } + + //---------------------------------------------------------------------- + public static function getWidth($version) + { + return self::$capacity[$version][Constants::QRCAP_WIDTH]; + } + + //---------------------------------------------------------------------- + public static function getRemainder($version) + { + return self::$capacity[$version][Constants::QRCAP_REMINDER]; + } + + //---------------------------------------------------------------------- + public static function getMinimumVersion($size, $level) + { + + for($i=1; $i<= Constants::QRSPEC_VERSION_MAX; $i++) { + $words = self::$capacity[$i][Constants::QRCAP_WORDS] - self::$capacity[$i][Constants::QRCAP_EC][$level]; + if($words >= $size) + return $i; + } + + return -1; + } + + //###################################################################### + + public static $lengthTableBits = array( + array(10, 12, 14), + array( 9, 11, 13), + array( 8, 16, 16), + array( 8, 10, 12) + ); + + //---------------------------------------------------------------------- + public static function lengthIndicator($mode, $version) + { + if ($mode == Constants::QR_MODE_STRUCTURE) + return 0; + + if ($version <= 9) { + $l = 0; + } else if ($version <= 26) { + $l = 1; + } else { + $l = 2; + } + + return self::$lengthTableBits[$mode][$l]; + } + + //---------------------------------------------------------------------- + public static function maximumWords($mode, $version) + { + if($mode == Constants::QR_MODE_STRUCTURE) + return 3; + + if($version <= 9) { + $l = 0; + } else if($version <= 26) { + $l = 1; + } else { + $l = 2; + } + + $bits = self::$lengthTableBits[$mode][$l]; + $words = (1 << $bits) - 1; + + if($mode == Constants::QR_MODE_KANJI) { + $words *= 2; // the number of bytes is required + } + + return $words; + } + + // Error correction code ----------------------------------------------- + // Table of the error correction code (Reed-Solomon block) + // See Table 12-16 (pp.30-36), JIS X0510:2004. + + public static $eccTable = array( + array(array( 0, 0), array( 0, 0), array( 0, 0), array( 0, 0)), + array(array( 1, 0), array( 1, 0), array( 1, 0), array( 1, 0)), // 1 + array(array( 1, 0), array( 1, 0), array( 1, 0), array( 1, 0)), + array(array( 1, 0), array( 1, 0), array( 2, 0), array( 2, 0)), + array(array( 1, 0), array( 2, 0), array( 2, 0), array( 4, 0)), + array(array( 1, 0), array( 2, 0), array( 2, 2), array( 2, 2)), // 5 + array(array( 2, 0), array( 4, 0), array( 4, 0), array( 4, 0)), + array(array( 2, 0), array( 4, 0), array( 2, 4), array( 4, 1)), + array(array( 2, 0), array( 2, 2), array( 4, 2), array( 4, 2)), + array(array( 2, 0), array( 3, 2), array( 4, 4), array( 4, 4)), + array(array( 2, 2), array( 4, 1), array( 6, 2), array( 6, 2)), //10 + array(array( 4, 0), array( 1, 4), array( 4, 4), array( 3, 8)), + array(array( 2, 2), array( 6, 2), array( 4, 6), array( 7, 4)), + array(array( 4, 0), array( 8, 1), array( 8, 4), array(12, 4)), + array(array( 3, 1), array( 4, 5), array(11, 5), array(11, 5)), + array(array( 5, 1), array( 5, 5), array( 5, 7), array(11, 7)), //15 + array(array( 5, 1), array( 7, 3), array(15, 2), array( 3, 13)), + array(array( 1, 5), array(10, 1), array( 1, 15), array( 2, 17)), + array(array( 5, 1), array( 9, 4), array(17, 1), array( 2, 19)), + array(array( 3, 4), array( 3, 11), array(17, 4), array( 9, 16)), + array(array( 3, 5), array( 3, 13), array(15, 5), array(15, 10)), //20 + array(array( 4, 4), array(17, 0), array(17, 6), array(19, 6)), + array(array( 2, 7), array(17, 0), array( 7, 16), array(34, 0)), + array(array( 4, 5), array( 4, 14), array(11, 14), array(16, 14)), + array(array( 6, 4), array( 6, 14), array(11, 16), array(30, 2)), + array(array( 8, 4), array( 8, 13), array( 7, 22), array(22, 13)), //25 + array(array(10, 2), array(19, 4), array(28, 6), array(33, 4)), + array(array( 8, 4), array(22, 3), array( 8, 26), array(12, 28)), + array(array( 3, 10), array( 3, 23), array( 4, 31), array(11, 31)), + array(array( 7, 7), array(21, 7), array( 1, 37), array(19, 26)), + array(array( 5, 10), array(19, 10), array(15, 25), array(23, 25)), //30 + array(array(13, 3), array( 2, 29), array(42, 1), array(23, 28)), + array(array(17, 0), array(10, 23), array(10, 35), array(19, 35)), + array(array(17, 1), array(14, 21), array(29, 19), array(11, 46)), + array(array(13, 6), array(14, 23), array(44, 7), array(59, 1)), + array(array(12, 7), array(12, 26), array(39, 14), array(22, 41)), //35 + array(array( 6, 14), array( 6, 34), array(46, 10), array( 2, 64)), + array(array(17, 4), array(29, 14), array(49, 10), array(24, 46)), + array(array( 4, 18), array(13, 32), array(48, 14), array(42, 32)), + array(array(20, 4), array(40, 7), array(43, 22), array(10, 67)), + array(array(19, 6), array(18, 31), array(34, 34), array(20, 61)),//40 + ); + + //---------------------------------------------------------------------- + // CACHEABLE!!! + + public static function getEccSpec($version, $level, array &$spec) + { + if (count($spec) < 5) { + $spec = array(0,0,0,0,0); + } + + $b1 = self::$eccTable[$version][$level][0]; + $b2 = self::$eccTable[$version][$level][1]; + $data = self::getDataLength($version, $level); + $ecc = self::getECCLength($version, $level); + + if($b2 == 0) { + $spec[0] = $b1; + $spec[1] = (int)($data / $b1); + $spec[2] = (int)($ecc / $b1); + $spec[3] = 0; + $spec[4] = 0; + } else { + $spec[0] = $b1; + $spec[1] = (int)($data / ($b1 + $b2)); + $spec[2] = (int)($ecc / ($b1 + $b2)); + $spec[3] = $b2; + $spec[4] = $spec[1] + 1; + } + } + + // Alignment pattern --------------------------------------------------- + + // Positions of alignment patterns. + // This array includes only the second and the third position of the + // alignment patterns. Rest of them can be calculated from the distance + // between them. + + // See Table 1 in Appendix E (pp.71) of JIS X0510:2004. + + public static $alignmentPattern = array( + array( 0, 0), + array( 0, 0), array(18, 0), array(22, 0), array(26, 0), array(30, 0), // 1- 5 + array(34, 0), array(22, 38), array(24, 42), array(26, 46), array(28, 50), // 6-10 + array(30, 54), array(32, 58), array(34, 62), array(26, 46), array(26, 48), //11-15 + array(26, 50), array(30, 54), array(30, 56), array(30, 58), array(34, 62), //16-20 + array(28, 50), array(26, 50), array(30, 54), array(28, 54), array(32, 58), //21-25 + array(30, 58), array(34, 62), array(26, 50), array(30, 54), array(26, 52), //26-30 + array(30, 56), array(34, 60), array(30, 58), array(34, 62), array(30, 54), //31-35 + array(24, 50), array(28, 54), array(32, 58), array(26, 54), array(30, 58), //35-40 + ); + + + /** -------------------------------------------------------------------- + * Put an alignment marker. + * @param frame + * @param width + * @param ox,oy center coordinate of the pattern + */ + public static function putAlignmentMarker(array &$frame, $ox, $oy) + { + $finder = array( + "\xa1\xa1\xa1\xa1\xa1", + "\xa1\xa0\xa0\xa0\xa1", + "\xa1\xa0\xa1\xa0\xa1", + "\xa1\xa0\xa0\xa0\xa1", + "\xa1\xa1\xa1\xa1\xa1" + ); + + $yStart = $oy-2; + $xStart = $ox-2; + + for($y=0; $y<5; $y++) { + QRstr::set($frame, $xStart, $yStart+$y, $finder[$y]); + } + } + + //---------------------------------------------------------------------- + public static function putAlignmentPattern($version, &$frame, $width) + { + if($version < 2) + return; + + $d = self::$alignmentPattern[$version][1] - self::$alignmentPattern[$version][0]; + if($d < 0) { + $w = 2; + } else { + $w = (int)(($width - self::$alignmentPattern[$version][0]) / $d + 2); + } + + if($w * $w - 3 == 1) { + $x = self::$alignmentPattern[$version][0]; + $y = self::$alignmentPattern[$version][0]; + self::putAlignmentMarker($frame, $x, $y); + return; + } + + $cx = self::$alignmentPattern[$version][0]; + for($x=1; $x<$w - 1; $x++) { + self::putAlignmentMarker($frame, 6, $cx); + self::putAlignmentMarker($frame, $cx, 6); + $cx += $d; + } + + $cy = self::$alignmentPattern[$version][0]; + for($y=0; $y<$w-1; $y++) { + $cx = self::$alignmentPattern[$version][0]; + for($x=0; $x<$w-1; $x++) { + self::putAlignmentMarker($frame, $cx, $cy); + $cx += $d; + } + $cy += $d; + } + } + + // Version information pattern ----------------------------------------- + + // Version information pattern (BCH coded). + // See Table 1 in Appendix D (pp.68) of JIS X0510:2004. + + // size: [Constants::QRSPEC_VERSION_MAX - 6] + + public static $versionPattern = array( + 0x07c94, 0x085bc, 0x09a99, 0x0a4d3, 0x0bbf6, 0x0c762, 0x0d847, 0x0e60d, + 0x0f928, 0x10b78, 0x1145d, 0x12a17, 0x13532, 0x149a6, 0x15683, 0x168c9, + 0x177ec, 0x18ec4, 0x191e1, 0x1afab, 0x1b08e, 0x1cc1a, 0x1d33f, 0x1ed75, + 0x1f250, 0x209d5, 0x216f0, 0x228ba, 0x2379f, 0x24b0b, 0x2542e, 0x26a64, + 0x27541, 0x28c69 + ); + + //---------------------------------------------------------------------- + public static function getVersionPattern($version) + { + if($version < 7 || $version > Constants::QRSPEC_VERSION_MAX) + return 0; + + return self::$versionPattern[$version -7]; + } + + // Format information -------------------------------------------------- + // See calcFormatInfo in tests/test_qrspec.c (orginal qrencode c lib) + + public static $formatInfo = array( + array(0x77c4, 0x72f3, 0x7daa, 0x789d, 0x662f, 0x6318, 0x6c41, 0x6976), + array(0x5412, 0x5125, 0x5e7c, 0x5b4b, 0x45f9, 0x40ce, 0x4f97, 0x4aa0), + array(0x355f, 0x3068, 0x3f31, 0x3a06, 0x24b4, 0x2183, 0x2eda, 0x2bed), + array(0x1689, 0x13be, 0x1ce7, 0x19d0, 0x0762, 0x0255, 0x0d0c, 0x083b) + ); + + public static function getFormatInfo($mask, $level) + { + if($mask < 0 || $mask > 7) + return 0; + + if($level < 0 || $level > 3) + return 0; + + return self::$formatInfo[$level][$mask]; + } + + // Frame --------------------------------------------------------------- + // Cache of initial frames. + + public static $frames = array(); + + /** -------------------------------------------------------------------- + * Put a finder pattern. + * @param frame + * @param width + * @param ox,oy upper-left coordinate of the pattern + */ + public static function putFinderPattern(&$frame, $ox, $oy) + { + $finder = array( + "\xc1\xc1\xc1\xc1\xc1\xc1\xc1", + "\xc1\xc0\xc0\xc0\xc0\xc0\xc1", + "\xc1\xc0\xc1\xc1\xc1\xc0\xc1", + "\xc1\xc0\xc1\xc1\xc1\xc0\xc1", + "\xc1\xc0\xc1\xc1\xc1\xc0\xc1", + "\xc1\xc0\xc0\xc0\xc0\xc0\xc1", + "\xc1\xc1\xc1\xc1\xc1\xc1\xc1" + ); + + for($y=0; $y<7; $y++) { + QRstr::set($frame, $ox, $oy+$y, $finder[$y]); + } + } + + //---------------------------------------------------------------------- + public static function createFrame($version) + { + $width = self::$capacity[$version][Constants::QRCAP_WIDTH]; + $frameLine = str_repeat ("\0", $width); + $frame = array_fill(0, $width, $frameLine); + + // Finder pattern + self::putFinderPattern($frame, 0, 0); + self::putFinderPattern($frame, $width - 7, 0); + self::putFinderPattern($frame, 0, $width - 7); + + // Separator + $yOffset = $width - 7; + + for($y=0; $y<7; $y++) { + $frame[$y][7] = "\xc0"; + $frame[$y][$width - 8] = "\xc0"; + $frame[$yOffset][7] = "\xc0"; + $yOffset++; + } + + $setPattern = str_repeat("\xc0", 8); + + QRstr::set($frame, 0, 7, $setPattern); + QRstr::set($frame, $width-8, 7, $setPattern); + QRstr::set($frame, 0, $width - 8, $setPattern); + + // Format info + $setPattern = str_repeat("\x84", 9); + QRstr::set($frame, 0, 8, $setPattern); + QRstr::set($frame, $width - 8, 8, $setPattern, 8); + + $yOffset = $width - 8; + + for($y=0; $y<8; $y++,$yOffset++) { + $frame[$y][8] = "\x84"; + $frame[$yOffset][8] = "\x84"; + } + + // Timing pattern + + for($i=1; $i<$width-15; $i++) { + $frame[6][7+$i] = chr(0x90 | ($i & 1)); + $frame[7+$i][6] = chr(0x90 | ($i & 1)); + } + + // Alignment pattern + self::putAlignmentPattern($version, $frame, $width); + + // Version information + if($version >= 7) { + $vinf = self::getVersionPattern($version); + + $v = $vinf; + + for($x=0; $x<6; $x++) { + for($y=0; $y<3; $y++) { + $frame[($width - 11)+$y][$x] = chr(0x88 | ($v & 1)); + $v = $v >> 1; + } + } + + $v = $vinf; + for($y=0; $y<6; $y++) { + for($x=0; $x<3; $x++) { + $frame[$y][$x+($width - 11)] = chr(0x88 | ($v & 1)); + $v = $v >> 1; + } + } + } + + // and a little bit... + $frame[$width - 8][8] = "\x81"; + + return $frame; + } + + //---------------------------------------------------------------------- + public static function debug($frame, $binary_mode = false) + { + if ($binary_mode) { + + foreach ($frame as &$frameLine) { + $frameLine = join('  ', explode('0', $frameLine)); + $frameLine = join('██', explode('1', $frameLine)); + } + + ?> + +


            '; + echo join("
            ", $frame); + echo '






    '; + + } else { + + foreach ($frame as &$frameLine) { + $frameLine = join(' ', explode("\xc0", $frameLine)); + $frameLine = join('', explode("\xc1", $frameLine)); + $frameLine = join(' ', explode("\xa0", $frameLine)); + $frameLine = join('', explode("\xa1", $frameLine)); + $frameLine = join('', explode("\x84", $frameLine)); //format 0 + $frameLine = join('', explode("\x85", $frameLine)); //format 1 + $frameLine = join('', explode("\x81", $frameLine)); //special bit + $frameLine = join(' ', explode("\x90", $frameLine)); //clock 0 + $frameLine = join('', explode("\x91", $frameLine)); //clock 1 + $frameLine = join(' ', explode("\x88", $frameLine)); //version + $frameLine = join('', explode("\x89", $frameLine)); //version + $frameLine = join('♦', explode("\x01", $frameLine)); + $frameLine = join('⋅', explode("\0", $frameLine)); + } + + ?> + + "; + echo join("
    ", $frame); + echo "
    "; + + } + } + + //---------------------------------------------------------------------- + public static function serial($frame) + { + return gzcompress(join("\n", $frame), 9); + } + + //---------------------------------------------------------------------- + public static function unserial($code) + { + return explode("\n", gzuncompress($code)); + } + + //---------------------------------------------------------------------- + public static function newFrame($version) + { + if($version < 1 || $version > Constants::QRSPEC_VERSION_MAX) + return null; + + if(!isset(self::$frames[$version])) { + + $fileName = Constants::QR_CACHE_DIR.'frame_'.$version.'.dat'; + + if (Constants::QR_CACHEABLE) { + if (file_exists($fileName)) { + self::$frames[$version] = self::unserial(file_get_contents($fileName)); + } else { + self::$frames[$version] = self::createFrame($version); + file_put_contents($fileName, self::serial(self::$frames[$version])); + } + } else { + self::$frames[$version] = self::createFrame($version); + } + } + + if(is_null(self::$frames[$version])) + return null; + + return self::$frames[$version]; + } + + //---------------------------------------------------------------------- + public static function rsBlockNum($spec) { return $spec[0] + $spec[3]; } + public static function rsBlockNum1($spec) { return $spec[0]; } + public static function rsDataCodes1($spec) { return $spec[1]; } + public static function rsEccCodes1($spec) { return $spec[2]; } + public static function rsBlockNum2($spec) { return $spec[3]; } + public static function rsDataCodes2($spec) { return $spec[4]; } + public static function rsEccCodes2($spec) { return $spec[2]; } + public static function rsDataLength($spec) { return ($spec[0] * $spec[1]) + ($spec[3] * $spec[4]); } + public static function rsEccLength($spec) { return ($spec[0] + $spec[3]) * $spec[2]; } + +} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRsplit.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRsplit.php new file mode 100644 index 000000000..805140a97 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRsplit.php @@ -0,0 +1,316 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * The following data / specifications are taken from + * "Two dimensional symbol -- QR-code -- Basic Specification" (JIS X0510:2004) + * or + * "Automatic identification and data capture techniques -- + * QR Code 2005 bar code symbology specification" (ISO/IEC 18004:2006) + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +use Exception; + +class QRsplit { + + public $dataStr = ''; + public $input; + public $modeHint; + + //---------------------------------------------------------------------- + public function __construct($dataStr, $input, $modeHint) + { + $this->dataStr = $dataStr; + $this->input = $input; + $this->modeHint = $modeHint; + } + + //---------------------------------------------------------------------- + public static function isdigitat($str, $pos) + { + if ($pos >= strlen($str)) + return false; + + return ((ord($str[$pos]) >= ord('0'))&&(ord($str[$pos]) <= ord('9'))); + } + + //---------------------------------------------------------------------- + public static function isalnumat($str, $pos) + { + if ($pos >= strlen($str)) + return false; + + return (QRinput::lookAnTable(ord($str[$pos])) >= 0); + } + + //---------------------------------------------------------------------- + public function identifyMode($pos) + { + if ($pos >= strlen($this->dataStr)) + return Constants::QR_MODE_NUL; + + $c = $this->dataStr[$pos]; + + if(self::isdigitat($this->dataStr, $pos)) { + return Constants::QR_MODE_NUM; + } else if(self::isalnumat($this->dataStr, $pos)) { + return Constants::QR_MODE_AN; + } else if($this->modeHint == Constants::QR_MODE_KANJI) { + + if ($pos+1 < strlen($this->dataStr)) + { + $d = $this->dataStr[$pos+1]; + $word = (ord($c) << 8) | ord($d); + if(($word >= 0x8140 && $word <= 0x9ffc) || ($word >= 0xe040 && $word <= 0xebbf)) { + return Constants::QR_MODE_KANJI; + } + } + } + + return Constants::QR_MODE_8; + } + + //---------------------------------------------------------------------- + public function eatNum() + { + $ln = QRspec::lengthIndicator(Constants::QR_MODE_NUM, $this->input->getVersion()); + + $p = 0; + while(self::isdigitat($this->dataStr, $p)) { + $p++; + } + + $run = $p; + $mode = $this->identifyMode($p); + + if($mode == Constants::QR_MODE_8) { + $dif = QRinput::estimateBitsModeNum($run) + 4 + $ln + + QRinput::estimateBitsMode8(1) // + 4 + l8 + - QRinput::estimateBitsMode8($run + 1); // - 4 - l8 + if($dif > 0) { + return $this->eat8(); + } + } + if($mode == Constants::QR_MODE_AN) { + $dif = QRinput::estimateBitsModeNum($run) + 4 + $ln + + QRinput::estimateBitsModeAn(1) // + 4 + la + - QRinput::estimateBitsModeAn($run + 1);// - 4 - la + if($dif > 0) { + return $this->eatAn(); + } + } + + $ret = $this->input->append(Constants::QR_MODE_NUM, $run, str_split($this->dataStr)); + if($ret < 0) + return -1; + + return $run; + } + + //---------------------------------------------------------------------- + public function eatAn() + { + $la = QRspec::lengthIndicator(Constants::QR_MODE_AN, $this->input->getVersion()); + $ln = QRspec::lengthIndicator(Constants::QR_MODE_NUM, $this->input->getVersion()); + + $p = 0; + + while(self::isalnumat($this->dataStr, $p)) { + if(self::isdigitat($this->dataStr, $p)) { + $q = $p; + while(self::isdigitat($this->dataStr, $q)) { + $q++; + } + + $dif = QRinput::estimateBitsModeAn($p) // + 4 + la + + QRinput::estimateBitsModeNum($q - $p) + 4 + $ln + - QRinput::estimateBitsModeAn($q); // - 4 - la + + if($dif < 0) { + break; + } else { + $p = $q; + } + } else { + $p++; + } + } + + $run = $p; + + if(!self::isalnumat($this->dataStr, $p)) { + $dif = QRinput::estimateBitsModeAn($run) + 4 + $la + + QRinput::estimateBitsMode8(1) // + 4 + l8 + - QRinput::estimateBitsMode8($run + 1); // - 4 - l8 + if($dif > 0) { + return $this->eat8(); + } + } + + $ret = $this->input->append(Constants::QR_MODE_AN, $run, str_split($this->dataStr)); + if($ret < 0) + return -1; + + return $run; + } + + //---------------------------------------------------------------------- + public function eatKanji() + { + $p = 0; + + while($this->identifyMode($p) == Constants::QR_MODE_KANJI) { + $p += 2; + } + + $ret = $this->input->append(Constants::QR_MODE_KANJI, $p, str_split($this->dataStr)); + if($ret < 0) + return -1; + + return $ret; + } + + //---------------------------------------------------------------------- + public function eat8() + { + $la = QRspec::lengthIndicator(Constants::QR_MODE_AN, $this->input->getVersion()); + $ln = QRspec::lengthIndicator(Constants::QR_MODE_NUM, $this->input->getVersion()); + + $p = 1; + $dataStrLen = strlen($this->dataStr); + + while($p < $dataStrLen) { + + $mode = $this->identifyMode($p); + if($mode == Constants::QR_MODE_KANJI) { + break; + } + if($mode == Constants::QR_MODE_NUM) { + $q = $p; + while(self::isdigitat($this->dataStr, $q)) { + $q++; + } + $dif = QRinput::estimateBitsMode8($p) // + 4 + l8 + + QRinput::estimateBitsModeNum($q - $p) + 4 + $ln + - QRinput::estimateBitsMode8($q); // - 4 - l8 + if($dif < 0) { + break; + } else { + $p = $q; + } + } else if($mode == Constants::QR_MODE_AN) { + $q = $p; + while(self::isalnumat($this->dataStr, $q)) { + $q++; + } + $dif = QRinput::estimateBitsMode8($p) // + 4 + l8 + + QRinput::estimateBitsModeAn($q - $p) + 4 + $la + - QRinput::estimateBitsMode8($q); // - 4 - l8 + if($dif < 0) { + break; + } else { + $p = $q; + } + } else { + $p++; + } + } + + $run = $p; + $ret = $this->input->append(Constants::QR_MODE_8, $run, str_split($this->dataStr)); + + if($ret < 0) + return -1; + + return $run; + } + + //---------------------------------------------------------------------- + public function splitString() + { + while (strlen($this->dataStr) > 0) + { + if($this->dataStr == '') + return 0; + + $mode = $this->identifyMode(0); + + switch ($mode) { + case Constants::QR_MODE_NUM: $length = $this->eatNum(); break; + case Constants::QR_MODE_AN: $length = $this->eatAn(); break; + case Constants::QR_MODE_KANJI: + if ($hint == Constants::QR_MODE_KANJI) + $length = $this->eatKanji(); + else $length = $this->eat8(); + break; + default: $length = $this->eat8(); break; + + } + + if($length == 0) return 0; + if($length < 0) return -1; + + $this->dataStr = substr($this->dataStr, $length); + } + } + + //---------------------------------------------------------------------- + public function toUpper() + { + $stringLen = strlen($this->dataStr); + $p = 0; + + while ($p<$stringLen) { + $mode = self::identifyMode(substr($this->dataStr, $p), $this->modeHint); + if($mode == Constants::QR_MODE_KANJI) { + $p += 2; + } else { + if (ord($this->dataStr[$p]) >= ord('a') && ord($this->dataStr[$p]) <= ord('z')) { + $this->dataStr[$p] = chr(ord($this->dataStr[$p]) - 32); + } + $p++; + } + } + + return $this->dataStr; + } + + //---------------------------------------------------------------------- + public static function splitStringToQRinput($string, QRinput $input, $modeHint, $casesensitive = true) + { + if(is_null($string) || $string == '\0' || $string == '') { + throw new Exception('empty string!!!'); + } + + $split = new QRsplit($string, $input, $modeHint); + + if(!$casesensitive) + $split->toUpper(); + + return $split->splitString(); + } +} diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRstr.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRstr.php new file mode 100644 index 000000000..64c4bd5c6 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRstr.php @@ -0,0 +1,35 @@ + + * + * PHP QR Code is distributed under LGPL 3 + * Copyright (C) 2010 Dominik Dzienia + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRstr { + public static function set(&$srctab, $x, $y, $repl, $replLen = false) { + $srctab[$y] = substr_replace($srctab[$y], ($replLen !== false)?substr($repl,0,$replLen):$repl, $x, ($replLen !== false)?$replLen:strlen($repl)); + } +} \ No newline at end of file diff --git a/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRtools.php b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRtools.php new file mode 100644 index 000000000..7c75a6e2a --- /dev/null +++ b/vendor/aferrandini/phpqrcode/lib/PHPQRCode/QRtools.php @@ -0,0 +1,171 @@ + + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or any later version. + * + * This library is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +namespace PHPQRCode; + +class QRtools { + + //---------------------------------------------------------------------- + public static function binarize($frame) + { + $len = count($frame); + foreach ($frame as &$frameLine) { + + for($i=0; $i<$len; $i++) { + $frameLine[$i] = (ord($frameLine[$i])&1)?'1':'0'; + } + } + + return $frame; + } + + //---------------------------------------------------------------------- + public static function tcpdfBarcodeArray($code, $mode = 'QR,L', $tcPdfVersion = '4.5.037') + { + $barcode_array = array(); + + if (!is_array($mode)) + $mode = explode(',', $mode); + + $eccLevel = 'L'; + + if (count($mode) > 1) { + $eccLevel = $mode[1]; + } + + $qrTab = QRcode::text($code, false, $eccLevel); + $size = count($qrTab); + + $barcode_array['num_rows'] = $size; + $barcode_array['num_cols'] = $size; + $barcode_array['bcode'] = array(); + + foreach ($qrTab as $line) { + $arrAdd = array(); + foreach(str_split($line) as $char) + $arrAdd[] = ($char=='1')?1:0; + $barcode_array['bcode'][] = $arrAdd; + } + + return $barcode_array; + } + + //---------------------------------------------------------------------- + public static function clearCache() + { + self::$frames = array(); + } + + //---------------------------------------------------------------------- + public static function buildCache() + { + QRtools::markTime('before_build_cache'); + + $mask = new QRmask(); + for ($a=1; $a <= Constants::QRSPEC_VERSION_MAX; $a++) { + $frame = QRspec::newFrame($a); + if (Constants::QR_IMAGE) { + $fileName = Constants::QR_CACHE_DIR.'frame_'.$a.'.png'; + QRimage::png(self::binarize($frame), $fileName, 1, 0); + } + + $width = count($frame); + $bitMask = array_fill(0, $width, array_fill(0, $width, 0)); + for ($maskNo=0; $maskNo<8; $maskNo++) + $mask->makeMaskNo($maskNo, $width, $frame, $bitMask, true); + } + + QRtools::markTime('after_build_cache'); + } + + //---------------------------------------------------------------------- + public static function log($outfile, $err) + { + if (Constants::QR_LOG_DIR !== false) { + if ($err != '') { + if ($outfile !== false) { + file_put_contents(Constants::QR_LOG_DIR.basename($outfile).'-errors.txt', date('Y-m-d H:i:s').': '.$err, FILE_APPEND); + } else { + file_put_contents(Constants::QR_LOG_DIR.'errors.txt', date('Y-m-d H:i:s').': '.$err, FILE_APPEND); + } + } + } + } + + //---------------------------------------------------------------------- + public static function dumpMask($frame) + { + $width = count($frame); + for($y=0;$y<$width;$y++) { + for($x=0;$x<$width;$x++) { + echo ord($frame[$y][$x]).','; + } + } + } + + //---------------------------------------------------------------------- + public static function markTime($markerId) + { + list($usec, $sec) = explode(" ", microtime()); + $time = ((float)$usec + (float)$sec); + + if (!isset($GLOBALS['qr_time_bench'])) + $GLOBALS['qr_time_bench'] = array(); + + $GLOBALS['qr_time_bench'][$markerId] = $time; + } + + //---------------------------------------------------------------------- + public static function timeBenchmark() + { + self::markTime('finish'); + + $lastTime = 0; + $startTime = 0; + $p = 0; + + echo ' + + '; + + foreach($GLOBALS['qr_time_bench'] as $markerId=>$thisTime) { + if ($p > 0) { + echo ''; + } else { + $startTime = $thisTime; + } + + $p++; + $lastTime = $thisTime; + } + + echo ' + + +
    BENCHMARK
    till '.$markerId.': '.number_format($thisTime-$lastTime, 6).'s
    TOTAL: '.number_format($lastTime-$startTime, 6).'s
    '; + } + +} + +QRtools::markTime('start'); diff --git a/vendor/aferrandini/phpqrcode/readme.md b/vendor/aferrandini/phpqrcode/readme.md new file mode 100644 index 000000000..e8f2f5ab3 --- /dev/null +++ b/vendor/aferrandini/phpqrcode/readme.md @@ -0,0 +1,37 @@ +# PHP QRCode Library + +To install this library please follow the next steps: + +## Install the library using `composer`: + +Add the required module to your `composer.json` file: + + { + "require": { + ... + "aferrandini/phpqrcode": "1.0.1" + ... + } + } + +Then run the command `composer update`. + + +## Usage + +Sample code: + + \PHPQRCode\QRcode::png("Test", "/tmp/qrcode.png", 'L', 4, 2); + +This code will generate a PNG file on '/tmp/qrcode.png' with a QRCode that contains the word 'Test'. + +## Acknowledgements + +This library is an import of PHP QR Code by Dominik Dzienia that you can find at http://phpqrcode.sourceforge.net + +Based on C libqrencode library (ver. 3.1.1), Copyright (C) 2006-2010 by Kentaro Fukuchi +http://megaui.net/fukuchi/works/qrencode/index.en.html + +QR Code is registered trademarks of DENSO WAVE INCORPORATED in JAPAN and other countries. + +Reed-Solomon code encoder is written by Phil Karn, KA9Q. Copyright (C) 2002, 2003, 2004, 2006 Phil Karn, KA9Q diff --git a/vendor/autoload.php b/vendor/autoload.php new file mode 100644 index 000000000..0c1c7c4bf --- /dev/null +++ b/vendor/autoload.php @@ -0,0 +1,7 @@ + + * Jordi Boggiano + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace Composer\Autoload; + +/** + * ClassLoader implements a PSR-0, PSR-4 and classmap class loader. + * + * $loader = new \Composer\Autoload\ClassLoader(); + * + * // register classes with namespaces + * $loader->add('Symfony\Component', __DIR__.'/component'); + * $loader->add('Symfony', __DIR__.'/framework'); + * + * // activate the autoloader + * $loader->register(); + * + * // to enable searching the include path (eg. for PEAR packages) + * $loader->setUseIncludePath(true); + * + * In this example, if you try to use a class in the Symfony\Component + * namespace or one of its children (Symfony\Component\Console for instance), + * the autoloader will first look for the class under the component/ + * directory, and it will then fallback to the framework/ directory if not + * found before giving up. + * + * This class is loosely based on the Symfony UniversalClassLoader. + * + * @author Fabien Potencier + * @author Jordi Boggiano + * @see http://www.php-fig.org/psr/psr-0/ + * @see http://www.php-fig.org/psr/psr-4/ + */ +class ClassLoader +{ + // PSR-4 + private $prefixLengthsPsr4 = array(); + private $prefixDirsPsr4 = array(); + private $fallbackDirsPsr4 = array(); + + // PSR-0 + private $prefixesPsr0 = array(); + private $fallbackDirsPsr0 = array(); + + private $useIncludePath = false; + private $classMap = array(); + private $classMapAuthoritative = false; + private $missingClasses = array(); + + public function getPrefixes() + { + if (!empty($this->prefixesPsr0)) { + return call_user_func_array('array_merge', $this->prefixesPsr0); + } + + return array(); + } + + public function getPrefixesPsr4() + { + return $this->prefixDirsPsr4; + } + + public function getFallbackDirs() + { + return $this->fallbackDirsPsr0; + } + + public function getFallbackDirsPsr4() + { + return $this->fallbackDirsPsr4; + } + + public function getClassMap() + { + return $this->classMap; + } + + /** + * @param array $classMap Class to filename map + */ + public function addClassMap(array $classMap) + { + if ($this->classMap) { + $this->classMap = array_merge($this->classMap, $classMap); + } else { + $this->classMap = $classMap; + } + } + + /** + * Registers a set of PSR-0 directories for a given prefix, either + * appending or prepending to the ones previously set for this prefix. + * + * @param string $prefix The prefix + * @param array|string $paths The PSR-0 root directories + * @param bool $prepend Whether to prepend the directories + */ + public function add($prefix, $paths, $prepend = false) + { + if (!$prefix) { + if ($prepend) { + $this->fallbackDirsPsr0 = array_merge( + (array) $paths, + $this->fallbackDirsPsr0 + ); + } else { + $this->fallbackDirsPsr0 = array_merge( + $this->fallbackDirsPsr0, + (array) $paths + ); + } + + return; + } + + $first = $prefix[0]; + if (!isset($this->prefixesPsr0[$first][$prefix])) { + $this->prefixesPsr0[$first][$prefix] = (array) $paths; + + return; + } + if ($prepend) { + $this->prefixesPsr0[$first][$prefix] = array_merge( + (array) $paths, + $this->prefixesPsr0[$first][$prefix] + ); + } else { + $this->prefixesPsr0[$first][$prefix] = array_merge( + $this->prefixesPsr0[$first][$prefix], + (array) $paths + ); + } + } + + /** + * Registers a set of PSR-4 directories for a given namespace, either + * appending or prepending to the ones previously set for this namespace. + * + * @param string $prefix The prefix/namespace, with trailing '\\' + * @param array|string $paths The PSR-4 base directories + * @param bool $prepend Whether to prepend the directories + * + * @throws \InvalidArgumentException + */ + public function addPsr4($prefix, $paths, $prepend = false) + { + if (!$prefix) { + // Register directories for the root namespace. + if ($prepend) { + $this->fallbackDirsPsr4 = array_merge( + (array) $paths, + $this->fallbackDirsPsr4 + ); + } else { + $this->fallbackDirsPsr4 = array_merge( + $this->fallbackDirsPsr4, + (array) $paths + ); + } + } elseif (!isset($this->prefixDirsPsr4[$prefix])) { + // Register directories for a new namespace. + $length = strlen($prefix); + if ('\\' !== $prefix[$length - 1]) { + throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator."); + } + $this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length; + $this->prefixDirsPsr4[$prefix] = (array) $paths; + } elseif ($prepend) { + // Prepend directories for an already registered namespace. + $this->prefixDirsPsr4[$prefix] = array_merge( + (array) $paths, + $this->prefixDirsPsr4[$prefix] + ); + } else { + // Append directories for an already registered namespace. + $this->prefixDirsPsr4[$prefix] = array_merge( + $this->prefixDirsPsr4[$prefix], + (array) $paths + ); + } + } + + /** + * Registers a set of PSR-0 directories for a given prefix, + * replacing any others previously set for this prefix. + * + * @param string $prefix The prefix + * @param array|string $paths The PSR-0 base directories + */ + public function set($prefix, $paths) + { + if (!$prefix) { + $this->fallbackDirsPsr0 = (array) $paths; + } else { + $this->prefixesPsr0[$prefix[0]][$prefix] = (array) $paths; + } + } + + /** + * Registers a set of PSR-4 directories for a given namespace, + * replacing any others previously set for this namespace. + * + * @param string $prefix The prefix/namespace, with trailing '\\' + * @param array|string $paths The PSR-4 base directories + * + * @throws \InvalidArgumentException + */ + public function setPsr4($prefix, $paths) + { + if (!$prefix) { + $this->fallbackDirsPsr4 = (array) $paths; + } else { + $length = strlen($prefix); + if ('\\' !== $prefix[$length - 1]) { + throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator."); + } + $this->prefixLengthsPsr4[$prefix[0]][$prefix] = $length; + $this->prefixDirsPsr4[$prefix] = (array) $paths; + } + } + + /** + * Turns on searching the include path for class files. + * + * @param bool $useIncludePath + */ + public function setUseIncludePath($useIncludePath) + { + $this->useIncludePath = $useIncludePath; + } + + /** + * Can be used to check if the autoloader uses the include path to check + * for classes. + * + * @return bool + */ + public function getUseIncludePath() + { + return $this->useIncludePath; + } + + /** + * Turns off searching the prefix and fallback directories for classes + * that have not been registered with the class map. + * + * @param bool $classMapAuthoritative + */ + public function setClassMapAuthoritative($classMapAuthoritative) + { + $this->classMapAuthoritative = $classMapAuthoritative; + } + + /** + * Should class lookup fail if not found in the current class map? + * + * @return bool + */ + public function isClassMapAuthoritative() + { + return $this->classMapAuthoritative; + } + + /** + * Registers this instance as an autoloader. + * + * @param bool $prepend Whether to prepend the autoloader or not + */ + public function register($prepend = false) + { + spl_autoload_register(array($this, 'loadClass'), true, $prepend); + } + + /** + * Unregisters this instance as an autoloader. + */ + public function unregister() + { + spl_autoload_unregister(array($this, 'loadClass')); + } + + /** + * Loads the given class or interface. + * + * @param string $class The name of the class + * @return bool|null True if loaded, null otherwise + */ + public function loadClass($class) + { + if ($file = $this->findFile($class)) { + includeFile($file); + + return true; + } + } + + /** + * Finds the path to the file where the class is defined. + * + * @param string $class The name of the class + * + * @return string|false The path if found, false otherwise + */ + public function findFile($class) + { + // work around for PHP 5.3.0 - 5.3.2 https://bugs.php.net/50731 + if ('\\' == $class[0]) { + $class = substr($class, 1); + } + + // class map lookup + if (isset($this->classMap[$class])) { + return $this->classMap[$class]; + } + if ($this->classMapAuthoritative || isset($this->missingClasses[$class])) { + return false; + } + + $file = $this->findFileWithExtension($class, '.php'); + + // Search for Hack files if we are running on HHVM + if (false === $file && defined('HHVM_VERSION')) { + $file = $this->findFileWithExtension($class, '.hh'); + } + + if (false === $file) { + // Remember that this class does not exist. + $this->missingClasses[$class] = true; + } + + return $file; + } + + private function findFileWithExtension($class, $ext) + { + // PSR-4 lookup + $logicalPathPsr4 = strtr($class, '\\', DIRECTORY_SEPARATOR) . $ext; + + $first = $class[0]; + if (isset($this->prefixLengthsPsr4[$first])) { + foreach ($this->prefixLengthsPsr4[$first] as $prefix => $length) { + if (0 === strpos($class, $prefix)) { + foreach ($this->prefixDirsPsr4[$prefix] as $dir) { + if (file_exists($file = $dir . DIRECTORY_SEPARATOR . substr($logicalPathPsr4, $length))) { + return $file; + } + } + } + } + } + + // PSR-4 fallback dirs + foreach ($this->fallbackDirsPsr4 as $dir) { + if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr4)) { + return $file; + } + } + + // PSR-0 lookup + if (false !== $pos = strrpos($class, '\\')) { + // namespaced class name + $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1) + . strtr(substr($logicalPathPsr4, $pos + 1), '_', DIRECTORY_SEPARATOR); + } else { + // PEAR-like class name + $logicalPathPsr0 = strtr($class, '_', DIRECTORY_SEPARATOR) . $ext; + } + + if (isset($this->prefixesPsr0[$first])) { + foreach ($this->prefixesPsr0[$first] as $prefix => $dirs) { + if (0 === strpos($class, $prefix)) { + foreach ($dirs as $dir) { + if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) { + return $file; + } + } + } + } + } + + // PSR-0 fallback dirs + foreach ($this->fallbackDirsPsr0 as $dir) { + if (file_exists($file = $dir . DIRECTORY_SEPARATOR . $logicalPathPsr0)) { + return $file; + } + } + + // PSR-0 include paths. + if ($this->useIncludePath && $file = stream_resolve_include_path($logicalPathPsr0)) { + return $file; + } + + return false; + } +} + +/** + * Scope isolated include. + * + * Prevents access to $this/self from included files. + */ +function includeFile($file) +{ + include $file; +} diff --git a/vendor/composer/LICENSE b/vendor/composer/LICENSE new file mode 100644 index 000000000..1a2812488 --- /dev/null +++ b/vendor/composer/LICENSE @@ -0,0 +1,21 @@ + +Copyright (c) 2016 Nils Adermann, Jordi Boggiano + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is furnished +to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. + diff --git a/vendor/composer/autoload_classmap.php b/vendor/composer/autoload_classmap.php new file mode 100644 index 000000000..4caa0fb2e --- /dev/null +++ b/vendor/composer/autoload_classmap.php @@ -0,0 +1,256 @@ + $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode.php', + 'PHPQRCode\\Autoloader' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/Autoloader.php', + 'PHPQRCode\\Constants' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/Constants.php', + 'PHPQRCode\\FrameFiller' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/FrameFiller.php', + 'PHPQRCode\\QRbitstream' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRbitstream.php', + 'PHPQRCode\\QRcode' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRcode.php', + 'PHPQRCode\\QRencode' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRencode.php', + 'PHPQRCode\\QRimage' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRimage.php', + 'PHPQRCode\\QRinput' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRinput.php', + 'PHPQRCode\\QRinputItem' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRinputItem.php', + 'PHPQRCode\\QRmask' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRmask.php', + 'PHPQRCode\\QRrawcode' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrawcode.php', + 'PHPQRCode\\QRrs' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrs.php', + 'PHPQRCode\\QRrsItem' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrsItem.php', + 'PHPQRCode\\QRrsblock' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrsblock.php', + 'PHPQRCode\\QRspec' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRspec.php', + 'PHPQRCode\\QRsplit' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRsplit.php', + 'PHPQRCode\\QRstr' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRstr.php', + 'PHPQRCode\\QRtools' => $vendorDir . '/aferrandini/phpqrcode/lib/PHPQRCode/QRtools.php', + 'PclZip' => $vendorDir . '/pclzip/pclzip/pclzip.lib.php', + 'Qiniu\\Auth' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Auth.php', + 'Qiniu\\Config' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Config.php', + 'Qiniu\\Etag' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Etag.php', + 'Qiniu\\Http\\Client' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Http/Client.php', + 'Qiniu\\Http\\Error' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Http/Error.php', + 'Qiniu\\Http\\Request' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Http/Request.php', + 'Qiniu\\Http\\Response' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Http/Response.php', + 'Qiniu\\Processing\\ImageUrlBuilder' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Processing/ImageUrlBuilder.php', + 'Qiniu\\Processing\\Operation' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Processing/Operation.php', + 'Qiniu\\Processing\\PersistentFop' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Processing/PersistentFop.php', + 'Qiniu\\Storage\\BucketManager' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Storage/BucketManager.php', + 'Qiniu\\Storage\\FormUploader' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Storage/FormUploader.php', + 'Qiniu\\Storage\\ResumeUploader' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Storage/ResumeUploader.php', + 'Qiniu\\Storage\\UploadManager' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Storage/UploadManager.php', + 'Qiniu\\Zone' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/Zone.php', + 'Wechat\\Lib\\Cache' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/Lib/Cache.php', + 'Wechat\\Lib\\Common' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/Lib/Common.php', + 'Wechat\\Lib\\Tools' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/Lib/Tools.php', + 'Wechat\\Loader' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/Loader.php', + 'Wechat\\WechatCard' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatCard.php', + 'Wechat\\WechatCustom' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatCustom.php', + 'Wechat\\WechatDevice' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatDevice.php', + 'Wechat\\WechatExtends' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatExtends.php', + 'Wechat\\WechatMedia' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatMedia.php', + 'Wechat\\WechatMenu' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatMenu.php', + 'Wechat\\WechatOauth' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatOauth.php', + 'Wechat\\WechatPay' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatPay.php', + 'Wechat\\WechatPoi' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatPoi.php', + 'Wechat\\WechatReceive' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatReceive.php', + 'Wechat\\WechatScript' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatScript.php', + 'Wechat\\WechatService' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatService.php', + 'Wechat\\WechatUser' => $vendorDir . '/zoujingli/wechat-php-sdk/Wechat/WechatUser.php', + 'Workerman\\Autoloader' => $vendorDir . '/workerman/workerman/Autoloader.php', + 'Workerman\\Connection\\AsyncTcpConnection' => $vendorDir . '/workerman/workerman/Connection/AsyncTcpConnection.php', + 'Workerman\\Connection\\ConnectionInterface' => $vendorDir . '/workerman/workerman/Connection/ConnectionInterface.php', + 'Workerman\\Connection\\TcpConnection' => $vendorDir . '/workerman/workerman/Connection/TcpConnection.php', + 'Workerman\\Connection\\UdpConnection' => $vendorDir . '/workerman/workerman/Connection/UdpConnection.php', + 'Workerman\\Events\\Ev' => $vendorDir . '/workerman/workerman/Events/Ev.php', + 'Workerman\\Events\\Event' => $vendorDir . '/workerman/workerman/Events/Event.php', + 'Workerman\\Events\\EventInterface' => $vendorDir . '/workerman/workerman/Events/EventInterface.php', + 'Workerman\\Events\\Libevent' => $vendorDir . '/workerman/workerman/Events/Libevent.php', + 'Workerman\\Events\\React' => $vendorDir . '/workerman/workerman/Events/React.php', + 'Workerman\\Events\\React\\ExtEventLoop' => $vendorDir . '/workerman/workerman/Events/React/ExtEventLoop.php', + 'Workerman\\Events\\React\\LibEventLoop' => $vendorDir . '/workerman/workerman/Events/React/LibEventLoop.php', + 'Workerman\\Events\\React\\StreamSelectLoop' => $vendorDir . '/workerman/workerman/Events/React/StreamSelectLoop.php', + 'Workerman\\Events\\Select' => $vendorDir . '/workerman/workerman/Events/Select.php', + 'Workerman\\Lib\\Timer' => $vendorDir . '/workerman/workerman/Lib/Timer.php', + 'Workerman\\Protocols\\Frame' => $vendorDir . '/workerman/workerman/Protocols/Frame.php', + 'Workerman\\Protocols\\Http' => $vendorDir . '/workerman/workerman/Protocols/Http.php', + 'Workerman\\Protocols\\HttpCache' => $vendorDir . '/workerman/workerman/Protocols/Http.php', + 'Workerman\\Protocols\\ProtocolInterface' => $vendorDir . '/workerman/workerman/Protocols/ProtocolInterface.php', + 'Workerman\\Protocols\\Text' => $vendorDir . '/workerman/workerman/Protocols/Text.php', + 'Workerman\\Protocols\\Websocket' => $vendorDir . '/workerman/workerman/Protocols/Websocket.php', + 'Workerman\\Protocols\\Ws' => $vendorDir . '/workerman/workerman/Protocols/Ws.php', + 'Workerman\\WebServer' => $vendorDir . '/workerman/workerman/WebServer.php', + 'Workerman\\Worker' => $vendorDir . '/workerman/workerman/Worker.php', + 'think\\App' => $baseDir . '/thinkphp/library/think/App.php', + 'think\\Build' => $baseDir . '/thinkphp/library/think/Build.php', + 'think\\Cache' => $baseDir . '/thinkphp/library/think/Cache.php', + 'think\\Collection' => $baseDir . '/thinkphp/library/think/Collection.php', + 'think\\Config' => $baseDir . '/thinkphp/library/think/Config.php', + 'think\\Console' => $baseDir . '/thinkphp/library/think/Console.php', + 'think\\Controller' => $baseDir . '/thinkphp/library/think/Controller.php', + 'think\\Cookie' => $baseDir . '/thinkphp/library/think/Cookie.php', + 'think\\Db' => $baseDir . '/thinkphp/library/think/Db.php', + 'think\\Debug' => $baseDir . '/thinkphp/library/think/Debug.php', + 'think\\Env' => $baseDir . '/thinkphp/library/think/Env.php', + 'think\\Error' => $baseDir . '/thinkphp/library/think/Error.php', + 'think\\Exception' => $baseDir . '/thinkphp/library/think/Exception.php', + 'think\\File' => $baseDir . '/thinkphp/library/think/File.php', + 'think\\Hook' => $baseDir . '/thinkphp/library/think/Hook.php', + 'think\\Lang' => $baseDir . '/thinkphp/library/think/Lang.php', + 'think\\Loader' => $baseDir . '/thinkphp/library/think/Loader.php', + 'think\\Log' => $baseDir . '/thinkphp/library/think/Log.php', + 'think\\Model' => $baseDir . '/thinkphp/library/think/Model.php', + 'think\\Paginator' => $baseDir . '/thinkphp/library/think/Paginator.php', + 'think\\Process' => $baseDir . '/thinkphp/library/think/Process.php', + 'think\\Queue' => $vendorDir . '/topthink/think-queue/src/Queue.php', + 'think\\Request' => $baseDir . '/thinkphp/library/think/Request.php', + 'think\\Response' => $baseDir . '/thinkphp/library/think/Response.php', + 'think\\Route' => $baseDir . '/thinkphp/library/think/Route.php', + 'think\\Session' => $baseDir . '/thinkphp/library/think/Session.php', + 'think\\Template' => $baseDir . '/thinkphp/library/think/Template.php', + 'think\\Url' => $baseDir . '/thinkphp/library/think/Url.php', + 'think\\Validate' => $baseDir . '/thinkphp/library/think/Validate.php', + 'think\\View' => $baseDir . '/thinkphp/library/think/View.php', + 'think\\cache\\Driver' => $baseDir . '/thinkphp/library/think/cache/Driver.php', + 'think\\cache\\driver\\File' => $baseDir . '/thinkphp/library/think/cache/driver/File.php', + 'think\\cache\\driver\\Lite' => $baseDir . '/thinkphp/library/think/cache/driver/Lite.php', + 'think\\cache\\driver\\Memcache' => $baseDir . '/thinkphp/library/think/cache/driver/Memcache.php', + 'think\\cache\\driver\\Memcached' => $baseDir . '/thinkphp/library/think/cache/driver/Memcached.php', + 'think\\cache\\driver\\Redis' => $baseDir . '/thinkphp/library/think/cache/driver/Redis.php', + 'think\\cache\\driver\\Sqlite' => $baseDir . '/thinkphp/library/think/cache/driver/Sqlite.php', + 'think\\cache\\driver\\Wincache' => $baseDir . '/thinkphp/library/think/cache/driver/Wincache.php', + 'think\\cache\\driver\\Xcache' => $baseDir . '/thinkphp/library/think/cache/driver/Xcache.php', + 'think\\captcha\\Captcha' => $vendorDir . '/topthink/think-captcha/src/Captcha.php', + 'think\\captcha\\CaptchaController' => $vendorDir . '/topthink/think-captcha/src/CaptchaController.php', + 'think\\composer\\Plugin' => $vendorDir . '/topthink/think-installer/src/Plugin.php', + 'think\\composer\\ThinkExtend' => $vendorDir . '/topthink/think-installer/src/ThinkExtend.php', + 'think\\composer\\ThinkFramework' => $vendorDir . '/topthink/think-installer/src/ThinkFramework.php', + 'think\\composer\\ThinkTesting' => $vendorDir . '/topthink/think-installer/src/ThinkTesting.php', + 'think\\config\\driver\\Ini' => $baseDir . '/thinkphp/library/think/config/driver/Ini.php', + 'think\\config\\driver\\Json' => $baseDir . '/thinkphp/library/think/config/driver/Json.php', + 'think\\config\\driver\\Xml' => $baseDir . '/thinkphp/library/think/config/driver/Xml.php', + 'think\\console\\Command' => $baseDir . '/thinkphp/library/think/console/Command.php', + 'think\\console\\Input' => $baseDir . '/thinkphp/library/think/console/Input.php', + 'think\\console\\Output' => $baseDir . '/thinkphp/library/think/console/Output.php', + 'think\\console\\command\\Build' => $baseDir . '/thinkphp/library/think/console/command/Build.php', + 'think\\console\\command\\Clear' => $baseDir . '/thinkphp/library/think/console/command/Clear.php', + 'think\\console\\command\\Help' => $baseDir . '/thinkphp/library/think/console/command/Help.php', + 'think\\console\\command\\Lists' => $baseDir . '/thinkphp/library/think/console/command/Lists.php', + 'think\\console\\command\\Make' => $baseDir . '/thinkphp/library/think/console/command/Make.php', + 'think\\console\\command\\make\\Controller' => $baseDir . '/thinkphp/library/think/console/command/make/Controller.php', + 'think\\console\\command\\make\\Model' => $baseDir . '/thinkphp/library/think/console/command/make/Model.php', + 'think\\console\\command\\optimize\\Autoload' => $baseDir . '/thinkphp/library/think/console/command/optimize/Autoload.php', + 'think\\console\\command\\optimize\\Config' => $baseDir . '/thinkphp/library/think/console/command/optimize/Config.php', + 'think\\console\\command\\optimize\\Route' => $baseDir . '/thinkphp/library/think/console/command/optimize/Route.php', + 'think\\console\\command\\optimize\\Schema' => $baseDir . '/thinkphp/library/think/console/command/optimize/Schema.php', + 'think\\console\\input\\Argument' => $baseDir . '/thinkphp/library/think/console/input/Argument.php', + 'think\\console\\input\\Definition' => $baseDir . '/thinkphp/library/think/console/input/Definition.php', + 'think\\console\\input\\Option' => $baseDir . '/thinkphp/library/think/console/input/Option.php', + 'think\\console\\output\\Ask' => $baseDir . '/thinkphp/library/think/console/output/Ask.php', + 'think\\console\\output\\Descriptor' => $baseDir . '/thinkphp/library/think/console/output/Descriptor.php', + 'think\\console\\output\\Formatter' => $baseDir . '/thinkphp/library/think/console/output/Formatter.php', + 'think\\console\\output\\Question' => $baseDir . '/thinkphp/library/think/console/output/Question.php', + 'think\\console\\output\\descriptor\\Console' => $baseDir . '/thinkphp/library/think/console/output/descriptor/Console.php', + 'think\\console\\output\\driver\\Buffer' => $baseDir . '/thinkphp/library/think/console/output/driver/Buffer.php', + 'think\\console\\output\\driver\\Console' => $baseDir . '/thinkphp/library/think/console/output/driver/Console.php', + 'think\\console\\output\\driver\\Nothing' => $baseDir . '/thinkphp/library/think/console/output/driver/Nothing.php', + 'think\\console\\output\\formatter\\Stack' => $baseDir . '/thinkphp/library/think/console/output/formatter/Stack.php', + 'think\\console\\output\\formatter\\Style' => $baseDir . '/thinkphp/library/think/console/output/formatter/Style.php', + 'think\\console\\output\\question\\Choice' => $baseDir . '/thinkphp/library/think/console/output/question/Choice.php', + 'think\\console\\output\\question\\Confirmation' => $baseDir . '/thinkphp/library/think/console/output/question/Confirmation.php', + 'think\\controller\\Rest' => $baseDir . '/thinkphp/library/think/controller/Rest.php', + 'think\\controller\\Yar' => $baseDir . '/thinkphp/library/think/controller/Yar.php', + 'think\\db\\Builder' => $baseDir . '/thinkphp/library/think/db/Builder.php', + 'think\\db\\Connection' => $baseDir . '/thinkphp/library/think/db/Connection.php', + 'think\\db\\Query' => $baseDir . '/thinkphp/library/think/db/Query.php', + 'think\\db\\builder\\Mysql' => $baseDir . '/thinkphp/library/think/db/builder/Mysql.php', + 'think\\db\\builder\\Pgsql' => $baseDir . '/thinkphp/library/think/db/builder/Pgsql.php', + 'think\\db\\builder\\Sqlite' => $baseDir . '/thinkphp/library/think/db/builder/Sqlite.php', + 'think\\db\\builder\\Sqlsrv' => $baseDir . '/thinkphp/library/think/db/builder/Sqlsrv.php', + 'think\\db\\connector\\Mysql' => $baseDir . '/thinkphp/library/think/db/connector/Mysql.php', + 'think\\db\\connector\\Pgsql' => $baseDir . '/thinkphp/library/think/db/connector/Pgsql.php', + 'think\\db\\connector\\Sqlite' => $baseDir . '/thinkphp/library/think/db/connector/Sqlite.php', + 'think\\db\\connector\\Sqlsrv' => $baseDir . '/thinkphp/library/think/db/connector/Sqlsrv.php', + 'think\\db\\exception\\BindParamException' => $baseDir . '/thinkphp/library/think/db/exception/BindParamException.php', + 'think\\db\\exception\\DataNotFoundException' => $baseDir . '/thinkphp/library/think/db/exception/DataNotFoundException.php', + 'think\\db\\exception\\ModelNotFoundException' => $baseDir . '/thinkphp/library/think/db/exception/ModelNotFoundException.php', + 'think\\debug\\Console' => $baseDir . '/thinkphp/library/think/debug/Console.php', + 'think\\debug\\Html' => $baseDir . '/thinkphp/library/think/debug/Html.php', + 'think\\exception\\ClassNotFoundException' => $baseDir . '/thinkphp/library/think/exception/ClassNotFoundException.php', + 'think\\exception\\DbException' => $baseDir . '/thinkphp/library/think/exception/DbException.php', + 'think\\exception\\ErrorException' => $baseDir . '/thinkphp/library/think/exception/ErrorException.php', + 'think\\exception\\Handle' => $baseDir . '/thinkphp/library/think/exception/Handle.php', + 'think\\exception\\HttpException' => $baseDir . '/thinkphp/library/think/exception/HttpException.php', + 'think\\exception\\HttpResponseException' => $baseDir . '/thinkphp/library/think/exception/HttpResponseException.php', + 'think\\exception\\PDOException' => $baseDir . '/thinkphp/library/think/exception/PDOException.php', + 'think\\exception\\RouteNotFoundException' => $baseDir . '/thinkphp/library/think/exception/RouteNotFoundException.php', + 'think\\exception\\TemplateNotFoundException' => $baseDir . '/thinkphp/library/think/exception/TemplateNotFoundException.php', + 'think\\exception\\ThrowableError' => $baseDir . '/thinkphp/library/think/exception/ThrowableError.php', + 'think\\exception\\ValidateException' => $baseDir . '/thinkphp/library/think/exception/ValidateException.php', + 'think\\helper\\Arr' => $vendorDir . '/topthink/think-helper/src/Arr.php', + 'think\\helper\\Hash' => $vendorDir . '/topthink/think-helper/src/Hash.php', + 'think\\helper\\Str' => $vendorDir . '/topthink/think-helper/src/Str.php', + 'think\\helper\\Time' => $vendorDir . '/topthink/think-helper/src/Time.php', + 'think\\helper\\hash\\Bcrypt' => $vendorDir . '/topthink/think-helper/src/hash/Bcrypt.php', + 'think\\helper\\hash\\Md5' => $vendorDir . '/topthink/think-helper/src/hash/Md5.php', + 'think\\log\\driver\\File' => $baseDir . '/thinkphp/library/think/log/driver/File.php', + 'think\\log\\driver\\Socket' => $baseDir . '/thinkphp/library/think/log/driver/Socket.php', + 'think\\log\\driver\\Test' => $baseDir . '/thinkphp/library/think/log/driver/Test.php', + 'think\\model\\Collection' => $baseDir . '/thinkphp/library/think/model/Collection.php', + 'think\\model\\Merge' => $baseDir . '/thinkphp/library/think/model/Merge.php', + 'think\\model\\Pivot' => $baseDir . '/thinkphp/library/think/model/Pivot.php', + 'think\\model\\Relation' => $baseDir . '/thinkphp/library/think/model/Relation.php', + 'think\\model\\relation\\BelongsTo' => $baseDir . '/thinkphp/library/think/model/relation/BelongsTo.php', + 'think\\model\\relation\\BelongsToMany' => $baseDir . '/thinkphp/library/think/model/relation/BelongsToMany.php', + 'think\\model\\relation\\HasMany' => $baseDir . '/thinkphp/library/think/model/relation/HasMany.php', + 'think\\model\\relation\\HasManyThrough' => $baseDir . '/thinkphp/library/think/model/relation/HasManyThrough.php', + 'think\\model\\relation\\HasOne' => $baseDir . '/thinkphp/library/think/model/relation/HasOne.php', + 'think\\model\\relation\\MorphMany' => $baseDir . '/thinkphp/library/think/model/relation/MorphMany.php', + 'think\\model\\relation\\MorphTo' => $baseDir . '/thinkphp/library/think/model/relation/MorphTo.php', + 'think\\model\\relation\\OneToOne' => $baseDir . '/thinkphp/library/think/model/relation/OneToOne.php', + 'think\\mongo\\Builder' => $vendorDir . '/topthink/think-mongo/src/Builder.php', + 'think\\mongo\\Connection' => $vendorDir . '/topthink/think-mongo/src/Connection.php', + 'think\\mongo\\Query' => $vendorDir . '/topthink/think-mongo/src/Query.php', + 'think\\paginator\\driver\\Bootstrap' => $baseDir . '/thinkphp/library/think/paginator/driver/Bootstrap.php', + 'think\\process\\Builder' => $baseDir . '/thinkphp/library/think/process/Builder.php', + 'think\\process\\Utils' => $baseDir . '/thinkphp/library/think/process/Utils.php', + 'think\\process\\exception\\Failed' => $baseDir . '/thinkphp/library/think/process/exception/Failed.php', + 'think\\process\\exception\\Timeout' => $baseDir . '/thinkphp/library/think/process/exception/Timeout.php', + 'think\\process\\pipes\\Pipes' => $baseDir . '/thinkphp/library/think/process/pipes/Pipes.php', + 'think\\process\\pipes\\Unix' => $baseDir . '/thinkphp/library/think/process/pipes/Unix.php', + 'think\\process\\pipes\\Windows' => $baseDir . '/thinkphp/library/think/process/pipes/Windows.php', + 'think\\queue\\CallQueuedHandler' => $vendorDir . '/topthink/think-queue/src/queue/CallQueuedHandler.php', + 'think\\queue\\Connector' => $vendorDir . '/topthink/think-queue/src/queue/Connector.php', + 'think\\queue\\Job' => $vendorDir . '/topthink/think-queue/src/queue/Job.php', + 'think\\queue\\Listener' => $vendorDir . '/topthink/think-queue/src/queue/Listener.php', + 'think\\queue\\Queueable' => $vendorDir . '/topthink/think-queue/src/queue/Queueable.php', + 'think\\queue\\ShouldQueue' => $vendorDir . '/topthink/think-queue/src/queue/ShouldQueue.php', + 'think\\queue\\Worker' => $vendorDir . '/topthink/think-queue/src/queue/Worker.php', + 'think\\queue\\command\\Listen' => $vendorDir . '/topthink/think-queue/src/queue/command/Listen.php', + 'think\\queue\\command\\Restart' => $vendorDir . '/topthink/think-queue/src/queue/command/Restart.php', + 'think\\queue\\command\\Subscribe' => $vendorDir . '/topthink/think-queue/src/queue/command/Subscribe.php', + 'think\\queue\\command\\Work' => $vendorDir . '/topthink/think-queue/src/queue/command/Work.php', + 'think\\queue\\connector\\Database' => $vendorDir . '/topthink/think-queue/src/queue/connector/Database.php', + 'think\\queue\\connector\\Redis' => $vendorDir . '/topthink/think-queue/src/queue/connector/Redis.php', + 'think\\queue\\connector\\Sync' => $vendorDir . '/topthink/think-queue/src/queue/connector/Sync.php', + 'think\\queue\\connector\\Topthink' => $vendorDir . '/topthink/think-queue/src/queue/connector/Topthink.php', + 'think\\queue\\job\\Database' => $vendorDir . '/topthink/think-queue/src/queue/job/Database.php', + 'think\\queue\\job\\Redis' => $vendorDir . '/topthink/think-queue/src/queue/job/Redis.php', + 'think\\queue\\job\\Sync' => $vendorDir . '/topthink/think-queue/src/queue/job/Sync.php', + 'think\\queue\\job\\Topthink' => $vendorDir . '/topthink/think-queue/src/queue/job/Topthink.php', + 'think\\response\\Json' => $baseDir . '/thinkphp/library/think/response/Json.php', + 'think\\response\\Jsonp' => $baseDir . '/thinkphp/library/think/response/Jsonp.php', + 'think\\response\\Redirect' => $baseDir . '/thinkphp/library/think/response/Redirect.php', + 'think\\response\\View' => $baseDir . '/thinkphp/library/think/response/View.php', + 'think\\response\\Xml' => $baseDir . '/thinkphp/library/think/response/Xml.php', + 'think\\session\\driver\\Memcache' => $baseDir . '/thinkphp/library/think/session/driver/Memcache.php', + 'think\\session\\driver\\Memcached' => $baseDir . '/thinkphp/library/think/session/driver/Memcached.php', + 'think\\session\\driver\\Redis' => $baseDir . '/thinkphp/library/think/session/driver/Redis.php', + 'think\\template\\TagLib' => $baseDir . '/thinkphp/library/think/template/TagLib.php', + 'think\\template\\driver\\File' => $baseDir . '/thinkphp/library/think/template/driver/File.php', + 'think\\template\\taglib\\Cx' => $baseDir . '/thinkphp/library/think/template/taglib/Cx.php', + 'think\\view\\driver\\Php' => $baseDir . '/thinkphp/library/think/view/driver/Php.php', + 'think\\view\\driver\\Think' => $baseDir . '/thinkphp/library/think/view/driver/Think.php', + 'think\\worker\\Server' => $vendorDir . '/topthink/think-worker/src/Server.php', +); diff --git a/vendor/composer/autoload_files.php b/vendor/composer/autoload_files.php new file mode 100644 index 000000000..35feae109 --- /dev/null +++ b/vendor/composer/autoload_files.php @@ -0,0 +1,13 @@ + $vendorDir . '/topthink/think-helper/src/helper.php', + '841780ea2e1d6545ea3a253239d59c05' => $vendorDir . '/qiniu/php-sdk/src/Qiniu/functions.php', + '1cfd2761b63b0a29ed23657ea394cb2d' => $vendorDir . '/topthink/think-captcha/src/helper.php', + 'cc56288302d9df745d97c934d6a6e5f0' => $vendorDir . '/topthink/think-queue/src/common.php', +); diff --git a/vendor/composer/autoload_namespaces.php b/vendor/composer/autoload_namespaces.php new file mode 100644 index 000000000..ba7e24cd7 --- /dev/null +++ b/vendor/composer/autoload_namespaces.php @@ -0,0 +1,10 @@ + array($vendorDir . '/aferrandini/phpqrcode/lib'), +); diff --git a/vendor/composer/autoload_psr4.php b/vendor/composer/autoload_psr4.php new file mode 100644 index 000000000..1dc897235 --- /dev/null +++ b/vendor/composer/autoload_psr4.php @@ -0,0 +1,18 @@ + array($vendorDir . '/topthink/think-worker/src'), + 'think\\mongo\\' => array($vendorDir . '/topthink/think-mongo/src'), + 'think\\helper\\' => array($vendorDir . '/topthink/think-helper/src'), + 'think\\composer\\' => array($vendorDir . '/topthink/think-installer/src'), + 'think\\captcha\\' => array($vendorDir . '/topthink/think-captcha/src'), + 'think\\' => array($baseDir . '/thinkphp/library/think', $vendorDir . '/topthink/think-queue/src'), + 'Workerman\\' => array($vendorDir . '/workerman/workerman'), + 'Wechat\\' => array($vendorDir . '/zoujingli/wechat-php-sdk/Wechat'), + 'Qiniu\\' => array($vendorDir . '/qiniu/php-sdk/src/Qiniu'), +); diff --git a/vendor/composer/autoload_real.php b/vendor/composer/autoload_real.php new file mode 100644 index 000000000..d4cc4f02e --- /dev/null +++ b/vendor/composer/autoload_real.php @@ -0,0 +1,70 @@ += 50600 && !defined('HHVM_VERSION'); + if ($useStaticLoader) { + require_once __DIR__ . '/autoload_static.php'; + + call_user_func(\Composer\Autoload\ComposerStaticInit9e22b278b1b4ad2907e604cae5a45125::getInitializer($loader)); + } else { + $map = require __DIR__ . '/autoload_namespaces.php'; + foreach ($map as $namespace => $path) { + $loader->set($namespace, $path); + } + + $map = require __DIR__ . '/autoload_psr4.php'; + foreach ($map as $namespace => $path) { + $loader->setPsr4($namespace, $path); + } + + $classMap = require __DIR__ . '/autoload_classmap.php'; + if ($classMap) { + $loader->addClassMap($classMap); + } + } + + $loader->register(true); + + if ($useStaticLoader) { + $includeFiles = Composer\Autoload\ComposerStaticInit9e22b278b1b4ad2907e604cae5a45125::$files; + } else { + $includeFiles = require __DIR__ . '/autoload_files.php'; + } + foreach ($includeFiles as $fileIdentifier => $file) { + composerRequire9e22b278b1b4ad2907e604cae5a45125($fileIdentifier, $file); + } + + return $loader; + } +} + +function composerRequire9e22b278b1b4ad2907e604cae5a45125($fileIdentifier, $file) +{ + if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) { + require $file; + + $GLOBALS['__composer_autoload_files'][$fileIdentifier] = true; + } +} diff --git a/vendor/composer/autoload_static.php b/vendor/composer/autoload_static.php new file mode 100644 index 000000000..aa4b5ab8a --- /dev/null +++ b/vendor/composer/autoload_static.php @@ -0,0 +1,347 @@ + __DIR__ . '/..' . '/topthink/think-helper/src/helper.php', + '841780ea2e1d6545ea3a253239d59c05' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/functions.php', + '1cfd2761b63b0a29ed23657ea394cb2d' => __DIR__ . '/..' . '/topthink/think-captcha/src/helper.php', + 'cc56288302d9df745d97c934d6a6e5f0' => __DIR__ . '/..' . '/topthink/think-queue/src/common.php', + ); + + public static $prefixLengthsPsr4 = array ( + 't' => + array ( + 'think\\worker\\' => 13, + 'think\\mongo\\' => 12, + 'think\\helper\\' => 13, + 'think\\composer\\' => 15, + 'think\\captcha\\' => 14, + 'think\\' => 6, + ), + 'W' => + array ( + 'Workerman\\' => 10, + 'Wechat\\' => 7, + ), + 'Q' => + array ( + 'Qiniu\\' => 6, + ), + ); + + public static $prefixDirsPsr4 = array ( + 'think\\worker\\' => + array ( + 0 => __DIR__ . '/..' . '/topthink/think-worker/src', + ), + 'think\\mongo\\' => + array ( + 0 => __DIR__ . '/..' . '/topthink/think-mongo/src', + ), + 'think\\helper\\' => + array ( + 0 => __DIR__ . '/..' . '/topthink/think-helper/src', + ), + 'think\\composer\\' => + array ( + 0 => __DIR__ . '/..' . '/topthink/think-installer/src', + ), + 'think\\captcha\\' => + array ( + 0 => __DIR__ . '/..' . '/topthink/think-captcha/src', + ), + 'think\\' => + array ( + 0 => __DIR__ . '/../..' . '/thinkphp/library/think', + 1 => __DIR__ . '/..' . '/topthink/think-queue/src', + ), + 'Workerman\\' => + array ( + 0 => __DIR__ . '/..' . '/workerman/workerman', + ), + 'Wechat\\' => + array ( + 0 => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat', + ), + 'Qiniu\\' => + array ( + 0 => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu', + ), + ); + + public static $prefixesPsr0 = array ( + 'P' => + array ( + 'PHPQRCode' => + array ( + 0 => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib', + ), + ), + ); + + public static $classMap = array ( + 'PHPQRCode' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode.php', + 'PHPQRCode\\Autoloader' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/Autoloader.php', + 'PHPQRCode\\Constants' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/Constants.php', + 'PHPQRCode\\FrameFiller' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/FrameFiller.php', + 'PHPQRCode\\QRbitstream' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRbitstream.php', + 'PHPQRCode\\QRcode' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRcode.php', + 'PHPQRCode\\QRencode' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRencode.php', + 'PHPQRCode\\QRimage' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRimage.php', + 'PHPQRCode\\QRinput' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRinput.php', + 'PHPQRCode\\QRinputItem' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRinputItem.php', + 'PHPQRCode\\QRmask' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRmask.php', + 'PHPQRCode\\QRrawcode' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrawcode.php', + 'PHPQRCode\\QRrs' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrs.php', + 'PHPQRCode\\QRrsItem' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrsItem.php', + 'PHPQRCode\\QRrsblock' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRrsblock.php', + 'PHPQRCode\\QRspec' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRspec.php', + 'PHPQRCode\\QRsplit' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRsplit.php', + 'PHPQRCode\\QRstr' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRstr.php', + 'PHPQRCode\\QRtools' => __DIR__ . '/..' . '/aferrandini/phpqrcode/lib/PHPQRCode/QRtools.php', + 'PclZip' => __DIR__ . '/..' . '/pclzip/pclzip/pclzip.lib.php', + 'Qiniu\\Auth' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Auth.php', + 'Qiniu\\Config' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Config.php', + 'Qiniu\\Etag' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Etag.php', + 'Qiniu\\Http\\Client' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Http/Client.php', + 'Qiniu\\Http\\Error' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Http/Error.php', + 'Qiniu\\Http\\Request' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Http/Request.php', + 'Qiniu\\Http\\Response' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Http/Response.php', + 'Qiniu\\Processing\\ImageUrlBuilder' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Processing/ImageUrlBuilder.php', + 'Qiniu\\Processing\\Operation' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Processing/Operation.php', + 'Qiniu\\Processing\\PersistentFop' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Processing/PersistentFop.php', + 'Qiniu\\Storage\\BucketManager' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Storage/BucketManager.php', + 'Qiniu\\Storage\\FormUploader' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Storage/FormUploader.php', + 'Qiniu\\Storage\\ResumeUploader' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Storage/ResumeUploader.php', + 'Qiniu\\Storage\\UploadManager' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Storage/UploadManager.php', + 'Qiniu\\Zone' => __DIR__ . '/..' . '/qiniu/php-sdk/src/Qiniu/Zone.php', + 'Wechat\\Lib\\Cache' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/Lib/Cache.php', + 'Wechat\\Lib\\Common' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/Lib/Common.php', + 'Wechat\\Lib\\Tools' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/Lib/Tools.php', + 'Wechat\\Loader' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/Loader.php', + 'Wechat\\WechatCard' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatCard.php', + 'Wechat\\WechatCustom' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatCustom.php', + 'Wechat\\WechatDevice' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatDevice.php', + 'Wechat\\WechatExtends' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatExtends.php', + 'Wechat\\WechatMedia' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatMedia.php', + 'Wechat\\WechatMenu' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatMenu.php', + 'Wechat\\WechatOauth' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatOauth.php', + 'Wechat\\WechatPay' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatPay.php', + 'Wechat\\WechatPoi' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatPoi.php', + 'Wechat\\WechatReceive' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatReceive.php', + 'Wechat\\WechatScript' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatScript.php', + 'Wechat\\WechatService' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatService.php', + 'Wechat\\WechatUser' => __DIR__ . '/..' . '/zoujingli/wechat-php-sdk/Wechat/WechatUser.php', + 'Workerman\\Autoloader' => __DIR__ . '/..' . '/workerman/workerman/Autoloader.php', + 'Workerman\\Connection\\AsyncTcpConnection' => __DIR__ . '/..' . '/workerman/workerman/Connection/AsyncTcpConnection.php', + 'Workerman\\Connection\\ConnectionInterface' => __DIR__ . '/..' . '/workerman/workerman/Connection/ConnectionInterface.php', + 'Workerman\\Connection\\TcpConnection' => __DIR__ . '/..' . '/workerman/workerman/Connection/TcpConnection.php', + 'Workerman\\Connection\\UdpConnection' => __DIR__ . '/..' . '/workerman/workerman/Connection/UdpConnection.php', + 'Workerman\\Events\\Ev' => __DIR__ . '/..' . '/workerman/workerman/Events/Ev.php', + 'Workerman\\Events\\Event' => __DIR__ . '/..' . '/workerman/workerman/Events/Event.php', + 'Workerman\\Events\\EventInterface' => __DIR__ . '/..' . '/workerman/workerman/Events/EventInterface.php', + 'Workerman\\Events\\Libevent' => __DIR__ . '/..' . '/workerman/workerman/Events/Libevent.php', + 'Workerman\\Events\\React' => __DIR__ . '/..' . '/workerman/workerman/Events/React.php', + 'Workerman\\Events\\React\\ExtEventLoop' => __DIR__ . '/..' . '/workerman/workerman/Events/React/ExtEventLoop.php', + 'Workerman\\Events\\React\\LibEventLoop' => __DIR__ . '/..' . '/workerman/workerman/Events/React/LibEventLoop.php', + 'Workerman\\Events\\React\\StreamSelectLoop' => __DIR__ . '/..' . '/workerman/workerman/Events/React/StreamSelectLoop.php', + 'Workerman\\Events\\Select' => __DIR__ . '/..' . '/workerman/workerman/Events/Select.php', + 'Workerman\\Lib\\Timer' => __DIR__ . '/..' . '/workerman/workerman/Lib/Timer.php', + 'Workerman\\Protocols\\Frame' => __DIR__ . '/..' . '/workerman/workerman/Protocols/Frame.php', + 'Workerman\\Protocols\\Http' => __DIR__ . '/..' . '/workerman/workerman/Protocols/Http.php', + 'Workerman\\Protocols\\HttpCache' => __DIR__ . '/..' . '/workerman/workerman/Protocols/Http.php', + 'Workerman\\Protocols\\ProtocolInterface' => __DIR__ . '/..' . '/workerman/workerman/Protocols/ProtocolInterface.php', + 'Workerman\\Protocols\\Text' => __DIR__ . '/..' . '/workerman/workerman/Protocols/Text.php', + 'Workerman\\Protocols\\Websocket' => __DIR__ . '/..' . '/workerman/workerman/Protocols/Websocket.php', + 'Workerman\\Protocols\\Ws' => __DIR__ . '/..' . '/workerman/workerman/Protocols/Ws.php', + 'Workerman\\WebServer' => __DIR__ . '/..' . '/workerman/workerman/WebServer.php', + 'Workerman\\Worker' => __DIR__ . '/..' . '/workerman/workerman/Worker.php', + 'think\\App' => __DIR__ . '/../..' . '/thinkphp/library/think/App.php', + 'think\\Build' => __DIR__ . '/../..' . '/thinkphp/library/think/Build.php', + 'think\\Cache' => __DIR__ . '/../..' . '/thinkphp/library/think/Cache.php', + 'think\\Collection' => __DIR__ . '/../..' . '/thinkphp/library/think/Collection.php', + 'think\\Config' => __DIR__ . '/../..' . '/thinkphp/library/think/Config.php', + 'think\\Console' => __DIR__ . '/../..' . '/thinkphp/library/think/Console.php', + 'think\\Controller' => __DIR__ . '/../..' . '/thinkphp/library/think/Controller.php', + 'think\\Cookie' => __DIR__ . '/../..' . '/thinkphp/library/think/Cookie.php', + 'think\\Db' => __DIR__ . '/../..' . '/thinkphp/library/think/Db.php', + 'think\\Debug' => __DIR__ . '/../..' . '/thinkphp/library/think/Debug.php', + 'think\\Env' => __DIR__ . '/../..' . '/thinkphp/library/think/Env.php', + 'think\\Error' => __DIR__ . '/../..' . '/thinkphp/library/think/Error.php', + 'think\\Exception' => __DIR__ . '/../..' . '/thinkphp/library/think/Exception.php', + 'think\\File' => __DIR__ . '/../..' . '/thinkphp/library/think/File.php', + 'think\\Hook' => __DIR__ . '/../..' . '/thinkphp/library/think/Hook.php', + 'think\\Lang' => __DIR__ . '/../..' . '/thinkphp/library/think/Lang.php', + 'think\\Loader' => __DIR__ . '/../..' . '/thinkphp/library/think/Loader.php', + 'think\\Log' => __DIR__ . '/../..' . '/thinkphp/library/think/Log.php', + 'think\\Model' => __DIR__ . '/../..' . '/thinkphp/library/think/Model.php', + 'think\\Paginator' => __DIR__ . '/../..' . '/thinkphp/library/think/Paginator.php', + 'think\\Process' => __DIR__ . '/../..' . '/thinkphp/library/think/Process.php', + 'think\\Queue' => __DIR__ . '/..' . '/topthink/think-queue/src/Queue.php', + 'think\\Request' => __DIR__ . '/../..' . '/thinkphp/library/think/Request.php', + 'think\\Response' => __DIR__ . '/../..' . '/thinkphp/library/think/Response.php', + 'think\\Route' => __DIR__ . '/../..' . '/thinkphp/library/think/Route.php', + 'think\\Session' => __DIR__ . '/../..' . '/thinkphp/library/think/Session.php', + 'think\\Template' => __DIR__ . '/../..' . '/thinkphp/library/think/Template.php', + 'think\\Url' => __DIR__ . '/../..' . '/thinkphp/library/think/Url.php', + 'think\\Validate' => __DIR__ . '/../..' . '/thinkphp/library/think/Validate.php', + 'think\\View' => __DIR__ . '/../..' . '/thinkphp/library/think/View.php', + 'think\\cache\\Driver' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/Driver.php', + 'think\\cache\\driver\\File' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/File.php', + 'think\\cache\\driver\\Lite' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Lite.php', + 'think\\cache\\driver\\Memcache' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Memcache.php', + 'think\\cache\\driver\\Memcached' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Memcached.php', + 'think\\cache\\driver\\Redis' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Redis.php', + 'think\\cache\\driver\\Sqlite' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Sqlite.php', + 'think\\cache\\driver\\Wincache' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Wincache.php', + 'think\\cache\\driver\\Xcache' => __DIR__ . '/../..' . '/thinkphp/library/think/cache/driver/Xcache.php', + 'think\\captcha\\Captcha' => __DIR__ . '/..' . '/topthink/think-captcha/src/Captcha.php', + 'think\\captcha\\CaptchaController' => __DIR__ . '/..' . '/topthink/think-captcha/src/CaptchaController.php', + 'think\\composer\\Plugin' => __DIR__ . '/..' . '/topthink/think-installer/src/Plugin.php', + 'think\\composer\\ThinkExtend' => __DIR__ . '/..' . '/topthink/think-installer/src/ThinkExtend.php', + 'think\\composer\\ThinkFramework' => __DIR__ . '/..' . '/topthink/think-installer/src/ThinkFramework.php', + 'think\\composer\\ThinkTesting' => __DIR__ . '/..' . '/topthink/think-installer/src/ThinkTesting.php', + 'think\\config\\driver\\Ini' => __DIR__ . '/../..' . '/thinkphp/library/think/config/driver/Ini.php', + 'think\\config\\driver\\Json' => __DIR__ . '/../..' . '/thinkphp/library/think/config/driver/Json.php', + 'think\\config\\driver\\Xml' => __DIR__ . '/../..' . '/thinkphp/library/think/config/driver/Xml.php', + 'think\\console\\Command' => __DIR__ . '/../..' . '/thinkphp/library/think/console/Command.php', + 'think\\console\\Input' => __DIR__ . '/../..' . '/thinkphp/library/think/console/Input.php', + 'think\\console\\Output' => __DIR__ . '/../..' . '/thinkphp/library/think/console/Output.php', + 'think\\console\\command\\Build' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/Build.php', + 'think\\console\\command\\Clear' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/Clear.php', + 'think\\console\\command\\Help' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/Help.php', + 'think\\console\\command\\Lists' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/Lists.php', + 'think\\console\\command\\Make' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/Make.php', + 'think\\console\\command\\make\\Controller' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/make/Controller.php', + 'think\\console\\command\\make\\Model' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/make/Model.php', + 'think\\console\\command\\optimize\\Autoload' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/optimize/Autoload.php', + 'think\\console\\command\\optimize\\Config' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/optimize/Config.php', + 'think\\console\\command\\optimize\\Route' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/optimize/Route.php', + 'think\\console\\command\\optimize\\Schema' => __DIR__ . '/../..' . '/thinkphp/library/think/console/command/optimize/Schema.php', + 'think\\console\\input\\Argument' => __DIR__ . '/../..' . '/thinkphp/library/think/console/input/Argument.php', + 'think\\console\\input\\Definition' => __DIR__ . '/../..' . '/thinkphp/library/think/console/input/Definition.php', + 'think\\console\\input\\Option' => __DIR__ . '/../..' . '/thinkphp/library/think/console/input/Option.php', + 'think\\console\\output\\Ask' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/Ask.php', + 'think\\console\\output\\Descriptor' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/Descriptor.php', + 'think\\console\\output\\Formatter' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/Formatter.php', + 'think\\console\\output\\Question' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/Question.php', + 'think\\console\\output\\descriptor\\Console' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/descriptor/Console.php', + 'think\\console\\output\\driver\\Buffer' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/driver/Buffer.php', + 'think\\console\\output\\driver\\Console' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/driver/Console.php', + 'think\\console\\output\\driver\\Nothing' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/driver/Nothing.php', + 'think\\console\\output\\formatter\\Stack' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/formatter/Stack.php', + 'think\\console\\output\\formatter\\Style' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/formatter/Style.php', + 'think\\console\\output\\question\\Choice' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/question/Choice.php', + 'think\\console\\output\\question\\Confirmation' => __DIR__ . '/../..' . '/thinkphp/library/think/console/output/question/Confirmation.php', + 'think\\controller\\Rest' => __DIR__ . '/../..' . '/thinkphp/library/think/controller/Rest.php', + 'think\\controller\\Yar' => __DIR__ . '/../..' . '/thinkphp/library/think/controller/Yar.php', + 'think\\db\\Builder' => __DIR__ . '/../..' . '/thinkphp/library/think/db/Builder.php', + 'think\\db\\Connection' => __DIR__ . '/../..' . '/thinkphp/library/think/db/Connection.php', + 'think\\db\\Query' => __DIR__ . '/../..' . '/thinkphp/library/think/db/Query.php', + 'think\\db\\builder\\Mysql' => __DIR__ . '/../..' . '/thinkphp/library/think/db/builder/Mysql.php', + 'think\\db\\builder\\Pgsql' => __DIR__ . '/../..' . '/thinkphp/library/think/db/builder/Pgsql.php', + 'think\\db\\builder\\Sqlite' => __DIR__ . '/../..' . '/thinkphp/library/think/db/builder/Sqlite.php', + 'think\\db\\builder\\Sqlsrv' => __DIR__ . '/../..' . '/thinkphp/library/think/db/builder/Sqlsrv.php', + 'think\\db\\connector\\Mysql' => __DIR__ . '/../..' . '/thinkphp/library/think/db/connector/Mysql.php', + 'think\\db\\connector\\Pgsql' => __DIR__ . '/../..' . '/thinkphp/library/think/db/connector/Pgsql.php', + 'think\\db\\connector\\Sqlite' => __DIR__ . '/../..' . '/thinkphp/library/think/db/connector/Sqlite.php', + 'think\\db\\connector\\Sqlsrv' => __DIR__ . '/../..' . '/thinkphp/library/think/db/connector/Sqlsrv.php', + 'think\\db\\exception\\BindParamException' => __DIR__ . '/../..' . '/thinkphp/library/think/db/exception/BindParamException.php', + 'think\\db\\exception\\DataNotFoundException' => __DIR__ . '/../..' . '/thinkphp/library/think/db/exception/DataNotFoundException.php', + 'think\\db\\exception\\ModelNotFoundException' => __DIR__ . '/../..' . '/thinkphp/library/think/db/exception/ModelNotFoundException.php', + 'think\\debug\\Console' => __DIR__ . '/../..' . '/thinkphp/library/think/debug/Console.php', + 'think\\debug\\Html' => __DIR__ . '/../..' . '/thinkphp/library/think/debug/Html.php', + 'think\\exception\\ClassNotFoundException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/ClassNotFoundException.php', + 'think\\exception\\DbException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/DbException.php', + 'think\\exception\\ErrorException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/ErrorException.php', + 'think\\exception\\Handle' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/Handle.php', + 'think\\exception\\HttpException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/HttpException.php', + 'think\\exception\\HttpResponseException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/HttpResponseException.php', + 'think\\exception\\PDOException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/PDOException.php', + 'think\\exception\\RouteNotFoundException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/RouteNotFoundException.php', + 'think\\exception\\TemplateNotFoundException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/TemplateNotFoundException.php', + 'think\\exception\\ThrowableError' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/ThrowableError.php', + 'think\\exception\\ValidateException' => __DIR__ . '/../..' . '/thinkphp/library/think/exception/ValidateException.php', + 'think\\helper\\Arr' => __DIR__ . '/..' . '/topthink/think-helper/src/Arr.php', + 'think\\helper\\Hash' => __DIR__ . '/..' . '/topthink/think-helper/src/Hash.php', + 'think\\helper\\Str' => __DIR__ . '/..' . '/topthink/think-helper/src/Str.php', + 'think\\helper\\Time' => __DIR__ . '/..' . '/topthink/think-helper/src/Time.php', + 'think\\helper\\hash\\Bcrypt' => __DIR__ . '/..' . '/topthink/think-helper/src/hash/Bcrypt.php', + 'think\\helper\\hash\\Md5' => __DIR__ . '/..' . '/topthink/think-helper/src/hash/Md5.php', + 'think\\log\\driver\\File' => __DIR__ . '/../..' . '/thinkphp/library/think/log/driver/File.php', + 'think\\log\\driver\\Socket' => __DIR__ . '/../..' . '/thinkphp/library/think/log/driver/Socket.php', + 'think\\log\\driver\\Test' => __DIR__ . '/../..' . '/thinkphp/library/think/log/driver/Test.php', + 'think\\model\\Collection' => __DIR__ . '/../..' . '/thinkphp/library/think/model/Collection.php', + 'think\\model\\Merge' => __DIR__ . '/../..' . '/thinkphp/library/think/model/Merge.php', + 'think\\model\\Pivot' => __DIR__ . '/../..' . '/thinkphp/library/think/model/Pivot.php', + 'think\\model\\Relation' => __DIR__ . '/../..' . '/thinkphp/library/think/model/Relation.php', + 'think\\model\\relation\\BelongsTo' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/BelongsTo.php', + 'think\\model\\relation\\BelongsToMany' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/BelongsToMany.php', + 'think\\model\\relation\\HasMany' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/HasMany.php', + 'think\\model\\relation\\HasManyThrough' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/HasManyThrough.php', + 'think\\model\\relation\\HasOne' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/HasOne.php', + 'think\\model\\relation\\MorphMany' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/MorphMany.php', + 'think\\model\\relation\\MorphTo' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/MorphTo.php', + 'think\\model\\relation\\OneToOne' => __DIR__ . '/../..' . '/thinkphp/library/think/model/relation/OneToOne.php', + 'think\\mongo\\Builder' => __DIR__ . '/..' . '/topthink/think-mongo/src/Builder.php', + 'think\\mongo\\Connection' => __DIR__ . '/..' . '/topthink/think-mongo/src/Connection.php', + 'think\\mongo\\Query' => __DIR__ . '/..' . '/topthink/think-mongo/src/Query.php', + 'think\\paginator\\driver\\Bootstrap' => __DIR__ . '/../..' . '/thinkphp/library/think/paginator/driver/Bootstrap.php', + 'think\\process\\Builder' => __DIR__ . '/../..' . '/thinkphp/library/think/process/Builder.php', + 'think\\process\\Utils' => __DIR__ . '/../..' . '/thinkphp/library/think/process/Utils.php', + 'think\\process\\exception\\Failed' => __DIR__ . '/../..' . '/thinkphp/library/think/process/exception/Failed.php', + 'think\\process\\exception\\Timeout' => __DIR__ . '/../..' . '/thinkphp/library/think/process/exception/Timeout.php', + 'think\\process\\pipes\\Pipes' => __DIR__ . '/../..' . '/thinkphp/library/think/process/pipes/Pipes.php', + 'think\\process\\pipes\\Unix' => __DIR__ . '/../..' . '/thinkphp/library/think/process/pipes/Unix.php', + 'think\\process\\pipes\\Windows' => __DIR__ . '/../..' . '/thinkphp/library/think/process/pipes/Windows.php', + 'think\\queue\\CallQueuedHandler' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/CallQueuedHandler.php', + 'think\\queue\\Connector' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/Connector.php', + 'think\\queue\\Job' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/Job.php', + 'think\\queue\\Listener' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/Listener.php', + 'think\\queue\\Queueable' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/Queueable.php', + 'think\\queue\\ShouldQueue' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/ShouldQueue.php', + 'think\\queue\\Worker' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/Worker.php', + 'think\\queue\\command\\Listen' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/command/Listen.php', + 'think\\queue\\command\\Restart' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/command/Restart.php', + 'think\\queue\\command\\Subscribe' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/command/Subscribe.php', + 'think\\queue\\command\\Work' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/command/Work.php', + 'think\\queue\\connector\\Database' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/connector/Database.php', + 'think\\queue\\connector\\Redis' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/connector/Redis.php', + 'think\\queue\\connector\\Sync' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/connector/Sync.php', + 'think\\queue\\connector\\Topthink' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/connector/Topthink.php', + 'think\\queue\\job\\Database' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/job/Database.php', + 'think\\queue\\job\\Redis' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/job/Redis.php', + 'think\\queue\\job\\Sync' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/job/Sync.php', + 'think\\queue\\job\\Topthink' => __DIR__ . '/..' . '/topthink/think-queue/src/queue/job/Topthink.php', + 'think\\response\\Json' => __DIR__ . '/../..' . '/thinkphp/library/think/response/Json.php', + 'think\\response\\Jsonp' => __DIR__ . '/../..' . '/thinkphp/library/think/response/Jsonp.php', + 'think\\response\\Redirect' => __DIR__ . '/../..' . '/thinkphp/library/think/response/Redirect.php', + 'think\\response\\View' => __DIR__ . '/../..' . '/thinkphp/library/think/response/View.php', + 'think\\response\\Xml' => __DIR__ . '/../..' . '/thinkphp/library/think/response/Xml.php', + 'think\\session\\driver\\Memcache' => __DIR__ . '/../..' . '/thinkphp/library/think/session/driver/Memcache.php', + 'think\\session\\driver\\Memcached' => __DIR__ . '/../..' . '/thinkphp/library/think/session/driver/Memcached.php', + 'think\\session\\driver\\Redis' => __DIR__ . '/../..' . '/thinkphp/library/think/session/driver/Redis.php', + 'think\\template\\TagLib' => __DIR__ . '/../..' . '/thinkphp/library/think/template/TagLib.php', + 'think\\template\\driver\\File' => __DIR__ . '/../..' . '/thinkphp/library/think/template/driver/File.php', + 'think\\template\\taglib\\Cx' => __DIR__ . '/../..' . '/thinkphp/library/think/template/taglib/Cx.php', + 'think\\view\\driver\\Php' => __DIR__ . '/../..' . '/thinkphp/library/think/view/driver/Php.php', + 'think\\view\\driver\\Think' => __DIR__ . '/../..' . '/thinkphp/library/think/view/driver/Think.php', + 'think\\worker\\Server' => __DIR__ . '/..' . '/topthink/think-worker/src/Server.php', + ); + + public static function getInitializer(ClassLoader $loader) + { + return \Closure::bind(function () use ($loader) { + $loader->prefixLengthsPsr4 = ComposerStaticInit9e22b278b1b4ad2907e604cae5a45125::$prefixLengthsPsr4; + $loader->prefixDirsPsr4 = ComposerStaticInit9e22b278b1b4ad2907e604cae5a45125::$prefixDirsPsr4; + $loader->prefixesPsr0 = ComposerStaticInit9e22b278b1b4ad2907e604cae5a45125::$prefixesPsr0; + $loader->classMap = ComposerStaticInit9e22b278b1b4ad2907e604cae5a45125::$classMap; + + }, null, ClassLoader::class); + } +} diff --git a/vendor/composer/installed.json b/vendor/composer/installed.json new file mode 100644 index 000000000..7a019e8b6 --- /dev/null +++ b/vendor/composer/installed.json @@ -0,0 +1,518 @@ +[ + { + "name": "topthink/think-installer", + "version": "v1.0.11", + "version_normalized": "1.0.11.0", + "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" + }, + "time": "2016-12-01 09:08:45", + "type": "composer-plugin", + "extra": { + "class": "think\\composer\\Plugin" + }, + "installation-source": "dist", + "autoload": { + "psr-4": { + "think\\composer\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ] + }, + { + "name": "pclzip/pclzip", + "version": "2.8.2", + "version_normalized": "2.8.2.0", + "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": "" + }, + "time": "2014-06-05 11:42:24", + "type": "library", + "installation-source": "dist", + "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" + ] + }, + { + "name": "zoujingli/wechat-php-sdk", + "version": "dev-master", + "version_normalized": "9999999-dev", + "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" + }, + "time": "2017-02-07 02:34:38", + "type": "project", + "installation-source": "dist", + "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" + ] + }, + { + "name": "qiniu/php-sdk", + "version": "v7.1.3", + "version_normalized": "7.1.3.0", + "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" + }, + "time": "2016-11-18 02:57:31", + "type": "library", + "installation-source": "dist", + "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" + ] + }, + { + "name": "aferrandini/phpqrcode", + "version": "1.0.1", + "version_normalized": "1.0.1.0", + "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" + }, + "time": "2013-07-08 09:39:08", + "type": "library", + "installation-source": "dist", + "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" + ] + }, + { + "name": "topthink/framework", + "version": "v5.0.5", + "version_normalized": "5.0.5.0", + "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.*" + }, + "time": "2017-01-23 05:59:21", + "type": "think-framework", + "installation-source": "dist", + "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" + ] + }, + { + "name": "topthink/think-captcha", + "version": "v1.0.7", + "version_normalized": "1.0.7.0", + "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": "" + }, + "time": "2016-07-06 01:47:11", + "type": "library", + "installation-source": "dist", + "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" + }, + { + "name": "topthink/think-mongo", + "version": "v1.5", + "version_normalized": "1.5.0.0", + "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": "" + }, + "time": "2017-02-06 06:05:55", + "type": "library", + "installation-source": "dist", + "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" + }, + { + "name": "workerman/workerman", + "version": "v3.3.7", + "version_normalized": "3.3.7.0", + "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." + }, + "time": "2017-02-02 02:52:58", + "type": "library", + "installation-source": "dist", + "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" + ] + }, + { + "name": "topthink/think-worker", + "version": "v1.0.1", + "version_normalized": "1.0.1.0", + "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" + }, + "time": "2016-10-08 06:07:03", + "type": "library", + "installation-source": "dist", + "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" + }, + { + "name": "topthink/think-helper", + "version": "v1.0.5", + "version_normalized": "1.0.5.0", + "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": "" + }, + "time": "2016-12-01 07:08:40", + "type": "library", + "installation-source": "dist", + "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" + }, + { + "name": "topthink/think-queue", + "version": "v1.1.2", + "version_normalized": "1.1.2.0", + "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" + }, + "time": "2016-12-01 04:29:39", + "type": "think-extend", + "extra": { + "think-config": { + "queue": "src/config.php" + } + }, + "installation-source": "dist", + "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" + } +] diff --git a/vendor/pclzip/pclzip/composer.json b/vendor/pclzip/pclzip/composer.json new file mode 100644 index 000000000..a4df7fb6e --- /dev/null +++ b/vendor/pclzip/pclzip/composer.json @@ -0,0 +1,16 @@ +{ + "name": "pclzip/pclzip", + "type": "library", + "description": "A PHP library that offers compression and extraction functions for Zip formatted archives", + "keywords": ["php", "zip"], + "homepage": "http://www.phpconcept.net/pclzip", + "license": ["LGPL-2.1"], + "authors": [ + { + "name": "Vincent Blavet" + } + ], + "autoload": { + "classmap": ["pclzip.lib.php"] + } +} diff --git a/vendor/pclzip/pclzip/gnu-lgpl.txt b/vendor/pclzip/pclzip/gnu-lgpl.txt new file mode 100644 index 000000000..583509c7e --- /dev/null +++ b/vendor/pclzip/pclzip/gnu-lgpl.txt @@ -0,0 +1,504 @@ + GNU LESSER GENERAL PUBLIC LICENSE + Version 2.1, February 1999 + + Copyright (C) 1991, 1999 Free Software Foundation, Inc. + 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + +[This is the first released version of the Lesser GPL. It also counts + as the successor of the GNU Library Public License, version 2, hence + the version number 2.1.] + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +Licenses are intended to guarantee your freedom to share and change +free software--to make sure the software is free for all its users. + + This license, the Lesser General Public License, applies to some +specially designated software packages--typically libraries--of the +Free Software Foundation and other authors who decide to use it. You +can use it too, but we suggest you first think carefully about whether +this license or the ordinary General Public License is the better +strategy to use in any particular case, based on the explanations below. + + When we speak of free software, we are referring to freedom of use, +not price. Our General Public Licenses are designed to make sure that +you have the freedom to distribute copies of free software (and charge +for this service if you wish); that you receive source code or can get +it if you want it; that you can change the software and use pieces of +it in new free programs; and that you are informed that you can do +these things. + + To protect your rights, we need to make restrictions that forbid +distributors to deny you these rights or to ask you to surrender these +rights. These restrictions translate to certain responsibilities for +you if you distribute copies of the library or if you modify it. + + For example, if you distribute copies of the library, whether gratis +or for a fee, you must give the recipients all the rights that we gave +you. You must make sure that they, too, receive or can get the source +code. If you link other code with the library, you must provide +complete object files to the recipients, so that they can relink them +with the library after making changes to the library and recompiling +it. And you must show them these terms so they know their rights. + + We protect your rights with a two-step method: (1) we copyright the +library, and (2) we offer you this license, which gives you legal +permission to copy, distribute and/or modify the library. + + To protect each distributor, we want to make it very clear that +there is no warranty for the free library. Also, if the library is +modified by someone else and passed on, the recipients should know +that what they have is not the original version, so that the original +author's reputation will not be affected by problems that might be +introduced by others. + + Finally, software patents pose a constant threat to the existence of +any free program. We wish to make sure that a company cannot +effectively restrict the users of a free program by obtaining a +restrictive license from a patent holder. Therefore, we insist that +any patent license obtained for a version of the library must be +consistent with the full freedom of use specified in this license. + + Most GNU software, including some libraries, is covered by the +ordinary GNU General Public License. This license, the GNU Lesser +General Public License, applies to certain designated libraries, and +is quite different from the ordinary General Public License. We use +this license for certain libraries in order to permit linking those +libraries into non-free programs. + + When a program is linked with a library, whether statically or using +a shared library, the combination of the two is legally speaking a +combined work, a derivative of the original library. The ordinary +General Public License therefore permits such linking only if the +entire combination fits its criteria of freedom. The Lesser General +Public License permits more lax criteria for linking other code with +the library. + + We call this license the "Lesser" General Public License because it +does Less to protect the user's freedom than the ordinary General +Public License. It also provides other free software developers Less +of an advantage over competing non-free programs. These disadvantages +are the reason we use the ordinary General Public License for many +libraries. However, the Lesser license provides advantages in certain +special circumstances. + + For example, on rare occasions, there may be a special need to +encourage the widest possible use of a certain library, so that it becomes +a de-facto standard. To achieve this, non-free programs must be +allowed to use the library. A more frequent case is that a free +library does the same job as widely used non-free libraries. In this +case, there is little to gain by limiting the free library to free +software only, so we use the Lesser General Public License. + + In other cases, permission to use a particular library in non-free +programs enables a greater number of people to use a large body of +free software. For example, permission to use the GNU C Library in +non-free programs enables many more people to use the whole GNU +operating system, as well as its variant, the GNU/Linux operating +system. + + Although the Lesser General Public License is Less protective of the +users' freedom, it does ensure that the user of a program that is +linked with the Library has the freedom and the wherewithal to run +that program using a modified version of the Library. + + The precise terms and conditions for copying, distribution and +modification follow. Pay close attention to the difference between a +"work based on the library" and a "work that uses the library". The +former contains code derived from the library, whereas the latter must +be combined with the library in order to run. + + GNU LESSER GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License Agreement applies to any software library or other +program which contains a notice placed by the copyright holder or +other authorized party saying it may be distributed under the terms of +this Lesser General Public License (also called "this License"). +Each licensee is addressed as "you". + + A "library" means a collection of software functions and/or data +prepared so as to be conveniently linked with application programs +(which use some of those functions and data) to form executables. + + The "Library", below, refers to any such software library or work +which has been distributed under these terms. A "work based on the +Library" means either the Library or any derivative work under +copyright law: that is to say, a work containing the Library or a +portion of it, either verbatim or with modifications and/or translated +straightforwardly into another language. (Hereinafter, translation is +included without limitation in the term "modification".) + + "Source code" for a work means the preferred form of the work for +making modifications to it. For a library, complete source code means +all the source code for all modules it contains, plus any associated +interface definition files, plus the scripts used to control compilation +and installation of the library. + + Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running a program using the Library is not restricted, and output from +such a program is covered only if its contents constitute a work based +on the Library (independent of the use of the Library in a tool for +writing it). Whether that is true depends on what the Library does +and what the program that uses the Library does. + + 1. You may copy and distribute verbatim copies of the Library's +complete source code as you receive it, in any medium, provided that +you conspicuously and appropriately publish on each copy an +appropriate copyright notice and disclaimer of warranty; keep intact +all the notices that refer to this License and to the absence of any +warranty; and distribute a copy of this License along with the +Library. + + You may charge a fee for the physical act of transferring a copy, +and you may at your option offer warranty protection in exchange for a +fee. + + 2. You may modify your copy or copies of the Library or any portion +of it, thus forming a work based on the Library, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) The modified work must itself be a software library. + + b) You must cause the files modified to carry prominent notices + stating that you changed the files and the date of any change. + + c) You must cause the whole of the work to be licensed at no + charge to all third parties under the terms of this License. + + d) If a facility in the modified Library refers to a function or a + table of data to be supplied by an application program that uses + the facility, other than as an argument passed when the facility + is invoked, then you must make a good faith effort to ensure that, + in the event an application does not supply such function or + table, the facility still operates, and performs whatever part of + its purpose remains meaningful. + + (For example, a function in a library to compute square roots has + a purpose that is entirely well-defined independent of the + application. Therefore, Subsection 2d requires that any + application-supplied function or table used by this function must + be optional: if the application does not supply it, the square + root function must still compute square roots.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Library, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Library, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote +it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Library. + +In addition, mere aggregation of another work not based on the Library +with the Library (or with a work based on the Library) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may opt to apply the terms of the ordinary GNU General Public +License instead of this License to a given copy of the Library. To do +this, you must alter all the notices that refer to this License, so +that they refer to the ordinary GNU General Public License, version 2, +instead of to this License. (If a newer version than version 2 of the +ordinary GNU General Public License has appeared, then you can specify +that version instead if you wish.) Do not make any other change in +these notices. + + Once this change is made in a given copy, it is irreversible for +that copy, so the ordinary GNU General Public License applies to all +subsequent copies and derivative works made from that copy. + + This option is useful when you wish to copy part of the code of +the Library into a program that is not a library. + + 4. You may copy and distribute the Library (or a portion or +derivative of it, under Section 2) in object code or executable form +under the terms of Sections 1 and 2 above provided that you accompany +it with the complete corresponding machine-readable source code, which +must be distributed under the terms of Sections 1 and 2 above on a +medium customarily used for software interchange. + + If distribution of object code is made by offering access to copy +from a designated place, then offering equivalent access to copy the +source code from the same place satisfies the requirement to +distribute the source code, even though third parties are not +compelled to copy the source along with the object code. + + 5. A program that contains no derivative of any portion of the +Library, but is designed to work with the Library by being compiled or +linked with it, is called a "work that uses the Library". Such a +work, in isolation, is not a derivative work of the Library, and +therefore falls outside the scope of this License. + + However, linking a "work that uses the Library" with the Library +creates an executable that is a derivative of the Library (because it +contains portions of the Library), rather than a "work that uses the +library". The executable is therefore covered by this License. +Section 6 states terms for distribution of such executables. + + When a "work that uses the Library" uses material from a header file +that is part of the Library, the object code for the work may be a +derivative work of the Library even though the source code is not. +Whether this is true is especially significant if the work can be +linked without the Library, or if the work is itself a library. The +threshold for this to be true is not precisely defined by law. + + If such an object file uses only numerical parameters, data +structure layouts and accessors, and small macros and small inline +functions (ten lines or less in length), then the use of the object +file is unrestricted, regardless of whether it is legally a derivative +work. (Executables containing this object code plus portions of the +Library will still fall under Section 6.) + + Otherwise, if the work is a derivative of the Library, you may +distribute the object code for the work under the terms of Section 6. +Any executables containing that work also fall under Section 6, +whether or not they are linked directly with the Library itself. + + 6. As an exception to the Sections above, you may also combine or +link a "work that uses the Library" with the Library to produce a +work containing portions of the Library, and distribute that work +under terms of your choice, provided that the terms permit +modification of the work for the customer's own use and reverse +engineering for debugging such modifications. + + You must give prominent notice with each copy of the work that the +Library is used in it and that the Library and its use are covered by +this License. You must supply a copy of this License. If the work +during execution displays copyright notices, you must include the +copyright notice for the Library among them, as well as a reference +directing the user to the copy of this License. Also, you must do one +of these things: + + a) Accompany the work with the complete corresponding + machine-readable source code for the Library including whatever + changes were used in the work (which must be distributed under + Sections 1 and 2 above); and, if the work is an executable linked + with the Library, with the complete machine-readable "work that + uses the Library", as object code and/or source code, so that the + user can modify the Library and then relink to produce a modified + executable containing the modified Library. (It is understood + that the user who changes the contents of definitions files in the + Library will not necessarily be able to recompile the application + to use the modified definitions.) + + b) Use a suitable shared library mechanism for linking with the + Library. A suitable mechanism is one that (1) uses at run time a + copy of the library already present on the user's computer system, + rather than copying library functions into the executable, and (2) + will operate properly with a modified version of the library, if + the user installs one, as long as the modified version is + interface-compatible with the version that the work was made with. + + c) Accompany the work with a written offer, valid for at + least three years, to give the same user the materials + specified in Subsection 6a, above, for a charge no more + than the cost of performing this distribution. + + d) If distribution of the work is made by offering access to copy + from a designated place, offer equivalent access to copy the above + specified materials from the same place. + + e) Verify that the user has already received a copy of these + materials or that you have already sent this user a copy. + + For an executable, the required form of the "work that uses the +Library" must include any data and utility programs needed for +reproducing the executable from it. However, as a special exception, +the materials to be distributed need not include anything that is +normally distributed (in either source or binary form) with the major +components (compiler, kernel, and so on) of the operating system on +which the executable runs, unless that component itself accompanies +the executable. + + It may happen that this requirement contradicts the license +restrictions of other proprietary libraries that do not normally +accompany the operating system. Such a contradiction means you cannot +use both them and the Library together in an executable that you +distribute. + + 7. You may place library facilities that are a work based on the +Library side-by-side in a single library together with other library +facilities not covered by this License, and distribute such a combined +library, provided that the separate distribution of the work based on +the Library and of the other library facilities is otherwise +permitted, and provided that you do these two things: + + a) Accompany the combined library with a copy of the same work + based on the Library, uncombined with any other library + facilities. This must be distributed under the terms of the + Sections above. + + b) Give prominent notice with the combined library of the fact + that part of it is a work based on the Library, and explaining + where to find the accompanying uncombined form of the same work. + + 8. You may not copy, modify, sublicense, link with, or distribute +the Library except as expressly provided under this License. Any +attempt otherwise to copy, modify, sublicense, link with, or +distribute the Library is void, and will automatically terminate your +rights under this License. However, parties who have received copies, +or rights, from you under this License will not have their licenses +terminated so long as such parties remain in full compliance. + + 9. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Library or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Library (or any work based on the +Library), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Library or works based on it. + + 10. Each time you redistribute the Library (or any work based on the +Library), the recipient automatically receives a license from the +original licensor to copy, distribute, link with or modify the Library +subject to these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties with +this License. + + 11. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Library at all. For example, if a patent +license would not permit royalty-free redistribution of the Library by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Library. + +If any portion of this section is held invalid or unenforceable under any +particular circumstance, the balance of the section is intended to apply, +and the section as a whole is intended to apply in other circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 12. If the distribution and/or use of the Library is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Library under this License may add +an explicit geographical distribution limitation excluding those countries, +so that distribution is permitted only in or among countries not thus +excluded. In such case, this License incorporates the limitation as if +written in the body of this License. + + 13. The Free Software Foundation may publish revised and/or new +versions of the Lesser General Public License from time to time. +Such new versions will be similar in spirit to the present version, +but may differ in detail to address new problems or concerns. + +Each version is given a distinguishing version number. If the Library +specifies a version number of this License which applies to it and +"any later version", you have the option of following the terms and +conditions either of that version or of any later version published by +the Free Software Foundation. If the Library does not specify a +license version number, you may choose any version ever published by +the Free Software Foundation. + + 14. If you wish to incorporate parts of the Library into other free +programs whose distribution conditions are incompatible with these, +write to the author to ask for permission. For software which is +copyrighted by the Free Software Foundation, write to the Free +Software Foundation; we sometimes make exceptions for this. Our +decision will be guided by the two goals of preserving the free status +of all derivatives of our free software and of promoting the sharing +and reuse of software generally. + + NO WARRANTY + + 15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO +WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW. +EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR +OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY +KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE +LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME +THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN +WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY +AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU +FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR +CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE +LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING +RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A +FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF +SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH +DAMAGES. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Libraries + + If you develop a new library, and you want it to be of the greatest +possible use to the public, we recommend making it free software that +everyone can redistribute and change. You can do so by permitting +redistribution under these terms (or, alternatively, under the terms of the +ordinary General Public License). + + To apply these terms, attach the following notices to the library. It is +safest to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least the +"copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, write to the Free Software + Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + +Also add information on how to contact you by electronic and paper mail. + +You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the library, if +necessary. Here is a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the + library `Frob' (a library for tweaking knobs) written by James Random Hacker. + + , 1 April 1990 + Ty Coon, President of Vice + +That's all there is to it! + + diff --git a/vendor/pclzip/pclzip/pclzip.lib.php b/vendor/pclzip/pclzip/pclzip.lib.php new file mode 100644 index 000000000..33be0d786 --- /dev/null +++ b/vendor/pclzip/pclzip/pclzip.lib.php @@ -0,0 +1,5414 @@ +zipname = $p_zipname; + $this->zip_fd = 0; + $this->magic_quotes_status = -1; + + // ----- Return + return; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : + // create($p_filelist, $p_add_dir="", $p_remove_dir="") + // create($p_filelist, $p_option, $p_option_value, ...) + // Description : + // This method supports two different synopsis. The first one is historical. + // This method creates a Zip Archive. The Zip file is created in the + // filesystem. The files and directories indicated in $p_filelist + // are added in the archive. See the parameters description for the + // supported format of $p_filelist. + // When a directory is in the list, the directory and its content is added + // in the archive. + // In this synopsis, the function takes an optional variable list of + // options. See bellow the supported options. + // Parameters : + // $p_filelist : An array containing file or directory names, or + // a string containing one filename or one directory name, or + // a string containing a list of filenames and/or directory + // names separated by spaces. + // $p_add_dir : A path to add before the real path of the archived file, + // in order to have it memorized in the archive. + // $p_remove_dir : A path to remove from the real path of the file to archive, + // in order to have a shorter path memorized in the archive. + // When $p_add_dir and $p_remove_dir are set, $p_remove_dir + // is removed first, before $p_add_dir is added. + // Options : + // PCLZIP_OPT_ADD_PATH : + // PCLZIP_OPT_REMOVE_PATH : + // PCLZIP_OPT_REMOVE_ALL_PATH : + // PCLZIP_OPT_COMMENT : + // PCLZIP_CB_PRE_ADD : + // PCLZIP_CB_POST_ADD : + // Return Values : + // 0 on failure, + // The list of the added files, with a status of the add action. + // (see PclZip::listContent() for list entry format) + // -------------------------------------------------------------------------------- + public function create($p_filelist) + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Set default values + $v_options = array(); + $v_options[PCLZIP_OPT_NO_COMPRESSION] = false; + + // ----- Look for variable options arguments + $v_size = func_num_args(); + + // ----- Look for arguments + if ($v_size > 1) { + // ----- Get the arguments + $v_arg_list = func_get_args(); + + // ----- Remove from the options list the first argument + array_shift($v_arg_list); + $v_size--; + + // ----- Look for first arg + if ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) { + + // ----- Parse the options + $v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options, array( + PCLZIP_OPT_REMOVE_PATH => 'optional', + PCLZIP_OPT_REMOVE_ALL_PATH => 'optional', + PCLZIP_OPT_ADD_PATH => 'optional', + PCLZIP_CB_PRE_ADD => 'optional', + PCLZIP_CB_POST_ADD => 'optional', + PCLZIP_OPT_NO_COMPRESSION => 'optional', + PCLZIP_OPT_COMMENT => 'optional', + PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional', + PCLZIP_OPT_TEMP_FILE_ON => 'optional', + PCLZIP_OPT_TEMP_FILE_OFF => 'optional' + //, PCLZIP_OPT_CRYPT => 'optional' + )); + if ($v_result != 1) { + return 0; + } + + // ----- Look for 2 args + // Here we need to support the first historic synopsis of the + // method. + } else { + + // ----- Get the first argument + $v_options[PCLZIP_OPT_ADD_PATH] = $v_arg_list[0]; + + // ----- Look for the optional second argument + if ($v_size == 2) { + $v_options[PCLZIP_OPT_REMOVE_PATH] = $v_arg_list[1]; + } elseif ($v_size > 2) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments"); + + return 0; + } + } + } + + // ----- Look for default option values + $this->privOptionDefaultThreshold($v_options); + + // ----- Init + $v_string_list = array(); + $v_att_list = array(); + $v_filedescr_list = array(); + $p_result_list = array(); + + // ----- Look if the $p_filelist is really an array + if (is_array($p_filelist)) { + + // ----- Look if the first element is also an array + // This will mean that this is a file description entry + if (isset($p_filelist[0]) && is_array($p_filelist[0])) { + $v_att_list = $p_filelist; + + // ----- The list is a list of string names + } else { + $v_string_list = $p_filelist; + } + + // ----- Look if the $p_filelist is a string + } elseif (is_string($p_filelist)) { + // ----- Create a list from the string + $v_string_list = explode(PCLZIP_SEPARATOR, $p_filelist); + + // ----- Invalid variable type for $p_filelist + } else { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type p_filelist"); + + return 0; + } + + // ----- Reformat the string list + if (sizeof($v_string_list) != 0) { + foreach ($v_string_list as $v_string) { + if ($v_string != '') { + $v_att_list[][PCLZIP_ATT_FILE_NAME] = $v_string; + } else { + } + } + } + + // ----- For each file in the list check the attributes + $v_supported_attributes = array( + PCLZIP_ATT_FILE_NAME => 'mandatory', + PCLZIP_ATT_FILE_NEW_SHORT_NAME => 'optional', + PCLZIP_ATT_FILE_NEW_FULL_NAME => 'optional', + PCLZIP_ATT_FILE_MTIME => 'optional', + PCLZIP_ATT_FILE_CONTENT => 'optional', + PCLZIP_ATT_FILE_COMMENT => 'optional' + ); + foreach ($v_att_list as $v_entry) { + $v_result = $this->privFileDescrParseAtt($v_entry, $v_filedescr_list[], $v_options, $v_supported_attributes); + if ($v_result != 1) { + return 0; + } + } + + // ----- Expand the filelist (expand directories) + $v_result = $this->privFileDescrExpand($v_filedescr_list, $v_options); + if ($v_result != 1) { + return 0; + } + + // ----- Call the create fct + $v_result = $this->privCreate($v_filedescr_list, $p_result_list, $v_options); + if ($v_result != 1) { + return 0; + } + + // ----- Return + return $p_result_list; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : + // add($p_filelist, $p_add_dir="", $p_remove_dir="") + // add($p_filelist, $p_option, $p_option_value, ...) + // Description : + // This method supports two synopsis. The first one is historical. + // This methods add the list of files in an existing archive. + // If a file with the same name already exists, it is added at the end of the + // archive, the first one is still present. + // If the archive does not exist, it is created. + // Parameters : + // $p_filelist : An array containing file or directory names, or + // a string containing one filename or one directory name, or + // a string containing a list of filenames and/or directory + // names separated by spaces. + // $p_add_dir : A path to add before the real path of the archived file, + // in order to have it memorized in the archive. + // $p_remove_dir : A path to remove from the real path of the file to archive, + // in order to have a shorter path memorized in the archive. + // When $p_add_dir and $p_remove_dir are set, $p_remove_dir + // is removed first, before $p_add_dir is added. + // Options : + // PCLZIP_OPT_ADD_PATH : + // PCLZIP_OPT_REMOVE_PATH : + // PCLZIP_OPT_REMOVE_ALL_PATH : + // PCLZIP_OPT_COMMENT : + // PCLZIP_OPT_ADD_COMMENT : + // PCLZIP_OPT_PREPEND_COMMENT : + // PCLZIP_CB_PRE_ADD : + // PCLZIP_CB_POST_ADD : + // Return Values : + // 0 on failure, + // The list of the added files, with a status of the add action. + // (see PclZip::listContent() for list entry format) + // -------------------------------------------------------------------------------- + public function add($p_filelist) + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Set default values + $v_options = array(); + $v_options[PCLZIP_OPT_NO_COMPRESSION] = false; + + // ----- Look for variable options arguments + $v_size = func_num_args(); + + // ----- Look for arguments + if ($v_size > 1) { + // ----- Get the arguments + $v_arg_list = func_get_args(); + + // ----- Remove form the options list the first argument + array_shift($v_arg_list); + $v_size--; + + // ----- Look for first arg + if ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) { + + // ----- Parse the options + $v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options, array( + PCLZIP_OPT_REMOVE_PATH => 'optional', + PCLZIP_OPT_REMOVE_ALL_PATH => 'optional', + PCLZIP_OPT_ADD_PATH => 'optional', + PCLZIP_CB_PRE_ADD => 'optional', + PCLZIP_CB_POST_ADD => 'optional', + PCLZIP_OPT_NO_COMPRESSION => 'optional', + PCLZIP_OPT_COMMENT => 'optional', + PCLZIP_OPT_ADD_COMMENT => 'optional', + PCLZIP_OPT_PREPEND_COMMENT => 'optional', + PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional', + PCLZIP_OPT_TEMP_FILE_ON => 'optional', + PCLZIP_OPT_TEMP_FILE_OFF => 'optional' + //, PCLZIP_OPT_CRYPT => 'optional' + )); + if ($v_result != 1) { + return 0; + } + + // ----- Look for 2 args + // Here we need to support the first historic synopsis of the + // method. + } else { + + // ----- Get the first argument + $v_options[PCLZIP_OPT_ADD_PATH] = $v_add_path = $v_arg_list[0]; + + // ----- Look for the optional second argument + if ($v_size == 2) { + $v_options[PCLZIP_OPT_REMOVE_PATH] = $v_arg_list[1]; + } elseif ($v_size > 2) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments"); + + // ----- Return + return 0; + } + } + } + + // ----- Look for default option values + $this->privOptionDefaultThreshold($v_options); + + // ----- Init + $v_string_list = array(); + $v_att_list = array(); + $v_filedescr_list = array(); + $p_result_list = array(); + + // ----- Look if the $p_filelist is really an array + if (is_array($p_filelist)) { + + // ----- Look if the first element is also an array + // This will mean that this is a file description entry + if (isset($p_filelist[0]) && is_array($p_filelist[0])) { + $v_att_list = $p_filelist; + + // ----- The list is a list of string names + } else { + $v_string_list = $p_filelist; + } + + // ----- Look if the $p_filelist is a string + } elseif (is_string($p_filelist)) { + // ----- Create a list from the string + $v_string_list = explode(PCLZIP_SEPARATOR, $p_filelist); + + // ----- Invalid variable type for $p_filelist + } else { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type '" . gettype($p_filelist) . "' for p_filelist"); + + return 0; + } + + // ----- Reformat the string list + if (sizeof($v_string_list) != 0) { + foreach ($v_string_list as $v_string) { + $v_att_list[][PCLZIP_ATT_FILE_NAME] = $v_string; + } + } + + // ----- For each file in the list check the attributes + $v_supported_attributes = array( + PCLZIP_ATT_FILE_NAME => 'mandatory', + PCLZIP_ATT_FILE_NEW_SHORT_NAME => 'optional', + PCLZIP_ATT_FILE_NEW_FULL_NAME => 'optional', + PCLZIP_ATT_FILE_MTIME => 'optional', + PCLZIP_ATT_FILE_CONTENT => 'optional', + PCLZIP_ATT_FILE_COMMENT => 'optional' + ); + foreach ($v_att_list as $v_entry) { + $v_result = $this->privFileDescrParseAtt($v_entry, $v_filedescr_list[], $v_options, $v_supported_attributes); + if ($v_result != 1) { + return 0; + } + } + + // ----- Expand the filelist (expand directories) + $v_result = $this->privFileDescrExpand($v_filedescr_list, $v_options); + if ($v_result != 1) { + return 0; + } + + // ----- Call the create fct + $v_result = $this->privAdd($v_filedescr_list, $p_result_list, $v_options); + if ($v_result != 1) { + return 0; + } + + // ----- Return + return $p_result_list; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : listContent() + // Description : + // This public method, gives the list of the files and directories, with their + // properties. + // The properties of each entries in the list are (used also in other functions) : + // filename : Name of the file. For a create or add action it is the filename + // given by the user. For an extract function it is the filename + // of the extracted file. + // stored_filename : Name of the file / directory stored in the archive. + // size : Size of the stored file. + // compressed_size : Size of the file's data compressed in the archive + // (without the headers overhead) + // mtime : Last known modification date of the file (UNIX timestamp) + // comment : Comment associated with the file + // folder : true | false + // index : index of the file in the archive + // status : status of the action (depending of the action) : + // Values are : + // ok : OK ! + // filtered : the file / dir is not extracted (filtered by user) + // already_a_directory : the file can not be extracted because a + // directory with the same name already exists + // write_protected : the file can not be extracted because a file + // with the same name already exists and is + // write protected + // newer_exist : the file was not extracted because a newer file exists + // path_creation_fail : the file is not extracted because the folder + // does not exist and can not be created + // write_error : the file was not extracted because there was a + // error while writing the file + // read_error : the file was not extracted because there was a error + // while reading the file + // invalid_header : the file was not extracted because of an archive + // format error (bad file header) + // Note that each time a method can continue operating when there + // is an action error on a file, the error is only logged in the file status. + // Return Values : + // 0 on an unrecoverable failure, + // The list of the files in the archive. + // -------------------------------------------------------------------------------- + public function listContent() + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Check archive + if (!$this->privCheckFormat()) { + return (0); + } + + // ----- Call the extracting fct + $p_list = array(); + if (($v_result = $this->privList($p_list)) != 1) { + unset($p_list); + + return (0); + } + + // ----- Return + return $p_list; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : + // extract($p_path="./", $p_remove_path="") + // extract([$p_option, $p_option_value, ...]) + // Description : + // This method supports two synopsis. The first one is historical. + // This method extract all the files / directories from the archive to the + // folder indicated in $p_path. + // If you want to ignore the 'root' part of path of the memorized files + // you can indicate this in the optional $p_remove_path parameter. + // By default, if a newer file with the same name already exists, the + // file is not extracted. + // + // If both PCLZIP_OPT_PATH and PCLZIP_OPT_ADD_PATH aoptions + // are used, the path indicated in PCLZIP_OPT_ADD_PATH is append + // at the end of the path value of PCLZIP_OPT_PATH. + // Parameters : + // $p_path : Path where the files and directories are to be extracted + // $p_remove_path : First part ('root' part) of the memorized path + // (if any similar) to remove while extracting. + // Options : + // PCLZIP_OPT_PATH : + // PCLZIP_OPT_ADD_PATH : + // PCLZIP_OPT_REMOVE_PATH : + // PCLZIP_OPT_REMOVE_ALL_PATH : + // PCLZIP_CB_PRE_EXTRACT : + // PCLZIP_CB_POST_EXTRACT : + // Return Values : + // 0 or a negative value on failure, + // The list of the extracted files, with a status of the action. + // (see PclZip::listContent() for list entry format) + // -------------------------------------------------------------------------------- + public function extract() + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Check archive + if (!$this->privCheckFormat()) { + return (0); + } + + // ----- Set default values + $v_options = array(); + // $v_path = "./"; + $v_path = ''; + $v_remove_path = ""; + $v_remove_all_path = false; + + // ----- Look for variable options arguments + $v_size = func_num_args(); + + // ----- Default values for option + $v_options[PCLZIP_OPT_EXTRACT_AS_STRING] = false; + + // ----- Look for arguments + if ($v_size > 0) { + // ----- Get the arguments + $v_arg_list = func_get_args(); + + // ----- Look for first arg + if ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) { + + // ----- Parse the options + $v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options, array( + PCLZIP_OPT_PATH => 'optional', + PCLZIP_OPT_REMOVE_PATH => 'optional', + PCLZIP_OPT_REMOVE_ALL_PATH => 'optional', + PCLZIP_OPT_ADD_PATH => 'optional', + PCLZIP_CB_PRE_EXTRACT => 'optional', + PCLZIP_CB_POST_EXTRACT => 'optional', + PCLZIP_OPT_SET_CHMOD => 'optional', + PCLZIP_OPT_BY_NAME => 'optional', + PCLZIP_OPT_BY_EREG => 'optional', + PCLZIP_OPT_BY_PREG => 'optional', + PCLZIP_OPT_BY_INDEX => 'optional', + PCLZIP_OPT_EXTRACT_AS_STRING => 'optional', + PCLZIP_OPT_EXTRACT_IN_OUTPUT => 'optional', + PCLZIP_OPT_REPLACE_NEWER => 'optional', + PCLZIP_OPT_STOP_ON_ERROR => 'optional', + PCLZIP_OPT_EXTRACT_DIR_RESTRICTION => 'optional', + PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional', + PCLZIP_OPT_TEMP_FILE_ON => 'optional', + PCLZIP_OPT_TEMP_FILE_OFF => 'optional' + )); + if ($v_result != 1) { + return 0; + } + + // ----- Set the arguments + if (isset($v_options[PCLZIP_OPT_PATH])) { + $v_path = $v_options[PCLZIP_OPT_PATH]; + } + if (isset($v_options[PCLZIP_OPT_REMOVE_PATH])) { + $v_remove_path = $v_options[PCLZIP_OPT_REMOVE_PATH]; + } + if (isset($v_options[PCLZIP_OPT_REMOVE_ALL_PATH])) { + $v_remove_all_path = $v_options[PCLZIP_OPT_REMOVE_ALL_PATH]; + } + if (isset($v_options[PCLZIP_OPT_ADD_PATH])) { + // ----- Check for '/' in last path char + if ((strlen($v_path) > 0) && (substr($v_path, -1) != '/')) { + $v_path .= '/'; + } + $v_path .= $v_options[PCLZIP_OPT_ADD_PATH]; + } + + // ----- Look for 2 args + // Here we need to support the first historic synopsis of the + // method. + } else { + + // ----- Get the first argument + $v_path = $v_arg_list[0]; + + // ----- Look for the optional second argument + if ($v_size == 2) { + $v_remove_path = $v_arg_list[1]; + } elseif ($v_size > 2) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments"); + + // ----- Return + return 0; + } + } + } + + // ----- Look for default option values + $this->privOptionDefaultThreshold($v_options); + + // ----- Trace + + // ----- Call the extracting fct + $p_list = array(); + $v_result = $this->privExtractByRule($p_list, $v_path, $v_remove_path, $v_remove_all_path, $v_options); + if ($v_result < 1) { + unset($p_list); + + return (0); + } + + // ----- Return + return $p_list; + } + // -------------------------------------------------------------------------------- + + + // -------------------------------------------------------------------------------- + // Function : + // extractByIndex($p_index, $p_path="./", $p_remove_path="") + // extractByIndex($p_index, [$p_option, $p_option_value, ...]) + // Description : + // This method supports two synopsis. The first one is historical. + // This method is doing a partial extract of the archive. + // The extracted files or folders are identified by their index in the + // archive (from 0 to n). + // Note that if the index identify a folder, only the folder entry is + // extracted, not all the files included in the archive. + // Parameters : + // $p_index : A single index (integer) or a string of indexes of files to + // extract. The form of the string is "0,4-6,8-12" with only numbers + // and '-' for range or ',' to separate ranges. No spaces or ';' + // are allowed. + // $p_path : Path where the files and directories are to be extracted + // $p_remove_path : First part ('root' part) of the memorized path + // (if any similar) to remove while extracting. + // Options : + // PCLZIP_OPT_PATH : + // PCLZIP_OPT_ADD_PATH : + // PCLZIP_OPT_REMOVE_PATH : + // PCLZIP_OPT_REMOVE_ALL_PATH : + // PCLZIP_OPT_EXTRACT_AS_STRING : The files are extracted as strings and + // not as files. + // The resulting content is in a new field 'content' in the file + // structure. + // This option must be used alone (any other options are ignored). + // PCLZIP_CB_PRE_EXTRACT : + // PCLZIP_CB_POST_EXTRACT : + // Return Values : + // 0 on failure, + // The list of the extracted files, with a status of the action. + // (see PclZip::listContent() for list entry format) + // -------------------------------------------------------------------------------- + //function extractByIndex($p_index, options...) + public function extractByIndex($p_index) + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Check archive + if (!$this->privCheckFormat()) { + return (0); + } + + // ----- Set default values + $v_options = array(); + // $v_path = "./"; + $v_path = ''; + $v_remove_path = ""; + $v_remove_all_path = false; + + // ----- Look for variable options arguments + $v_size = func_num_args(); + + // ----- Default values for option + $v_options[PCLZIP_OPT_EXTRACT_AS_STRING] = false; + + // ----- Look for arguments + if ($v_size > 1) { + // ----- Get the arguments + $v_arg_list = func_get_args(); + + // ----- Remove form the options list the first argument + array_shift($v_arg_list); + $v_size--; + + // ----- Look for first arg + if ((is_integer($v_arg_list[0])) && ($v_arg_list[0] > 77000)) { + + // ----- Parse the options + $v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options, array( + PCLZIP_OPT_PATH => 'optional', + PCLZIP_OPT_REMOVE_PATH => 'optional', + PCLZIP_OPT_REMOVE_ALL_PATH => 'optional', + PCLZIP_OPT_EXTRACT_AS_STRING => 'optional', + PCLZIP_OPT_ADD_PATH => 'optional', + PCLZIP_CB_PRE_EXTRACT => 'optional', + PCLZIP_CB_POST_EXTRACT => 'optional', + PCLZIP_OPT_SET_CHMOD => 'optional', + PCLZIP_OPT_REPLACE_NEWER => 'optional', + PCLZIP_OPT_STOP_ON_ERROR => 'optional', + PCLZIP_OPT_EXTRACT_DIR_RESTRICTION => 'optional', + PCLZIP_OPT_TEMP_FILE_THRESHOLD => 'optional', + PCLZIP_OPT_TEMP_FILE_ON => 'optional', + PCLZIP_OPT_TEMP_FILE_OFF => 'optional' + )); + if ($v_result != 1) { + return 0; + } + + // ----- Set the arguments + if (isset($v_options[PCLZIP_OPT_PATH])) { + $v_path = $v_options[PCLZIP_OPT_PATH]; + } + if (isset($v_options[PCLZIP_OPT_REMOVE_PATH])) { + $v_remove_path = $v_options[PCLZIP_OPT_REMOVE_PATH]; + } + if (isset($v_options[PCLZIP_OPT_REMOVE_ALL_PATH])) { + $v_remove_all_path = $v_options[PCLZIP_OPT_REMOVE_ALL_PATH]; + } + if (isset($v_options[PCLZIP_OPT_ADD_PATH])) { + // ----- Check for '/' in last path char + if ((strlen($v_path) > 0) && (substr($v_path, -1) != '/')) { + $v_path .= '/'; + } + $v_path .= $v_options[PCLZIP_OPT_ADD_PATH]; + } + if (!isset($v_options[PCLZIP_OPT_EXTRACT_AS_STRING])) { + $v_options[PCLZIP_OPT_EXTRACT_AS_STRING] = false; + } else { + } + + // ----- Look for 2 args + // Here we need to support the first historic synopsis of the + // method. + } else { + + // ----- Get the first argument + $v_path = $v_arg_list[0]; + + // ----- Look for the optional second argument + if ($v_size == 2) { + $v_remove_path = $v_arg_list[1]; + } elseif ($v_size > 2) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid number / type of arguments"); + + // ----- Return + return 0; + } + } + } + + // ----- Trace + + // ----- Trick + // Here I want to reuse extractByRule(), so I need to parse the $p_index + // with privParseOptions() + $v_arg_trick = array( + PCLZIP_OPT_BY_INDEX, + $p_index + ); + $v_options_trick = array(); + $v_result = $this->privParseOptions($v_arg_trick, sizeof($v_arg_trick), $v_options_trick, array( + PCLZIP_OPT_BY_INDEX => 'optional' + )); + if ($v_result != 1) { + return 0; + } + $v_options[PCLZIP_OPT_BY_INDEX] = $v_options_trick[PCLZIP_OPT_BY_INDEX]; + + // ----- Look for default option values + $this->privOptionDefaultThreshold($v_options); + + // ----- Call the extracting fct + if (($v_result = $this->privExtractByRule($p_list, $v_path, $v_remove_path, $v_remove_all_path, $v_options)) < 1) { + return (0); + } + + // ----- Return + return $p_list; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : + // delete([$p_option, $p_option_value, ...]) + // Description : + // This method removes files from the archive. + // If no parameters are given, then all the archive is emptied. + // Parameters : + // None or optional arguments. + // Options : + // PCLZIP_OPT_BY_INDEX : + // PCLZIP_OPT_BY_NAME : + // PCLZIP_OPT_BY_EREG : + // PCLZIP_OPT_BY_PREG : + // Return Values : + // 0 on failure, + // The list of the files which are still present in the archive. + // (see PclZip::listContent() for list entry format) + // -------------------------------------------------------------------------------- + public function delete() + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Check archive + if (!$this->privCheckFormat()) { + return (0); + } + + // ----- Set default values + $v_options = array(); + + // ----- Look for variable options arguments + $v_size = func_num_args(); + + // ----- Look for arguments + if ($v_size > 0) { + // ----- Get the arguments + $v_arg_list = func_get_args(); + + // ----- Parse the options + $v_result = $this->privParseOptions($v_arg_list, $v_size, $v_options, array( + PCLZIP_OPT_BY_NAME => 'optional', + PCLZIP_OPT_BY_EREG => 'optional', + PCLZIP_OPT_BY_PREG => 'optional', + PCLZIP_OPT_BY_INDEX => 'optional' + )); + if ($v_result != 1) { + return 0; + } + } + + // ----- Magic quotes trick + $this->privDisableMagicQuotes(); + + // ----- Call the delete fct + $v_list = array(); + if (($v_result = $this->privDeleteByRule($v_list, $v_options)) != 1) { + $this->privSwapBackMagicQuotes(); + unset($v_list); + + return (0); + } + + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_list; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : deleteByIndex() + // Description : + // ***** Deprecated ***** + // delete(PCLZIP_OPT_BY_INDEX, $p_index) should be prefered. + // -------------------------------------------------------------------------------- + public function deleteByIndex($p_index) + { + + $p_list = $this->delete(PCLZIP_OPT_BY_INDEX, $p_index); + + // ----- Return + return $p_list; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : properties() + // Description : + // This method gives the properties of the archive. + // The properties are : + // nb : Number of files in the archive + // comment : Comment associated with the archive file + // status : not_exist, ok + // Parameters : + // None + // Return Values : + // 0 on failure, + // An array with the archive properties. + // -------------------------------------------------------------------------------- + public function properties() + { + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Magic quotes trick + $this->privDisableMagicQuotes(); + + // ----- Check archive + if (!$this->privCheckFormat()) { + $this->privSwapBackMagicQuotes(); + + return (0); + } + + // ----- Default properties + $v_prop = array(); + $v_prop['comment'] = ''; + $v_prop['nb'] = 0; + $v_prop['status'] = 'not_exist'; + + // ----- Look if file exists + if (@is_file($this->zipname)) { + // ----- Open the zip file + if (($this->zip_fd = @fopen($this->zipname, 'rb')) == 0) { + $this->privSwapBackMagicQuotes(); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive \'' . $this->zipname . '\' in binary read mode'); + + // ----- Return + return 0; + } + + // ----- Read the central directory informations + $v_central_dir = array(); + if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1) { + $this->privSwapBackMagicQuotes(); + + return 0; + } + + // ----- Close the zip file + $this->privCloseFd(); + + // ----- Set the user attributes + $v_prop['comment'] = $v_central_dir['comment']; + $v_prop['nb'] = $v_central_dir['entries']; + $v_prop['status'] = 'ok'; + } + + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_prop; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : duplicate() + // Description : + // This method creates an archive by copying the content of an other one. If + // the archive already exist, it is replaced by the new one without any warning. + // Parameters : + // $p_archive : The filename of a valid archive, or + // a valid PclZip object. + // Return Values : + // 1 on success. + // 0 or a negative value on error (error code). + // -------------------------------------------------------------------------------- + public function duplicate($p_archive) + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Look if the $p_archive is a PclZip object + if ((is_object($p_archive)) && (get_class($p_archive) == 'pclzip')) { + + // ----- Duplicate the archive + $v_result = $this->privDuplicate($p_archive->zipname); + + // ----- Look if the $p_archive is a string (so a filename) + } elseif (is_string($p_archive)) { + + // ----- Check that $p_archive is a valid zip file + // TBC : Should also check the archive format + if (!is_file($p_archive)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "No file with filename '" . $p_archive . "'"); + $v_result = PCLZIP_ERR_MISSING_FILE; + } else { + // ----- Duplicate the archive + $v_result = $this->privDuplicate($p_archive); + } + + // ----- Invalid variable + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type p_archive_to_add"); + $v_result = PCLZIP_ERR_INVALID_PARAMETER; + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : merge() + // Description : + // This method merge the $p_archive_to_add archive at the end of the current + // one ($this). + // If the archive ($this) does not exist, the merge becomes a duplicate. + // If the $p_archive_to_add archive does not exist, the merge is a success. + // Parameters : + // $p_archive_to_add : It can be directly the filename of a valid zip archive, + // or a PclZip object archive. + // Return Values : + // 1 on success, + // 0 or negative values on error (see below). + // -------------------------------------------------------------------------------- + public function merge($p_archive_to_add) + { + $v_result = 1; + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Check archive + if (!$this->privCheckFormat()) { + return (0); + } + + // ----- Look if the $p_archive_to_add is a PclZip object + if ((is_object($p_archive_to_add)) && (get_class($p_archive_to_add) == 'pclzip')) { + + // ----- Merge the archive + $v_result = $this->privMerge($p_archive_to_add); + + // ----- Look if the $p_archive_to_add is a string (so a filename) + } elseif (is_string($p_archive_to_add)) { + + // ----- Create a temporary archive + $v_object_archive = new PclZip($p_archive_to_add); + + // ----- Merge the archive + $v_result = $this->privMerge($v_object_archive); + + // ----- Invalid variable + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid variable type p_archive_to_add"); + $v_result = PCLZIP_ERR_INVALID_PARAMETER; + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : errorCode() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function errorCode() + { + if (PCLZIP_ERROR_EXTERNAL == 1) { + return (PclErrorCode()); + } else { + return ($this->error_code); + } + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : errorName() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function errorName($p_with_code = false) + { + $v_name = array( + PCLZIP_ERR_NO_ERROR => 'PCLZIP_ERR_NO_ERROR', + PCLZIP_ERR_WRITE_OPEN_FAIL => 'PCLZIP_ERR_WRITE_OPEN_FAIL', + PCLZIP_ERR_READ_OPEN_FAIL => 'PCLZIP_ERR_READ_OPEN_FAIL', + PCLZIP_ERR_INVALID_PARAMETER => 'PCLZIP_ERR_INVALID_PARAMETER', + PCLZIP_ERR_MISSING_FILE => 'PCLZIP_ERR_MISSING_FILE', + PCLZIP_ERR_FILENAME_TOO_LONG => 'PCLZIP_ERR_FILENAME_TOO_LONG', + PCLZIP_ERR_INVALID_ZIP => 'PCLZIP_ERR_INVALID_ZIP', + PCLZIP_ERR_BAD_EXTRACTED_FILE => 'PCLZIP_ERR_BAD_EXTRACTED_FILE', + PCLZIP_ERR_DIR_CREATE_FAIL => 'PCLZIP_ERR_DIR_CREATE_FAIL', + PCLZIP_ERR_BAD_EXTENSION => 'PCLZIP_ERR_BAD_EXTENSION', + PCLZIP_ERR_BAD_FORMAT => 'PCLZIP_ERR_BAD_FORMAT', + PCLZIP_ERR_DELETE_FILE_FAIL => 'PCLZIP_ERR_DELETE_FILE_FAIL', + PCLZIP_ERR_RENAME_FILE_FAIL => 'PCLZIP_ERR_RENAME_FILE_FAIL', + PCLZIP_ERR_BAD_CHECKSUM => 'PCLZIP_ERR_BAD_CHECKSUM', + PCLZIP_ERR_INVALID_ARCHIVE_ZIP => 'PCLZIP_ERR_INVALID_ARCHIVE_ZIP', + PCLZIP_ERR_MISSING_OPTION_VALUE => 'PCLZIP_ERR_MISSING_OPTION_VALUE', + PCLZIP_ERR_INVALID_OPTION_VALUE => 'PCLZIP_ERR_INVALID_OPTION_VALUE', + PCLZIP_ERR_UNSUPPORTED_COMPRESSION => 'PCLZIP_ERR_UNSUPPORTED_COMPRESSION', + PCLZIP_ERR_UNSUPPORTED_ENCRYPTION => 'PCLZIP_ERR_UNSUPPORTED_ENCRYPTION', + PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE => 'PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE', + PCLZIP_ERR_DIRECTORY_RESTRICTION => 'PCLZIP_ERR_DIRECTORY_RESTRICTION' + ); + + if (isset($v_name[$this->error_code])) { + $v_value = $v_name[$this->error_code]; + } else { + $v_value = 'NoName'; + } + + if ($p_with_code) { + return ($v_value . ' (' . $this->error_code . ')'); + } else { + return ($v_value); + } + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : errorInfo() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function errorInfo($p_full = false) + { + if (PCLZIP_ERROR_EXTERNAL == 1) { + return (PclErrorString()); + } else { + if ($p_full) { + return ($this->errorName(true) . " : " . $this->error_string); + } else { + return ($this->error_string . " [code " . $this->error_code . "]"); + } + } + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // ***** UNDER THIS LINE ARE DEFINED PRIVATE INTERNAL FUNCTIONS ***** + // ***** ***** + // ***** THESES FUNCTIONS MUST NOT BE USED DIRECTLY ***** + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privCheckFormat() + // Description : + // This method check that the archive exists and is a valid zip archive. + // Several level of check exists. (futur) + // Parameters : + // $p_level : Level of check. Default 0. + // 0 : Check the first bytes (magic codes) (default value)) + // 1 : 0 + Check the central directory (futur) + // 2 : 1 + Check each file header (futur) + // Return Values : + // true on success, + // false on error, the error code is set. + // -------------------------------------------------------------------------------- + public function privCheckFormat($p_level = 0) + { + $v_result = true; + + // ----- Reset the file system cache + clearstatcache(); + + // ----- Reset the error handler + $this->privErrorReset(); + + // ----- Look if the file exits + if (!is_file($this->zipname)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "Missing archive file '" . $this->zipname . "'"); + + return (false); + } + + // ----- Check that the file is readeable + if (!is_readable($this->zipname)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, "Unable to read archive '" . $this->zipname . "'"); + + return (false); + } + + // ----- Check the magic code + // TBC + + // ----- Check the central header + // TBC + + // ----- Check each file header + // TBC + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privParseOptions() + // Description : + // This internal methods reads the variable list of arguments ($p_options_list, + // $p_size) and generate an array with the options and values ($v_result_list). + // $v_requested_options contains the options that can be present and those that + // must be present. + // $v_requested_options is an array, with the option value as key, and 'optional', + // or 'mandatory' as value. + // Parameters : + // See above. + // Return Values : + // 1 on success. + // 0 on failure. + // -------------------------------------------------------------------------------- + public function privParseOptions(&$p_options_list, $p_size, &$v_result_list, $v_requested_options = false) + { + $v_result = 1; + + // ----- Read the options + $i = 0; + while ($i < $p_size) { + + // ----- Check if the option is supported + if (!isset($v_requested_options[$p_options_list[$i]])) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid optional parameter '" . $p_options_list[$i] . "' for this method"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Look for next option + switch ($p_options_list[$i]) { + // ----- Look for options that request a path value + case PCLZIP_OPT_PATH: + case PCLZIP_OPT_REMOVE_PATH: + case PCLZIP_OPT_ADD_PATH: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + $v_result_list[$p_options_list[$i]] = PclZipUtilTranslateWinPath($p_options_list[$i + 1], false); + $i++; + break; + + case PCLZIP_OPT_TEMP_FILE_THRESHOLD: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + return PclZip::errorCode(); + } + + // ----- Check for incompatible options + if (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_OFF])) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '" . PclZipUtilOptionText($p_options_list[$i]) . "' can not be used with option 'PCLZIP_OPT_TEMP_FILE_OFF'"); + + return PclZip::errorCode(); + } + + // ----- Check the value + $v_value = $p_options_list[$i + 1]; + if ((!is_integer($v_value)) || ($v_value < 0)) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Integer expected for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + return PclZip::errorCode(); + } + + // ----- Get the value (and convert it in bytes) + $v_result_list[$p_options_list[$i]] = $v_value * 1048576; + $i++; + break; + + case PCLZIP_OPT_TEMP_FILE_ON: + // ----- Check for incompatible options + if (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_OFF])) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '" . PclZipUtilOptionText($p_options_list[$i]) . "' can not be used with option 'PCLZIP_OPT_TEMP_FILE_OFF'"); + + return PclZip::errorCode(); + } + + $v_result_list[$p_options_list[$i]] = true; + break; + + case PCLZIP_OPT_TEMP_FILE_OFF: + // ----- Check for incompatible options + if (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_ON])) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '" . PclZipUtilOptionText($p_options_list[$i]) . "' can not be used with option 'PCLZIP_OPT_TEMP_FILE_ON'"); + + return PclZip::errorCode(); + } + // ----- Check for incompatible options + if (isset($v_result_list[PCLZIP_OPT_TEMP_FILE_THRESHOLD])) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Option '" . PclZipUtilOptionText($p_options_list[$i]) . "' can not be used with option 'PCLZIP_OPT_TEMP_FILE_THRESHOLD'"); + + return PclZip::errorCode(); + } + + $v_result_list[$p_options_list[$i]] = true; + break; + + case PCLZIP_OPT_EXTRACT_DIR_RESTRICTION: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + if (is_string($p_options_list[$i + 1]) && ($p_options_list[$i + 1] != '')) { + $v_result_list[$p_options_list[$i]] = PclZipUtilTranslateWinPath($p_options_list[$i + 1], false); + $i++; + } else { + } + break; + + // ----- Look for options that request an array of string for value + case PCLZIP_OPT_BY_NAME: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + if (is_string($p_options_list[$i + 1])) { + $v_result_list[$p_options_list[$i]][0] = $p_options_list[$i + 1]; + } elseif (is_array($p_options_list[$i + 1])) { + $v_result_list[$p_options_list[$i]] = $p_options_list[$i + 1]; + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Wrong parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + $i++; + break; + + // ----- Look for options that request an EREG or PREG expression + case PCLZIP_OPT_BY_EREG: + $p_options_list[$i] = PCLZIP_OPT_BY_PREG; + // ereg() is deprecated starting with PHP 5.3. Move PCLZIP_OPT_BY_EREG + // to PCLZIP_OPT_BY_PREG + case PCLZIP_OPT_BY_PREG: + //case PCLZIP_OPT_CRYPT : + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + if (is_string($p_options_list[$i + 1])) { + $v_result_list[$p_options_list[$i]] = $p_options_list[$i + 1]; + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Wrong parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + $i++; + break; + + // ----- Look for options that takes a string + case PCLZIP_OPT_COMMENT: + case PCLZIP_OPT_ADD_COMMENT: + case PCLZIP_OPT_PREPEND_COMMENT: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + if (is_string($p_options_list[$i + 1])) { + $v_result_list[$p_options_list[$i]] = $p_options_list[$i + 1]; + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Wrong parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + $i++; + break; + + // ----- Look for options that request an array of index + case PCLZIP_OPT_BY_INDEX: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + $v_work_list = array(); + if (is_string($p_options_list[$i + 1])) { + + // ----- Remove spaces + $p_options_list[$i + 1] = strtr($p_options_list[$i + 1], ' ', ''); + + // ----- Parse items + $v_work_list = explode(",", $p_options_list[$i + 1]); + } elseif (is_integer($p_options_list[$i + 1])) { + $v_work_list[0] = $p_options_list[$i + 1] . '-' . $p_options_list[$i + 1]; + } elseif (is_array($p_options_list[$i + 1])) { + $v_work_list = $p_options_list[$i + 1]; + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Value must be integer, string or array for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Reduce the index list + // each index item in the list must be a couple with a start and + // an end value : [0,3], [5-5], [8-10], ... + // ----- Check the format of each item + $v_sort_flag = false; + $v_sort_value = 0; + for ($j = 0; $j < sizeof($v_work_list); $j++) { + // ----- Explode the item + $v_item_list = explode("-", $v_work_list[$j]); + $v_size_item_list = sizeof($v_item_list); + + // ----- TBC : Here we might check that each item is a + // real integer ... + + // ----- Look for single value + if ($v_size_item_list == 1) { + // ----- Set the option value + $v_result_list[$p_options_list[$i]][$j]['start'] = $v_item_list[0]; + $v_result_list[$p_options_list[$i]][$j]['end'] = $v_item_list[0]; + } elseif ($v_size_item_list == 2) { + // ----- Set the option value + $v_result_list[$p_options_list[$i]][$j]['start'] = $v_item_list[0]; + $v_result_list[$p_options_list[$i]][$j]['end'] = $v_item_list[1]; + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Too many values in index range for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Look for list sort + if ($v_result_list[$p_options_list[$i]][$j]['start'] < $v_sort_value) { + $v_sort_flag = true; + + // ----- TBC : An automatic sort should be writen ... + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Invalid order of index range for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + $v_sort_value = $v_result_list[$p_options_list[$i]][$j]['start']; + } + + // ----- Sort the items + if ($v_sort_flag) { + // TBC : To Be Completed + } + + // ----- Next option + $i++; + break; + + // ----- Look for options that request no value + case PCLZIP_OPT_REMOVE_ALL_PATH: + case PCLZIP_OPT_EXTRACT_AS_STRING: + case PCLZIP_OPT_NO_COMPRESSION: + case PCLZIP_OPT_EXTRACT_IN_OUTPUT: + case PCLZIP_OPT_REPLACE_NEWER: + case PCLZIP_OPT_STOP_ON_ERROR: + $v_result_list[$p_options_list[$i]] = true; + break; + + // ----- Look for options that request an octal value + case PCLZIP_OPT_SET_CHMOD: + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + $v_result_list[$p_options_list[$i]] = $p_options_list[$i + 1]; + $i++; + break; + + // ----- Look for options that request a call-back + case PCLZIP_CB_PRE_EXTRACT: + case PCLZIP_CB_POST_EXTRACT: + case PCLZIP_CB_PRE_ADD: + case PCLZIP_CB_POST_ADD: + /* for futur use + case PCLZIP_CB_PRE_DELETE : + case PCLZIP_CB_POST_DELETE : + case PCLZIP_CB_PRE_LIST : + case PCLZIP_CB_POST_LIST : + */ + // ----- Check the number of parameters + if (($i + 1) >= $p_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_OPTION_VALUE, "Missing parameter value for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Get the value + $v_function_name = $p_options_list[$i + 1]; + + // ----- Check that the value is a valid existing function + if (!function_exists($v_function_name)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_OPTION_VALUE, "Function '" . $v_function_name . "()' is not an existing function for option '" . PclZipUtilOptionText($p_options_list[$i]) . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Set the attribute + $v_result_list[$p_options_list[$i]] = $v_function_name; + $i++; + break; + + default: + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Unknown parameter '" . $p_options_list[$i] . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Next options + $i++; + } + + // ----- Look for mandatory options + if ($v_requested_options !== false) { + for ($key = reset($v_requested_options); $key = key($v_requested_options); $key = next($v_requested_options)) { + // ----- Look for mandatory option + if ($v_requested_options[$key] == 'mandatory') { + // ----- Look if present + if (!isset($v_result_list[$key])) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Missing mandatory parameter " . PclZipUtilOptionText($key) . "(" . $key . ")"); + + // ----- Return + return PclZip::errorCode(); + } + } + } + } + + // ----- Look for default values + if (!isset($v_result_list[PCLZIP_OPT_TEMP_FILE_THRESHOLD])) { + + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privOptionDefaultThreshold() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privOptionDefaultThreshold(&$p_options) + { + $v_result = 1; + + if (isset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD]) || isset($p_options[PCLZIP_OPT_TEMP_FILE_OFF])) { + return $v_result; + } + + // ----- Get 'memory_limit' configuration value + $v_memory_limit = ini_get('memory_limit'); + $v_memory_limit = trim($v_memory_limit); + $last = strtolower(substr($v_memory_limit, -1)); + + if ($last == 'g') { + //$v_memory_limit = $v_memory_limit*1024*1024*1024; + $v_memory_limit = $v_memory_limit * 1073741824; + } + if ($last == 'm') { + //$v_memory_limit = $v_memory_limit*1024*1024; + $v_memory_limit = $v_memory_limit * 1048576; + } + if ($last == 'k') { + $v_memory_limit = $v_memory_limit * 1024; + } + + $p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] = floor($v_memory_limit * PCLZIP_TEMPORARY_FILE_RATIO); + + // ----- Sanity check : No threshold if value lower than 1M + if ($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] < 1048576) { + unset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD]); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privFileDescrParseAtt() + // Description : + // Parameters : + // Return Values : + // 1 on success. + // 0 on failure. + // -------------------------------------------------------------------------------- + public function privFileDescrParseAtt(&$p_file_list, &$p_filedescr, $v_options, $v_requested_options = false) + { + $v_result = 1; + + // ----- For each file in the list check the attributes + foreach ($p_file_list as $v_key => $v_value) { + + // ----- Check if the option is supported + if (!isset($v_requested_options[$v_key])) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid file attribute '" . $v_key . "' for this file"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Look for attribute + switch ($v_key) { + case PCLZIP_ATT_FILE_NAME: + if (!is_string($v_value)) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type " . gettype($v_value) . ". String expected for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + + $p_filedescr['filename'] = PclZipUtilPathReduction($v_value); + + if ($p_filedescr['filename'] == '') { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid empty filename for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + + break; + + case PCLZIP_ATT_FILE_NEW_SHORT_NAME: + if (!is_string($v_value)) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type " . gettype($v_value) . ". String expected for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + + $p_filedescr['new_short_name'] = PclZipUtilPathReduction($v_value); + + if ($p_filedescr['new_short_name'] == '') { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid empty short filename for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + break; + + case PCLZIP_ATT_FILE_NEW_FULL_NAME: + if (!is_string($v_value)) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type " . gettype($v_value) . ". String expected for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + + $p_filedescr['new_full_name'] = PclZipUtilPathReduction($v_value); + + if ($p_filedescr['new_full_name'] == '') { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid empty full filename for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + break; + + // ----- Look for options that takes a string + case PCLZIP_ATT_FILE_COMMENT: + if (!is_string($v_value)) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type " . gettype($v_value) . ". String expected for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + + $p_filedescr['comment'] = $v_value; + break; + + case PCLZIP_ATT_FILE_MTIME: + if (!is_integer($v_value)) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE, "Invalid type " . gettype($v_value) . ". Integer expected for attribute '" . PclZipUtilOptionText($v_key) . "'"); + + return PclZip::errorCode(); + } + + $p_filedescr['mtime'] = $v_value; + break; + + case PCLZIP_ATT_FILE_CONTENT: + $p_filedescr['content'] = $v_value; + break; + + default: + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Unknown parameter '" . $v_key . "'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Look for mandatory options + if ($v_requested_options !== false) { + for ($key = reset($v_requested_options); $key = key($v_requested_options); $key = next($v_requested_options)) { + // ----- Look for mandatory option + if ($v_requested_options[$key] == 'mandatory') { + // ----- Look if present + if (!isset($p_file_list[$key])) { + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Missing mandatory parameter " . PclZipUtilOptionText($key) . "(" . $key . ")"); + + return PclZip::errorCode(); + } + } + } + } + + // end foreach + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privFileDescrExpand() + // Description : + // This method look for each item of the list to see if its a file, a folder + // or a string to be added as file. For any other type of files (link, other) + // just ignore the item. + // Then prepare the information that will be stored for that file. + // When its a folder, expand the folder with all the files that are in that + // folder (recursively). + // Parameters : + // Return Values : + // 1 on success. + // 0 on failure. + // -------------------------------------------------------------------------------- + public function privFileDescrExpand(&$p_filedescr_list, &$p_options) + { + $v_result = 1; + + // ----- Create a result list + $v_result_list = array(); + + // ----- Look each entry + for ($i = 0; $i < sizeof($p_filedescr_list); $i++) { + + // ----- Get filedescr + $v_descr = $p_filedescr_list[$i]; + + // ----- Reduce the filename + $v_descr['filename'] = PclZipUtilTranslateWinPath($v_descr['filename'], false); + $v_descr['filename'] = PclZipUtilPathReduction($v_descr['filename']); + + // ----- Look for real file or folder + if (file_exists($v_descr['filename'])) { + if (@is_file($v_descr['filename'])) { + $v_descr['type'] = 'file'; + } elseif (@is_dir($v_descr['filename'])) { + $v_descr['type'] = 'folder'; + } elseif (@is_link($v_descr['filename'])) { + // skip + continue; + } else { + // skip + continue; + } + + // ----- Look for string added as file + } elseif (isset($v_descr['content'])) { + $v_descr['type'] = 'virtual_file'; + + // ----- Missing file + } else { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "File '" . $v_descr['filename'] . "' does not exist"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Calculate the stored filename + $this->privCalculateStoredFilename($v_descr, $p_options); + + // ----- Add the descriptor in result list + $v_result_list[sizeof($v_result_list)] = $v_descr; + + // ----- Look for folder + if ($v_descr['type'] == 'folder') { + // ----- List of items in folder + $v_dirlist_descr = array(); + $v_dirlist_nb = 0; + if ($v_folder_handler = @opendir($v_descr['filename'])) { + while (($v_item_handler = @readdir($v_folder_handler)) !== false) { + + // ----- Skip '.' and '..' + if (($v_item_handler == '.') || ($v_item_handler == '..')) { + continue; + } + + // ----- Compose the full filename + $v_dirlist_descr[$v_dirlist_nb]['filename'] = $v_descr['filename'] . '/' . $v_item_handler; + + // ----- Look for different stored filename + // Because the name of the folder was changed, the name of the + // files/sub-folders also change + if (($v_descr['stored_filename'] != $v_descr['filename']) && (!isset($p_options[PCLZIP_OPT_REMOVE_ALL_PATH]))) { + if ($v_descr['stored_filename'] != '') { + $v_dirlist_descr[$v_dirlist_nb]['new_full_name'] = $v_descr['stored_filename'] . '/' . $v_item_handler; + } else { + $v_dirlist_descr[$v_dirlist_nb]['new_full_name'] = $v_item_handler; + } + } + + $v_dirlist_nb++; + } + + @closedir($v_folder_handler); + } else { + // TBC : unable to open folder in read mode + } + + // ----- Expand each element of the list + if ($v_dirlist_nb != 0) { + // ----- Expand + if (($v_result = $this->privFileDescrExpand($v_dirlist_descr, $p_options)) != 1) { + return $v_result; + } + + // ----- Concat the resulting list + $v_result_list = array_merge($v_result_list, $v_dirlist_descr); + } else { + } + + // ----- Free local array + unset($v_dirlist_descr); + } + } + + // ----- Get the result list + $p_filedescr_list = $v_result_list; + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privCreate() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privCreate($p_filedescr_list, &$p_result_list, &$p_options) + { + $v_result = 1; + $v_list_detail = array(); + + // ----- Magic quotes trick + $this->privDisableMagicQuotes(); + + // ----- Open the file in write mode + if (($v_result = $this->privOpenFd('wb')) != 1) { + // ----- Return + return $v_result; + } + + // ----- Add the list of files + $v_result = $this->privAddList($p_filedescr_list, $p_result_list, $p_options); + + // ----- Close + $this->privCloseFd(); + + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privAdd() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privAdd($p_filedescr_list, &$p_result_list, &$p_options) + { + $v_result = 1; + $v_list_detail = array(); + + // ----- Look if the archive exists or is empty + if ((!is_file($this->zipname)) || (filesize($this->zipname) == 0)) { + + // ----- Do a create + $v_result = $this->privCreate($p_filedescr_list, $p_result_list, $p_options); + + // ----- Return + return $v_result; + } + // ----- Magic quotes trick + $this->privDisableMagicQuotes(); + + // ----- Open the zip file + if (($v_result = $this->privOpenFd('rb')) != 1) { + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + + // ----- Read the central directory informations + $v_central_dir = array(); + if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Go to beginning of File + @rewind($this->zip_fd); + + // ----- Creates a temporay file + $v_zip_temp_name = PCLZIP_TEMPORARY_DIR . uniqid('pclzip-') . '.tmp'; + + // ----- Open the temporary file in write mode + if (($v_zip_temp_fd = @fopen($v_zip_temp_name, 'wb')) == 0) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \'' . $v_zip_temp_name . '\' in binary write mode'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Copy the files from the archive to the temporary file + // TBC : Here I should better append the file and go back to erase the central dir + $v_size = $v_central_dir['offset']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = fread($this->zip_fd, $v_read_size); + @fwrite($v_zip_temp_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Swap the file descriptor + // Here is a trick : I swap the temporary fd with the zip fd, in order to use + // the following methods on the temporary fil and not the real archive + $v_swap = $this->zip_fd; + $this->zip_fd = $v_zip_temp_fd; + $v_zip_temp_fd = $v_swap; + + // ----- Add the files + $v_header_list = array(); + if (($v_result = $this->privAddFileList($p_filedescr_list, $v_header_list, $p_options)) != 1) { + fclose($v_zip_temp_fd); + $this->privCloseFd(); + @unlink($v_zip_temp_name); + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + + // ----- Store the offset of the central dir + $v_offset = @ftell($this->zip_fd); + + // ----- Copy the block of file headers from the old archive + $v_size = $v_central_dir['size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($v_zip_temp_fd, $v_read_size); + @fwrite($this->zip_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Create the Central Dir files header + for ($i = 0, $v_count = 0; $i < sizeof($v_header_list); $i++) { + // ----- Create the file header + if ($v_header_list[$i]['status'] == 'ok') { + if (($v_result = $this->privWriteCentralFileHeader($v_header_list[$i])) != 1) { + fclose($v_zip_temp_fd); + $this->privCloseFd(); + @unlink($v_zip_temp_name); + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + $v_count++; + } + + // ----- Transform the header to a 'usable' info + $this->privConvertHeader2FileInfo($v_header_list[$i], $p_result_list[$i]); + } + + // ----- Zip file comment + $v_comment = $v_central_dir['comment']; + if (isset($p_options[PCLZIP_OPT_COMMENT])) { + $v_comment = $p_options[PCLZIP_OPT_COMMENT]; + } + if (isset($p_options[PCLZIP_OPT_ADD_COMMENT])) { + $v_comment = $v_comment . $p_options[PCLZIP_OPT_ADD_COMMENT]; + } + if (isset($p_options[PCLZIP_OPT_PREPEND_COMMENT])) { + $v_comment = $p_options[PCLZIP_OPT_PREPEND_COMMENT] . $v_comment; + } + + // ----- Calculate the size of the central header + $v_size = @ftell($this->zip_fd) - $v_offset; + + // ----- Create the central dir footer + if (($v_result = $this->privWriteCentralHeader($v_count + $v_central_dir['entries'], $v_size, $v_offset, $v_comment)) != 1) { + // ----- Reset the file list + unset($v_header_list); + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + + // ----- Swap back the file descriptor + $v_swap = $this->zip_fd; + $this->zip_fd = $v_zip_temp_fd; + $v_zip_temp_fd = $v_swap; + + // ----- Close + $this->privCloseFd(); + + // ----- Close the temporary file + @fclose($v_zip_temp_fd); + + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Delete the zip file + // TBC : I should test the result ... + @unlink($this->zipname); + + // ----- Rename the temporary file + // TBC : I should test the result ... + //@rename($v_zip_temp_name, $this->zipname); + PclZipUtilRename($v_zip_temp_name, $this->zipname); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privOpenFd() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function privOpenFd($p_mode) + { + $v_result = 1; + + // ----- Look if already open + if ($this->zip_fd != 0) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Zip file \'' . $this->zipname . '\' already open'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Open the zip file + if (($this->zip_fd = @fopen($this->zipname, $p_mode)) == 0) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive \'' . $this->zipname . '\' in ' . $p_mode . ' mode'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privCloseFd() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function privCloseFd() + { + $v_result = 1; + + if ($this->zip_fd != 0) { + @fclose($this->zip_fd); + } + $this->zip_fd = 0; + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privAddList() + // Description : + // $p_add_dir and $p_remove_dir will give the ability to memorize a path which is + // different from the real path of the file. This is usefull if you want to have PclTar + // running in any directory, and memorize relative path from an other directory. + // Parameters : + // $p_list : An array containing the file or directory names to add in the tar + // $p_result_list : list of added files with their properties (specially the status field) + // $p_add_dir : Path to add in the filename path archived + // $p_remove_dir : Path to remove in the filename path archived + // Return Values : + // -------------------------------------------------------------------------------- + // function privAddList($p_list, &$p_result_list, $p_add_dir, $p_remove_dir, $p_remove_all_dir, &$p_options) + public function privAddList($p_filedescr_list, &$p_result_list, &$p_options) + { + $v_result = 1; + + // ----- Add the files + $v_header_list = array(); + if (($v_result = $this->privAddFileList($p_filedescr_list, $v_header_list, $p_options)) != 1) { + // ----- Return + return $v_result; + } + + // ----- Store the offset of the central dir + $v_offset = @ftell($this->zip_fd); + + // ----- Create the Central Dir files header + for ($i = 0, $v_count = 0; $i < sizeof($v_header_list); $i++) { + // ----- Create the file header + if ($v_header_list[$i]['status'] == 'ok') { + if (($v_result = $this->privWriteCentralFileHeader($v_header_list[$i])) != 1) { + // ----- Return + return $v_result; + } + $v_count++; + } + + // ----- Transform the header to a 'usable' info + $this->privConvertHeader2FileInfo($v_header_list[$i], $p_result_list[$i]); + } + + // ----- Zip file comment + $v_comment = ''; + if (isset($p_options[PCLZIP_OPT_COMMENT])) { + $v_comment = $p_options[PCLZIP_OPT_COMMENT]; + } + + // ----- Calculate the size of the central header + $v_size = @ftell($this->zip_fd) - $v_offset; + + // ----- Create the central dir footer + if (($v_result = $this->privWriteCentralHeader($v_count, $v_size, $v_offset, $v_comment)) != 1) { + // ----- Reset the file list + unset($v_header_list); + + // ----- Return + return $v_result; + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privAddFileList() + // Description : + // Parameters : + // $p_filedescr_list : An array containing the file description + // or directory names to add in the zip + // $p_result_list : list of added files with their properties (specially the status field) + // Return Values : + // -------------------------------------------------------------------------------- + public function privAddFileList($p_filedescr_list, &$p_result_list, &$p_options) + { + $v_result = 1; + $v_header = array(); + + // ----- Recuperate the current number of elt in list + $v_nb = sizeof($p_result_list); + + // ----- Loop on the files + for ($j = 0; ($j < sizeof($p_filedescr_list)) && ($v_result == 1); $j++) { + // ----- Format the filename + $p_filedescr_list[$j]['filename'] = PclZipUtilTranslateWinPath($p_filedescr_list[$j]['filename'], false); + + // ----- Skip empty file names + // TBC : Can this be possible ? not checked in DescrParseAtt ? + if ($p_filedescr_list[$j]['filename'] == "") { + continue; + } + + // ----- Check the filename + if (($p_filedescr_list[$j]['type'] != 'virtual_file') && (!file_exists($p_filedescr_list[$j]['filename']))) { + PclZip::privErrorLog(PCLZIP_ERR_MISSING_FILE, "File '" . $p_filedescr_list[$j]['filename'] . "' does not exist"); + + return PclZip::errorCode(); + } + + // ----- Look if it is a file or a dir with no all path remove option + // or a dir with all its path removed + // if ( (is_file($p_filedescr_list[$j]['filename'])) + // || ( is_dir($p_filedescr_list[$j]['filename']) + if (($p_filedescr_list[$j]['type'] == 'file') || ($p_filedescr_list[$j]['type'] == 'virtual_file') || (($p_filedescr_list[$j]['type'] == 'folder') && (!isset($p_options[PCLZIP_OPT_REMOVE_ALL_PATH]) || !$p_options[PCLZIP_OPT_REMOVE_ALL_PATH]))) { + + // ----- Add the file + $v_result = $this->privAddFile($p_filedescr_list[$j], $v_header, $p_options); + if ($v_result != 1) { + return $v_result; + } + + // ----- Store the file infos + $p_result_list[$v_nb++] = $v_header; + } + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privAddFile() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privAddFile($p_filedescr, &$p_header, &$p_options) + { + $v_result = 1; + + // ----- Working variable + $p_filename = $p_filedescr['filename']; + + // TBC : Already done in the fileAtt check ... ? + if ($p_filename == "") { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_PARAMETER, "Invalid file list parameter (invalid or empty list)"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Look for a stored different filename + /* TBC : Removed + if (isset($p_filedescr['stored_filename'])) { + $v_stored_filename = $p_filedescr['stored_filename']; + } else { + $v_stored_filename = $p_filedescr['stored_filename']; + } + */ + + // ----- Set the file properties + clearstatcache(); + $p_header['version'] = 20; + $p_header['version_extracted'] = 10; + $p_header['flag'] = 0; + $p_header['compression'] = 0; + $p_header['crc'] = 0; + $p_header['compressed_size'] = 0; + $p_header['filename_len'] = strlen($p_filename); + $p_header['extra_len'] = 0; + $p_header['disk'] = 0; + $p_header['internal'] = 0; + $p_header['offset'] = 0; + $p_header['filename'] = $p_filename; + // TBC : Removed $p_header['stored_filename'] = $v_stored_filename; + $p_header['stored_filename'] = $p_filedescr['stored_filename']; + $p_header['extra'] = ''; + $p_header['status'] = 'ok'; + $p_header['index'] = -1; + + // ----- Look for regular file + if ($p_filedescr['type'] == 'file') { + $p_header['external'] = 0x00000000; + $p_header['size'] = filesize($p_filename); + + // ----- Look for regular folder + } elseif ($p_filedescr['type'] == 'folder') { + $p_header['external'] = 0x00000010; + $p_header['mtime'] = filemtime($p_filename); + $p_header['size'] = filesize($p_filename); + + // ----- Look for virtual file + } elseif ($p_filedescr['type'] == 'virtual_file') { + $p_header['external'] = 0x00000000; + $p_header['size'] = strlen($p_filedescr['content']); + } + + // ----- Look for filetime + if (isset($p_filedescr['mtime'])) { + $p_header['mtime'] = $p_filedescr['mtime']; + } elseif ($p_filedescr['type'] == 'virtual_file') { + $p_header['mtime'] = time(); + } else { + $p_header['mtime'] = filemtime($p_filename); + } + + // ------ Look for file comment + if (isset($p_filedescr['comment'])) { + $p_header['comment_len'] = strlen($p_filedescr['comment']); + $p_header['comment'] = $p_filedescr['comment']; + } else { + $p_header['comment_len'] = 0; + $p_header['comment'] = ''; + } + + // ----- Look for pre-add callback + if (isset($p_options[PCLZIP_CB_PRE_ADD])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_header, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_PRE_ADD].'(PCLZIP_CB_PRE_ADD, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_PRE_ADD](PCLZIP_CB_PRE_ADD, $v_local_header); + if ($v_result == 0) { + // ----- Change the file status + $p_header['status'] = "skipped"; + $v_result = 1; + } + + // ----- Update the informations + // Only some fields can be modified + if ($p_header['stored_filename'] != $v_local_header['stored_filename']) { + $p_header['stored_filename'] = PclZipUtilPathReduction($v_local_header['stored_filename']); + } + } + + // ----- Look for empty stored filename + if ($p_header['stored_filename'] == "") { + $p_header['status'] = "filtered"; + } + + // ----- Check the path length + if (strlen($p_header['stored_filename']) > 0xFF) { + $p_header['status'] = 'filename_too_long'; + } + + // ----- Look if no error, or file not skipped + if ($p_header['status'] == 'ok') { + + // ----- Look for a file + if ($p_filedescr['type'] == 'file') { + // ----- Look for using temporary file to zip + if ((!isset($p_options[PCLZIP_OPT_TEMP_FILE_OFF])) && (isset($p_options[PCLZIP_OPT_TEMP_FILE_ON]) || (isset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD]) && ($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] <= $p_header['size'])))) { + $v_result = $this->privAddFileUsingTempFile($p_filedescr, $p_header, $p_options); + if ($v_result < PCLZIP_ERR_NO_ERROR) { + return $v_result; + } + + // ----- Use "in memory" zip algo + } else { + + // ----- Open the source file + if (($v_file = @fopen($p_filename, "rb")) == 0) { + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, "Unable to open file '$p_filename' in binary read mode"); + + return PclZip::errorCode(); + } + + // ----- Read the file content + $v_content = @fread($v_file, $p_header['size']); + + // ----- Close the file + @fclose($v_file); + + // ----- Calculate the CRC + $p_header['crc'] = @crc32($v_content); + + // ----- Look for no compression + if ($p_options[PCLZIP_OPT_NO_COMPRESSION]) { + // ----- Set header parameters + $p_header['compressed_size'] = $p_header['size']; + $p_header['compression'] = 0; + + // ----- Look for normal compression + } else { + // ----- Compress the content + $v_content = @gzdeflate($v_content); + + // ----- Set header parameters + $p_header['compressed_size'] = strlen($v_content); + $p_header['compression'] = 8; + } + + // ----- Call the header generation + if (($v_result = $this->privWriteFileHeader($p_header)) != 1) { + @fclose($v_file); + + return $v_result; + } + + // ----- Write the compressed (or not) content + @fwrite($this->zip_fd, $v_content, $p_header['compressed_size']); + + } + + // ----- Look for a virtual file (a file from string) + } elseif ($p_filedescr['type'] == 'virtual_file') { + + $v_content = $p_filedescr['content']; + + // ----- Calculate the CRC + $p_header['crc'] = @crc32($v_content); + + // ----- Look for no compression + if ($p_options[PCLZIP_OPT_NO_COMPRESSION]) { + // ----- Set header parameters + $p_header['compressed_size'] = $p_header['size']; + $p_header['compression'] = 0; + + // ----- Look for normal compression + } else { + // ----- Compress the content + $v_content = @gzdeflate($v_content); + + // ----- Set header parameters + $p_header['compressed_size'] = strlen($v_content); + $p_header['compression'] = 8; + } + + // ----- Call the header generation + if (($v_result = $this->privWriteFileHeader($p_header)) != 1) { + @fclose($v_file); + + return $v_result; + } + + // ----- Write the compressed (or not) content + @fwrite($this->zip_fd, $v_content, $p_header['compressed_size']); + + // ----- Look for a directory + } elseif ($p_filedescr['type'] == 'folder') { + // ----- Look for directory last '/' + if (@substr($p_header['stored_filename'], -1) != '/') { + $p_header['stored_filename'] .= '/'; + } + + // ----- Set the file properties + $p_header['size'] = 0; + //$p_header['external'] = 0x41FF0010; // Value for a folder : to be checked + $p_header['external'] = 0x00000010; // Value for a folder : to be checked + + // ----- Call the header generation + if (($v_result = $this->privWriteFileHeader($p_header)) != 1) { + return $v_result; + } + } + } + + // ----- Look for post-add callback + if (isset($p_options[PCLZIP_CB_POST_ADD])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_header, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_POST_ADD].'(PCLZIP_CB_POST_ADD, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_POST_ADD](PCLZIP_CB_POST_ADD, $v_local_header); + if ($v_result == 0) { + // ----- Ignored + $v_result = 1; + } + + // ----- Update the informations + // Nothing can be modified + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privAddFileUsingTempFile() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privAddFileUsingTempFile($p_filedescr, &$p_header, &$p_options) + { + $v_result = PCLZIP_ERR_NO_ERROR; + + // ----- Working variable + $p_filename = $p_filedescr['filename']; + + // ----- Open the source file + if (($v_file = @fopen($p_filename, "rb")) == 0) { + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, "Unable to open file '$p_filename' in binary read mode"); + + return PclZip::errorCode(); + } + + // ----- Creates a compressed temporary file + $v_gzip_temp_name = PCLZIP_TEMPORARY_DIR . uniqid('pclzip-') . '.gz'; + if (($v_file_compressed = @gzopen($v_gzip_temp_name, "wb")) == 0) { + fclose($v_file); + PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL, 'Unable to open temporary file \'' . $v_gzip_temp_name . '\' in binary write mode'); + + return PclZip::errorCode(); + } + + // ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks + $v_size = filesize($p_filename); + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($v_file, $v_read_size); + //$v_binary_data = pack('a'.$v_read_size, $v_buffer); + @gzputs($v_file_compressed, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Close the file + @fclose($v_file); + @gzclose($v_file_compressed); + + // ----- Check the minimum file size + if (filesize($v_gzip_temp_name) < 18) { + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'gzip temporary file \'' . $v_gzip_temp_name . '\' has invalid filesize - should be minimum 18 bytes'); + + return PclZip::errorCode(); + } + + // ----- Extract the compressed attributes + if (($v_file_compressed = @fopen($v_gzip_temp_name, "rb")) == 0) { + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \'' . $v_gzip_temp_name . '\' in binary read mode'); + + return PclZip::errorCode(); + } + + // ----- Read the gzip file header + $v_binary_data = @fread($v_file_compressed, 10); + $v_data_header = unpack('a1id1/a1id2/a1cm/a1flag/Vmtime/a1xfl/a1os', $v_binary_data); + + // ----- Check some parameters + $v_data_header['os'] = bin2hex($v_data_header['os']); + + // ----- Read the gzip file footer + @fseek($v_file_compressed, filesize($v_gzip_temp_name) - 8); + $v_binary_data = @fread($v_file_compressed, 8); + $v_data_footer = unpack('Vcrc/Vcompressed_size', $v_binary_data); + + // ----- Set the attributes + $p_header['compression'] = ord($v_data_header['cm']); + //$p_header['mtime'] = $v_data_header['mtime']; + $p_header['crc'] = $v_data_footer['crc']; + $p_header['compressed_size'] = filesize($v_gzip_temp_name) - 18; + + // ----- Close the file + @fclose($v_file_compressed); + + // ----- Call the header generation + if (($v_result = $this->privWriteFileHeader($p_header)) != 1) { + return $v_result; + } + + // ----- Add the compressed data + if (($v_file_compressed = @fopen($v_gzip_temp_name, "rb")) == 0) { + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \'' . $v_gzip_temp_name . '\' in binary read mode'); + + return PclZip::errorCode(); + } + + // ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks + fseek($v_file_compressed, 10); + $v_size = $p_header['compressed_size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($v_file_compressed, $v_read_size); + //$v_binary_data = pack('a'.$v_read_size, $v_buffer); + @fwrite($this->zip_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Close the file + @fclose($v_file_compressed); + + // ----- Unlink the temporary file + @unlink($v_gzip_temp_name); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privCalculateStoredFilename() + // Description : + // Based on file descriptor properties and global options, this method + // calculate the filename that will be stored in the archive. + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privCalculateStoredFilename(&$p_filedescr, &$p_options) + { + $v_result = 1; + + // ----- Working variables + $p_filename = $p_filedescr['filename']; + if (isset($p_options[PCLZIP_OPT_ADD_PATH])) { + $p_add_dir = $p_options[PCLZIP_OPT_ADD_PATH]; + } else { + $p_add_dir = ''; + } + if (isset($p_options[PCLZIP_OPT_REMOVE_PATH])) { + $p_remove_dir = $p_options[PCLZIP_OPT_REMOVE_PATH]; + } else { + $p_remove_dir = ''; + } + if (isset($p_options[PCLZIP_OPT_REMOVE_ALL_PATH])) { + $p_remove_all_dir = $p_options[PCLZIP_OPT_REMOVE_ALL_PATH]; + } else { + $p_remove_all_dir = 0; + } + + // ----- Look for full name change + if (isset($p_filedescr['new_full_name'])) { + // ----- Remove drive letter if any + $v_stored_filename = PclZipUtilTranslateWinPath($p_filedescr['new_full_name']); + + // ----- Look for path and/or short name change + } else { + + // ----- Look for short name change + // Its when we cahnge just the filename but not the path + if (isset($p_filedescr['new_short_name'])) { + $v_path_info = pathinfo($p_filename); + $v_dir = ''; + if ($v_path_info['dirname'] != '') { + $v_dir = $v_path_info['dirname'] . '/'; + } + $v_stored_filename = $v_dir . $p_filedescr['new_short_name']; + } else { + // ----- Calculate the stored filename + $v_stored_filename = $p_filename; + } + + // ----- Look for all path to remove + if ($p_remove_all_dir) { + $v_stored_filename = basename($p_filename); + + // ----- Look for partial path remove + } elseif ($p_remove_dir != "") { + if (substr($p_remove_dir, -1) != '/') { + $p_remove_dir .= "/"; + } + + if ((substr($p_filename, 0, 2) == "./") || (substr($p_remove_dir, 0, 2) == "./")) { + + if ((substr($p_filename, 0, 2) == "./") && (substr($p_remove_dir, 0, 2) != "./")) { + $p_remove_dir = "./" . $p_remove_dir; + } + if ((substr($p_filename, 0, 2) != "./") && (substr($p_remove_dir, 0, 2) == "./")) { + $p_remove_dir = substr($p_remove_dir, 2); + } + } + + $v_compare = PclZipUtilPathInclusion($p_remove_dir, $v_stored_filename); + if ($v_compare > 0) { + if ($v_compare == 2) { + $v_stored_filename = ""; + } else { + $v_stored_filename = substr($v_stored_filename, strlen($p_remove_dir)); + } + } + } + + // ----- Remove drive letter if any + $v_stored_filename = PclZipUtilTranslateWinPath($v_stored_filename); + + // ----- Look for path to add + if ($p_add_dir != "") { + if (substr($p_add_dir, -1) == "/") { + $v_stored_filename = $p_add_dir . $v_stored_filename; + } else { + $v_stored_filename = $p_add_dir . "/" . $v_stored_filename; + } + } + } + + // ----- Filename (reduce the path of stored name) + $v_stored_filename = PclZipUtilPathReduction($v_stored_filename); + $p_filedescr['stored_filename'] = $v_stored_filename; + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privWriteFileHeader() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privWriteFileHeader(&$p_header) + { + $v_result = 1; + + // ----- Store the offset position of the file + $p_header['offset'] = ftell($this->zip_fd); + + // ----- Transform UNIX mtime to DOS format mdate/mtime + $v_date = getdate($p_header['mtime']); + $v_mtime = ($v_date['hours'] << 11) + ($v_date['minutes'] << 5) + $v_date['seconds'] / 2; + $v_mdate = (($v_date['year'] - 1980) << 9) + ($v_date['mon'] << 5) + $v_date['mday']; + + // ----- Packed data + $v_binary_data = pack("VvvvvvVVVvv", 0x04034b50, $p_header['version_extracted'], $p_header['flag'], $p_header['compression'], $v_mtime, $v_mdate, $p_header['crc'], $p_header['compressed_size'], $p_header['size'], strlen($p_header['stored_filename']), $p_header['extra_len']); + + // ----- Write the first 148 bytes of the header in the archive + fputs($this->zip_fd, $v_binary_data, 30); + + // ----- Write the variable fields + if (strlen($p_header['stored_filename']) != 0) { + fputs($this->zip_fd, $p_header['stored_filename'], strlen($p_header['stored_filename'])); + } + if ($p_header['extra_len'] != 0) { + fputs($this->zip_fd, $p_header['extra'], $p_header['extra_len']); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privWriteCentralFileHeader() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privWriteCentralFileHeader(&$p_header) + { + $v_result = 1; + + // TBC + //for (reset($p_header); $key = key($p_header); next($p_header)) { + //} + + // ----- Transform UNIX mtime to DOS format mdate/mtime + $v_date = getdate($p_header['mtime']); + $v_mtime = ($v_date['hours'] << 11) + ($v_date['minutes'] << 5) + $v_date['seconds'] / 2; + $v_mdate = (($v_date['year'] - 1980) << 9) + ($v_date['mon'] << 5) + $v_date['mday']; + + // ----- Packed data + $v_binary_data = pack("VvvvvvvVVVvvvvvVV", 0x02014b50, $p_header['version'], $p_header['version_extracted'], $p_header['flag'], $p_header['compression'], $v_mtime, $v_mdate, $p_header['crc'], $p_header['compressed_size'], $p_header['size'], strlen($p_header['stored_filename']), $p_header['extra_len'], $p_header['comment_len'], $p_header['disk'], $p_header['internal'], $p_header['external'], $p_header['offset']); + + // ----- Write the 42 bytes of the header in the zip file + fputs($this->zip_fd, $v_binary_data, 46); + + // ----- Write the variable fields + if (strlen($p_header['stored_filename']) != 0) { + fputs($this->zip_fd, $p_header['stored_filename'], strlen($p_header['stored_filename'])); + } + if ($p_header['extra_len'] != 0) { + fputs($this->zip_fd, $p_header['extra'], $p_header['extra_len']); + } + if ($p_header['comment_len'] != 0) { + fputs($this->zip_fd, $p_header['comment'], $p_header['comment_len']); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privWriteCentralHeader() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privWriteCentralHeader($p_nb_entries, $p_size, $p_offset, $p_comment) + { + $v_result = 1; + + // ----- Packed data + $v_binary_data = pack("VvvvvVVv", 0x06054b50, 0, 0, $p_nb_entries, $p_nb_entries, $p_size, $p_offset, strlen($p_comment)); + + // ----- Write the 22 bytes of the header in the zip file + fputs($this->zip_fd, $v_binary_data, 22); + + // ----- Write the variable fields + if (strlen($p_comment) != 0) { + fputs($this->zip_fd, $p_comment, strlen($p_comment)); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privList() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privList(&$p_list) + { + $v_result = 1; + + // ----- Magic quotes trick + $this->privDisableMagicQuotes(); + + // ----- Open the zip file + if (($this->zip_fd = @fopen($this->zipname, 'rb')) == 0) { + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive \'' . $this->zipname . '\' in binary read mode'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read the central directory informations + $v_central_dir = array(); + if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1) { + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Go to beginning of Central Dir + @rewind($this->zip_fd); + if (@fseek($this->zip_fd, $v_central_dir['offset'])) { + $this->privSwapBackMagicQuotes(); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read each entry + for ($i = 0; $i < $v_central_dir['entries']; $i++) { + // ----- Read the file header + if (($v_result = $this->privReadCentralFileHeader($v_header)) != 1) { + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + $v_header['index'] = $i; + + // ----- Get the only interesting attributes + $this->privConvertHeader2FileInfo($v_header, $p_list[$i]); + unset($v_header); + } + + // ----- Close the zip file + $this->privCloseFd(); + + // ----- Magic quotes trick + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privConvertHeader2FileInfo() + // Description : + // This function takes the file informations from the central directory + // entries and extract the interesting parameters that will be given back. + // The resulting file infos are set in the array $p_info + // $p_info['filename'] : Filename with full path. Given by user (add), + // extracted in the filesystem (extract). + // $p_info['stored_filename'] : Stored filename in the archive. + // $p_info['size'] = Size of the file. + // $p_info['compressed_size'] = Compressed size of the file. + // $p_info['mtime'] = Last modification date of the file. + // $p_info['comment'] = Comment associated with the file. + // $p_info['folder'] = true/false : indicates if the entry is a folder or not. + // $p_info['status'] = status of the action on the file. + // $p_info['crc'] = CRC of the file content. + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privConvertHeader2FileInfo($p_header, &$p_info) + { + $v_result = 1; + + // ----- Get the interesting attributes + $v_temp_path = PclZipUtilPathReduction($p_header['filename']); + $p_info['filename'] = $v_temp_path; + $v_temp_path = PclZipUtilPathReduction($p_header['stored_filename']); + $p_info['stored_filename'] = $v_temp_path; + $p_info['size'] = $p_header['size']; + $p_info['compressed_size'] = $p_header['compressed_size']; + $p_info['mtime'] = $p_header['mtime']; + $p_info['comment'] = $p_header['comment']; + $p_info['folder'] = (($p_header['external'] & 0x00000010) == 0x00000010); + $p_info['index'] = $p_header['index']; + $p_info['status'] = $p_header['status']; + $p_info['crc'] = $p_header['crc']; + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privExtractByRule() + // Description : + // Extract a file or directory depending of rules (by index, by name, ...) + // Parameters : + // $p_file_list : An array where will be placed the properties of each + // extracted file + // $p_path : Path to add while writing the extracted files + // $p_remove_path : Path to remove (from the file memorized path) while writing the + // extracted files. If the path does not match the file path, + // the file is extracted with its memorized path. + // $p_remove_path does not apply to 'list' mode. + // $p_path and $p_remove_path are commulative. + // Return Values : + // 1 on success,0 or less on error (see error code list) + // -------------------------------------------------------------------------------- + public function privExtractByRule(&$p_file_list, $p_path, $p_remove_path, $p_remove_all_path, &$p_options) + { + $v_result = 1; + + // ----- Magic quotes trick + $this->privDisableMagicQuotes(); + + // ----- Check the path + if (($p_path == "") || ((substr($p_path, 0, 1) != "/") && (substr($p_path, 0, 3) != "../") && (substr($p_path, 1, 2) != ":/"))) { + $p_path = "./" . $p_path; + } + + // ----- Reduce the path last (and duplicated) '/' + if (($p_path != "./") && ($p_path != "/")) { + // ----- Look for the path end '/' + while (substr($p_path, -1) == "/") { + $p_path = substr($p_path, 0, strlen($p_path) - 1); + } + } + + // ----- Look for path to remove format (should end by /) + if (($p_remove_path != "") && (substr($p_remove_path, -1) != '/')) { + $p_remove_path .= '/'; + } + $p_remove_path_size = strlen($p_remove_path); + + // ----- Open the zip file + if (($v_result = $this->privOpenFd('rb')) != 1) { + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Read the central directory informations + $v_central_dir = array(); + if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Start at beginning of Central Dir + $v_pos_entry = $v_central_dir['offset']; + + // ----- Read each entry + $j_start = 0; + for ($i = 0, $v_nb_extracted = 0; $i < $v_central_dir['entries']; $i++) { + + // ----- Read next Central dir entry + @rewind($this->zip_fd); + if (@fseek($this->zip_fd, $v_pos_entry)) { + // ----- Close the zip file + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read the file header + $v_header = array(); + if (($v_result = $this->privReadCentralFileHeader($v_header)) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Store the index + $v_header['index'] = $i; + + // ----- Store the file position + $v_pos_entry = ftell($this->zip_fd); + + // ----- Look for the specific extract rules + $v_extract = false; + + // ----- Look for extract by name rule + if ((isset($p_options[PCLZIP_OPT_BY_NAME])) && ($p_options[PCLZIP_OPT_BY_NAME] != 0)) { + + // ----- Look if the filename is in the list + for ($j = 0; ($j < sizeof($p_options[PCLZIP_OPT_BY_NAME])) && (!$v_extract); $j++) { + + // ----- Look for a directory + if (substr($p_options[PCLZIP_OPT_BY_NAME][$j], -1) == "/") { + + // ----- Look if the directory is in the filename path + if ((strlen($v_header['stored_filename']) > strlen($p_options[PCLZIP_OPT_BY_NAME][$j])) && (substr($v_header['stored_filename'], 0, strlen($p_options[PCLZIP_OPT_BY_NAME][$j])) == $p_options[PCLZIP_OPT_BY_NAME][$j])) { + $v_extract = true; + } + + // ----- Look for a filename + } elseif ($v_header['stored_filename'] == $p_options[PCLZIP_OPT_BY_NAME][$j]) { + $v_extract = true; + } + } + // ----- Look for extract by ereg rule + // ereg() is deprecated with PHP 5.3 + /* + elseif ( (isset($p_options[PCLZIP_OPT_BY_EREG])) + && ($p_options[PCLZIP_OPT_BY_EREG] != "")) { + + if (ereg($p_options[PCLZIP_OPT_BY_EREG], $v_header['stored_filename'])) { + $v_extract = true; + } + } + */ + + // ----- Look for extract by preg rule + } elseif ((isset($p_options[PCLZIP_OPT_BY_PREG])) && ($p_options[PCLZIP_OPT_BY_PREG] != "")) { + + if (preg_match($p_options[PCLZIP_OPT_BY_PREG], $v_header['stored_filename'])) { + $v_extract = true; + } + + // ----- Look for extract by index rule + } elseif ((isset($p_options[PCLZIP_OPT_BY_INDEX])) && ($p_options[PCLZIP_OPT_BY_INDEX] != 0)) { + + // ----- Look if the index is in the list + for ($j = $j_start; ($j < sizeof($p_options[PCLZIP_OPT_BY_INDEX])) && (!$v_extract); $j++) { + + if (($i >= $p_options[PCLZIP_OPT_BY_INDEX][$j]['start']) && ($i <= $p_options[PCLZIP_OPT_BY_INDEX][$j]['end'])) { + $v_extract = true; + } + if ($i >= $p_options[PCLZIP_OPT_BY_INDEX][$j]['end']) { + $j_start = $j + 1; + } + + if ($p_options[PCLZIP_OPT_BY_INDEX][$j]['start'] > $i) { + break; + } + } + + // ----- Look for no rule, which means extract all the archive + } else { + $v_extract = true; + } + + // ----- Check compression method + if (($v_extract) && (($v_header['compression'] != 8) && ($v_header['compression'] != 0))) { + $v_header['status'] = 'unsupported_compression'; + + // ----- Look for PCLZIP_OPT_STOP_ON_ERROR + if ((isset($p_options[PCLZIP_OPT_STOP_ON_ERROR])) && ($p_options[PCLZIP_OPT_STOP_ON_ERROR] === true)) { + + $this->privSwapBackMagicQuotes(); + + PclZip::privErrorLog(PCLZIP_ERR_UNSUPPORTED_COMPRESSION, "Filename '" . $v_header['stored_filename'] . "' is " . "compressed by an unsupported compression " . "method (" . $v_header['compression'] . ") "); + + return PclZip::errorCode(); + } + } + + // ----- Check encrypted files + if (($v_extract) && (($v_header['flag'] & 1) == 1)) { + $v_header['status'] = 'unsupported_encryption'; + + // ----- Look for PCLZIP_OPT_STOP_ON_ERROR + if ((isset($p_options[PCLZIP_OPT_STOP_ON_ERROR])) && ($p_options[PCLZIP_OPT_STOP_ON_ERROR] === true)) { + + $this->privSwapBackMagicQuotes(); + + PclZip::privErrorLog(PCLZIP_ERR_UNSUPPORTED_ENCRYPTION, "Unsupported encryption for " . " filename '" . $v_header['stored_filename'] . "'"); + + return PclZip::errorCode(); + } + } + + // ----- Look for real extraction + if (($v_extract) && ($v_header['status'] != 'ok')) { + $v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted++]); + if ($v_result != 1) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + $v_extract = false; + } + + // ----- Look for real extraction + if ($v_extract) { + + // ----- Go to the file position + @rewind($this->zip_fd); + if (@fseek($this->zip_fd, $v_header['offset'])) { + // ----- Close the zip file + $this->privCloseFd(); + + $this->privSwapBackMagicQuotes(); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Look for extraction as string + if ($p_options[PCLZIP_OPT_EXTRACT_AS_STRING]) { + + $v_string = ''; + + // ----- Extracting the file + $v_result1 = $this->privExtractFileAsString($v_header, $v_string, $p_options); + if ($v_result1 < 1) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result1; + } + + // ----- Get the only interesting attributes + if (($v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted])) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Set the file content + $p_file_list[$v_nb_extracted]['content'] = $v_string; + + // ----- Next extracted file + $v_nb_extracted++; + + // ----- Look for user callback abort + if ($v_result1 == 2) { + break; + } + + // ----- Look for extraction in standard output + } elseif ((isset($p_options[PCLZIP_OPT_EXTRACT_IN_OUTPUT])) && ($p_options[PCLZIP_OPT_EXTRACT_IN_OUTPUT])) { + // ----- Extracting the file in standard output + $v_result1 = $this->privExtractFileInOutput($v_header, $p_options); + if ($v_result1 < 1) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result1; + } + + // ----- Get the only interesting attributes + if (($v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted++])) != 1) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Look for user callback abort + if ($v_result1 == 2) { + break; + } + + // ----- Look for normal extraction + } else { + // ----- Extracting the file + $v_result1 = $this->privExtractFile($v_header, $p_path, $p_remove_path, $p_remove_all_path, $p_options); + if ($v_result1 < 1) { + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result1; + } + + // ----- Get the only interesting attributes + if (($v_result = $this->privConvertHeader2FileInfo($v_header, $p_file_list[$v_nb_extracted++])) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + return $v_result; + } + + // ----- Look for user callback abort + if ($v_result1 == 2) { + break; + } + } + } + } + + // ----- Close the zip file + $this->privCloseFd(); + $this->privSwapBackMagicQuotes(); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privExtractFile() + // Description : + // Parameters : + // Return Values : + // + // 1 : ... ? + // PCLZIP_ERR_USER_ABORTED(2) : User ask for extraction stop in callback + // -------------------------------------------------------------------------------- + public function privExtractFile(&$p_entry, $p_path, $p_remove_path, $p_remove_all_path, &$p_options) + { + $v_result = 1; + + // ----- Read the file header + if (($v_result = $this->privReadFileHeader($v_header)) != 1) { + // ----- Return + return $v_result; + } + + // ----- Check that the file header is coherent with $p_entry info + if ($this->privCheckFileHeaders($v_header, $p_entry) != 1) { + // TBC + } + + // ----- Look for all path to remove + if ($p_remove_all_path == true) { + // ----- Look for folder entry that not need to be extracted + if (($p_entry['external'] & 0x00000010) == 0x00000010) { + + $p_entry['status'] = "filtered"; + + return $v_result; + } + + // ----- Get the basename of the path + $p_entry['filename'] = basename($p_entry['filename']); + + // ----- Look for path to remove + } elseif ($p_remove_path != "") { + if (PclZipUtilPathInclusion($p_remove_path, $p_entry['filename']) == 2) { + + // ----- Change the file status + $p_entry['status'] = "filtered"; + + // ----- Return + return $v_result; + } + + $p_remove_path_size = strlen($p_remove_path); + if (substr($p_entry['filename'], 0, $p_remove_path_size) == $p_remove_path) { + + // ----- Remove the path + $p_entry['filename'] = substr($p_entry['filename'], $p_remove_path_size); + + } + } + + // ----- Add the path + if ($p_path != '') { + $p_entry['filename'] = $p_path . "/" . $p_entry['filename']; + } + + // ----- Check a base_dir_restriction + if (isset($p_options[PCLZIP_OPT_EXTRACT_DIR_RESTRICTION])) { + $v_inclusion = PclZipUtilPathInclusion($p_options[PCLZIP_OPT_EXTRACT_DIR_RESTRICTION], $p_entry['filename']); + if ($v_inclusion == 0) { + + PclZip::privErrorLog(PCLZIP_ERR_DIRECTORY_RESTRICTION, "Filename '" . $p_entry['filename'] . "' is " . "outside PCLZIP_OPT_EXTRACT_DIR_RESTRICTION"); + + return PclZip::errorCode(); + } + } + + // ----- Look for pre-extract callback + if (isset($p_options[PCLZIP_CB_PRE_EXTRACT])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_entry, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_PRE_EXTRACT].'(PCLZIP_CB_PRE_EXTRACT, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_PRE_EXTRACT](PCLZIP_CB_PRE_EXTRACT, $v_local_header); + if ($v_result == 0) { + // ----- Change the file status + $p_entry['status'] = "skipped"; + $v_result = 1; + } + + // ----- Look for abort result + if ($v_result == 2) { + // ----- This status is internal and will be changed in 'skipped' + $p_entry['status'] = "aborted"; + $v_result = PCLZIP_ERR_USER_ABORTED; + } + + // ----- Update the informations + // Only some fields can be modified + $p_entry['filename'] = $v_local_header['filename']; + } + + // ----- Look if extraction should be done + if ($p_entry['status'] == 'ok') { + + // ----- Look for specific actions while the file exist + if (file_exists($p_entry['filename'])) { + + // ----- Look if file is a directory + if (is_dir($p_entry['filename'])) { + + // ----- Change the file status + $p_entry['status'] = "already_a_directory"; + + // ----- Look for PCLZIP_OPT_STOP_ON_ERROR + // For historical reason first PclZip implementation does not stop + // when this kind of error occurs. + if ((isset($p_options[PCLZIP_OPT_STOP_ON_ERROR])) && ($p_options[PCLZIP_OPT_STOP_ON_ERROR] === true)) { + + PclZip::privErrorLog(PCLZIP_ERR_ALREADY_A_DIRECTORY, "Filename '" . $p_entry['filename'] . "' is " . "already used by an existing directory"); + + return PclZip::errorCode(); + } + + // ----- Look if file is write protected + } elseif (!is_writeable($p_entry['filename'])) { + + // ----- Change the file status + $p_entry['status'] = "write_protected"; + + // ----- Look for PCLZIP_OPT_STOP_ON_ERROR + // For historical reason first PclZip implementation does not stop + // when this kind of error occurs. + if ((isset($p_options[PCLZIP_OPT_STOP_ON_ERROR])) && ($p_options[PCLZIP_OPT_STOP_ON_ERROR] === true)) { + + PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL, "Filename '" . $p_entry['filename'] . "' exists " . "and is write protected"); + + return PclZip::errorCode(); + } + + // ----- Look if the extracted file is older + } elseif (filemtime($p_entry['filename']) > $p_entry['mtime']) { + // ----- Change the file status + if ((isset($p_options[PCLZIP_OPT_REPLACE_NEWER])) && ($p_options[PCLZIP_OPT_REPLACE_NEWER] === true)) { + } else { + $p_entry['status'] = "newer_exist"; + + // ----- Look for PCLZIP_OPT_STOP_ON_ERROR + // For historical reason first PclZip implementation does not stop + // when this kind of error occurs. + if ((isset($p_options[PCLZIP_OPT_STOP_ON_ERROR])) && ($p_options[PCLZIP_OPT_STOP_ON_ERROR] === true)) { + + PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL, "Newer version of '" . $p_entry['filename'] . "' exists " . "and option PCLZIP_OPT_REPLACE_NEWER is not selected"); + + return PclZip::errorCode(); + } + } + } else { + } + + // ----- Check the directory availability and create it if necessary + } else { + if ((($p_entry['external'] & 0x00000010) == 0x00000010) || (substr($p_entry['filename'], -1) == '/')) { + $v_dir_to_check = $p_entry['filename']; + } elseif (!strstr($p_entry['filename'], "/")) { + $v_dir_to_check = ""; + } else { + $v_dir_to_check = dirname($p_entry['filename']); + } + + if (($v_result = $this->privDirCheck($v_dir_to_check, (($p_entry['external'] & 0x00000010) == 0x00000010))) != 1) { + + // ----- Change the file status + $p_entry['status'] = "path_creation_fail"; + + // ----- Return + //return $v_result; + $v_result = 1; + } + } + } + + // ----- Look if extraction should be done + if ($p_entry['status'] == 'ok') { + + // ----- Do the extraction (if not a folder) + if (!(($p_entry['external'] & 0x00000010) == 0x00000010)) { + // ----- Look for not compressed file + if ($p_entry['compression'] == 0) { + + // ----- Opening destination file + if (($v_dest_file = @fopen($p_entry['filename'], 'wb')) == 0) { + + // ----- Change the file status + $p_entry['status'] = "write_error"; + + // ----- Return + return $v_result; + } + + // ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks + $v_size = $p_entry['compressed_size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($this->zip_fd, $v_read_size); + /* Try to speed up the code + $v_binary_data = pack('a'.$v_read_size, $v_buffer); + @fwrite($v_dest_file, $v_binary_data, $v_read_size); + */ + @fwrite($v_dest_file, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Closing the destination file + fclose($v_dest_file); + + // ----- Change the file mtime + touch($p_entry['filename'], $p_entry['mtime']); + + } else { + // ----- TBC + // Need to be finished + if (($p_entry['flag'] & 1) == 1) { + PclZip::privErrorLog(PCLZIP_ERR_UNSUPPORTED_ENCRYPTION, 'File \'' . $p_entry['filename'] . '\' is encrypted. Encrypted files are not supported.'); + + return PclZip::errorCode(); + } + + // ----- Look for using temporary file to unzip + if ((!isset($p_options[PCLZIP_OPT_TEMP_FILE_OFF])) && (isset($p_options[PCLZIP_OPT_TEMP_FILE_ON]) || (isset($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD]) && ($p_options[PCLZIP_OPT_TEMP_FILE_THRESHOLD] <= $p_entry['size'])))) { + $v_result = $this->privExtractFileUsingTempFile($p_entry, $p_options); + if ($v_result < PCLZIP_ERR_NO_ERROR) { + return $v_result; + } + + // ----- Look for extract in memory + } else { + + // ----- Read the compressed file in a buffer (one shot) + $v_buffer = @fread($this->zip_fd, $p_entry['compressed_size']); + + // ----- Decompress the file + $v_file_content = @gzinflate($v_buffer); + unset($v_buffer); + if ($v_file_content === false) { + + // ----- Change the file status + // TBC + $p_entry['status'] = "error"; + + return $v_result; + } + + // ----- Opening destination file + if (($v_dest_file = @fopen($p_entry['filename'], 'wb')) == 0) { + + // ----- Change the file status + $p_entry['status'] = "write_error"; + + return $v_result; + } + + // ----- Write the uncompressed data + @fwrite($v_dest_file, $v_file_content, $p_entry['size']); + unset($v_file_content); + + // ----- Closing the destination file + @fclose($v_dest_file); + + } + + // ----- Change the file mtime + @touch($p_entry['filename'], $p_entry['mtime']); + } + + // ----- Look for chmod option + if (isset($p_options[PCLZIP_OPT_SET_CHMOD])) { + + // ----- Change the mode of the file + @chmod($p_entry['filename'], $p_options[PCLZIP_OPT_SET_CHMOD]); + } + + } + } + + // ----- Change abort status + if ($p_entry['status'] == "aborted") { + $p_entry['status'] = "skipped"; + + // ----- Look for post-extract callback + } elseif (isset($p_options[PCLZIP_CB_POST_EXTRACT])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_entry, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_POST_EXTRACT].'(PCLZIP_CB_POST_EXTRACT, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_POST_EXTRACT](PCLZIP_CB_POST_EXTRACT, $v_local_header); + + // ----- Look for abort result + if ($v_result == 2) { + $v_result = PCLZIP_ERR_USER_ABORTED; + } + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privExtractFileUsingTempFile() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privExtractFileUsingTempFile(&$p_entry, &$p_options) + { + $v_result = 1; + + // ----- Creates a temporary file + $v_gzip_temp_name = PCLZIP_TEMPORARY_DIR . uniqid('pclzip-') . '.gz'; + if (($v_dest_file = @fopen($v_gzip_temp_name, "wb")) == 0) { + fclose($v_file); + PclZip::privErrorLog(PCLZIP_ERR_WRITE_OPEN_FAIL, 'Unable to open temporary file \'' . $v_gzip_temp_name . '\' in binary write mode'); + + return PclZip::errorCode(); + } + + // ----- Write gz file format header + $v_binary_data = pack('va1a1Va1a1', 0x8b1f, Chr($p_entry['compression']), Chr(0x00), time(), Chr(0x00), Chr(3)); + @fwrite($v_dest_file, $v_binary_data, 10); + + // ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks + $v_size = $p_entry['compressed_size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($this->zip_fd, $v_read_size); + //$v_binary_data = pack('a'.$v_read_size, $v_buffer); + @fwrite($v_dest_file, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Write gz file format footer + $v_binary_data = pack('VV', $p_entry['crc'], $p_entry['size']); + @fwrite($v_dest_file, $v_binary_data, 8); + + // ----- Close the temporary file + @fclose($v_dest_file); + + // ----- Opening destination file + if (($v_dest_file = @fopen($p_entry['filename'], 'wb')) == 0) { + $p_entry['status'] = "write_error"; + + return $v_result; + } + + // ----- Open the temporary gz file + if (($v_src_file = @gzopen($v_gzip_temp_name, 'rb')) == 0) { + @fclose($v_dest_file); + $p_entry['status'] = "read_error"; + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \'' . $v_gzip_temp_name . '\' in binary read mode'); + + return PclZip::errorCode(); + } + + // ----- Read the file by PCLZIP_READ_BLOCK_SIZE octets blocks + $v_size = $p_entry['size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @gzread($v_src_file, $v_read_size); + //$v_binary_data = pack('a'.$v_read_size, $v_buffer); + @fwrite($v_dest_file, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + @fclose($v_dest_file); + @gzclose($v_src_file); + + // ----- Delete the temporary file + @unlink($v_gzip_temp_name); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privExtractFileInOutput() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privExtractFileInOutput(&$p_entry, &$p_options) + { + $v_result = 1; + + // ----- Read the file header + if (($v_result = $this->privReadFileHeader($v_header)) != 1) { + return $v_result; + } + + // ----- Check that the file header is coherent with $p_entry info + if ($this->privCheckFileHeaders($v_header, $p_entry) != 1) { + // TBC + } + + // ----- Look for pre-extract callback + if (isset($p_options[PCLZIP_CB_PRE_EXTRACT])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_entry, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_PRE_EXTRACT].'(PCLZIP_CB_PRE_EXTRACT, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_PRE_EXTRACT](PCLZIP_CB_PRE_EXTRACT, $v_local_header); + if ($v_result == 0) { + // ----- Change the file status + $p_entry['status'] = "skipped"; + $v_result = 1; + } + + // ----- Look for abort result + if ($v_result == 2) { + // ----- This status is internal and will be changed in 'skipped' + $p_entry['status'] = "aborted"; + $v_result = PCLZIP_ERR_USER_ABORTED; + } + + // ----- Update the informations + // Only some fields can be modified + $p_entry['filename'] = $v_local_header['filename']; + } + + // ----- Trace + + // ----- Look if extraction should be done + if ($p_entry['status'] == 'ok') { + + // ----- Do the extraction (if not a folder) + if (!(($p_entry['external'] & 0x00000010) == 0x00000010)) { + // ----- Look for not compressed file + if ($p_entry['compressed_size'] == $p_entry['size']) { + + // ----- Read the file in a buffer (one shot) + $v_buffer = @fread($this->zip_fd, $p_entry['compressed_size']); + + // ----- Send the file to the output + echo $v_buffer; + unset($v_buffer); + } else { + + // ----- Read the compressed file in a buffer (one shot) + $v_buffer = @fread($this->zip_fd, $p_entry['compressed_size']); + + // ----- Decompress the file + $v_file_content = gzinflate($v_buffer); + unset($v_buffer); + + // ----- Send the file to the output + echo $v_file_content; + unset($v_file_content); + } + } + } + + // ----- Change abort status + if ($p_entry['status'] == "aborted") { + $p_entry['status'] = "skipped"; + + // ----- Look for post-extract callback + } elseif (isset($p_options[PCLZIP_CB_POST_EXTRACT])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_entry, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_POST_EXTRACT].'(PCLZIP_CB_POST_EXTRACT, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_POST_EXTRACT](PCLZIP_CB_POST_EXTRACT, $v_local_header); + + // ----- Look for abort result + if ($v_result == 2) { + $v_result = PCLZIP_ERR_USER_ABORTED; + } + } + + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privExtractFileAsString() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privExtractFileAsString(&$p_entry, &$p_string, &$p_options) + { + $v_result = 1; + + // ----- Read the file header + $v_header = array(); + if (($v_result = $this->privReadFileHeader($v_header)) != 1) { + // ----- Return + return $v_result; + } + + // ----- Check that the file header is coherent with $p_entry info + if ($this->privCheckFileHeaders($v_header, $p_entry) != 1) { + // TBC + } + + // ----- Look for pre-extract callback + if (isset($p_options[PCLZIP_CB_PRE_EXTRACT])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_entry, $v_local_header); + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_PRE_EXTRACT].'(PCLZIP_CB_PRE_EXTRACT, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_PRE_EXTRACT](PCLZIP_CB_PRE_EXTRACT, $v_local_header); + if ($v_result == 0) { + // ----- Change the file status + $p_entry['status'] = "skipped"; + $v_result = 1; + } + + // ----- Look for abort result + if ($v_result == 2) { + // ----- This status is internal and will be changed in 'skipped' + $p_entry['status'] = "aborted"; + $v_result = PCLZIP_ERR_USER_ABORTED; + } + + // ----- Update the informations + // Only some fields can be modified + $p_entry['filename'] = $v_local_header['filename']; + } + + // ----- Look if extraction should be done + if ($p_entry['status'] == 'ok') { + + // ----- Do the extraction (if not a folder) + if (!(($p_entry['external'] & 0x00000010) == 0x00000010)) { + // ----- Look for not compressed file + // if ($p_entry['compressed_size'] == $p_entry['size']) + if ($p_entry['compression'] == 0) { + + // ----- Reading the file + $p_string = @fread($this->zip_fd, $p_entry['compressed_size']); + } else { + + // ----- Reading the file + $v_data = @fread($this->zip_fd, $p_entry['compressed_size']); + + // ----- Decompress the file + if (($p_string = @gzinflate($v_data)) === false) { + // TBC + } + } + + // ----- Trace + } else { + // TBC : error : can not extract a folder in a string + } + + } + + // ----- Change abort status + if ($p_entry['status'] == "aborted") { + $p_entry['status'] = "skipped"; + + // ----- Look for post-extract callback + } elseif (isset($p_options[PCLZIP_CB_POST_EXTRACT])) { + + // ----- Generate a local information + $v_local_header = array(); + $this->privConvertHeader2FileInfo($p_entry, $v_local_header); + + // ----- Swap the content to header + $v_local_header['content'] = $p_string; + $p_string = ''; + + // ----- Call the callback + // Here I do not use call_user_func() because I need to send a reference to the + // header. + // eval('$v_result = '.$p_options[PCLZIP_CB_POST_EXTRACT].'(PCLZIP_CB_POST_EXTRACT, $v_local_header);'); + $v_result = $p_options[PCLZIP_CB_POST_EXTRACT](PCLZIP_CB_POST_EXTRACT, $v_local_header); + + // ----- Swap back the content to header + $p_string = $v_local_header['content']; + unset($v_local_header['content']); + + // ----- Look for abort result + if ($v_result == 2) { + $v_result = PCLZIP_ERR_USER_ABORTED; + } + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privReadFileHeader() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privReadFileHeader(&$p_header) + { + $v_result = 1; + + // ----- Read the 4 bytes signature + $v_binary_data = @fread($this->zip_fd, 4); + $v_data = unpack('Vid', $v_binary_data); + + // ----- Check signature + if ($v_data['id'] != 0x04034b50) { + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Invalid archive structure'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read the first 42 bytes of the header + $v_binary_data = fread($this->zip_fd, 26); + + // ----- Look for invalid block size + if (strlen($v_binary_data) != 26) { + $p_header['filename'] = ""; + $p_header['status'] = "invalid_header"; + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Invalid block size : " . strlen($v_binary_data)); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Extract the values + $v_data = unpack('vversion/vflag/vcompression/vmtime/vmdate/Vcrc/Vcompressed_size/Vsize/vfilename_len/vextra_len', $v_binary_data); + + // ----- Get filename + $p_header['filename'] = fread($this->zip_fd, $v_data['filename_len']); + + // ----- Get extra_fields + if ($v_data['extra_len'] != 0) { + $p_header['extra'] = fread($this->zip_fd, $v_data['extra_len']); + } else { + $p_header['extra'] = ''; + } + + // ----- Extract properties + $p_header['version_extracted'] = $v_data['version']; + $p_header['compression'] = $v_data['compression']; + $p_header['size'] = $v_data['size']; + $p_header['compressed_size'] = $v_data['compressed_size']; + $p_header['crc'] = $v_data['crc']; + $p_header['flag'] = $v_data['flag']; + $p_header['filename_len'] = $v_data['filename_len']; + + // ----- Recuperate date in UNIX format + $p_header['mdate'] = $v_data['mdate']; + $p_header['mtime'] = $v_data['mtime']; + if ($p_header['mdate'] && $p_header['mtime']) { + // ----- Extract time + $v_hour = ($p_header['mtime'] & 0xF800) >> 11; + $v_minute = ($p_header['mtime'] & 0x07E0) >> 5; + $v_seconde = ($p_header['mtime'] & 0x001F) * 2; + + // ----- Extract date + $v_year = (($p_header['mdate'] & 0xFE00) >> 9) + 1980; + $v_month = ($p_header['mdate'] & 0x01E0) >> 5; + $v_day = $p_header['mdate'] & 0x001F; + + // ----- Get UNIX date format + $p_header['mtime'] = @mktime($v_hour, $v_minute, $v_seconde, $v_month, $v_day, $v_year); + + } else { + $p_header['mtime'] = time(); + } + + // TBC + //for (reset($v_data); $key = key($v_data); next($v_data)) { + //} + + // ----- Set the stored filename + $p_header['stored_filename'] = $p_header['filename']; + + // ----- Set the status field + $p_header['status'] = "ok"; + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privReadCentralFileHeader() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privReadCentralFileHeader(&$p_header) + { + $v_result = 1; + + // ----- Read the 4 bytes signature + $v_binary_data = @fread($this->zip_fd, 4); + $v_data = unpack('Vid', $v_binary_data); + + // ----- Check signature + if ($v_data['id'] != 0x02014b50) { + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Invalid archive structure'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read the first 42 bytes of the header + $v_binary_data = fread($this->zip_fd, 42); + + // ----- Look for invalid block size + if (strlen($v_binary_data) != 42) { + $p_header['filename'] = ""; + $p_header['status'] = "invalid_header"; + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Invalid block size : " . strlen($v_binary_data)); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Extract the values + $p_header = unpack('vversion/vversion_extracted/vflag/vcompression/vmtime/vmdate/Vcrc/Vcompressed_size/Vsize/vfilename_len/vextra_len/vcomment_len/vdisk/vinternal/Vexternal/Voffset', $v_binary_data); + + // ----- Get filename + if ($p_header['filename_len'] != 0) { + $p_header['filename'] = fread($this->zip_fd, $p_header['filename_len']); + } else { + $p_header['filename'] = ''; + } + + // ----- Get extra + if ($p_header['extra_len'] != 0) { + $p_header['extra'] = fread($this->zip_fd, $p_header['extra_len']); + } else { + $p_header['extra'] = ''; + } + + // ----- Get comment + if ($p_header['comment_len'] != 0) { + $p_header['comment'] = fread($this->zip_fd, $p_header['comment_len']); + } else { + $p_header['comment'] = ''; + } + + // ----- Extract properties + + // ----- Recuperate date in UNIX format + //if ($p_header['mdate'] && $p_header['mtime']) + // TBC : bug : this was ignoring time with 0/0/0 + if (1) { + // ----- Extract time + $v_hour = ($p_header['mtime'] & 0xF800) >> 11; + $v_minute = ($p_header['mtime'] & 0x07E0) >> 5; + $v_seconde = ($p_header['mtime'] & 0x001F) * 2; + + // ----- Extract date + $v_year = (($p_header['mdate'] & 0xFE00) >> 9) + 1980; + $v_month = ($p_header['mdate'] & 0x01E0) >> 5; + $v_day = $p_header['mdate'] & 0x001F; + + // ----- Get UNIX date format + $p_header['mtime'] = @mktime($v_hour, $v_minute, $v_seconde, $v_month, $v_day, $v_year); + + } else { + $p_header['mtime'] = time(); + } + + // ----- Set the stored filename + $p_header['stored_filename'] = $p_header['filename']; + + // ----- Set default status to ok + $p_header['status'] = 'ok'; + + // ----- Look if it is a directory + if (substr($p_header['filename'], -1) == '/') { + //$p_header['external'] = 0x41FF0010; + $p_header['external'] = 0x00000010; + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privCheckFileHeaders() + // Description : + // Parameters : + // Return Values : + // 1 on success, + // 0 on error; + // -------------------------------------------------------------------------------- + public function privCheckFileHeaders(&$p_local_header, &$p_central_header) + { + $v_result = 1; + + // ----- Check the static values + // TBC + if ($p_local_header['filename'] != $p_central_header['filename']) { + } + if ($p_local_header['version_extracted'] != $p_central_header['version_extracted']) { + } + if ($p_local_header['flag'] != $p_central_header['flag']) { + } + if ($p_local_header['compression'] != $p_central_header['compression']) { + } + if ($p_local_header['mtime'] != $p_central_header['mtime']) { + } + if ($p_local_header['filename_len'] != $p_central_header['filename_len']) { + } + + // ----- Look for flag bit 3 + if (($p_local_header['flag'] & 8) == 8) { + $p_local_header['size'] = $p_central_header['size']; + $p_local_header['compressed_size'] = $p_central_header['compressed_size']; + $p_local_header['crc'] = $p_central_header['crc']; + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privReadEndCentralDir() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privReadEndCentralDir(&$p_central_dir) + { + $v_result = 1; + + // ----- Go to the end of the zip file + $v_size = filesize($this->zipname); + @fseek($this->zip_fd, $v_size); + if (@ftell($this->zip_fd) != $v_size) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Unable to go to the end of the archive \'' . $this->zipname . '\''); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- First try : look if this is an archive with no commentaries (most of the time) + // in this case the end of central dir is at 22 bytes of the file end + $v_found = 0; + if ($v_size > 26) { + @fseek($this->zip_fd, $v_size - 22); + if (($v_pos = @ftell($this->zip_fd)) != ($v_size - 22)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Unable to seek back to the middle of the archive \'' . $this->zipname . '\''); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read for bytes + $v_binary_data = @fread($this->zip_fd, 4); + $v_data = @unpack('Vid', $v_binary_data); + + // ----- Check signature + if ($v_data['id'] == 0x06054b50) { + $v_found = 1; + } + + $v_pos = ftell($this->zip_fd); + } + + // ----- Go back to the maximum possible size of the Central Dir End Record + if (!$v_found) { + $v_maximum_size = 65557; // 0xFFFF + 22; + if ($v_maximum_size > $v_size) { + $v_maximum_size = $v_size; + } + @fseek($this->zip_fd, $v_size - $v_maximum_size); + if (@ftell($this->zip_fd) != ($v_size - $v_maximum_size)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'Unable to seek back to the middle of the archive \'' . $this->zipname . '\''); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read byte per byte in order to find the signature + $v_pos = ftell($this->zip_fd); + $v_bytes = 0x00000000; + while ($v_pos < $v_size) { + // ----- Read a byte + $v_byte = @fread($this->zip_fd, 1); + + // ----- Add the byte + //$v_bytes = ($v_bytes << 8) | Ord($v_byte); + // Note we mask the old value down such that once shifted we can never end up with more than a 32bit number + // Otherwise on systems where we have 64bit integers the check below for the magic number will fail. + $v_bytes = (($v_bytes & 0xFFFFFF) << 8) | Ord($v_byte); + + // ----- Compare the bytes + if ($v_bytes == 0x504b0506) { + $v_pos++; + break; + } + + $v_pos++; + } + + // ----- Look if not found end of central dir + if ($v_pos == $v_size) { + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Unable to find End of Central Dir Record signature"); + + // ----- Return + return PclZip::errorCode(); + } + } + + // ----- Read the first 18 bytes of the header + $v_binary_data = fread($this->zip_fd, 18); + + // ----- Look for invalid block size + if (strlen($v_binary_data) != 18) { + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, "Invalid End of Central Dir Record size : " . strlen($v_binary_data)); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Extract the values + $v_data = unpack('vdisk/vdisk_start/vdisk_entries/ventries/Vsize/Voffset/vcomment_size', $v_binary_data); + + // ----- Check the global size + if (($v_pos + $v_data['comment_size'] + 18) != $v_size) { + + // ----- Removed in release 2.2 see readme file + // The check of the file size is a little too strict. + // Some bugs where found when a zip is encrypted/decrypted with 'crypt'. + // While decrypted, zip has training 0 bytes + if (0) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_BAD_FORMAT, 'The central dir is not at the end of the archive.' . ' Some trailing bytes exists after the archive.'); + + // ----- Return + return PclZip::errorCode(); + } + } + + // ----- Get comment + if ($v_data['comment_size'] != 0) { + $p_central_dir['comment'] = fread($this->zip_fd, $v_data['comment_size']); + } else { + $p_central_dir['comment'] = ''; + } + + $p_central_dir['entries'] = $v_data['entries']; + $p_central_dir['disk_entries'] = $v_data['disk_entries']; + $p_central_dir['offset'] = $v_data['offset']; + $p_central_dir['size'] = $v_data['size']; + $p_central_dir['disk'] = $v_data['disk']; + $p_central_dir['disk_start'] = $v_data['disk_start']; + + // TBC + //for (reset($p_central_dir); $key = key($p_central_dir); next($p_central_dir)) { + //} + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privDeleteByRule() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privDeleteByRule(&$p_result_list, &$p_options) + { + $v_result = 1; + $v_list_detail = array(); + + // ----- Open the zip file + if (($v_result = $this->privOpenFd('rb')) != 1) { + // ----- Return + return $v_result; + } + + // ----- Read the central directory informations + $v_central_dir = array(); + if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1) { + $this->privCloseFd(); + + return $v_result; + } + + // ----- Go to beginning of File + @rewind($this->zip_fd); + + // ----- Scan all the files + // ----- Start at beginning of Central Dir + $v_pos_entry = $v_central_dir['offset']; + @rewind($this->zip_fd); + if (@fseek($this->zip_fd, $v_pos_entry)) { + // ----- Close the zip file + $this->privCloseFd(); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read each entry + $v_header_list = array(); + $j_start = 0; + for ($i = 0, $v_nb_extracted = 0; $i < $v_central_dir['entries']; $i++) { + + // ----- Read the file header + $v_header_list[$v_nb_extracted] = array(); + if (($v_result = $this->privReadCentralFileHeader($v_header_list[$v_nb_extracted])) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + + return $v_result; + } + + // ----- Store the index + $v_header_list[$v_nb_extracted]['index'] = $i; + + // ----- Look for the specific extract rules + $v_found = false; + + // ----- Look for extract by name rule + if ((isset($p_options[PCLZIP_OPT_BY_NAME])) && ($p_options[PCLZIP_OPT_BY_NAME] != 0)) { + + // ----- Look if the filename is in the list + for ($j = 0; ($j < sizeof($p_options[PCLZIP_OPT_BY_NAME])) && (!$v_found); $j++) { + + // ----- Look for a directory + if (substr($p_options[PCLZIP_OPT_BY_NAME][$j], -1) == "/") { + + // ----- Look if the directory is in the filename path + if ((strlen($v_header_list[$v_nb_extracted]['stored_filename']) > strlen($p_options[PCLZIP_OPT_BY_NAME][$j])) && (substr($v_header_list[$v_nb_extracted]['stored_filename'], 0, strlen($p_options[PCLZIP_OPT_BY_NAME][$j])) == $p_options[PCLZIP_OPT_BY_NAME][$j])) { + $v_found = true; + } elseif ((($v_header_list[$v_nb_extracted]['external'] & 0x00000010) == 0x00000010) /* Indicates a folder */ && ($v_header_list[$v_nb_extracted]['stored_filename'] . '/' == $p_options[PCLZIP_OPT_BY_NAME][$j])) { + $v_found = true; + } + + // ----- Look for a filename + } elseif ($v_header_list[$v_nb_extracted]['stored_filename'] == $p_options[PCLZIP_OPT_BY_NAME][$j]) { + $v_found = true; + } + } + + // ----- Look for extract by ereg rule + // ereg() is deprecated with PHP 5.3 + /* + elseif ( (isset($p_options[PCLZIP_OPT_BY_EREG])) + && ($p_options[PCLZIP_OPT_BY_EREG] != "")) { + + if (ereg($p_options[PCLZIP_OPT_BY_EREG], $v_header_list[$v_nb_extracted]['stored_filename'])) { + $v_found = true; + } + } + */ + + // ----- Look for extract by preg rule + } elseif ((isset($p_options[PCLZIP_OPT_BY_PREG])) && ($p_options[PCLZIP_OPT_BY_PREG] != "")) { + + if (preg_match($p_options[PCLZIP_OPT_BY_PREG], $v_header_list[$v_nb_extracted]['stored_filename'])) { + $v_found = true; + } + + // ----- Look for extract by index rule + } elseif ((isset($p_options[PCLZIP_OPT_BY_INDEX])) && ($p_options[PCLZIP_OPT_BY_INDEX] != 0)) { + + // ----- Look if the index is in the list + for ($j = $j_start; ($j < sizeof($p_options[PCLZIP_OPT_BY_INDEX])) && (!$v_found); $j++) { + + if (($i >= $p_options[PCLZIP_OPT_BY_INDEX][$j]['start']) && ($i <= $p_options[PCLZIP_OPT_BY_INDEX][$j]['end'])) { + $v_found = true; + } + if ($i >= $p_options[PCLZIP_OPT_BY_INDEX][$j]['end']) { + $j_start = $j + 1; + } + + if ($p_options[PCLZIP_OPT_BY_INDEX][$j]['start'] > $i) { + break; + } + } + } else { + $v_found = true; + } + + // ----- Look for deletion + if ($v_found) { + unset($v_header_list[$v_nb_extracted]); + } else { + $v_nb_extracted++; + } + } + + // ----- Look if something need to be deleted + if ($v_nb_extracted > 0) { + + // ----- Creates a temporay file + $v_zip_temp_name = PCLZIP_TEMPORARY_DIR . uniqid('pclzip-') . '.tmp'; + + // ----- Creates a temporary zip archive + $v_temp_zip = new PclZip($v_zip_temp_name); + + // ----- Open the temporary zip file in write mode + if (($v_result = $v_temp_zip->privOpenFd('wb')) != 1) { + $this->privCloseFd(); + + // ----- Return + return $v_result; + } + + // ----- Look which file need to be kept + for ($i = 0; $i < sizeof($v_header_list); $i++) { + + // ----- Calculate the position of the header + @rewind($this->zip_fd); + if (@fseek($this->zip_fd, $v_header_list[$i]['offset'])) { + // ----- Close the zip file + $this->privCloseFd(); + $v_temp_zip->privCloseFd(); + @unlink($v_zip_temp_name); + + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_INVALID_ARCHIVE_ZIP, 'Invalid archive size'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Read the file header + $v_local_header = array(); + if (($v_result = $this->privReadFileHeader($v_local_header)) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $v_temp_zip->privCloseFd(); + @unlink($v_zip_temp_name); + + // ----- Return + return $v_result; + } + + // ----- Check that local file header is same as central file header + if ($this->privCheckFileHeaders($v_local_header, $v_header_list[$i]) != 1) { + // TBC + } + unset($v_local_header); + + // ----- Write the file header + if (($v_result = $v_temp_zip->privWriteFileHeader($v_header_list[$i])) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $v_temp_zip->privCloseFd(); + @unlink($v_zip_temp_name); + + // ----- Return + return $v_result; + } + + // ----- Read/write the data block + if (($v_result = PclZipUtilCopyBlock($this->zip_fd, $v_temp_zip->zip_fd, $v_header_list[$i]['compressed_size'])) != 1) { + // ----- Close the zip file + $this->privCloseFd(); + $v_temp_zip->privCloseFd(); + @unlink($v_zip_temp_name); + + // ----- Return + return $v_result; + } + } + + // ----- Store the offset of the central dir + $v_offset = @ftell($v_temp_zip->zip_fd); + + // ----- Re-Create the Central Dir files header + for ($i = 0; $i < sizeof($v_header_list); $i++) { + // ----- Create the file header + if (($v_result = $v_temp_zip->privWriteCentralFileHeader($v_header_list[$i])) != 1) { + $v_temp_zip->privCloseFd(); + $this->privCloseFd(); + @unlink($v_zip_temp_name); + + // ----- Return + return $v_result; + } + + // ----- Transform the header to a 'usable' info + $v_temp_zip->privConvertHeader2FileInfo($v_header_list[$i], $p_result_list[$i]); + } + + // ----- Zip file comment + $v_comment = ''; + if (isset($p_options[PCLZIP_OPT_COMMENT])) { + $v_comment = $p_options[PCLZIP_OPT_COMMENT]; + } + + // ----- Calculate the size of the central header + $v_size = @ftell($v_temp_zip->zip_fd) - $v_offset; + + // ----- Create the central dir footer + if (($v_result = $v_temp_zip->privWriteCentralHeader(sizeof($v_header_list), $v_size, $v_offset, $v_comment)) != 1) { + // ----- Reset the file list + unset($v_header_list); + $v_temp_zip->privCloseFd(); + $this->privCloseFd(); + @unlink($v_zip_temp_name); + + // ----- Return + return $v_result; + } + + // ----- Close + $v_temp_zip->privCloseFd(); + $this->privCloseFd(); + + // ----- Delete the zip file + // TBC : I should test the result ... + @unlink($this->zipname); + + // ----- Rename the temporary file + // TBC : I should test the result ... + //@rename($v_zip_temp_name, $this->zipname); + PclZipUtilRename($v_zip_temp_name, $this->zipname); + + // ----- Destroy the temporary archive + unset($v_temp_zip); + + // ----- Remove every files : reset the file + } elseif ($v_central_dir['entries'] != 0) { + $this->privCloseFd(); + + if (($v_result = $this->privOpenFd('wb')) != 1) { + return $v_result; + } + + if (($v_result = $this->privWriteCentralHeader(0, 0, 0, '')) != 1) { + return $v_result; + } + + $this->privCloseFd(); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privDirCheck() + // Description : + // Check if a directory exists, if not it creates it and all the parents directory + // which may be useful. + // Parameters : + // $p_dir : Directory path to check. + // Return Values : + // 1 : OK + // -1 : Unable to create directory + // -------------------------------------------------------------------------------- + public function privDirCheck($p_dir, $p_is_dir = false) + { + $v_result = 1; + + // ----- Remove the final '/' + if (($p_is_dir) && (substr($p_dir, -1) == '/')) { + $p_dir = substr($p_dir, 0, strlen($p_dir) - 1); + } + + // ----- Check the directory availability + if ((is_dir($p_dir)) || ($p_dir == "")) { + return 1; + } + + // ----- Extract parent directory + $p_parent_dir = dirname($p_dir); + + // ----- Just a check + if ($p_parent_dir != $p_dir) { + // ----- Look for parent directory + if ($p_parent_dir != "") { + if (($v_result = $this->privDirCheck($p_parent_dir)) != 1) { + return $v_result; + } + } + } + + // ----- Create the directory + if (!@mkdir($p_dir, 0777)) { + // ----- Error log + PclZip::privErrorLog(PCLZIP_ERR_DIR_CREATE_FAIL, "Unable to create directory '$p_dir'"); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privMerge() + // Description : + // If $p_archive_to_add does not exist, the function exit with a success result. + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privMerge(&$p_archive_to_add) + { + $v_result = 1; + + // ----- Look if the archive_to_add exists + if (!is_file($p_archive_to_add->zipname)) { + + // ----- Nothing to merge, so merge is a success + $v_result = 1; + + // ----- Return + return $v_result; + } + + // ----- Look if the archive exists + if (!is_file($this->zipname)) { + + // ----- Do a duplicate + $v_result = $this->privDuplicate($p_archive_to_add->zipname); + + // ----- Return + return $v_result; + } + + // ----- Open the zip file + if (($v_result = $this->privOpenFd('rb')) != 1) { + // ----- Return + return $v_result; + } + + // ----- Read the central directory informations + $v_central_dir = array(); + if (($v_result = $this->privReadEndCentralDir($v_central_dir)) != 1) { + $this->privCloseFd(); + + return $v_result; + } + + // ----- Go to beginning of File + @rewind($this->zip_fd); + + // ----- Open the archive_to_add file + if (($v_result = $p_archive_to_add->privOpenFd('rb')) != 1) { + $this->privCloseFd(); + + // ----- Return + return $v_result; + } + + // ----- Read the central directory informations + $v_central_dir_to_add = array(); + if (($v_result = $p_archive_to_add->privReadEndCentralDir($v_central_dir_to_add)) != 1) { + $this->privCloseFd(); + $p_archive_to_add->privCloseFd(); + + return $v_result; + } + + // ----- Go to beginning of File + @rewind($p_archive_to_add->zip_fd); + + // ----- Creates a temporay file + $v_zip_temp_name = PCLZIP_TEMPORARY_DIR . uniqid('pclzip-') . '.tmp'; + + // ----- Open the temporary file in write mode + if (($v_zip_temp_fd = @fopen($v_zip_temp_name, 'wb')) == 0) { + $this->privCloseFd(); + $p_archive_to_add->privCloseFd(); + + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open temporary file \'' . $v_zip_temp_name . '\' in binary write mode'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Copy the files from the archive to the temporary file + // TBC : Here I should better append the file and go back to erase the central dir + $v_size = $v_central_dir['offset']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = fread($this->zip_fd, $v_read_size); + @fwrite($v_zip_temp_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Copy the files from the archive_to_add into the temporary file + $v_size = $v_central_dir_to_add['offset']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = fread($p_archive_to_add->zip_fd, $v_read_size); + @fwrite($v_zip_temp_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Store the offset of the central dir + $v_offset = @ftell($v_zip_temp_fd); + + // ----- Copy the block of file headers from the old archive + $v_size = $v_central_dir['size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($this->zip_fd, $v_read_size); + @fwrite($v_zip_temp_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Copy the block of file headers from the archive_to_add + $v_size = $v_central_dir_to_add['size']; + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($p_archive_to_add->zip_fd, $v_read_size); + @fwrite($v_zip_temp_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Merge the file comments + $v_comment = $v_central_dir['comment'] . ' ' . $v_central_dir_to_add['comment']; + + // ----- Calculate the size of the (new) central header + $v_size = @ftell($v_zip_temp_fd) - $v_offset; + + // ----- Swap the file descriptor + // Here is a trick : I swap the temporary fd with the zip fd, in order to use + // the following methods on the temporary fil and not the real archive fd + $v_swap = $this->zip_fd; + $this->zip_fd = $v_zip_temp_fd; + $v_zip_temp_fd = $v_swap; + + // ----- Create the central dir footer + if (($v_result = $this->privWriteCentralHeader($v_central_dir['entries'] + $v_central_dir_to_add['entries'], $v_size, $v_offset, $v_comment)) != 1) { + $this->privCloseFd(); + $p_archive_to_add->privCloseFd(); + @fclose($v_zip_temp_fd); + $this->zip_fd = null; + + // ----- Reset the file list + unset($v_header_list); + + // ----- Return + return $v_result; + } + + // ----- Swap back the file descriptor + $v_swap = $this->zip_fd; + $this->zip_fd = $v_zip_temp_fd; + $v_zip_temp_fd = $v_swap; + + // ----- Close + $this->privCloseFd(); + $p_archive_to_add->privCloseFd(); + + // ----- Close the temporary file + @fclose($v_zip_temp_fd); + + // ----- Delete the zip file + // TBC : I should test the result ... + @unlink($this->zipname); + + // ----- Rename the temporary file + // TBC : I should test the result ... + //@rename($v_zip_temp_name, $this->zipname); + PclZipUtilRename($v_zip_temp_name, $this->zipname); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privDuplicate() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privDuplicate($p_archive_filename) + { + $v_result = 1; + + // ----- Look if the $p_archive_filename exists + if (!is_file($p_archive_filename)) { + + // ----- Nothing to duplicate, so duplicate is a success. + $v_result = 1; + + // ----- Return + return $v_result; + } + + // ----- Open the zip file + if (($v_result = $this->privOpenFd('wb')) != 1) { + // ----- Return + return $v_result; + } + + // ----- Open the temporary file in write mode + if (($v_zip_temp_fd = @fopen($p_archive_filename, 'rb')) == 0) { + $this->privCloseFd(); + + PclZip::privErrorLog(PCLZIP_ERR_READ_OPEN_FAIL, 'Unable to open archive file \'' . $p_archive_filename . '\' in binary write mode'); + + // ----- Return + return PclZip::errorCode(); + } + + // ----- Copy the files from the archive to the temporary file + // TBC : Here I should better append the file and go back to erase the central dir + $v_size = filesize($p_archive_filename); + while ($v_size != 0) { + $v_read_size = ($v_size < PCLZIP_READ_BLOCK_SIZE ? $v_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = fread($v_zip_temp_fd, $v_read_size); + @fwrite($this->zip_fd, $v_buffer, $v_read_size); + $v_size -= $v_read_size; + } + + // ----- Close + $this->privCloseFd(); + + // ----- Close the temporary file + @fclose($v_zip_temp_fd); + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privErrorLog() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function privErrorLog($p_error_code = 0, $p_error_string = '') + { + if (PCLZIP_ERROR_EXTERNAL == 1) { + PclError($p_error_code, $p_error_string); + } else { + $this->error_code = $p_error_code; + $this->error_string = $p_error_string; + } + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privErrorReset() + // Description : + // Parameters : + // -------------------------------------------------------------------------------- + public function privErrorReset() + { + if (PCLZIP_ERROR_EXTERNAL == 1) { + PclErrorReset(); + } else { + $this->error_code = 0; + $this->error_string = ''; + } + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privDisableMagicQuotes() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privDisableMagicQuotes() + { + $v_result = 1; + + // ----- Look if function exists + if ((!function_exists("get_magic_quotes_runtime")) || (!function_exists("set_magic_quotes_runtime"))) { + return $v_result; + } + + // ----- Look if already done + if ($this->magic_quotes_status != -1) { + return $v_result; + } + + // ----- Get and memorize the magic_quote value + $this->magic_quotes_status = @get_magic_quotes_runtime(); + + // ----- Disable magic_quotes + if ($this->magic_quotes_status == 1) { + @set_magic_quotes_runtime(0); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- + + // -------------------------------------------------------------------------------- + // Function : privSwapBackMagicQuotes() + // Description : + // Parameters : + // Return Values : + // -------------------------------------------------------------------------------- + public function privSwapBackMagicQuotes() + { + $v_result = 1; + + // ----- Look if function exists + if ((!function_exists("get_magic_quotes_runtime")) || (!function_exists("set_magic_quotes_runtime"))) { + return $v_result; + } + + // ----- Look if something to do + if ($this->magic_quotes_status != -1) { + return $v_result; + } + + // ----- Swap back magic_quotes + if ($this->magic_quotes_status == 1) { + @set_magic_quotes_runtime($this->magic_quotes_status); + } + + // ----- Return + return $v_result; + } + // -------------------------------------------------------------------------------- +} + +// End of class +// -------------------------------------------------------------------------------- + +// -------------------------------------------------------------------------------- +// Function : PclZipUtilPathReduction() +// Description : +// Parameters : +// Return Values : +// -------------------------------------------------------------------------------- +function PclZipUtilPathReduction($p_dir) +{ + $v_result = ""; + + // ----- Look for not empty path + if ($p_dir != "") { + // ----- Explode path by directory names + $v_list = explode("/", $p_dir); + + // ----- Study directories from last to first + $v_skip = 0; + for ($i = sizeof($v_list) - 1; $i >= 0; $i--) { + // ----- Look for current path + if ($v_list[$i] == ".") { + // ----- Ignore this directory + // Should be the first $i=0, but no check is done + } elseif ($v_list[$i] == "..") { + $v_skip++; + } elseif ($v_list[$i] == "") { + // ----- First '/' i.e. root slash + if ($i == 0) { + $v_result = "/" . $v_result; + if ($v_skip > 0) { + // ----- It is an invalid path, so the path is not modified + // TBC + $v_result = $p_dir; + $v_skip = 0; + } + + // ----- Last '/' i.e. indicates a directory + } elseif ($i == (sizeof($v_list) - 1)) { + $v_result = $v_list[$i]; + + // ----- Double '/' inside the path + } else { + // ----- Ignore only the double '//' in path, + // but not the first and last '/' + } + } else { + // ----- Look for item to skip + if ($v_skip > 0) { + $v_skip--; + } else { + $v_result = $v_list[$i] . ($i != (sizeof($v_list) - 1) ? "/" . $v_result : ""); + } + } + } + + // ----- Look for skip + if ($v_skip > 0) { + while ($v_skip > 0) { + $v_result = '../' . $v_result; + $v_skip--; + } + } + } + + // ----- Return + return $v_result; +} +// -------------------------------------------------------------------------------- + +// -------------------------------------------------------------------------------- +// Function : PclZipUtilPathInclusion() +// Description : +// This function indicates if the path $p_path is under the $p_dir tree. Or, +// said in an other way, if the file or sub-dir $p_path is inside the dir +// $p_dir. +// The function indicates also if the path is exactly the same as the dir. +// This function supports path with duplicated '/' like '//', but does not +// support '.' or '..' statements. +// Parameters : +// Return Values : +// 0 if $p_path is not inside directory $p_dir +// 1 if $p_path is inside directory $p_dir +// 2 if $p_path is exactly the same as $p_dir +// -------------------------------------------------------------------------------- +function PclZipUtilPathInclusion($p_dir, $p_path) +{ + $v_result = 1; + + // ----- Look for path beginning by ./ + if (($p_dir == '.') || ((strlen($p_dir) >= 2) && (substr($p_dir, 0, 2) == './'))) { + $p_dir = PclZipUtilTranslateWinPath(getcwd(), false) . '/' . substr($p_dir, 1); + } + if (($p_path == '.') || ((strlen($p_path) >= 2) && (substr($p_path, 0, 2) == './'))) { + $p_path = PclZipUtilTranslateWinPath(getcwd(), false) . '/' . substr($p_path, 1); + } + + // ----- Explode dir and path by directory separator + $v_list_dir = explode("/", $p_dir); + $v_list_dir_size = sizeof($v_list_dir); + $v_list_path = explode("/", $p_path); + $v_list_path_size = sizeof($v_list_path); + + // ----- Study directories paths + $i = 0; + $j = 0; + while (($i < $v_list_dir_size) && ($j < $v_list_path_size) && ($v_result)) { + + // ----- Look for empty dir (path reduction) + if ($v_list_dir[$i] == '') { + $i++; + continue; + } + if ($v_list_path[$j] == '') { + $j++; + continue; + } + + // ----- Compare the items + if (($v_list_dir[$i] != $v_list_path[$j]) && ($v_list_dir[$i] != '') && ($v_list_path[$j] != '')) { + $v_result = 0; + } + + // ----- Next items + $i++; + $j++; + } + + // ----- Look if everything seems to be the same + if ($v_result) { + // ----- Skip all the empty items + while (($j < $v_list_path_size) && ($v_list_path[$j] == '')) { + $j++; + } + while (($i < $v_list_dir_size) && ($v_list_dir[$i] == '')) { + $i++; + } + + if (($i >= $v_list_dir_size) && ($j >= $v_list_path_size)) { + // ----- There are exactly the same + $v_result = 2; + } elseif ($i < $v_list_dir_size) { + // ----- The path is shorter than the dir + $v_result = 0; + } + } + + // ----- Return + return $v_result; +} +// -------------------------------------------------------------------------------- + +// -------------------------------------------------------------------------------- +// Function : PclZipUtilCopyBlock() +// Description : +// Parameters : +// $p_mode : read/write compression mode +// 0 : src & dest normal +// 1 : src gzip, dest normal +// 2 : src normal, dest gzip +// 3 : src & dest gzip +// Return Values : +// -------------------------------------------------------------------------------- +function PclZipUtilCopyBlock($p_src, $p_dest, $p_size, $p_mode = 0) +{ + $v_result = 1; + + if ($p_mode == 0) { + while ($p_size != 0) { + $v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($p_src, $v_read_size); + @fwrite($p_dest, $v_buffer, $v_read_size); + $p_size -= $v_read_size; + } + } elseif ($p_mode == 1) { + while ($p_size != 0) { + $v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @gzread($p_src, $v_read_size); + @fwrite($p_dest, $v_buffer, $v_read_size); + $p_size -= $v_read_size; + } + } elseif ($p_mode == 2) { + while ($p_size != 0) { + $v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @fread($p_src, $v_read_size); + @gzwrite($p_dest, $v_buffer, $v_read_size); + $p_size -= $v_read_size; + } + } elseif ($p_mode == 3) { + while ($p_size != 0) { + $v_read_size = ($p_size < PCLZIP_READ_BLOCK_SIZE ? $p_size : PCLZIP_READ_BLOCK_SIZE); + $v_buffer = @gzread($p_src, $v_read_size); + @gzwrite($p_dest, $v_buffer, $v_read_size); + $p_size -= $v_read_size; + } + } + + // ----- Return + return $v_result; +} +// -------------------------------------------------------------------------------- + +// -------------------------------------------------------------------------------- +// Function : PclZipUtilRename() +// Description : +// This function tries to do a simple rename() function. If it fails, it +// tries to copy the $p_src file in a new $p_dest file and then unlink the +// first one. +// Parameters : +// $p_src : Old filename +// $p_dest : New filename +// Return Values : +// 1 on success, 0 on failure. +// -------------------------------------------------------------------------------- +function PclZipUtilRename($p_src, $p_dest) +{ + $v_result = 1; + + // ----- Try to rename the files + if (!@rename($p_src, $p_dest)) { + + // ----- Try to copy & unlink the src + if (!@copy($p_src, $p_dest)) { + $v_result = 0; + } elseif (!@unlink($p_src)) { + $v_result = 0; + } + } + + // ----- Return + return $v_result; +} +// -------------------------------------------------------------------------------- + +// -------------------------------------------------------------------------------- +// Function : PclZipUtilOptionText() +// Description : +// Translate option value in text. Mainly for debug purpose. +// Parameters : +// $p_option : the option value. +// Return Values : +// The option text value. +// -------------------------------------------------------------------------------- +function PclZipUtilOptionText($p_option) +{ + + $v_list = get_defined_constants(); + for (reset($v_list); $v_key = key($v_list); next($v_list)) { + $v_prefix = substr($v_key, 0, 10); + if ((($v_prefix == 'PCLZIP_OPT') || ($v_prefix == 'PCLZIP_CB_') || ($v_prefix == 'PCLZIP_ATT')) && ($v_list[$v_key] == $p_option)) { + return $v_key; + } + } + + $v_result = 'Unknown'; + + return $v_result; +} +// -------------------------------------------------------------------------------- + +// -------------------------------------------------------------------------------- +// Function : PclZipUtilTranslateWinPath() +// Description : +// Translate windows path by replacing '\' by '/' and optionally removing +// drive letter. +// Parameters : +// $p_path : path to translate. +// $p_remove_disk_letter : true | false +// Return Values : +// The path translated. +// -------------------------------------------------------------------------------- +function PclZipUtilTranslateWinPath($p_path, $p_remove_disk_letter = true) +{ + if (stristr(php_uname(), 'windows')) { + // ----- Look for potential disk letter + if (($p_remove_disk_letter) && (($v_position = strpos($p_path, ':')) != false)) { + $p_path = substr($p_path, $v_position + 1); + } + // ----- Change potential windows directory separator + if ((strpos($p_path, '\\') > 0) || (substr($p_path, 0, 1) == '\\')) { + $p_path = strtr($p_path, '\\', '/'); + } + } + + return $p_path; +} +// -------------------------------------------------------------------------------- diff --git a/vendor/pclzip/pclzip/readme.txt b/vendor/pclzip/pclzip/readme.txt new file mode 100644 index 000000000..115c22edf --- /dev/null +++ b/vendor/pclzip/pclzip/readme.txt @@ -0,0 +1,421 @@ +// -------------------------------------------------------------------------------- +// PclZip 2.8.2 - readme.txt +// -------------------------------------------------------------------------------- +// License GNU/LGPL - August 2009 +// Vincent Blavet - vincent@phpconcept.net +// http://www.phpconcept.net +// -------------------------------------------------------------------------------- +// $Id: readme.txt,v 1.60 2009/09/30 20:35:21 vblavet Exp $ +// -------------------------------------------------------------------------------- + + + +0 - Sommaire +============ + 1 - Introduction + 2 - What's new + 3 - Corrected bugs + 4 - Known bugs or limitations + 5 - License + 6 - Warning + 7 - Documentation + 8 - Author + 9 - Contribute + +1 - Introduction +================ + + PclZip is a library that allow you to manage a Zip archive. + + Full documentation about PclZip can be found here : http://www.phpconcept.net/pclzip + +2 - What's new +============== + + Version 2.8.2 : + - PCLZIP_CB_PRE_EXTRACT and PCLZIP_CB_POST_EXTRACT are now supported with + extraction as a string (PCLZIP_OPT_EXTRACT_AS_STRING). The string + can also be modified in the post-extract call back. + **Bugs correction : + - PCLZIP_OPT_REMOVE_ALL_PATH was not working correctly + - Remove use of eval() and do direct call to callback functions + - Correct support of 64bits systems (Thanks to WordPress team) + + Version 2.8.1 : + - Move option PCLZIP_OPT_BY_EREG to PCLZIP_OPT_BY_PREG because ereg() is + deprecated in PHP 5.3. When using option PCLZIP_OPT_BY_EREG, PclZip will + automatically replace it by PCLZIP_OPT_BY_PREG. + + Version 2.8 : + - Improve extraction of zip archive for large files by using temporary files + This feature is working like the one defined in r2.7. + Options are renamed : PCLZIP_OPT_TEMP_FILE_ON, PCLZIP_OPT_TEMP_FILE_OFF, + PCLZIP_OPT_TEMP_FILE_THRESHOLD + - Add a ratio constant PCLZIP_TEMPORARY_FILE_RATIO to configure the auto + sense of temporary file use. + - Bug correction : Reduce filepath in returned file list to remove ennoying + './/' preambule in file path. + + Version 2.7 : + - Improve creation of zip archive for large files : + PclZip will now autosense the configured memory and use temporary files + when large file is suspected. + This feature can also ne triggered by manual options in create() and add() + methods. 'PCLZIP_OPT_ADD_TEMP_FILE_ON' force the use of temporary files, + 'PCLZIP_OPT_ADD_TEMP_FILE_OFF' disable the autosense technic, + 'PCLZIP_OPT_ADD_TEMP_FILE_THRESHOLD' allow for configuration of a size + threshold to use temporary files. + Using "temporary files" rather than "memory" might take more time, but + might give the ability to zip very large files : + Tested on my win laptop with a 88Mo file : + Zip "in-memory" : 18sec (max_execution_time=30, memory_limit=180Mo) + Zip "tmporary-files" : 23sec (max_execution_time=30, memory_limit=30Mo) + - Replace use of mktime() by time() to limit the E_STRICT error messages. + - Bug correction : When adding files with full windows path (drive letter) + PclZip is now working. Before, if the drive letter is not the default + path, PclZip was not able to add the file. + + Version 2.6 : + - Code optimisation + - New attributes PCLZIP_ATT_FILE_COMMENT gives the ability to + add a comment for a specific file. (Don't really know if this is usefull) + - New attribute PCLZIP_ATT_FILE_CONTENT gives the ability to add a string + as a file. + - New attribute PCLZIP_ATT_FILE_MTIME modify the timestamp associated with + a file. + - Correct a bug. Files archived with a timestamp with 0h0m0s were extracted + with current time + - Add CRC value in the informations returned back for each file after an + action. + - Add missing closedir() statement. + - When adding a folder, and removing the path of this folder, files were + incorrectly added with a '/' at the beginning. Which means files are + related to root in unix systems. Corrected. + - Add conditional if before constant definition. This will allow users + to redefine constants without changing the file, and then improve + upgrade of pclzip code for new versions. + + Version 2.5 : + - Introduce the ability to add file/folder with individual properties (file descriptor). + This gives for example the ability to change the filename of a zipped file. + . Able to add files individually + . Able to change full name + . Able to change short name + . Compatible with global options + - New attributes : PCLZIP_ATT_FILE_NAME, PCLZIP_ATT_FILE_NEW_SHORT_NAME, PCLZIP_ATT_FILE_NEW_FULL_NAME + - New error code : PCLZIP_ERR_INVALID_ATTRIBUTE_VALUE + - Add a security control feature. PclZip can extract any file in any folder + of a system. People may use this to upload a zip file and try to override + a system file. The PCLZIP_OPT_EXTRACT_DIR_RESTRICTION will give the + ability to forgive any directory transversal behavior. + - New PCLZIP_OPT_EXTRACT_DIR_RESTRICTION : check extraction path + - New error code : PCLZIP_ERR_DIRECTORY_RESTRICTION + - Modification in PclZipUtilPathInclusion() : dir and path beginning with ./ will be prepend + by current path (getcwd()) + + Version 2.4 : + - Code improvment : try to speed up the code by removing unusefull call to pack() + - Correct bug in delete() : delete() should be called with no argument. This was not + the case in 2.3. This is corrected in 2.4. + - Correct a bug in path_inclusion function. When the path has several '../../', the + result was bad. + - Add a check for magic_quotes_runtime configuration. If enabled, PclZip will + disable it while working and det it back to its original value. + This resolve a lots of bad formated archive errors. + - Bug correction : PclZip now correctly unzip file in some specific situation, + when compressed content has same size as uncompressed content. + - Bug correction : When selecting option 'PCLZIP_OPT_REMOVE_ALL_PATH', + directories are not any more created. + - Code improvment : correct unclosed opendir(), better handling of . and .. in + loops. + + + Version 2.3 : + - Correct a bug with PHP5 : affecting the value 0xFE49FFE0 to a variable does not + give the same result in PHP4 and PHP5 .... + + Version 2.2 : + - Try development of PCLZIP_OPT_CRYPT ..... + However this becomes to a stop. To crypt/decrypt I need to multiply 2 long integers, + the result (greater than a long) is not supported by PHP. Even the use of bcmath + functions does not help. I did not find yet a solution ...; + - Add missing '/' at end of directory entries + - Check is a file is encrypted or not. Returns status 'unsupported_encryption' and/or + error code PCLZIP_ERR_UNSUPPORTED_ENCRYPTION. + - Corrected : Bad "version need to extract" field in local file header + - Add private method privCheckFileHeaders() in order to check local and central + file headers. PclZip is now supporting purpose bit flag bit 3. Purpose bit flag bit 3 gives + the ability to have a local file header without size, compressed size and crc filled. + - Add a generic status 'error' for file status + - Add control of compression type. PclZip only support deflate compression method. + Before v2.2, PclZip does not check the compression method used in an archive while + extracting. With v2.2 PclZip returns a new error status for a file using an unsupported + compression method. New status is "unsupported_compression". New error code is + PCLZIP_ERR_UNSUPPORTED_COMPRESSION. + - Add optional attribute PCLZIP_OPT_STOP_ON_ERROR. This will stop the extract of files + when errors like 'a folder with same name exists' or 'a newer file exists' or + 'a write protected file' exists, rather than set a status for the concerning file + and resume the extract of the zip. + - Add optional attribute PCLZIP_OPT_REPLACE_NEWER. This will force, during an extract' the + replacement of the file, even if a newer version of the file exists. + Note that today if a file with the same name already exists but is older it will be + replaced by the extracted one. + - Improve PclZipUtilOption() + - Support of zip archive with trailing bytes. Before 2.2, PclZip checks that the central + directory structure is the last data in the archive. Crypt encryption/decryption of + zip archive put trailing 0 bytes after decryption. PclZip is now supporting this. + + Version 2.1 : + - Add the ability to abort the extraction by using a user callback function. + The user can now return the value '2' in its callback which indicates to stop the + extraction. For a pre call-back extract is stopped before the extration of the current + file. For a post call back, the extraction is stopped after. + - Add the ability to extract a file (or several files) directly in the standard output. + This is done by the new parameter PCLZIP_OPT_EXTRACT_IN_OUTPUT with method extract(). + - Add support for parameters PCLZIP_OPT_COMMENT, PCLZIP_OPT_ADD_COMMENT, + PCLZIP_OPT_PREPEND_COMMENT. This will create, replace, add, or prepend comments + in the zip archive. + - When merging two archives, the comments are not any more lost, but merged, with a + blank space separator. + - Corrected bug : Files are not deleted when all files are asked to be deleted. + - Corrected bug : Folders with name '0' made PclZip to abort the create or add feature. + + + Version 2.0 : + ***** Warning : Some new features may break the backward compatibility for your scripts. + Please carefully read the readme file. + - Add the ability to delete by Index, name and regular expression. This feature is + performed by the method delete(), which uses the optional parameters + PCLZIP_OPT_BY_INDEX, PCLZIP_OPT_BY_NAME, PCLZIP_OPT_BY_EREG or PCLZIP_OPT_BY_PREG. + - Add the ability to extract by regular expression. To extract by regexp you must use the method + extract(), with the option PCLZIP_OPT_BY_EREG or PCLZIP_OPT_BY_PREG + (depending if you want to use ereg() or preg_match() syntax) followed by the + regular expression pattern. + - Add the ability to extract by index, directly with the extract() method. This is a + code improvment of the extractByIndex() method. + - Add the ability to extract by name. To extract by name you must use the method + extract(), with the option PCLZIP_OPT_BY_NAME followed by the filename to + extract or an array of filenames to extract. To extract all a folder, use the folder + name rather than the filename with a '/' at the end. + - Add the ability to add files without compression. This is done with a new attribute + which is PCLZIP_OPT_NO_COMPRESSION. + - Add the attribute PCLZIP_OPT_EXTRACT_AS_STRING, which allow to extract a file directly + in a string without using any file (or temporary file). + - Add constant PCLZIP_SEPARATOR for static configuration of filename separators in a single string. + The default separator is now a comma (,) and not any more a blank space. + THIS BREAK THE BACKWARD COMPATIBILITY : Please check if this may have an impact with + your script. + - Improve algorythm performance by removing the use of temporary files when adding or + extracting files in an archive. + - Add (correct) detection of empty filename zipping. This can occurs when the removed + path is the same + as a zipped dir. The dir is not zipped (['status'] = filtered), only its content. + - Add better support for windows paths (thanks for help from manus@manusfreedom.com). + - Corrected bug : When the archive file already exists with size=0, the add() method + fails. Corrected in 2.0. + - Remove the use of OS_WINDOWS constant. Use php_uname() function rather. + - Control the order of index ranges in extract by index feature. + - Change the internal management of folders (better handling of internal flag). + + + Version 1.3 : + - Removing the double include check. This is now done by include_once() and require_once() + PHP directives. + - Changing the error handling mecanism : Remove the use of an external error library. + The former PclError...() functions are replaced by internal equivalent methods. + By changing the environment variable PCLZIP_ERROR_EXTERNAL you can still use the former library. + Introducing the use of constants for error codes rather than integer values. This will help + in futur improvment. + Introduction of error handling functions like errorCode(), errorName() and errorInfo(). + - Remove the deprecated use of calling function with arguments passed by reference. + - Add the calling of extract(), extractByIndex(), create() and add() functions + with variable options rather than fixed arguments. + - Add the ability to remove all the file path while extracting or adding, + without any need to specify the path to remove. + This is available for extract(), extractByIndex(), create() and add() functionS by using + the new variable options parameters : + - PCLZIP_OPT_REMOVE_ALL_PATH : by indicating this option while calling the fct. + - Ability to change the mode of a file after the extraction (chmod()). + This is available for extract() and extractByIndex() functionS by using + the new variable options parameters. + - PCLZIP_OPT_SET_CHMOD : by setting the value of this option. + - Ability to definition call-back options. These call-back will be called during the adding, + or the extracting of file (extract(), extractByIndex(), create() and add() functions) : + - PCLZIP_CB_PRE_EXTRACT : will be called before each extraction of a file. The user + can trigerred the change the filename of the extracted file. The user can triggered the + skip of the extraction. This is adding a 'skipped' status in the file list result value. + - PCLZIP_CB_POST_EXTRACT : will be called after each extraction of a file. + Nothing can be triggered from that point. + - PCLZIP_CB_PRE_ADD : will be called before each add of a file. The user + can trigerred the change the stored filename of the added file. The user can triggered the + skip of the add. This is adding a 'skipped' status in the file list result value. + - PCLZIP_CB_POST_ADD : will be called after each add of a file. + Nothing can be triggered from that point. + - Two status are added in the file list returned as function result : skipped & filename_too_long + 'skipped' is used when a call-back function ask for skipping the file. + 'filename_too_long' is used while adding a file with a too long filename to archive (the file is + not added) + - Adding the function PclZipUtilPathInclusion(), that check the inclusion of a path into + a directory. + - Add a check of the presence of the archive file before some actions (like list, ...) + - Add the initialisation of field "index" in header array. This means that by + default index will be -1 when not explicitly set by the methods. + + Version 1.2 : + - Adding a duplicate function. + - Adding a merge function. The merge function is a "quick merge" function, + it just append the content of an archive at the end of the first one. There + is no check for duplicate files or more recent files. + - Improve the search of the central directory end. + + Version 1.1.2 : + + - Changing the license of PclZip. PclZip is now released under the GNU / LGPL license + (see License section). + - Adding the optional support of a static temporary directory. You will need to configure + the constant PCLZIP_TEMPORARY_DIR if you want to use this feature. + - Improving the rename() function. In some cases rename() does not work (different + Filesystems), so it will be replaced by a copy() + unlink() functions. + + Version 1.1.1 : + + - Maintenance release, no new feature. + + Version 1.1 : + + - New method Add() : adding files in the archive + - New method ExtractByIndex() : partial extract of the archive, files are identified by + their index in the archive + - New method DeleteByIndex() : delete some files/folder entries from the archive, + files are identified by their index in the archive. + - Adding a test of the zlib extension presence. If not present abort the script. + + Version 1.0.1 : + + - No new feature + + +3 - Corrected bugs +================== + + Corrected in Version 2.0 : + - Corrected : During an extraction, if a call-back fucntion is used and try to skip + a file, all the extraction process is stopped. + + Corrected in Version 1.3 : + - Corrected : Support of static synopsis for method extract() is broken. + - Corrected : invalid size of archive content field (0xFF) should be (0xFFFF). + - Corrected : When an extract is done with a remove_path parameter, the entry for + the directory with exactly the same path is not skipped/filtered. + - Corrected : extractByIndex() and deleteByIndex() were not managing index in the + right way. For example indexes '1,3-5,11' will only extract files 1 and 11. This + is due to a sort of the index resulting table that puts 11 before 3-5 (sort on + string and not interger). The sort is temporarilly removed, this means that + you must provide a sorted list of index ranges. + + Corrected in Version 1.2 : + + - Nothing. + + Corrected in Version 1.1.2 : + + - Corrected : Winzip is unable to delete or add new files in a PclZip created archives. + + Corrected in Version 1.1.1 : + + - Corrected : When archived file is not compressed (0% compression), the + extract method fails. + + Corrected in Version 1.1 : + + - Corrected : Adding a complete tree of folder may result in a bad archive + creation. + + Corrected in Version 1.0.1 : + + - Corrected : Error while compressing files greater than PCLZIP_READ_BLOCK_SIZE (default=1024). + + +4 - Known bugs or limitations +============================= + + Please publish bugs reports in SourceForge : + http://sourceforge.net/tracker/?group_id=40254&atid=427564 + + In Version 2.x : + - PclZip does only support file uncompressed or compressed with deflate (compression method 8) + - PclZip does not support password protected zip archive + - Some concern were seen when changing mtime of a file while archiving. + Seems to be linked to Daylight Saving Time (PclTest_changing_mtime). + + In Version 1.2 : + + - merge() methods does not check for duplicate files or last date of modifications. + + In Version 1.1 : + + - Limitation : Using 'extract' fields in the file header in the zip archive is not supported. + - WinZip is unable to delete a single file in a PclZip created archive. It is also unable to + add a file in a PclZip created archive. (Corrected in v.1.2) + + In Version 1.0.1 : + + - Adding a complete tree of folder may result in a bad archive + creation. (Corrected in V.1.1). + - Path given to methods must be in the unix format (/) and not the Windows format (\). + Workaround : Use only / directory separators. + - PclZip is using temporary files that are sometime the name of the file with a .tmp or .gz + added suffix. Files with these names may already exist and may be overwritten. + Workaround : none. + - PclZip does not check if the zlib extension is present. If it is absent, the zip + file is not created and the lib abort without warning. + Workaround : enable the zlib extension on the php install + + In Version 1.0 : + + - Error while compressing files greater than PCLZIP_READ_BLOCK_SIZE (default=1024). + (Corrected in v.1.0.1) + - Limitation : Multi-disk zip archive are not supported. + + +5 - License +=========== + + Since version 1.1.2, PclZip Library is released under GNU/LGPL license. + This library is free, so you can use it at no cost. + + HOWEVER, if you release a script, an application, a library or any kind of + code using PclZip library (or a part of it), YOU MUST : + - Indicate in the documentation (or a readme file), that your work + uses PclZip Library, and make a reference to the author and the web site + http://www.phpconcept.net + - Gives the ability to the final user to update the PclZip libary. + + I will also appreciate that you send me a mail (vincent@phpconcept.net), just to + be aware that someone is using PclZip. + + For more information about GNU/LGPL license : http://www.gnu.org + +6 - Warning +================= + + This library and the associated files are non commercial, non professional work. + It should not have unexpected results. However if any damage is caused by this software + the author can not be responsible. + The use of this software is at the risk of the user. + +7 - Documentation +================= + PclZip User Manuel is available in English on PhpConcept : http://www.phpconcept.net/pclzip/man/en/index.php + A Russian translation was done by Feskov Kuzma : http://php.russofile.ru/ru/authors/unsort/zip/ + +8 - Author +========== + + This software was written by Vincent Blavet (vincent@phpconcept.net) on its leasure time. + +9 - Contribute +============== + If you want to contribute to the development of PclZip, please contact vincent@phpconcept.net. + If you can help in financing PhpConcept hosting service, please go to + http://www.phpconcept.net/soutien.php diff --git a/vendor/qiniu/php-sdk/.gitignore b/vendor/qiniu/php-sdk/.gitignore new file mode 100644 index 000000000..b4ad32034 --- /dev/null +++ b/vendor/qiniu/php-sdk/.gitignore @@ -0,0 +1,11 @@ +*.phar +*.zip +build/artifacts +phpunit.xml +phpunit.functional.xml +.DS_Store +.swp +.build +composer.lock +vendor +src/package.xml diff --git a/vendor/qiniu/php-sdk/.scrutinizer.yml b/vendor/qiniu/php-sdk/.scrutinizer.yml new file mode 100644 index 000000000..8d9304c52 --- /dev/null +++ b/vendor/qiniu/php-sdk/.scrutinizer.yml @@ -0,0 +1,35 @@ +filter: + excluded_paths: [tests/*] +checks: + php: + code_rating: true + remove_extra_empty_lines: true + remove_php_closing_tag: true + remove_trailing_whitespace: true + fix_use_statements: + remove_unused: true + preserve_multiple: false + preserve_blanklines: true + order_alphabetically: true + fix_php_opening_tag: true + fix_linefeed: true + fix_line_ending: true + fix_identation_4spaces: true + fix_doc_comments: true +tools: + external_code_coverage: + timeout: 1200 + runs: 3 + php_analyzer: true + php_code_coverage: false + php_code_sniffer: + config: + standard: PSR2 + filter: + paths: ['src'] + php_loc: + enabled: true + excluded_dirs: [vendor, tests] + php_cpd: + enabled: true + excluded_dirs: [vendor, tests] diff --git a/vendor/qiniu/php-sdk/.travis.yml b/vendor/qiniu/php-sdk/.travis.yml new file mode 100644 index 000000000..d05be1fd3 --- /dev/null +++ b/vendor/qiniu/php-sdk/.travis.yml @@ -0,0 +1,22 @@ +# using docker container +sudo: false + +language: php +php: + - 5.3 + - 5.4 + - 5.5 + - 5.6 + - 7.0 +before_script: + - export QINIU_TEST_ENV="travis" + - travis_retry composer self-update + - travis_retry composer install --no-interaction --prefer-source --dev +script: + - ./vendor/bin/phpcs --standard=PSR2 src + - ./vendor/bin/phpcs --standard=PSR2 examples + - ./vendor/bin/phpcs --standard=PSR2 tests + - ./vendor/bin/phpunit --coverage-text --coverage-clover=coverage.clover tests/Qiniu/Tests/ +after_script: + - wget https://scrutinizer-ci.com/ocular.phar + - php ocular.phar code-coverage:upload --format=php-clover coverage.clover diff --git a/vendor/qiniu/php-sdk/CHANGELOG.md b/vendor/qiniu/php-sdk/CHANGELOG.md new file mode 100644 index 000000000..f6a9ad0f7 --- /dev/null +++ b/vendor/qiniu/php-sdk/CHANGELOG.md @@ -0,0 +1,74 @@ +#Changelog + +## 7.1.3 (2016-11-18) +### 澧炲姞 +* move, copy鎿嶄綔澧炲姞force鍙傛暟 + +## 7.1.2 (2016-11-12) +### 淇 +* 鏄庣‘鎶涘嚭鑾峰彇鍚勫尯鍩熷煙鍚嶅け璐ユ椂鐨勬姤閿 + +## 7.1.1 (2016-11-02) +### 淇 +* 澶氬尯鍩熼厤缃枃浠跺瓨鍌ㄧ洰褰曚粠home淇敼鍒皌mp鐩綍 + + +## 7.1.0 (2016-10-22) +### 澧炲姞 +* 澶氬瓨鍌ㄥ尯鍩熺殑鏀寔 + +## 7.0.8 (2016-07-19) +### 澧炲姞 +* demo +* https url 鏀寔 +* deleteAfterDays 绛栫暐 +* 娣诲姞鍥剧墖澶勭悊閾炬帴缁熶竴鎷兼帴鏂规硶 by @SherlockRen + +## 7.0.7 (2016-01-12) +### 淇 +* PersistentFop鍙傛暟pipeline鍜宯otify_url澶辨晥 +* resume 妯″紡 close file inputstream + +## 7.0.6 (2015-12-05) +### 淇 +* php7.0 Json 瀵圭┖瀛楃涓茶В鏋愬崟鍏冩祴璇曟姤閿 +* 寮鍚畨鍏ㄦā寮忔垨鑰呰缃彲鎿嶄綔鐩綍鏍戞椂锛岃缃瓹URLOPT_FOLLOWLOCATION鎶ラ敊, by @twocabbages +* fetch 鏀寔涓嶆寚瀹歬ey, by @sinkcup + +## 7.0.5 (2015-10-29) +### 澧炲姞 +* 澧炲姞涓婁紶绛栫暐鏈灏忔枃浠跺ぇ灏忛檺鍒 fsizeMin +* 澧炲姞甯歌examples + +## 7.0.4 (2015-07-23) +### 淇 +* 涓浜涘湴鏂圭殑涓ユ牸姣旇緝妫鏌 +* resumeupload 澶囩敤鍦板潃澶辨晥 + +## 7.0.3 (2015-07-10) +### 淇敼 +* 澶歾one 鏀寔 + +## 7.0.2 (2015-04-18) +### 淇敼 +* fetch 鎺ュ彛杩斿洖鍐呭璋冩暣 +* pfop 鎺ュ彛璋冩暣 + +###淇 +* exception 绫昏皟鐢 + +## 7.0.1 (2015-03-27) +### 澧炲姞 +* 澧炲姞浠g爜娉ㄩ噴 + +## 7.0.0 (2015-02-03) + +### 澧炲姞 +* 绠鍖栦笂浼犳帴鍙 +* 鑷姩閫夋嫨鏂偣缁笂浼犺繕鏄洿浼 +* 閲嶆瀯浠g爜锛屾帴鍙e拰鍐呴儴缁撴瀯鏇存竻鏅 +* 鏀瑰彉mime +* 浠g爜瑕嗙洊搴︽姤鍛 +* policy鏀逛负array, 渚夸簬鐏垫椿澧炲姞锛屽苟鍔犲叆杩囨湡瀛楁妫鏌 +* 鏂囦欢鍒楄〃鏀寔鐩綍褰㈠紡 +* 鍒╃敤鍏冪紪绋嬫柟寮忔敮鎸 fop 鍜 pfop diff --git a/vendor/qiniu/php-sdk/CONTRIBUTING.md b/vendor/qiniu/php-sdk/CONTRIBUTING.md new file mode 100644 index 000000000..0466bf970 --- /dev/null +++ b/vendor/qiniu/php-sdk/CONTRIBUTING.md @@ -0,0 +1,30 @@ +# 璐$尞浠g爜鎸囧崡 + +鎴戜滑闈炲父娆㈣繋澶у鏉ヨ础鐚唬鐮侊紝鎴戜滑浼氬悜璐$尞鑰呰嚧浠ユ渶璇氭寶鐨勬暚鎰忋 + +涓鑸彲浠ラ氳繃鍦℅ithub涓婃彁浜Pull Request](https://github.com/qiniu/php-sdk)鏉ヨ础鐚唬鐮併 + +## Pull Request瑕佹眰 + +- **[PSR-2 缂栫爜椋庢牸鏍囧噯](https://github.com/php-fig/fig-standards/blob/master/accepted/PSR-2-coding-style-guide.md)** 銆傝閫氳繃椤圭洰涓殑code sniffer妫鏌ャ + +- **浠g爜鏍煎紡** 鎻愪氦鍓 璇锋寜 ./vendor/bin/phpcbf --standard=PSR2 杩涜鏍煎紡鍖栥 + +- **蹇呴』娣诲姞娴嬭瘯锛** - 濡傛灉娌℃湁娴嬭瘯锛堝崟鍏冩祴璇曘侀泦鎴愭祴璇曢兘鍙互锛夛紝閭d箞鎻愪氦鐨勮ˉ涓佹槸涓嶄細閫氳繃鐨勩 + +- **璁板緱鏇存柊鏂囨。** - 淇濊瘉`README.md`浠ュ強鍏朵粬鐩稿叧鏂囨。鍙婃椂鏇存柊锛屽拰浠g爜鐨勫彉鏇翠繚鎸佷竴鑷存с + +- **鑰冭檻鎴戜滑鐨勫彂甯冨懆鏈** - 鎴戜滑鐨勭増鏈彿浼氭湇浠嶽SemVer v2.0.0](http://semver.org/)锛屾垜浠粷瀵逛笉浼氶殢鎰忓彉鏇村澶栫殑API銆 + +- **鍒涘缓feature鍒嗘敮** - 鏈濂戒笉瑕佷粠浣犵殑master鍒嗘敮鎻愪氦 pull request銆 + +- **涓涓猣eature鎻愪氦涓涓猵ull璇锋眰** - 濡傛灉浣犵殑浠g爜鍙樻洿浜嗗涓搷浣滐紝閭e氨鎻愪氦澶氫釜pull璇锋眰鍚с + +- **娓呮櫚鐨刢ommit鍘嗗彶** - 淇濊瘉浣犵殑pull璇锋眰鐨勬瘡娆ommit鎿嶄綔閮芥槸鏈夋剰涔夌殑銆傚鏋滀綘寮鍙戜腑闇瑕佹墽琛屽娆$殑鍗虫椂commit鎿嶄綔锛岄偅涔堣鎶婂畠浠斁鍒颁竴璧峰啀鎻愪氦pull璇锋眰銆 + +## 杩愯娴嬭瘯 + +``` bash +./vendor/bin/phpunit tests/Qiniu/Tests/ + +``` diff --git a/vendor/qiniu/php-sdk/LICENSE b/vendor/qiniu/php-sdk/LICENSE new file mode 100644 index 000000000..ba646be91 --- /dev/null +++ b/vendor/qiniu/php-sdk/LICENSE @@ -0,0 +1,22 @@ +The MIT License (MIT) + +Copyright (c) 2014 Qiniu, Ltd. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + diff --git a/vendor/qiniu/php-sdk/README.md b/vendor/qiniu/php-sdk/README.md new file mode 100644 index 000000000..453eef4d1 --- /dev/null +++ b/vendor/qiniu/php-sdk/README.md @@ -0,0 +1,75 @@ +# Qiniu Resource Storage SDK for PHP +[![doxygen.io](http://doxygen.io/github.com/qiniu/php-sdk/?status.svg)](http://doxygen.io/github.com/qiniu/php-sdk/) +[![Software License](https://img.shields.io/badge/license-MIT-brightgreen.svg)](LICENSE) +[![Build Status](https://travis-ci.org/qiniu/php-sdk.svg)](https://travis-ci.org/qiniu/php-sdk) +[![Latest Stable Version](https://img.shields.io/packagist/v/qiniu/php-sdk.svg)](https://packagist.org/packages/qiniu/php-sdk) +[![Total Downloads](https://img.shields.io/packagist/dt/qiniu/php-sdk.svg)](https://packagist.org/packages/qiniu/php-sdk) +[![Scrutinizer Code Quality](https://scrutinizer-ci.com/g/qiniu/php-sdk/badges/quality-score.png?b=master)](https://scrutinizer-ci.com/g/qiniu/php-sdk/?branch=master) +[![Code Coverage](https://scrutinizer-ci.com/g/qiniu/php-sdk/badges/coverage.png?b=master)](https://scrutinizer-ci.com/g/qiniu/php-sdk/?branch=master) +[![Join Chat](https://badges.gitter.im/Join%20Chat.svg)](https://gitter.im/qiniu/php-sdk?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) +[![@qiniu on weibo](http://img.shields.io/badge/weibo-%40qiniutek-blue.svg)](http://weibo.com/qiniutek) + +## 瀹夎 + +* 閫氳繃composer锛岃繖鏄帹鑽愮殑鏂瑰紡锛屽彲浠ヤ娇鐢╟omposer.json 澹版槑渚濊禆锛屾垨鑰呰繍琛屼笅闈㈢殑鍛戒护銆係DK 鍖呭凡缁忔斁鍒拌繖閲 [`qiniu/php-sdk`][install-packagist] 銆 +```bash +$ composer require qiniu/php-sdk +``` +* 鐩存帴涓嬭浇瀹夎锛孲DK 娌℃湁渚濊禆鍏朵粬绗笁鏂瑰簱锛屼絾闇瑕佸弬鐓 composer鐨刟utoloader锛屽鍔犱竴涓嚜宸辩殑autoloader绋嬪簭銆 + +## 杩愯鐜 + +| Qiniu SDK鐗堟湰 | PHP 鐗堟湰 | +|:--------------------:|:---------------------------:| +| 7.x | cURL extension, 5.3 - 5.6,7.0 | +| 6.x | cURL extension, 5.2 - 5.6 | + +## 浣跨敤鏂规硶 + +### 涓婁紶 +```php +use Qiniu\Storage\UploadManager; +use Qiniu\Auth; +... + $upManager = new UploadManager(); + $auth = new Auth($accessKey, $secretKey); + $token = $auth->uploadToken($bucketName); + list($ret, $error) = $upManager->put($token, 'formput', 'hello world'); +... +``` + +## 娴嬭瘯 + +``` bash +$ ./vendor/bin/phpunit tests/Qiniu/Tests/ +``` + +## 甯歌闂 + +- $error淇濈暀浜嗚姹傚搷搴旂殑淇℃伅锛屽け璐ユ儏鍐典笅ret 涓簄one, 灏$error鍙互鎵撳嵃鍑烘潵锛屾彁浜ょ粰鎴戜滑銆 +- API 鐨勪娇鐢 demo 鍙互鍙傝 [鍗曞厓娴嬭瘯](https://github.com/qiniu/php-sdk/blob/master/tests)銆 + +## 浠g爜璐$尞 + +璇︽儏鍙傝僛浠g爜鎻愪氦鎸囧崡](https://github.com/qiniu/php-sdk/blob/master/CONTRIBUTING.md)銆 + +## 璐$尞璁板綍 + +- [鎵鏈夎础鐚匽(https://github.com/qiniu/php-sdk/contributors) + +## 鑱旂郴鎴戜滑 + +- 濡傛灉闇瑕佸府鍔╋紝璇锋彁浜ゅ伐鍗曪紙鍦╬ortal鍙充晶鐐瑰嚮鍜ㄨ鍜屽缓璁彁浜ゅ伐鍗曪紝鎴栬呯洿鎺ュ悜 support@qiniu.com 鍙戦侀偖浠讹級 +- 濡傛灉鏈変粈涔堥棶棰橈紝鍙互鍒伴棶绛旂ぞ鍖烘彁闂紝[闂瓟绀惧尯](http://qiniu.segmentfault.com/) +- 鏇磋缁嗙殑鏂囨。锛岃[瀹樻柟鏂囨。绔橾(http://developer.qiniu.com/) +- 濡傛灉鍙戠幇浜哹ug锛 娆㈣繋鎻愪氦 [issue](https://github.com/qiniu/php-sdk/issues) +- 濡傛灉鏈夊姛鑳介渶姹傦紝娆㈣繋鎻愪氦 [issue](https://github.com/qiniu/php-sdk/issues) +- 濡傛灉瑕佹彁浜や唬鐮侊紝娆㈣繋鎻愪氦 pull request +- 娆㈣繋鍏虫敞鎴戜滑鐨刐寰俊](http://www.qiniu.com/#weixin) [寰崥](http://weibo.com/qiniutek)锛屽強鏃惰幏鍙栧姩鎬佷俊鎭 + +## 浠g爜璁稿彲 + +The MIT License (MIT).璇︽儏瑙 [License鏂囦欢](https://github.com/qiniu/php-sdk/blob/master/LICENSE). + +[packagist]: http://packagist.org +[install-packagist]: https://packagist.org/packages/qiniu/php-sdk diff --git a/vendor/qiniu/php-sdk/autoload.php b/vendor/qiniu/php-sdk/autoload.php new file mode 100644 index 000000000..4379b91da --- /dev/null +++ b/vendor/qiniu/php-sdk/autoload.php @@ -0,0 +1,14 @@ +=5.3.3" + }, + "require-dev": { + "phpunit/phpunit": "~4.0", + "squizlabs/php_codesniffer": "~2.3" + }, + "autoload": { + "psr-4": {"Qiniu\\": "src/Qiniu"}, + "files": ["src/Qiniu/functions.php"] + } +} diff --git a/vendor/qiniu/php-sdk/examples/README.md b/vendor/qiniu/php-sdk/examples/README.md new file mode 100644 index 000000000..6cf8e30af --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/README.md @@ -0,0 +1,10 @@ +# examples + +杩欎簺 examples 鏃ㄥ湪甯姪浣犲揩閫熶簡瑙d娇鐢ㄤ竷鐗涚殑sdk銆傝繖浜沝emo閮芥槸鍙互鐩存帴杩愯鐨勶紝 浣嗘槸鍦ㄨ繍琛屼箣鍓嶉渶瑕佸~涓婃偍鑷繁鐨勫弬鏁般 + +姣斿锛 + +* `$bucket` 闇瑕佸~涓婃偍鎯虫搷浣滅殑 [bucket鍚嶅瓧](http://developer.qiniu.com/docs/v6/api/overview/concepts.html#bucket)銆 +* `$accessKey` 鍜 `$secretKey` 鍙互鍦ㄦ垜浠殑[绠$悊鍚庡彴](https://portal.qiniu.com/setting/key)鎵惧埌銆 +* 鍦ㄨ繘琛宍瑙嗛杞爜`锛 `鍘嬬缉鏂囦欢`绛夊紓姝ユ搷浣滄椂 闇瑕佷娇鐢ㄥ埌鐨勯槦鍒楀悕绉颁篃鍙互鍦ㄦ垜浠琜绠$悊鍚庡彴](https://portal.qiniu.com/mps/pipeline)鏂板缓銆 + diff --git a/vendor/qiniu/php-sdk/examples/auth.php b/vendor/qiniu/php-sdk/examples/auth.php new file mode 100644 index 000000000..1ddbec023 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/auth.php @@ -0,0 +1,11 @@ +stat($bucket, $key); +echo "\n====> $key stat : \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} + +//灏嗘枃浠朵粠鏂囦欢$key 澶嶅埗鍒版枃浠$key2銆 鍙互鍦ㄤ笉鍚宐ucket澶嶅埗 +$key2 = 'php-logo2.png'; +$err = $bucketMgr->copy($bucket, $key, $bucket, $key2); +echo "\n====> copy $key to $key2 : \n"; +if ($err !== null) { + var_dump($err); +} else { + echo "Success!"; +} + +//灏嗘枃浠朵粠鏂囦欢$key2 绉诲姩鍒版枃浠$key3銆 鍙互鍦ㄤ笉鍚宐ucket绉诲姩 +$key3 = 'php-logo3.png'; +$err = $bucketMgr->move($bucket, $key2, $bucket, $key3); +echo "\n====> move $key2 to $key3 : \n"; +if ($err !== null) { + var_dump($err); +} else { + echo "Success!"; +} + +//鍒犻櫎$bucket 涓殑鏂囦欢 $key +$err = $bucketMgr->delete($bucket, $key3); +echo "\n====> delete $key3 : \n"; +if ($err !== null) { + var_dump($err); +} else { + echo "Success!"; +} diff --git a/vendor/qiniu/php-sdk/examples/bucket_mgr_init.php b/vendor/qiniu/php-sdk/examples/bucket_mgr_init.php new file mode 100644 index 000000000..26bfcb1e5 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/bucket_mgr_init.php @@ -0,0 +1,14 @@ +verifyCallback($contentType, $authorization, $url, $callbackBody); + +if ($isQiniuCallback) { + $resp = array('ret' => 'success'); +} else { + $resp = array('ret' => 'failed'); +} + +echo json_encode($resp); diff --git a/vendor/qiniu/php-sdk/examples/download_token.php b/vendor/qiniu/php-sdk/examples/download_token.php new file mode 100644 index 000000000..0b7a0becc --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/download_token.php @@ -0,0 +1,17 @@ +/ +$baseUrl = 'http://sslayer.qiniudn.com/1.jpg?imageView2/1/h/500'; +// 瀵归摼鎺ヨ繘琛岀鍚 +$signedUrl = $auth->privateDownloadUrl($baseUrl); + +echo $signedUrl; diff --git a/vendor/qiniu/php-sdk/examples/fetch.php b/vendor/qiniu/php-sdk/examples/fetch.php new file mode 100644 index 000000000..d8f01db7b --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/fetch.php @@ -0,0 +1,23 @@ +fetch($url, $bucket, $key); +echo "=====> fetch $url to bucket: $bucket key: $key\n"; +if ($err !== null) { + var_dump($err); +} else { + echo 'Success'; +} diff --git a/vendor/qiniu/php-sdk/examples/file_copy.php b/vendor/qiniu/php-sdk/examples/file_copy.php new file mode 100644 index 000000000..a8e0c3657 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/file_copy.php @@ -0,0 +1,28 @@ +copy($bucket, $key, $bucket, $key2); +echo "\n====> copy $key to $key2 : \n"; +if ($err !== null) { + var_dump($err); +} else { + echo "Success!"; +} diff --git a/vendor/qiniu/php-sdk/examples/file_delete.php b/vendor/qiniu/php-sdk/examples/file_delete.php new file mode 100644 index 000000000..a24589d7f --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/file_delete.php @@ -0,0 +1,27 @@ +delete($bucket, $key); +echo "\n====> delete $key : \n"; +if ($err !== null) { + var_dump($err); +} else { + echo "Success!"; +} diff --git a/vendor/qiniu/php-sdk/examples/file_move.php b/vendor/qiniu/php-sdk/examples/file_move.php new file mode 100644 index 000000000..a8ff1d459 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/file_move.php @@ -0,0 +1,28 @@ +move($bucket, $key2, $bucket, $key3); +echo "\n====> move $key to $key2 : \n"; +if ($err !== null) { + var_dump($err); +} else { + echo "Success!"; +} diff --git a/vendor/qiniu/php-sdk/examples/file_stat.php b/vendor/qiniu/php-sdk/examples/file_stat.php new file mode 100644 index 000000000..1c884990a --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/file_stat.php @@ -0,0 +1,27 @@ +stat($bucket, $key); +echo "\n====> $key stat : \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/image_url_builder.php b/vendor/qiniu/php-sdk/examples/image_url_builder.php new file mode 100644 index 000000000..e99a72868 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/image_url_builder.php @@ -0,0 +1,74 @@ + + */ +$thumbLink = $imageUrlBuilder->thumbnail($url, 1, 100, 100); + +// 鍑芥暟鏂瑰紡璋冪敤 涔熷彲鎷兼帴澶氫釜鎿嶄綔鍙傛暟 鍥剧墖+姘村嵃 +$thumbLink2 = \Qiniu\thumbnail($url2, 1, 100, 100); +var_dump($thumbLink, $thumbLink2); + +/** + * 鍥剧墖姘村嵃 + * + * @param string $url 鍥剧墖閾炬帴 + * @param string $image 姘村嵃鍥剧墖閾炬帴 + * @param numeric $dissolve 閫忔槑搴 [鍙塢 + * @param string $gravity 姘村嵃浣嶇疆 [鍙塢 + * @param numeric $dx 妯酱杈硅窛 [鍙塢 + * @param numeric $dy 绾佃酱杈硅窛 [鍙塢 + * @param numeric $watermarkScale 鑷傚簲鍘熷浘鐨勭煭杈规瘮渚 [鍙塢 + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/watermark.html + * @return string + * @author Sherlock Ren + */ +$waterLink = $imageUrlBuilder->waterImg($url, $waterImage); +// 鍑芥暟璋冪敤鏂规硶 +//$waterLink = \Qiniu\waterImg($url, $waterImage); +var_dump($waterLink); + +/** + * 鏂囧瓧姘村嵃 + * + * @param string $url 鍥剧墖閾炬帴 + * @param string $text 鏂囧瓧 + * @param string $font 鏂囧瓧瀛椾綋 + * @param string $fontSize 鏂囧瓧瀛楀彿 + * @param string $fontColor 鏂囧瓧棰滆壊 [鍙塢 + * @param numeric $dissolve 閫忔槑搴 [鍙塢 + * @param string $gravity 姘村嵃浣嶇疆 [鍙塢 + * @param numeric $dx 妯酱杈硅窛 [鍙塢 + * @param numeric $dy 绾佃酱杈硅窛 [鍙塢 + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/watermark.html#text-watermark + * @return string + * @author Sherlock Ren + */ +$textLink = $imageUrlBuilder->waterText($url, '浣犵瀰鍟', '寰蒋闆呴粦', 300); +// 鍑芥暟璋冪敤鏂规硶 +// $textLink = \Qiniu\waterText($url, '浣犵瀰鍟', '寰蒋闆呴粦', 300); +var_dump($textLink); diff --git a/vendor/qiniu/php-sdk/examples/list_file.php b/vendor/qiniu/php-sdk/examples/list_file.php new file mode 100644 index 000000000..efe5a523f --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/list_file.php @@ -0,0 +1,34 @@ +listFiles($bucket, $prefix, $marker, $limit); +if ($err !== null) { + echo "\n====> list file err: \n"; + var_dump($err); +} else { + echo "Marker: $marker\n"; + echo "\nList Iterms====>\n"; + var_dump($iterms); +} diff --git a/vendor/qiniu/php-sdk/examples/mkzip.php b/vendor/qiniu/php-sdk/examples/mkzip.php new file mode 100644 index 000000000..6c8ac59d3 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/mkzip.php @@ -0,0 +1,42 @@ +execute($key, $fops); + +echo "\n====> pfop mkzip result: \n"; +if ($err != null) { + var_dump($err); +} else { + echo "PersistentFop Id: $id\n"; + + $res = "http://api.qiniu.com/status/get/prefop?id=$id"; + echo "Processing result: $res"; +} diff --git a/vendor/qiniu/php-sdk/examples/notify.php b/vendor/qiniu/php-sdk/examples/notify.php new file mode 100644 index 000000000..0cb8d9e69 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/notify.php @@ -0,0 +1,7 @@ +execute($key, $fops); +echo "\n====> pfop avthumb result: \n"; +if ($err != null) { + var_dump($err); +} else { + echo "PersistentFop Id: $id\n"; +} + +//鏌ヨ杞爜鐨勮繘搴﹀拰鐘舵 +list($ret, $err) = $pfop->status($id); +echo "\n====> pfop avthumb status: \n"; +if ($err != null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/persistent_fop_init.php b/vendor/qiniu/php-sdk/examples/persistent_fop_init.php new file mode 100644 index 000000000..2df01e9fe --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/persistent_fop_init.php @@ -0,0 +1,19 @@ +execute($key, $fops); +echo "\n====> pfop avthumb result: \n"; +if ($err != null) { + var_dump($err); +} else { + echo "PersistentFop Id: $id\n"; +} + +//鏌ヨ杞爜鐨勮繘搴﹀拰鐘舵 +list($ret, $err) = $pfop->status($id); +echo "\n====> pfop avthumb status: \n"; +if ($err != null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/pfop_watermark.php b/vendor/qiniu/php-sdk/examples/pfop_watermark.php new file mode 100644 index 000000000..4254773dd --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/pfop_watermark.php @@ -0,0 +1,46 @@ +execute($key, $fops); +echo "\n====> pfop avthumb result: \n"; +if ($err != null) { + var_dump($err); +} else { + echo "PersistentFop Id: $id\n"; +} + +//鏌ヨ杞爜鐨勮繘搴﹀拰鐘舵 +list($ret, $err) = $pfop->status($id); +echo "\n====> pfop avthumb status: \n"; +if ($err != null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/php-logo.png b/vendor/qiniu/php-sdk/examples/php-logo.png new file mode 100644 index 000000000..77e051fe4 Binary files /dev/null and b/vendor/qiniu/php-sdk/examples/php-logo.png differ diff --git a/vendor/qiniu/php-sdk/examples/qetag.php b/vendor/qiniu/php-sdk/examples/qetag.php new file mode 100644 index 000000000..e477dfff2 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/qetag.php @@ -0,0 +1,10 @@ +uploadToken($bucket); +$uploadMgr = new UploadManager(); + +//----------------------------------------upload demo1 ---------------------------------------- +// 涓婁紶瀛楃涓插埌涓冪墰 +list($ret, $err) = $uploadMgr->put($token, null, 'content string'); +echo "\n====> put result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} + + +//----------------------------------------upload demo2 ---------------------------------------- +// 涓婁紶鏂囦欢鍒颁竷鐗 +$filePath = './php-logo.png'; +$key = 'php-logo.png'; +list($ret, $err) = $uploadMgr->putFile($token, $key, $filePath); +echo "\n====> putFile result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} + + +//----------------------------------------upload demo3 ---------------------------------------- +// 涓婁紶鏂囦欢鍒颁竷鐗涘悗锛 涓冪墰灏嗘枃浠跺悕鍜屾枃浠跺ぇ灏忓洖璋冪粰涓氬姟鏈嶅姟鍣. +// 鍙弬鑰冩枃妗: http://developer.qiniu.com/docs/v6/api/reference/security/put-policy.html +$policy = array( + 'callbackUrl' => 'http://172.30.251.210/callback.php', + 'callbackBody' => 'filename=$(fname)&filesize=$(fsize)' +// 'callbackBodyType' => 'application/json', +// 'callbackBody' => '{"filename":$(fname), "filesize": $(fsize)}' //璁剧疆application/json鏍煎紡鍥炶皟 +); +$token = $auth->uploadToken($bucket, null, 3600, $policy); + + +list($ret, $err) = $uploadMgr->putFile($token, null, $key); +echo "\n====> putFile result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} + + +//----------------------------------------upload demo4 ---------------------------------------- +//涓婁紶瑙嗛锛屼笂浼犲畬鎴愬悗杩涜m3u8鐨勮浆鐮侊紝 骞剁粰瑙嗛鎵撴按鍗 +$wmImg = Qiniu\base64_urlSafeEncode('http://Bucket_Name.qiniudn.com/logo-s.png'); +$pfop = "avthumb/m3u8/wmImage/$wmImg"; + +//杞爜瀹屾垚鍚庡洖璋冨埌涓氬姟鏈嶅姟鍣ㄣ傦紙鍏綉鍙互璁块棶锛屽苟鐩稿簲200 OK锛 +$notifyUrl = 'http://notify.fake.com'; + +//鐙珛鐨勮浆鐮侀槦鍒楋細https://portal.qiniu.com/mps/pipeline +$pipeline = 'abc'; + +$policy = array( + 'persistentOps' => $pfop, + 'persistentNotifyUrl' => $notifyUrl, + 'persistentPipeline' => $pipeline +); +$token = $auth->uploadToken($bucket, null, 3600, $policy); + +list($ret, $err) = $uploadMgr->putFile($token, null, $key); +echo "\n====> putFile result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/upload.php b/vendor/qiniu/php-sdk/examples/upload.php new file mode 100644 index 000000000..e746a0a68 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/upload.php @@ -0,0 +1,39 @@ +uploadToken($bucket); + +// 瑕佷笂浼犳枃浠剁殑鏈湴璺緞 +$filePath = './php-logo.png'; + +// 涓婁紶鍒颁竷鐗涘悗淇濆瓨鐨勬枃浠跺悕 +$key = 'my-php-logo.png'; + +// 鍒濆鍖 UploadManager 瀵硅薄骞惰繘琛屾枃浠剁殑涓婁紶銆 +$uploadMgr = new UploadManager(); + +// 璋冪敤 UploadManager 鐨 putFile 鏂规硶杩涜鏂囦欢鐨勪笂浼犮 +list($ret, $err) = $uploadMgr->putFile($token, $key, $filePath); +echo "\n====> putFile result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/upload_and_callback.php b/vendor/qiniu/php-sdk/examples/upload_and_callback.php new file mode 100644 index 000000000..4b6cbf6c8 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/upload_and_callback.php @@ -0,0 +1,31 @@ + 'http://your.domain.com/callback.php', + 'callbackBody' => 'filename=$(fname)&filesize=$(fsize)' +); +$uptoken = $auth->uploadToken($bucket, null, 3600, $policy); + +//涓婁紶鏂囦欢鐨勬湰鍦拌矾寰 +$filePath = './php-logo.png'; + +$uploadMgr = new UploadManager(); + +list($ret, $err) = $uploadMgr->putFile($uptoken, null, $filePath); +echo "\n====> putFile result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/upload_and_pfop.php b/vendor/qiniu/php-sdk/examples/upload_and_pfop.php new file mode 100644 index 000000000..d123b8bb9 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/upload_and_pfop.php @@ -0,0 +1,40 @@ +uploadToken($bucket); +$uploadMgr = new UploadManager(); + +//涓婁紶瑙嗛锛屼笂浼犲畬鎴愬悗杩涜m3u8鐨勮浆鐮侊紝 骞剁粰瑙嗛鎵撴按鍗 +$wmImg = Qiniu\base64_urlSafeEncode('http://Bucket_Name.qiniudn.com/logo-s.png'); +$pfop = "avthumb/m3u8/wmImage/$wmImg"; + +//杞爜瀹屾垚鍚庨氱煡鍒颁綘鐨勪笟鍔℃湇鍔″櫒銆傦紙鍏綉鍙互璁块棶锛屽苟鐩稿簲200 OK锛 +$notifyUrl = 'http://notify.fake.com'; + +//鐙珛鐨勮浆鐮侀槦鍒楋細https://portal.qiniu.com/mps/pipeline +$pipeline = 'pipeline_name'; + +$policy = array( + 'persistentOps' => $pfop, + 'persistentNotifyUrl' => $notifyUrl, + 'persistentPipeline' => $pipeline +); +$token = $auth->uploadToken($bucket, null, 3600, $policy); + +list($ret, $err) = $uploadMgr->putFile($token, null, $key); +echo "\n====> putFile result: \n"; +if ($err !== null) { + var_dump($err); +} else { + var_dump($ret); +} diff --git a/vendor/qiniu/php-sdk/examples/upload_mgr_init.php b/vendor/qiniu/php-sdk/examples/upload_mgr_init.php new file mode 100644 index 000000000..3459ef1d7 --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/upload_mgr_init.php @@ -0,0 +1,18 @@ +uploadToken($bucket); + +// 鏋勫缓 UploadManager 瀵硅薄 +$uploadMgr = new UploadManager(); diff --git a/vendor/qiniu/php-sdk/examples/upload_token.php b/vendor/qiniu/php-sdk/examples/upload_token.php new file mode 100644 index 000000000..55e37da9a --- /dev/null +++ b/vendor/qiniu/php-sdk/examples/upload_token.php @@ -0,0 +1,13 @@ +uploadToken($bucket); + +echo $upToken; diff --git a/vendor/qiniu/php-sdk/phpunit.xml.dist b/vendor/qiniu/php-sdk/phpunit.xml.dist new file mode 100644 index 000000000..72ff67f97 --- /dev/null +++ b/vendor/qiniu/php-sdk/phpunit.xml.dist @@ -0,0 +1,19 @@ + + + + + tests + + + + diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Auth.php b/vendor/qiniu/php-sdk/src/Qiniu/Auth.php new file mode 100644 index 000000000..b98bc55c2 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Auth.php @@ -0,0 +1,163 @@ +accessKey = $accessKey; + $this->secretKey = $secretKey; + } + + public function getAccessKey() + { + return $this->accessKey; + } + + public function sign($data) + { + $hmac = hash_hmac('sha1', $data, $this->secretKey, true); + return $this->accessKey . ':' . \Qiniu\base64_urlSafeEncode($hmac); + } + + public function signWithData($data) + { + $data = \Qiniu\base64_urlSafeEncode($data); + return $this->sign($data) . ':' . $data; + } + + public function signRequest($urlString, $body, $contentType = null) + { + $url = parse_url($urlString); + $data = ''; + if (array_key_exists('path', $url)) { + $data = $url['path']; + } + if (array_key_exists('query', $url)) { + $data .= '?' . $url['query']; + } + $data .= "\n"; + + if ($body !== null && $contentType === 'application/x-www-form-urlencoded') { + $data .= $body; + } + return $this->sign($data); + } + + public function verifyCallback($contentType, $originAuthorization, $url, $body) + { + $authorization = 'QBox ' . $this->signRequest($url, $body, $contentType); + return $originAuthorization === $authorization; + } + + public function privateDownloadUrl($baseUrl, $expires = 3600) + { + $deadline = time() + $expires; + + $pos = strpos($baseUrl, '?'); + if ($pos !== false) { + $baseUrl .= '&e='; + } else { + $baseUrl .= '?e='; + } + $baseUrl .= $deadline; + + $token = $this->sign($baseUrl); + return "$baseUrl&token=$token"; + } + + public function uploadToken( + $bucket, + $key = null, + $expires = 3600, + $policy = null, + $strictPolicy = true, + Zone $zone = null + ) { + $deadline = time() + $expires; + $scope = $bucket; + if ($key !== null) { + $scope .= ':' . $key; + } + $args = array(); + $args = self::copyPolicy($args, $policy, $strictPolicy); + $args['scope'] = $scope; + $args['deadline'] = $deadline; + + if ($zone === null) { + $zone = new Zone(); + } + + list($upHosts, $err) = $zone->getUpHosts($this->accessKey, $bucket); + if ($err === null) { + $args['upHosts'] = $upHosts; + } + + $b = json_encode($args); + return $this->signWithData($b); + } + + /** + *涓婁紶绛栫暐锛屽弬鏁拌鏍艰瑙 + *http://developer.qiniu.com/docs/v6/api/reference/security/put-policy.html + */ + private static $policyFields = array( + 'callbackUrl', + 'callbackBody', + 'callbackHost', + 'callbackBodyType', + 'callbackFetchKey', + + 'returnUrl', + 'returnBody', + + 'endUser', + 'saveKey', + 'insertOnly', + + 'detectMime', + 'mimeLimit', + 'fsizeMin', + 'fsizeLimit', + + 'persistentOps', + 'persistentNotifyUrl', + 'persistentPipeline', + + 'deleteAfterDays', + + 'upHosts', + ); + + private static $deprecatedPolicyFields = array( + 'asyncOps', + ); + + private static function copyPolicy(&$policy, $originPolicy, $strictPolicy) + { + if ($originPolicy === null) { + return array(); + } + foreach ($originPolicy as $key => $value) { + if (in_array((string) $key, self::$deprecatedPolicyFields, true)) { + throw new \InvalidArgumentException("{$key} has deprecated"); + } + if (!$strictPolicy || in_array((string) $key, self::$policyFields, true)) { + $policy[$key] = $value; + } + } + return $policy; + } + + public function authorization($url, $body = null, $contentType = null) + { + $authorization = 'QBox ' . $this->signRequest($url, $body, $contentType); + return array('Authorization' => $authorization); + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Config.php b/vendor/qiniu/php-sdk/src/Qiniu/Config.php new file mode 100644 index 000000000..564bf29ca --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Config.php @@ -0,0 +1,25 @@ +zone = new Zone(); + // } + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Etag.php b/vendor/qiniu/php-sdk/src/Qiniu/Etag.php new file mode 100644 index 000000000..dfeb4e97e --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Etag.php @@ -0,0 +1,76 @@ + $val) { + array_push($data, '--' . $mimeBoundary); + array_push($data, "Content-Disposition: form-data; name=\"$key\""); + array_push($data, ''); + array_push($data, $val); + } + + array_push($data, '--' . $mimeBoundary); + $mimeType = empty($mimeType) ? 'application/octet-stream' : $mimeType; + $fileName = self::escapeQuotes($fileName); + array_push($data, "Content-Disposition: form-data; name=\"$name\"; filename=\"$fileName\""); + array_push($data, "Content-Type: $mimeType"); + array_push($data, ''); + array_push($data, $fileBody); + + array_push($data, '--' . $mimeBoundary . '--'); + array_push($data, ''); + + $body = implode("\r\n", $data); + $contentType = 'multipart/form-data; boundary=' . $mimeBoundary; + $headers['Content-Type'] = $contentType; + $request = new Request('POST', $url, $headers, $body); + return self::sendRequest($request); + } + + private static function userAgent() + { + $sdkInfo = "QiniuPHP/" . Config::SDK_VER; + + $systemInfo = php_uname("s"); + $machineInfo = php_uname("m"); + + $envInfo = "($systemInfo/$machineInfo)"; + + $phpVer = phpversion(); + + $ua = "$sdkInfo $envInfo PHP/$phpVer"; + return $ua; + } + + public static function sendRequest($request) + { + $t1 = microtime(true); + $ch = curl_init(); + $options = array( + CURLOPT_USERAGENT => self::userAgent(), + CURLOPT_RETURNTRANSFER => true, + CURLOPT_SSL_VERIFYPEER => false, + CURLOPT_SSL_VERIFYHOST => false, + CURLOPT_HEADER => true, + CURLOPT_NOBODY => false, + CURLOPT_CUSTOMREQUEST => $request->method, + CURLOPT_URL => $request->url + ); + + // Handle open_basedir & safe mode + if (!ini_get('safe_mode') && !ini_get('open_basedir')) { + $options[CURLOPT_FOLLOWLOCATION] = true; + } + + if (!empty($request->headers)) { + $headers = array(); + foreach ($request->headers as $key => $val) { + array_push($headers, "$key: $val"); + } + $options[CURLOPT_HTTPHEADER] = $headers; + } + curl_setopt($ch, CURLOPT_HTTPHEADER, array('Expect:')); + + if (!empty($request->body)) { + $options[CURLOPT_POSTFIELDS] = $request->body; + } + curl_setopt_array($ch, $options); + $result = curl_exec($ch); + $t2 = microtime(true); + $duration = round($t2-$t1, 3); + $ret = curl_errno($ch); + if ($ret !== 0) { + $r = new Response(-1, $duration, array(), null, curl_error($ch)); + curl_close($ch); + return $r; + } + $code = curl_getinfo($ch, CURLINFO_HTTP_CODE); + $header_size = curl_getinfo($ch, CURLINFO_HEADER_SIZE); + $headers = self::parseHeaders(substr($result, 0, $header_size)); + $body = substr($result, $header_size); + curl_close($ch); + return new Response($code, $duration, $headers, $body, null); + } + + private static function parseHeaders($raw) + { + $headers = array(); + $headerLines = explode("\r\n", $raw); + foreach ($headerLines as $line) { + $headerLine = trim($line); + $kv = explode(':', $headerLine); + if (count($kv) >1) { + $headers[$kv[0]] = trim($kv[1]); + } + } + return $headers; + } + + private static function escapeQuotes($str) + { + $find = array("\\", "\""); + $replace = array("\\\\", "\\\""); + return str_replace($find, $replace, $str); + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Http/Error.php b/vendor/qiniu/php-sdk/src/Qiniu/Http/Error.php new file mode 100644 index 000000000..a285a5d38 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Http/Error.php @@ -0,0 +1,29 @@ +url = $url; + $this->response = $response; + } + + public function code() + { + return $this->response->statusCode; + } + + public function getResponse() + { + return $this->response; + } + + public function message() + { + return $this->response->error; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Http/Request.php b/vendor/qiniu/php-sdk/src/Qiniu/Http/Request.php new file mode 100644 index 000000000..43b0bfdb2 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Http/Request.php @@ -0,0 +1,18 @@ +method = strtoupper($method); + $this->url = $url; + $this->headers = $headers; + $this->body = $body; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Http/Response.php b/vendor/qiniu/php-sdk/src/Qiniu/Http/Response.php new file mode 100644 index 000000000..0081ea1d5 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Http/Response.php @@ -0,0 +1,176 @@ + 'Continue', + 101 => 'Switching Protocols', + 102 => 'Processing', + 200 => 'OK', + 201 => 'Created', + 202 => 'Accepted', + 203 => 'Non-Authoritative Information', + 204 => 'No Content', + 205 => 'Reset Content', + 206 => 'Partial Content', + 207 => 'Multi-Status', + 208 => 'Already Reported', + 226 => 'IM Used', + 300 => 'Multiple Choices', + 301 => 'Moved Permanently', + 302 => 'Found', + 303 => 'See Other', + 304 => 'Not Modified', + 305 => 'Use Proxy', + 307 => 'Temporary Redirect', + 308 => 'Permanent Redirect', + 400 => 'Bad Request', + 401 => 'Unauthorized', + 402 => 'Payment Required', + 403 => 'Forbidden', + 404 => 'Not Found', + 405 => 'Method Not Allowed', + 406 => 'Not Acceptable', + 407 => 'Proxy Authentication Required', + 408 => 'Request Timeout', + 409 => 'Conflict', + 410 => 'Gone', + 411 => 'Length Required', + 412 => 'Precondition Failed', + 413 => 'Request Entity Too Large', + 414 => 'Request-URI Too Long', + 415 => 'Unsupported Media Type', + 416 => 'Requested Range Not Satisfiable', + 417 => 'Expectation Failed', + 422 => 'Unprocessable Entity', + 423 => 'Locked', + 424 => 'Failed Dependency', + 425 => 'Reserved for WebDAV advanced collections expired proposal', + 426 => 'Upgrade required', + 428 => 'Precondition Required', + 429 => 'Too Many Requests', + 431 => 'Request Header Fields Too Large', + 500 => 'Internal Server Error', + 501 => 'Not Implemented', + 502 => 'Bad Gateway', + 503 => 'Service Unavailable', + 504 => 'Gateway Timeout', + 505 => 'HTTP Version Not Supported', + 506 => 'Variant Also Negotiates (Experimental)', + 507 => 'Insufficient Storage', + 508 => 'Loop Detected', + 510 => 'Not Extended', + 511 => 'Network Authentication Required', + ); + + /** + * @param int $code 鐘舵佺爜 + * @param double $duration 璇锋眰鏃堕暱 + * @param array $headers 鍝嶅簲澶撮儴 + * @param string $body 鍝嶅簲鍐呭 + * @param string $error 閿欒鎻忚堪 + */ + public function __construct($code, $duration, array $headers = array(), $body = null, $error = null) + { + $this->statusCode = $code; + $this->duration = $duration; + $this->headers = $headers; + $this->body = $body; + $this->error = $error; + $this->jsonData = null; + if ($error !== null) { + return; + } + + if ($body === null) { + if ($code >= 400) { + $this->error = self::$statusTexts[$code]; + } + return; + } + if (self::isJson($headers)) { + try { + $jsonData = self::bodyJson($body); + if ($code >=400) { + $this->error = $body; + if ($jsonData['error'] !== null) { + $this->error = $jsonData['error']; + } + } + $this->jsonData = $jsonData; + } catch (\InvalidArgumentException $e) { + $this->error = $body; + if ($code >= 200 && $code < 300) { + $this->error = $e->getMessage(); + } + } + } elseif ($code >=400) { + $this->error = $body; + } + return; + } + + public function json() + { + return $this->jsonData; + } + + private static function bodyJson($body) + { + return \Qiniu\json_decode((string) $body, true, 512); + } + + public function xVia() + { + $via = $this->headers['X-Via']; + if ($via === null) { + $via = $this->headers['X-Px']; + } + if ($via === null) { + $via = $this->headers['Fw-Via']; + } + return $via; + } + + public function xLog() + { + return $this->headers['X-Log']; + } + + public function xReqId() + { + return $this->headers['X-Reqid']; + } + + public function ok() + { + return $this->statusCode >= 200 && $this->statusCode < 300 && $this->error === null; + } + + public function needRetry() + { + $code = $this->statusCode; + if ($code< 0 || ($code / 100 === 5 and $code !== 579) || $code === 996) { + return true; + } + } + + private static function isJson($headers) + { + return array_key_exists('Content-Type', $headers) && + strpos($headers['Content-Type'], 'application/json') === 0; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Processing/ImageUrlBuilder.php b/vendor/qiniu/php-sdk/src/Qiniu/Processing/ImageUrlBuilder.php new file mode 100644 index 000000000..4086e9ce7 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Processing/ImageUrlBuilder.php @@ -0,0 +1,282 @@ + + */ + public function thumbnail( + $url, + $mode, + $width, + $height, + $format = null, + $interlace = null, + $quality = null, + $ignoreError = 1 + ) { + + // url鍚堟硶鏁堥獙 + if (! $this->isUrl($url)) { + return $url; + } + + // 鍙傛暟鍚堟硶鎬ф晥楠 + if (! in_array(intval($mode), $this->modeArr, true)) { + return $url; + } + + if (! $width || ! $height) { + return $url; + } + + $thumbStr = 'imageView2/' . $mode . '/w/' . $width . '/h/' . $height . '/'; + + // 鎷兼帴杈撳嚭鏍煎紡 + if (! is_null($format) + && in_array($format, $this->formatArr) + ) { + $thumbStr .= 'format/' . $format . '/'; + } + + // 鎷兼帴娓愯繘鏄剧ず + if (! is_null($interlace) + && in_array(intval($interlace), array(0, 1), true) + ) { + $thumbStr .= 'interlace/' . $interlace . '/'; + } + + // 鎷兼帴鍥剧墖璐ㄩ噺 + if (! is_null($quality) + && intval($quality) >= 0 + && intval($quality) <= 100 + ) { + $thumbStr .= 'q/' . $quality . '/'; + } + + $thumbStr .= 'ignore-error/' . $ignoreError . '/'; + + // 濡傛灉鏈塹uery_string鐢▅绾垮垎鍓插疄鐜板鍙傛暟 + return $url . ($this->hasQuery($url) ? '|' : '?') . $thumbStr; + } + + /** + * 鍥剧墖姘村嵃 + * + * @param string $url 鍥剧墖閾炬帴 + * @param string $image 姘村嵃鍥剧墖閾炬帴 + * @param numeric $dissolve 閫忔槑搴 + * @param string $gravity 姘村嵃浣嶇疆 + * @param numeric $dx 妯酱杈硅窛 + * @param numeric $dy 绾佃酱杈硅窛 + * @param numeric $watermarkScale 鑷傚簲鍘熷浘鐨勭煭杈规瘮渚 + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/watermark.html + * @return string + * @author Sherlock Ren + */ + public function waterImg( + $url, + $image, + $dissolve = 100, + $gravity = 'SouthEast', + $dx = null, + $dy = null, + $watermarkScale = null + ) { + // url鍚堟硶鏁堥獙 + if (! $this->isUrl($url)) { + return $url; + } + + $waterStr = 'watermark/1/image/' . \Qiniu\base64_urlSafeEncode($image) . '/'; + + // 鎷兼帴姘村嵃閫忔槑搴 + if (is_numeric($dissolve) + && $dissolve <= 100 + ) { + $waterStr .= 'dissolve/' . $dissolve . '/'; + } + + // 鎷兼帴姘村嵃浣嶇疆 + if (in_array($gravity, $this->gravityArr, true)) { + $waterStr .= 'gravity/' . $gravity . '/'; + } + + // 鎷兼帴妯酱杈硅窛 + if (! is_null($dx) + && is_numeric($dx) + ) { + $waterStr .= 'dx/' . $dx . '/'; + } + + // 鎷兼帴绾佃酱杈硅窛 + if (! is_null($dy) + && is_numeric($dy) + ) { + $waterStr .= 'dy/' . $dy . '/'; + } + + // 鎷兼帴鑷傚簲鍘熷浘鐨勭煭杈规瘮渚 + if (! is_null($watermarkScale) + && is_numeric($watermarkScale) + && $watermarkScale > 0 + && $watermarkScale < 1 + ) { + $waterStr .= 'ws/' . $watermarkScale . '/'; + } + + // 濡傛灉鏈塹uery_string鐢▅绾垮垎鍓插疄鐜板鍙傛暟 + return $url . ($this->hasQuery($url) ? '|' : '?') . $waterStr; + } + + /** + * 鏂囧瓧姘村嵃 + * + * @param string $url 鍥剧墖閾炬帴 + * @param string $text 鏂囧瓧 + * @param string $font 鏂囧瓧瀛椾綋 + * @param string $fontSize 鏂囧瓧瀛楀彿 + * @param string $fontColor 鏂囧瓧棰滆壊 + * @param numeric $dissolve 閫忔槑搴 + * @param string $gravity 姘村嵃浣嶇疆 + * @param numeric $dx 妯酱杈硅窛 + * @param numeric $dy 绾佃酱杈硅窛 + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/watermark.html#text-watermark + * @return string + * @author Sherlock Ren + */ + public function waterText( + $url, + $text, + $font = '榛戜綋', + $fontSize = 0, + $fontColor = null, + $dissolve = 100, + $gravity = 'SouthEast', + $dx = null, + $dy = null + ) { + // url鍚堟硶鏁堥獙 + if (! $this->isUrl($url)) { + return $url; + } + + $waterStr = 'watermark/2/text/' + . \Qiniu\base64_urlSafeEncode($text) . '/font/' + . \Qiniu\base64_urlSafeEncode($font) . '/'; + + // 鎷兼帴鏂囧瓧澶у皬 + if (is_int($fontSize)) { + $waterStr .= 'fontsize/' . $fontSize . '/'; + } + + // 鎷兼帴鏂囧瓧棰滆壊 + if (! is_null($fontColor) + && $fontColor + ) { + $waterStr .= 'fill/' . \Qiniu\base64_urlSafeEncode($fontColor) . '/'; + } + + // 鎷兼帴姘村嵃閫忔槑搴 + if (is_numeric($dissolve) + && $dissolve <= 100 + ) { + $waterStr .= 'dissolve/' . $dissolve . '/'; + } + + // 鎷兼帴姘村嵃浣嶇疆 + if (in_array($gravity, $this->gravityArr, true)) { + $waterStr .= 'gravity/' . $gravity . '/'; + } + + // 鎷兼帴妯酱杈硅窛 + if (! is_null($dx) + && is_numeric($dx) + ) { + $waterStr .= 'dx/' . $dx . '/'; + } + + // 鎷兼帴绾佃酱杈硅窛 + if (! is_null($dy) + && is_numeric($dy) + ) { + $waterStr .= 'dy/' . $dy . '/'; + } + + // 濡傛灉鏈塹uery_string鐢▅绾垮垎鍓插疄鐜板鍙傛暟 + return $url . ($this->hasQuery($url) ? '|' : '?') . $waterStr; + } + + /** + * 鏁堥獙url鍚堟硶鎬 + * + * @param string $url url閾炬帴 + * @return string + * @author Sherlock Ren + */ + protected function isUrl($url) + { + $urlArr = parse_url($url); + + return $urlArr['scheme'] + && in_array($urlArr['scheme'], array('http', 'https')) + && $urlArr['host'] + && $urlArr['path']; + } + + /** + * 妫娴嬫槸鍚︽湁query + * + * @param string $url url閾炬帴 + * @return string + * @author Sherlock Ren + */ + protected function hasQuery($url) + { + $urlArr = parse_url($url); + + return ! empty($urlArr['query']); + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Processing/Operation.php b/vendor/qiniu/php-sdk/src/Qiniu/Processing/Operation.php new file mode 100644 index 000000000..02d7d7539 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Processing/Operation.php @@ -0,0 +1,60 @@ +auth = $auth; + $this->domain = $domain; + $this->token_expire = $token_expire; + } + + + /** + * 瀵硅祫婧愭枃浠惰繘琛屽鐞 + * + * @param $key 寰呭鐞嗙殑璧勬簮鏂囦欢鍚 + * @param $fops string|array fop鎿嶄綔锛屽娆op鎿嶄綔浠rray鐨勫舰寮忎紶鍏ャ + * eg. imageView2/1/w/200/h/200, imageMogr2/thumbnail/!75px + * + * @return array 鏂囦欢澶勭悊鍚庣殑缁撴灉鍙婇敊璇 + * + * @link http://developer.qiniu.com/docs/v6/api/reference/fop/ + */ + public function execute($key, $fops) + { + $url = $this->buildUrl($key, $fops); + $resp = Client::get($url); + if (!$resp->ok()) { + return array(null, new Error($url, $resp)); + } + if ($resp->json() !== null) { + return array($resp->json(), null); + } + return array($resp->body, null); + } + + public function buildUrl($key, $fops, $protocol = 'http') + { + if (is_array($fops)) { + $fops = implode('|', $fops); + } + + $url = $protocol."://$this->domain/$key?$fops"; + if ($this->auth !== null) { + $url = $this->auth->privateDownloadUrl($url, $this->token_expire); + } + + return $url; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Processing/PersistentFop.php b/vendor/qiniu/php-sdk/src/Qiniu/Processing/PersistentFop.php new file mode 100644 index 000000000..e385c4697 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Processing/PersistentFop.php @@ -0,0 +1,95 @@ +auth = $auth; + $this->bucket = $bucket; + $this->pipeline = $pipeline; + $this->notify_url = $notify_url; + $this->force = $force; + } + + /** + * 瀵硅祫婧愭枃浠惰繘琛屽紓姝ユ寔涔呭寲澶勭悊 + * + * @param $key 寰呭鐞嗙殑婧愭枃浠 + * @param $fops string|array 寰呭鐞嗙殑pfop鎿嶄綔锛屽涓猵fop鎿嶄綔浠rray鐨勫舰寮忎紶鍏ャ + * eg. avthumb/mp3/ab/192k, vframe/jpg/offset/7/w/480/h/360 + * + * @return array 杩斿洖鎸佷箙鍖栧鐞嗙殑persistentId, 鍜岃繑鍥炵殑閿欒銆 + * + * @link http://developer.qiniu.com/docs/v6/api/reference/fop/ + */ + public function execute($key, $fops) + { + if (is_array($fops)) { + $fops = implode(';', $fops); + } + $params = array('bucket' => $this->bucket, 'key' => $key, 'fops' => $fops); + \Qiniu\setWithoutEmpty($params, 'pipeline', $this->pipeline); + \Qiniu\setWithoutEmpty($params, 'notifyURL', $this->notify_url); + if ($this->force) { + $params['force'] = 1; + } + $data = http_build_query($params); + $url = Config::API_HOST . '/pfop/'; + $headers = $this->auth->authorization($url, $data, 'application/x-www-form-urlencoded'); + $headers['Content-Type'] = 'application/x-www-form-urlencoded'; + $response = Client::post($url, $data, $headers); + if (!$response->ok()) { + return array(null, new Error($url, $response)); + } + $r = $response->json(); + $id = $r['persistentId']; + return array($id, null); + } + + public static function status($id) + { + $url = Config::API_HOST . "/status/get/prefop?id=$id"; + $response = Client::get($url); + if (!$response->ok()) { + return array(null, new Error($url, $response)); + } + return array($response->json(), null); + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Storage/BucketManager.php b/vendor/qiniu/php-sdk/src/Qiniu/Storage/BucketManager.php new file mode 100644 index 000000000..0afb78138 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Storage/BucketManager.php @@ -0,0 +1,362 @@ +auth = $auth; + if ($zone === null) { + $this->zone = new Zone(); + } + } + + /** + * 鑾峰彇鎸囧畾璐﹀彿涓嬫墍鏈夌殑绌洪棿鍚嶃 + * + * @return string[] 鍖呭惈鎵鏈夌┖闂村悕 + */ + public function buckets() + { + return $this->rsGet('/buckets'); + } + + /** + * 鍒楀彇绌洪棿鐨勬枃浠跺垪琛 + * + * @param $bucket 绌洪棿鍚 + * @param $prefix 鍒椾妇鍓嶇紑 + * @param $marker 鍒椾妇鏍囪瘑绗 + * @param $limit 鍗曟鍒椾妇涓暟闄愬埗 + * @param $delimiter 鎸囧畾鐩綍鍒嗛殧绗 + * + * @return array 鍖呭惈鏂囦欢淇℃伅鐨勬暟缁勶紝绫讳技锛歔 + * { + * "hash" => "", + * "key" => "", + * "fsize" => "", + * "putTime" => "" + * }, + * ... + * ] + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/list.html + */ + public function listFiles($bucket, $prefix = null, $marker = null, $limit = 1000, $delimiter = null) + { + $query = array('bucket' => $bucket); + \Qiniu\setWithoutEmpty($query, 'prefix', $prefix); + \Qiniu\setWithoutEmpty($query, 'marker', $marker); + \Qiniu\setWithoutEmpty($query, 'limit', $limit); + \Qiniu\setWithoutEmpty($query, 'delimiter', $delimiter); + $url = Config::RSF_HOST . '/list?' . http_build_query($query); + list($ret, $error) = $this->get($url); + if ($ret === null) { + return array(null, null, $error); + } + $marker = array_key_exists('marker', $ret) ? $ret['marker'] : null; + return array($ret['items'], $marker, null); + } + + /** + * 鑾峰彇璧勬簮鐨勫厓淇℃伅锛屼絾涓嶈繑鍥炴枃浠跺唴瀹 + * + * @param $bucket 寰呰幏鍙栦俊鎭祫婧愭墍鍦ㄧ殑绌洪棿 + * @param $key 寰呰幏鍙栬祫婧愮殑鏂囦欢鍚 + * + * @return array 鍖呭惈鏂囦欢淇℃伅鐨勬暟缁勶紝绫讳技锛 + * [ + * "hash" => "", + * "key" => "", + * "fsize" => "", + * "putTime" => "" + * ] + * + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/stat.html + */ + public function stat($bucket, $key) + { + $path = '/stat/' . \Qiniu\entry($bucket, $key); + return $this->rsGet($path); + } + + /** + * 鍒犻櫎鎸囧畾璧勬簮 + * + * @param $bucket 寰呭垹闄よ祫婧愭墍鍦ㄧ殑绌洪棿 + * @param $key 寰呭垹闄よ祫婧愮殑鏂囦欢鍚 + * + * @return mixed 鎴愬姛杩斿洖NULL锛屽け璐ヨ繑鍥炲璞iniu\Http\Error + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/delete.html + */ + public function delete($bucket, $key) + { + $path = '/delete/' . \Qiniu\entry($bucket, $key); + list(, $error) = $this->rsPost($path); + return $error; + } + + + /** + * 缁欒祫婧愯繘琛岄噸鍛藉悕锛屾湰璐ㄤ负move鎿嶄綔銆 + * + * @param $bucket 寰呮搷浣滆祫婧愭墍鍦ㄧ┖闂 + * @param $oldname 寰呮搷浣滆祫婧愭枃浠跺悕 + * @param $newname 鐩爣璧勬簮鏂囦欢鍚 + * + * @return mixed 鎴愬姛杩斿洖NULL锛屽け璐ヨ繑鍥炲璞iniu\Http\Error + */ + public function rename($bucket, $oldname, $newname) + { + return $this->move($bucket, $oldname, $bucket, $newname); + } + + /** + * 缁欒祫婧愯繘琛岄噸鍛藉悕锛屾湰璐ㄤ负move鎿嶄綔銆 + * + * @param $from_bucket 寰呮搷浣滆祫婧愭墍鍦ㄧ┖闂 + * @param $from_key 寰呮搷浣滆祫婧愭枃浠跺悕 + * @param $to_bucket 鐩爣璧勬簮绌洪棿鍚 + * @param $to_key 鐩爣璧勬簮鏂囦欢鍚 + * + * @return mixed 鎴愬姛杩斿洖NULL锛屽け璐ヨ繑鍥炲璞iniu\Http\Error + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/copy.html + */ + public function copy($from_bucket, $from_key, $to_bucket, $to_key, $force = false) + { + $from = \Qiniu\entry($from_bucket, $from_key); + $to = \Qiniu\entry($to_bucket, $to_key); + $path = '/copy/' . $from . '/' . $to; + if ($force) { + $path .= '/force/true'; + } + list(, $error) = $this->rsPost($path); + return $error; + } + + /** + * 灏嗚祫婧愪粠涓涓┖闂村埌鍙︿竴涓┖闂 + * + * @param $from_bucket 寰呮搷浣滆祫婧愭墍鍦ㄧ┖闂 + * @param $from_key 寰呮搷浣滆祫婧愭枃浠跺悕 + * @param $to_bucket 鐩爣璧勬簮绌洪棿鍚 + * @param $to_key 鐩爣璧勬簮鏂囦欢鍚 + * + * @return mixed 鎴愬姛杩斿洖NULL锛屽け璐ヨ繑鍥炲璞iniu\Http\Error + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/move.html + */ + public function move($from_bucket, $from_key, $to_bucket, $to_key, $force = false) + { + $from = \Qiniu\entry($from_bucket, $from_key); + $to = \Qiniu\entry($to_bucket, $to_key); + $path = '/move/' . $from . '/' . $to; + if ($force) { + $path .= '/force/true'; + } + list(, $error) = $this->rsPost($path); + return $error; + } + + /** + * 涓诲姩淇敼鎸囧畾璧勬簮鐨勬枃浠剁被鍨 + * + * @param $bucket 寰呮搷浣滆祫婧愭墍鍦ㄧ┖闂 + * @param $key 寰呮搷浣滆祫婧愭枃浠跺悕 + * @param $mime 寰呮搷浣滄枃浠剁洰鏍噈imeType + * + * @return mixed 鎴愬姛杩斿洖NULL锛屽け璐ヨ繑鍥炲璞iniu\Http\Error + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/chgm.html + */ + public function changeMime($bucket, $key, $mime) + { + $resource = \Qiniu\entry($bucket, $key); + $encode_mime = \Qiniu\base64_urlSafeEncode($mime); + $path = '/chgm/' . $resource . '/mime/' .$encode_mime; + list(, $error) = $this->rsPost($path); + return $error; + } + + /** + * 浠庢寚瀹歎RL鎶撳彇璧勬簮锛屽苟灏嗚璧勬簮瀛樺偍鍒版寚瀹氱┖闂翠腑 + * + * @param $url 鎸囧畾鐨刄RL + * @param $bucket 鐩爣璧勬簮绌洪棿 + * @param $key 鐩爣璧勬簮鏂囦欢鍚 + * + * @return array 鍖呭惈宸叉媺鍙栫殑鏂囦欢淇℃伅銆 + * 鎴愬姛鏃讹細 [ + * [ + * "hash" => "", + * "key" => "" + * ], + * null + * ] + * + * 澶辫触鏃讹細 [ + * null, + * Qiniu/Http/Error + * ] + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/fetch.html + */ + public function fetch($url, $bucket, $key = null) + { + + $resource = \Qiniu\base64_urlSafeEncode($url); + $to = \Qiniu\entry($bucket, $key); + $path = '/fetch/' . $resource . '/to/' . $to; + + $ak = $this->auth->getAccessKey(); + $ioHost = $this->zone->getIoHost($ak, $bucket); + + $url = $ioHost . $path; + return $this->post($url, null); + } + + /** + * 浠庨暅鍍忔簮绔欐姄鍙栬祫婧愬埌绌洪棿涓紝濡傛灉绌洪棿涓凡缁忓瓨鍦紝鍒欒鐩栬璧勬簮 + * + * @param $bucket 寰呰幏鍙栬祫婧愭墍鍦ㄧ殑绌洪棿 + * @param $key 浠h幏鍙栬祫婧愭枃浠跺悕 + * + * @return mixed 鎴愬姛杩斿洖NULL锛屽け璐ヨ繑鍥炲璞iniu\Http\Error + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/prefetch.html + */ + public function prefetch($bucket, $key) + { + $resource = \Qiniu\entry($bucket, $key); + $path = '/prefetch/' . $resource; + + $ak = $this->auth->getAccessKey(); + $ioHost = $this->zone->getIoHost($ak, $bucket); + + $url = $ioHost . $path; + list(, $error) = $this->post($url, null); + return $error; + } + + /** + * 鍦ㄥ崟娆¤姹備腑杩涜澶氫釜璧勬簮绠$悊鎿嶄綔 + * + * @param $operations 璧勬簮绠$悊鎿嶄綔鏁扮粍 + * + * @return array 姣忎釜璧勬簮鐨勫鐞嗘儏鍐碉紝缁撴灉绫讳技锛 + * [ + * { "code" => , "data" => }, + * { "code" => }, + * { "code" => }, + * { "code" => }, + * { "code" => , "data" => { "error": "" } }, + * ... + * ] + * @link http://developer.qiniu.com/docs/v6/api/reference/rs/batch.html + */ + public function batch($operations) + { + $params = 'op=' . implode('&op=', $operations); + return $this->rsPost('/batch', $params); + } + + private function rsPost($path, $body = null) + { + $url = Config::RS_HOST . $path; + return $this->post($url, $body); + } + + private function rsGet($path) + { + $url = Config::RS_HOST . $path; + return $this->get($url); + } + + private function ioPost($path, $body = null) + { + $url = Config::IO_HOST . $path; + return $this->post($url, $body); + } + + private function get($url) + { + $headers = $this->auth->authorization($url); + $ret = Client::get($url, $headers); + if (!$ret->ok()) { + return array(null, new Error($url, $ret)); + } + return array($ret->json(), null); + } + + private function post($url, $body) + { + $headers = $this->auth->authorization($url, $body, 'application/x-www-form-urlencoded'); + $ret = Client::post($url, $body, $headers); + if (!$ret->ok()) { + return array(null, new Error($url, $ret)); + } + $r = ($ret->body === null) ? array() : $ret->json(); + return array($r, null); + } + + public static function buildBatchCopy($source_bucket, $key_pairs, $target_bucket) + { + return self::twoKeyBatch('copy', $source_bucket, $key_pairs, $target_bucket); + } + + + public static function buildBatchRename($bucket, $key_pairs) + { + return self::buildBatchMove($bucket, $key_pairs, $bucket); + } + + + public static function buildBatchMove($source_bucket, $key_pairs, $target_bucket) + { + return self::twoKeyBatch('move', $source_bucket, $key_pairs, $target_bucket); + } + + + public static function buildBatchDelete($bucket, $keys) + { + return self::oneKeyBatch('delete', $bucket, $keys); + } + + + public static function buildBatchStat($bucket, $keys) + { + return self::oneKeyBatch('stat', $bucket, $keys); + } + + private static function oneKeyBatch($operation, $bucket, $keys) + { + $data = array(); + foreach ($keys as $key) { + array_push($data, $operation . '/' . \Qiniu\entry($bucket, $key)); + } + return $data; + } + + private static function twoKeyBatch($operation, $source_bucket, $key_pairs, $target_bucket) + { + if ($target_bucket === null) { + $target_bucket = $source_bucket; + } + $data = array(); + foreach ($key_pairs as $from_key => $to_key) { + $from = \Qiniu\entry($source_bucket, $from_key); + $to = \Qiniu\entry($target_bucket, $to_key); + array_push($data, $operation . '/' . $from . '/' . $to); + } + return $data; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Storage/FormUploader.php b/vendor/qiniu/php-sdk/src/Qiniu/Storage/FormUploader.php new file mode 100644 index 000000000..2e1cae82d --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Storage/FormUploader.php @@ -0,0 +1,134 @@ + "", + * "key" => "" + * ] + */ + public static function put( + $upToken, + $key, + $data, + $config, + $params, + $mime, + $checkCrc + ) { + $fields = array('token' => $upToken); + if ($key === null) { + $fname = 'filename'; + } else { + $fname = $key; + $fields['key'] = $key; + } + if ($checkCrc) { + $fields['crc32'] = \Qiniu\crc32_data($data); + } + if ($params) { + foreach ($params as $k => $v) { + $fields[$k] = $v; + } + } + + list($upHost, $err) = $config->zone->getUpHostByToken($upToken); + if ($err != null) { + return array(null, $err); + } + + $response = Client::multipartPost($upHost, $fields, 'file', $fname, $data, $mime); + if (!$response->ok()) { + return array(null, new Error($upHost, $response)); + } + return array($response->json(), null); + } + + /** + * 涓婁紶鏂囦欢鍒颁竷鐗涳紝鍐呴儴浣跨敤 + * + * @param $upToken 涓婁紶鍑瘉 + * @param $key 涓婁紶鏂囦欢鍚 + * @param $filePath 涓婁紶鏂囦欢鐨勮矾寰 + * @param $params 鑷畾涔夊彉閲忥紝瑙勬牸鍙傝 + * http://developer.qiniu.com/docs/v6/api/overview/up/response/vars.html#xvar + * @param $mime 涓婁紶鏁版嵁鐨刴imeType + * @param $checkCrc 鏄惁鏍¢獙crc32 + * + * @return array 鍖呭惈宸蹭笂浼犳枃浠剁殑淇℃伅锛岀被浼硷細 + * [ + * "hash" => "", + * "key" => "" + * ] + */ + public static function putFile( + $upToken, + $key, + $filePath, + $config, + $params, + $mime, + $checkCrc + ) { + + $fields = array('token' => $upToken, 'file' => self::createFile($filePath, $mime)); + if ($key !== null) { + $fields['key'] = $key; + } + if ($checkCrc) { + $fields['crc32'] = \Qiniu\crc32_file($filePath); + } + if ($params) { + foreach ($params as $k => $v) { + $fields[$k] = $v; + } + } + $fields['key'] = $key; + $headers =array('Content-Type' => 'multipart/form-data'); + + list($upHost, $err) = $config->zone->getUpHostByToken($upToken); + if ($err != null) { + return array(null, $err); + } + + $response = client::post($upHost, $fields, $headers); + if (!$response->ok()) { + return array(null, new Error($upHost, $response)); + } + return array($response->json(), null); + } + + private static function createFile($filename, $mime) + { + // PHP 5.5 introduced a CurlFile object that deprecates the old @filename syntax + // See: https://wiki.php.net/rfc/curl-file-upload + if (function_exists('curl_file_create')) { + return curl_file_create($filename, $mime); + } + + // Use the old style if using an older version of PHP + $value = "@{$filename}"; + if (!empty($mime)) { + $value .= ';type=' . $mime; + } + + return $value; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Storage/ResumeUploader.php b/vendor/qiniu/php-sdk/src/Qiniu/Storage/ResumeUploader.php new file mode 100644 index 000000000..382f21c6b --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Storage/ResumeUploader.php @@ -0,0 +1,160 @@ +upToken = $upToken; + $this->key = $key; + $this->inputStream = $inputStream; + $this->size = $size; + $this->params = $params; + $this->mime = $mime; + $this->contexts = array(); + $this->config = $config; + + list($upHost, $err) = $config->zone->getUpHostByToken($upToken); + if ($err != null) { + throw new \Exception($err, 1); + } + $this->host = $upHost; + } + + /** + * 涓婁紶鎿嶄綔 + */ + public function upload() + { + $uploaded = 0; + while ($uploaded < $this->size) { + $blockSize = $this->blockSize($uploaded); + $data = fread($this->inputStream, $blockSize); + if ($data === false) { + throw new \Exception("file read failed", 1); + } + $crc = \Qiniu\crc32_data($data); + $response = $this->makeBlock($data, $blockSize); + $ret = null; + if ($response->ok() && $response->json() != null) { + $ret = $response->json(); + } + if ($response->statusCode < 0) { + list($bakHost, $err) = $this->config->zone->getBackupUpHostByToken($this->upToken); + if ($err != null) { + return array(null, $err); + } + $this->host = $bakHost; + } + if ($response->needRetry() || !isset($ret['crc32']) || $crc != $ret['crc32']) { + $response = $this->makeBlock($data, $blockSize); + $ret = $response->json(); + } + + if (! $response->ok() || !isset($ret['crc32'])|| $crc != $ret['crc32']) { + return array(null, new Error($this->currentUrl, $response)); + } + array_push($this->contexts, $ret['ctx']); + $uploaded += $blockSize; + } + return $this->makeFile(); + } + + /** + * 鍒涘缓鍧 + */ + private function makeBlock($block, $blockSize) + { + $url = $this->host . '/mkblk/' . $blockSize; + return $this->post($url, $block); + } + + private function fileUrl() + { + $url = $this->host . '/mkfile/' . $this->size; + $url .= '/mimeType/' . \Qiniu\base64_urlSafeEncode($this->mime); + if ($this->key != null) { + $url .= '/key/' . \Qiniu\base64_urlSafeEncode($this->key); + } + if (!empty($this->params)) { + foreach ($this->params as $key => $value) { + $val = \Qiniu\base64_urlSafeEncode($value); + $url .= "/$key/$val"; + } + } + return $url; + } + + /** + * 鍒涘缓鏂囦欢 + */ + private function makeFile() + { + $url = $this->fileUrl(); + $body = implode(',', $this->contexts); + $response = $this->post($url, $body); + if ($response->needRetry()) { + $response = $this->post($url, $body); + } + if (! $response->ok()) { + return array(null, new Error($this->currentUrl, $response)); + } + return array($response->json(), null); + } + + private function post($url, $data) + { + $this->currentUrl = $url; + $headers = array('Authorization' => 'UpToken ' . $this->upToken); + return Client::post($url, $data, $headers); + } + + private function blockSize($uploaded) + { + if ($this->size < $uploaded + Config::BLOCK_SIZE) { + return $this->size - $uploaded; + } + return Config::BLOCK_SIZE; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Storage/UploadManager.php b/vendor/qiniu/php-sdk/src/Qiniu/Storage/UploadManager.php new file mode 100644 index 000000000..f80a81a97 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Storage/UploadManager.php @@ -0,0 +1,141 @@ +config = $config; + } + + /** + * 涓婁紶浜岃繘鍒舵祦鍒颁竷鐗 + * + * @param $upToken 涓婁紶鍑瘉 + * @param $key 涓婁紶鏂囦欢鍚 + * @param $data 涓婁紶浜岃繘鍒舵祦 + * @param $params 鑷畾涔夊彉閲忥紝瑙勬牸鍙傝 + * http://developer.qiniu.com/docs/v6/api/overview/up/response/vars.html#xvar + * @param $mime 涓婁紶鏁版嵁鐨刴imeType + * @param $checkCrc 鏄惁鏍¢獙crc32 + * + * @return array 鍖呭惈宸蹭笂浼犳枃浠剁殑淇℃伅锛岀被浼硷細 + * [ + * "hash" => "", + * "key" => "" + * ] + */ + public function put( + $upToken, + $key, + $data, + $params = null, + $mime = 'application/octet-stream', + $checkCrc = false + ) { + $params = self::trimParams($params); + return FormUploader::put( + $upToken, + $key, + $data, + $this->config, + $params, + $mime, + $checkCrc + ); + } + + + /** + * 涓婁紶鏂囦欢鍒颁竷鐗 + * + * @param $upToken 涓婁紶鍑瘉 + * @param $key 涓婁紶鏂囦欢鍚 + * @param $filePath 涓婁紶鏂囦欢鐨勮矾寰 + * @param $params 鑷畾涔夊彉閲忥紝瑙勬牸鍙傝 + * http://developer.qiniu.com/docs/v6/api/overview/up/response/vars.html#xvar + * @param $mime 涓婁紶鏁版嵁鐨刴imeType + * @param $checkCrc 鏄惁鏍¢獙crc32 + * + * @return array 鍖呭惈宸蹭笂浼犳枃浠剁殑淇℃伅锛岀被浼硷細 + * [ + * "hash" => "", + * "key" => "" + * ] + */ + public function putFile( + $upToken, + $key, + $filePath, + $params = null, + $mime = 'application/octet-stream', + $checkCrc = false + ) { + $file = fopen($filePath, 'rb'); + if ($file === false) { + throw new \Exception("file can not open", 1); + } + $params = self::trimParams($params); + $stat = fstat($file); + $size = $stat['size']; + if ($size <= Config::BLOCK_SIZE) { + $data = fread($file, $size); + fclose($file); + if ($data === false) { + throw new \Exception("file can not read", 1); + } + return FormUploader::put( + $upToken, + $key, + $data, + $this->config, + $params, + $mime, + $checkCrc + ); + } + + $up = new ResumeUploader( + $upToken, + $key, + $file, + $size, + $params, + $mime, + $this->config + ); + $ret = $up->upload(); + fclose($file); + return $ret; + } + + public static function trimParams($params) + { + if ($params === null) { + return null; + } + $ret = array(); + foreach ($params as $k => $v) { + $pos = strpos($k, 'x:'); + if ($pos === 0 && !empty($v)) { + $ret[$k] = $v; + } + } + return $ret; + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/Zone.php b/vendor/qiniu/php-sdk/src/Qiniu/Zone.php new file mode 100644 index 000000000..3f58430bd --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/Zone.php @@ -0,0 +1,185 @@ +:: ==> + // array('deadline' => 'xxx', 'upHosts' => array(), 'ioHost' => 'xxx.com') + //) + public $hostCache; + public $scheme = 'http'; + + public function __construct($scheme = null) + { + $this->hostCache = array(); + if ($scheme != null) { + $this->scheme = $scheme; + } + } + + public function getUpHostByToken($uptoken) + { + list($ak, $bucket) = $this->unmarshalUpToken($uptoken); + list($upHosts, $err) = $this->getUpHosts($ak, $bucket); + return array($upHosts[0], $err); + } + + public function getBackupUpHostByToken($uptoken) + { + list($ak, $bucket) = $this->unmarshalUpToken($uptoken); + list($upHosts, $err) = $this->getUpHosts($ak, $bucket); + + $upHost = isset($upHosts[1]) ? $upHosts[1] : $upHosts[0]; + return array($upHost, $err); + } + + public function getIoHost($ak, $bucket) + { + list($bucketHosts,) = $this->getBucketHosts($ak, $bucket); + $ioHosts = $bucketHosts['ioHost']; + return $ioHosts[0]; + } + + public function getUpHosts($ak, $bucket) + { + list($bucketHosts, $err) = $this->getBucketHosts($ak, $bucket); + if ($err !== null) { + return array(null, $err); + } + + $upHosts = $bucketHosts['upHosts']; + return array($upHosts, null); + } + + private function unmarshalUpToken($uptoken) + { + $token = explode(':', $uptoken); + if (count($token) !== 3) { + throw new \Exception("Invalid Uptoken", 1); + } + + $ak = $token[0]; + $policy = base64_urlSafeDecode($token[2]); + $policy = json_decode($policy, true); + + $scope = $policy['scope']; + $bucket = $scope; + + if (strpos($scope, ':')) { + $scopes = explode(':', $scope); + $bucket = $scopes[0]; + } + + return array($ak, $bucket); + } + + public function getBucketHosts($ak, $bucket) + { + $key = $this->scheme . ":$ak:$bucket"; + + $bucketHosts = $this->getBucketHostsFromCache($key); + if (count($bucketHosts) > 0) { + return array($bucketHosts, null); + } + + list($hosts, $err) = $this->bucketHosts($ak, $bucket); + if ($err !== null) { + return array(null , $err); + } + + $schemeHosts = $hosts[$this->scheme]; + $bucketHosts = array( + 'upHosts' => $schemeHosts['up'], + 'ioHost' => $schemeHosts['io'], + 'deadline' => time() + $hosts['ttl'] + ); + + $this->setBucketHostsToCache($key, $bucketHosts); + return array($bucketHosts, null); + } + + private function getBucketHostsFromCache($key) + { + $ret = array(); + if (count($this->hostCache) === 0) { + $this->hostCacheFromFile(); + } + + if (!array_key_exists($key, $this->hostCache)) { + return $ret; + } + + if ($this->hostCache[$key]['deadline'] > time()) { + $ret = $this->hostCache[$key]; + } + + return $ret; + } + + private function setBucketHostsToCache($key, $val) + { + $this->hostCache[$key] = $val; + $this->hostCacheToFile(); + return; + } + + private function hostCacheFromFile() + { + + $path = $this->hostCacheFilePath(); + if (!file_exists($path)) { + return; + } + + $bucketHosts = file_get_contents($path); + $this->hostCache = json_decode($bucketHosts, true); + return; + } + + private function hostCacheToFile() + { + $path = $this->hostCacheFilePath(); + file_put_contents($path, json_encode($this->hostCache), LOCK_EX); + return; + } + + private function hostCacheFilePath() + { + return sys_get_temp_dir() . '/.qiniu_phpsdk_hostscache.json'; + } + + /* 璇锋眰鍖咃細 + * GET /v1/query?ak=&&bucket= + * 杩斿洖鍖咃細 + * + * 200 OK { + * "ttl": , // 鏈夋晥鏃堕棿 + * "http": { + * "up": [], + * "io": [], // 褰揵ucket涓篻lobal鏃讹紝鎴戜滑涓嶉渶瑕乮ohost, io缂虹渷 + * }, + * "https": { + * "up": [], + * "io": [], // 褰揵ucket涓篻lobal鏃讹紝鎴戜滑涓嶉渶瑕乮ohost, io缂虹渷 + * } + * } + **/ + private function bucketHosts($ak, $bucket) + { + $url = Config::UC_HOST . '/v1/query' . "?ak=$ak&bucket=$bucket"; + $ret = Client::Get($url); + if (!$ret->ok()) { + return array(null, new Error($url, $ret)); + } + $r = ($ret->body === null) ? array() : $ret->json(); + return array($r, null); + } +} diff --git a/vendor/qiniu/php-sdk/src/Qiniu/functions.php b/vendor/qiniu/php-sdk/src/Qiniu/functions.php new file mode 100644 index 000000000..8e2620523 --- /dev/null +++ b/vendor/qiniu/php-sdk/src/Qiniu/functions.php @@ -0,0 +1,241 @@ + 'JSON_ERROR_DEPTH - Maximum stack depth exceeded', + JSON_ERROR_STATE_MISMATCH => 'JSON_ERROR_STATE_MISMATCH - Underflow or the modes mismatch', + JSON_ERROR_CTRL_CHAR => 'JSON_ERROR_CTRL_CHAR - Unexpected control character found', + JSON_ERROR_SYNTAX => 'JSON_ERROR_SYNTAX - Syntax error, malformed JSON', + JSON_ERROR_UTF8 => 'JSON_ERROR_UTF8 - Malformed UTF-8 characters, possibly incorrectly encoded' + ); + + if (empty($json)) { + return null; + } + $data = \json_decode($json, $assoc, $depth); + + if (JSON_ERROR_NONE !== json_last_error()) { + $last = json_last_error(); + throw new \InvalidArgumentException( + 'Unable to parse JSON data: ' + . (isset($jsonErrors[$last]) + ? $jsonErrors[$last] + : 'Unknown error') + ); + } + + return $data; + } + + /** + * 璁$畻涓冪墰API涓殑鏁版嵁鏍煎紡 + * + * @param $bucket 寰呮搷浣滅殑绌洪棿鍚 + * @param $key 寰呮搷浣滅殑鏂囦欢鍚 + * + * @return string 绗﹀悎涓冪墰API瑙勬牸鐨勬暟鎹牸寮 + * @link http://developer.qiniu.com/docs/v6/api/reference/data-formats.html + */ + function entry($bucket, $key) + { + $en = $bucket; + if (!empty($key)) { + $en = $bucket . ':' . $key; + } + return base64_urlSafeEncode($en); + } + + /** + * array 杈呭姪鏂规硶锛屾棤鍊兼椂涓峴et + * + * @param $array 寰呮搷浣渁rray + * @param $key key + * @param $value value 涓簄ull鏃 涓嶈缃 + * + * @return array 鍘熸潵鐨刟rray锛屼究浜庤繛缁搷浣 + */ + function setWithoutEmpty(&$array, $key, $value) + { + if (!empty($value)) { + $array[$key] = $value; + } + return $array; + } + + /** + * 缂╃暐鍥鹃摼鎺ユ嫾鎺 + * + * @param string $url 鍥剧墖閾炬帴 + * @param int $mode 缂╃暐妯″紡 + * @param int $width 瀹藉害 + * @param int $height 闀垮害 + * @param string $format 杈撳嚭绫诲瀷 + * @param int $quality 鍥剧墖璐ㄩ噺 + * @param int $interlace 鏄惁鏀寔娓愯繘鏄剧ず + * @param int $ignoreError 蹇界暐缁撴灉 + * @return string + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/imageview2.html + * @author Sherlock Ren + */ + function thumbnail( + $url, + $mode, + $width, + $height, + $format = null, + $quality = null, + $interlace = null, + $ignoreError = 1 + ) { + static $imageUrlBuilder = null; + if (is_null($imageUrlBuilder)) { + $imageUrlBuilder = new \Qiniu\Processing\ImageUrlBuilder; + } + + return call_user_func_array(array($imageUrlBuilder, 'thumbnail'), func_get_args()); + } + + /** + * 鍥剧墖姘村嵃 + * + * @param string $url 鍥剧墖閾炬帴 + * @param string $image 姘村嵃鍥剧墖閾炬帴 + * @param numeric $dissolve 閫忔槑搴 + * @param string $gravity 姘村嵃浣嶇疆 + * @param numeric $dx 妯酱杈硅窛 + * @param numeric $dy 绾佃酱杈硅窛 + * @param numeric $watermarkScale 鑷傚簲鍘熷浘鐨勭煭杈规瘮渚 + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/watermark.html + * @return string + * @author Sherlock Ren + */ + function waterImg( + $url, + $image, + $dissolve = 100, + $gravity = 'SouthEast', + $dx = null, + $dy = null, + $watermarkScale = null + ) { + static $imageUrlBuilder = null; + if (is_null($imageUrlBuilder)) { + $imageUrlBuilder = new \Qiniu\Processing\ImageUrlBuilder; + } + + return call_user_func_array(array($imageUrlBuilder, 'waterImg'), func_get_args()); + } + + /** + * 鏂囧瓧姘村嵃 + * + * @param string $url 鍥剧墖閾炬帴 + * @param string $text 鏂囧瓧 + * @param string $font 鏂囧瓧瀛椾綋 + * @param string $fontSize 鏂囧瓧瀛楀彿 + * @param string $fontColor 鏂囧瓧棰滆壊 + * @param numeric $dissolve 閫忔槑搴 + * @param string $gravity 姘村嵃浣嶇疆 + * @param numeric $dx 妯酱杈硅窛 + * @param numeric $dy 绾佃酱杈硅窛 + * @link http://developer.qiniu.com/code/v6/api/kodo-api/image/watermark.html#text-watermark + * @return string + * @author Sherlock Ren + */ + function waterText( + $url, + $text, + $font = '榛戜綋', + $fontSize = 0, + $fontColor = null, + $dissolve = 100, + $gravity = 'SouthEast', + $dx = null, + $dy = null + ) { + static $imageUrlBuilder = null; + if (is_null($imageUrlBuilder)) { + $imageUrlBuilder = new \Qiniu\Processing\ImageUrlBuilder; + } + + return call_user_func_array(array($imageUrlBuilder, 'waterText'), func_get_args()); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/AuthTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/AuthTest.php new file mode 100644 index 000000000..9f4778c18 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/AuthTest.php @@ -0,0 +1,85 @@ +sign('test'); + $this->assertEquals('abcdefghklmnopq:mSNBTR7uS2crJsyFr2Amwv1LaYg=', $token); + } + + public function testSignWithData() + { + global $dummyAuth; + $token = $dummyAuth->signWithData('test'); + $this->assertEquals('abcdefghklmnopq:-jP8eEV9v48MkYiBGs81aDxl60E=:dGVzdA==', $token); + } + + public function testSignRequest() + { + global $dummyAuth; + $token = $dummyAuth->signRequest('http://www.qiniu.com?go=1', 'test', ''); + $this->assertEquals('abcdefghklmnopq:cFyRVoWrE3IugPIMP5YJFTO-O-Y=', $token); + $ctype = 'application/x-www-form-urlencoded'; + $token = $dummyAuth->signRequest('http://www.qiniu.com?go=1', 'test', $ctype); + $this->assertEquals($token, 'abcdefghklmnopq:svWRNcacOE-YMsc70nuIYdaa1e4='); + } + + public function testPrivateDownloadUrl() + { + global $dummyAuth; + $_SERVER['override_qiniu_auth_time'] = true; + $url = $dummyAuth->privateDownloadUrl('http://www.qiniu.com?go=1'); + $expect = 'http://www.qiniu.com?go=1&e=1234571490&token=abcdefghklmnopq:8vzBeLZ9W3E4kbBLFLW0Xe0u7v4='; + $this->assertEquals($expect, $url); + unset($_SERVER['override_qiniu_auth_time']); + } + + /** + * @expectedException InvalidArgumentException + * @expectedExceptionMessage asyncOps has deprecated + */ + public function testDeprecatedPolicy() + { + global $dummyAuth; + $token = $dummyAuth->uploadToken('1', null, 3600, array('asyncOps'=> 1)); + } + + public function testUploadToken() + { + global $dummyAuth; + $_SERVER['override_qiniu_auth_time'] = true; + $token = $dummyAuth->uploadToken('1', '2', 3600, array('endUser'=> 'y')); + // @codingStandardsIgnoreStart + $exp = 'abcdefghklmnopq:yyeexeUkPOROoTGvwBjJ0F0VLEo=:eyJlbmRVc2VyIjoieSIsInNjb3BlIjoiMToyIiwiZGVhZGxpbmUiOjEyMzQ1NzE0OTB9'; + // @codingStandardsIgnoreEnd + $this->assertEquals($exp, $token); + unset($_SERVER['override_qiniu_auth_time']); + } + + public function testVerifyCallback() + { + } + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/Base64Test.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/Base64Test.php new file mode 100644 index 000000000..6d6335307 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/Base64Test.php @@ -0,0 +1,14 @@ +assertEquals($a, \Qiniu\base64_urlSafeDecode($b)); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/BucketTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/BucketTest.php new file mode 100644 index 000000000..288e1bf58 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/BucketTest.php @@ -0,0 +1,214 @@ +bucketName = $bucketName; + $this->key = $key; + $this->key2 = $key2; + + global $testAuth; + $this->bucketManager = new BucketManager($testAuth); + + global $dummyAuth; + $this->dummyBucketManager = new BucketManager($dummyAuth); + } + + public function testBuckets() + { + + list($list, $error) = $this->bucketManager->buckets(); + $this->assertTrue(in_array($this->bucketName, $list)); + $this->assertNull($error); + + list($list2, $error) = $this->dummyBucketManager->buckets(); + $this->assertEquals(401, $error->code()); + $this->assertNull($list2); + $this->assertNotNull($error->message()); + } + + public function testList() + { + list($items, $marker, $error) = $this->bucketManager->listFiles($this->bucketName, null, null, 2); + $this->assertNotNull($items[0]); + $this->assertNotNull($marker); + } + + public function testStat() + { + list($stat, $error) = $this->bucketManager->stat($this->bucketName, $this->key); + $this->assertNotNull($stat); + $this->assertNull($error); + $this->assertNotNull($stat['hash']); + + list($stat, $error) = $this->bucketManager->stat($this->bucketName, 'nofile'); + $this->assertNull($stat); + $this->assertEquals(612, $error->code()); + $this->assertNotNull($error->message()); + + list($stat, $error) = $this->bucketManager->stat('nobucket', 'nofile'); + $this->assertNull($stat); + $this->assertEquals(631, $error->code()); + $this->assertNotNull($error->message()); + } + + public function testDelete() + { + $error = $this->bucketManager->delete($this->bucketName, 'del'); + $this->assertEquals(612, $error->code()); + } + + + public function testRename() + { + $key = 'renamefrom' . rand(); + $this->bucketManager->copy($this->bucketName, $this->key, $this->bucketName, $key); + $key2 = 'renameto' . $key; + $error = $this->bucketManager->rename($this->bucketName, $key, $key2); + $this->assertNull($error); + $error = $this->bucketManager->delete($this->bucketName, $key2); + $this->assertNull($error); + } + + + public function testCopy() + { + $key = 'copyto' . rand(); + $this->bucketManager->delete($this->bucketName, $key); + + $error = $this->bucketManager->copy( + $this->bucketName, + $this->key, + $this->bucketName, + $key + ); + $this->assertNull($error); + + //test force copy + $error = $this->bucketManager->copy( + $this->bucketName, + $this->key2, + $this->bucketName, + $key, + true + ); + $this->assertNull($error); + + list($key2Stat,) = $this->bucketManager->stat($this->bucketName, $this->key2); + list($key2CopiedStat,) = $this->bucketManager->stat($this->bucketName, $key); + + var_dump($key2Stat); + $this->assertEquals($key2Stat['hash'], $key2CopiedStat['hash']); + + $error = $this->bucketManager->delete($this->bucketName, $key); + $this->assertNull($error); + } + + + public function testChangeMime() + { + $error = $this->bucketManager->changeMime( + $this->bucketName, + 'php-sdk.html', + 'text/html' + ); + $this->assertNull($error); + } + + public function testPrefetch() + { + $error = $this->bucketManager->prefetch( + $this->bucketName, + 'php-sdk.html' + ); + $this->assertNull($error); + } + + public function testFetch() + { + list($ret, $error) = $this->bucketManager->fetch( + 'http://developer.qiniu.com/docs/v6/sdk/php-sdk.html', + $this->bucketName, + 'fetch.html' + ); + $this->assertArrayHasKey('hash', $ret); + $this->assertNull($error); + + list($ret, $error) = $this->bucketManager->fetch( + 'http://developer.qiniu.com/docs/v6/sdk/php-sdk.html', + $this->bucketName, + '' + ); + $this->assertArrayHasKey('key', $ret); + $this->assertNull($error); + + list($ret, $error) = $this->bucketManager->fetch( + 'http://developer.qiniu.com/docs/v6/sdk/php-sdk.html', + $this->bucketName + ); + $this->assertArrayHasKey('key', $ret); + $this->assertNull($error); + } + + public function testBatchCopy() + { + $key = 'copyto' . rand(); + $ops = BucketManager::buildBatchCopy( + $this->bucketName, + array($this->key => $key), + $this->bucketName + ); + list($ret, $error) = $this->bucketManager->batch($ops); + $this->assertEquals(200, $ret[0]['code']); + $ops = BucketManager::buildBatchDelete($this->bucketName, array($key)); + list($ret, $error) = $this->bucketManager->batch($ops); + $this->assertEquals(200, $ret[0]['code']); + } + + public function testBatchMove() + { + $key = 'movefrom'. rand(); + $this->bucketManager->copy($this->bucketName, $this->key, $this->bucketName, $key); + $key2 = $key . 'to'; + $ops = BucketManager::buildBatchMove( + $this->bucketName, + array($key => $key2), + $this->bucketName + ); + list($ret, $error) = $this->bucketManager->batch($ops); + $this->assertEquals(200, $ret[0]['code']); + $error = $this->bucketManager->delete($this->bucketName, $key2); + $this->assertNull($error); + } + + public function testBatchRename() + { + $key = 'rename' . rand(); + $this->bucketManager->copy($this->bucketName, $this->key, $this->bucketName, $key); + $key2 = $key . 'to'; + $ops = BucketManager::buildBatchRename($this->bucketName, array($key => $key2)); + list($ret, $error) = $this->bucketManager->batch($ops); + $this->assertEquals(200, $ret[0]['code']); + $error = $this->bucketManager->delete($this->bucketName, $key2); + $this->assertNull($error); + } + + public function testBatchStat() + { + $ops = BucketManager::buildBatchStat($this->bucketName, array('php-sdk.html')); + list($ret, $error) = $this->bucketManager->batch($ops); + $this->assertEquals(200, $ret[0]['code']); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/Crc32Test.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/Crc32Test.php new file mode 100644 index 000000000..bfb36da87 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/Crc32Test.php @@ -0,0 +1,21 @@ +assertEquals('1352841281', $b); + } + + public function testFile() + { + $b = \Qiniu\crc32_file(__file__); + $c = \Qiniu\crc32_file(__file__); + $this->assertEquals($c, $b); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/DownloadTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/DownloadTest.php new file mode 100644 index 000000000..82990f2a1 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/DownloadTest.php @@ -0,0 +1,25 @@ +privateDownloadUrl($base_url); + $response = Client::get($private_url); + $this->assertEquals(200, $response->statusCode); + } + + public function testFop() + { + global $testAuth; + $base_url = 'http://private-res.qiniudn.com/gogopher.jpg?exif'; + $private_url = $testAuth->privateDownloadUrl($base_url); + $response = Client::get($private_url); + $this->assertEquals(200, $response->statusCode); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/EtagTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/EtagTest.php new file mode 100644 index 000000000..d95857d61 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/EtagTest.php @@ -0,0 +1,52 @@ +assertEquals('Fto5o-5ea0sNMlW_75VgGJCv2AcJ', $r); + $this->assertNull($error); + } + + public function testLess4M() + { + $file = qiniuTempFile(3*1024*1024); + list($r, $error) = Etag::sum($file); + unlink($file); + $this->assertEquals('lrGEYiVvREEgGl_foQEnZ5a5BqwZ', $r); + $this->assertNull($error); + } + + public function test4M() + { + $file = qiniuTempFile(4*1024*1024); + list($r, $error) = Etag::sum($file); + unlink($file); + $this->assertEquals('lvOwUCzD-YVymzwJLRGZR3eD__GV', $r); + $this->assertNull($error); + } + + public function testMore4M() + { + $file = qiniuTempFile(5*1024*1024); + list($r, $error) = Etag::sum($file); + unlink($file); + $this->assertEquals('lqtEDHt7Yo5j1a2mjlB2Ds8DUYNM', $r); + $this->assertNull($error); + } + + public function test8M() + { + $file = qiniuTempFile(8*1024*1024); + list($r, $error) = Etag::sum($file); + unlink($file); + $this->assertEquals('ljpekgMJ9VSYlE8hMX06GIWXxfDI', $r); + $this->assertNull($error); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/FopTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/FopTest.php new file mode 100644 index 000000000..e1ea730e8 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/FopTest.php @@ -0,0 +1,37 @@ +execute('gogopher.jpg', 'exif'); + $this->assertNull($error); + $this->assertNotNull($exif); + } + + public function testExifPrivate() + { + global $testAuth; + $fop = new Operation('private-res.qiniudn.com', $testAuth); + list($exif, $error) = $fop->execute('noexif.jpg', 'exif'); + $this->assertNotNull($error); + $this->assertNull($exif); + } + + public function testbuildUrl() + { + $fops = 'imageView2/2/h/200'; + $fop = new Operation('testres.qiniudn.com'); + $url = $fop->buildUrl('gogopher.jpg', $fops); + $this->assertEquals($url, 'http://testres.qiniudn.com/gogopher.jpg?imageView2/2/h/200'); + + $fops = array('imageView2/2/h/200', 'imageInfo'); + $url = $fop->buildUrl('gogopher.jpg', $fops); + $this->assertEquals($url, 'http://testres.qiniudn.com/gogopher.jpg?imageView2/2/h/200|imageInfo'); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/FormUpTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/FormUpTest.php new file mode 100644 index 000000000..6f324c624 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/FormUpTest.php @@ -0,0 +1,59 @@ +bucketName = $bucketName; + + global $testAuth; + $this->auth = $testAuth; + $this->cfg = new Config(); + } + + public function testData() + { + $token = $this->auth->uploadToken($this->bucketName); + list($ret, $error) = FormUploader::put($token, 'formput', 'hello world', $this->cfg, null, 'text/plain', true); + $this->assertNull($error); + $this->assertNotNull($ret['hash']); + } + + public function testData2() + { + $upManager = new UploadManager(); + $token = $this->auth->uploadToken($this->bucketName); + list($ret, $error) = $upManager->put($token, 'formput', 'hello world', null, 'text/plain', true); + $this->assertNull($error); + $this->assertNotNull($ret['hash']); + } + + public function testFile() + { + $key = 'formPutFile'; + $token = $this->auth->uploadToken($this->bucketName, $key); + list($ret, $error) = FormUploader::putFile($token, $key, __file__, $this->cfg, null, 'text/plain', true); + $this->assertNull($error); + $this->assertNotNull($ret['hash']); + } + + public function testFile2() + { + $key = 'formPutFile'; + $token = $this->auth->uploadToken($this->bucketName, $key); + $upManager = new UploadManager(); + list($ret, $error) = $upManager->putFile($token, $key, __file__, null, 'text/plain', true); + $this->assertNull($error); + $this->assertNotNull($ret['hash']); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/HttpTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/HttpTest.php new file mode 100644 index 000000000..73eaf30f3 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/HttpTest.php @@ -0,0 +1,43 @@ +assertEquals($response->statusCode, 200); + $this->assertNotNull($response->body); + $this->assertNull($response->error); + } + + public function testGetQiniu() + { + $response = Client::get('up.qiniu.com'); + $this->assertEquals(404, $response->statusCode); + $this->assertNotNull($response->body); + $this->assertNotNull($response->xReqId()); + $this->assertNotNull($response->xLog()); + $this->assertNotNull($response->error); + } + + public function testPost() + { + $response = Client::post('baidu.com', null); + $this->assertEquals($response->statusCode, 200); + $this->assertNotNull($response->body); + $this->assertNull($response->error); + } + + public function testPostQiniu() + { + $response = Client::post('up.qiniu.com', null); + $this->assertEquals($response->statusCode, 400); + $this->assertNotNull($response->body); + $this->assertNotNull($response->xReqId()); + $this->assertNotNull($response->xLog()); + $this->assertNotNull($response->error); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ImageUrlBuilderTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ImageUrlBuilderTest.php new file mode 100644 index 000000000..26050f999 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ImageUrlBuilderTest.php @@ -0,0 +1,261 @@ + + */ +class ImageUrlBuilderTest extends \PHPUnit_Framework_TestCase +{ + /** + * 缂╃暐鍥炬祴璇 + * + * @test + * @return void + * @author Sherlock Ren + */ + public function testThumbutl() + { + $imageUrlBuilder = new \Qiniu\Processing\ImageUrlBuilder(); + $url = 'http://78re52.com1.z0.glb.clouddn.com/resource/gogopher.jpg'; + $url2 = $url . '?watermark/1/gravity/SouthEast/dx/0/dy/0/image/' + . 'aHR0cDovL2Fkcy1jZG4uY2h1Y2h1amllLmNvbS9Ga1R6bnpIY2RLdmRBUFc5cHZZZ3pTc21UY0tB'; + // 寮傚父娴嬭瘯 + $this->assertEquals($url, $imageUrlBuilder->thumbnail($url, 1, 0, 0)); + $this->assertEquals($url, \Qiniu\thumbnail($url, 1, 0, 0)); + + // 绠鍗曠缉鐣ユ祴璇 + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/ignore-error/1/', + $imageUrlBuilder->thumbnail($url, 1, 200, 200) + ); + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/ignore-error/1/', + \Qiniu\thumbnail($url, 1, 200, 200) + ); + + // 杈撳嚭鏍煎紡娴嬭瘯 + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/format/png/ignore-error/1/', + $imageUrlBuilder->thumbnail($url, 1, 200, 200, 'png') + ); + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/format/png/ignore-error/1/', + \Qiniu\thumbnail($url, 1, 200, 200, 'png') + ); + + // 娓愯繘鏄剧ず娴嬭瘯 + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/format/png/interlace/1/ignore-error/1/', + $imageUrlBuilder->thumbnail($url, 1, 200, 200, 'png', 1) + ); + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/format/png/ignore-error/1/', + \Qiniu\thumbnail($url, 1, 200, 200, 'png', 2) + ); + + // 鍥剧墖璐ㄩ噺娴嬭瘯 + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/format/png/interlace/1/q/80/ignore-error/1/', + $imageUrlBuilder->thumbnail($url, 1, 200, 200, 'png', 1, 80) + ); + $this->assertEquals( + $url . '?imageView2/1/w/200/h/200/format/png/interlace/1/ignore-error/1/', + \Qiniu\thumbnail($url, 1, 200, 200, 'png', 1, 101) + ); + + // 澶氬弬鏁版祴璇 + $this->assertEquals( + $url2 . '|imageView2/1/w/200/h/200/ignore-error/1/', + $imageUrlBuilder->thumbnail($url2, 1, 200, 200) + ); + $this->assertEquals( + $url2 . '|imageView2/1/w/200/h/200/ignore-error/1/', + \Qiniu\thumbnail($url2, 1, 200, 200) + ); + } + + /** + * 鍥剧墖姘村嵃娴嬭瘯 + * + * @test + * @param void + * @return void + * @author Sherlock Ren + */ + public function waterImgTest() + { + $imageUrlBuilder = new \Qiniu\Processing\ImageUrlBuilder(); + $url = 'http://78re52.com1.z0.glb.clouddn.com/resource/gogopher.jpg'; + $url2 = $url . '?imageView2/1/w/200/h/200/format/png/ignore-error/1/'; + $image = 'http://developer.qiniu.com/resource/logo-2.jpg'; + + // 姘村嵃绠鍗曟祴璇 + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + $imageUrlBuilder->waterImg($url, $image) + ); + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/gravity/SouthEast/', + $imageUrlBuilder->waterImg($url, $image, 101) + ); + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==/', + $imageUrlBuilder->waterImg($url, $image, 101, 'sdfsd') + ); + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + \Qiniu\waterImg($url, $image) + ); + + // 妯酱杈硅窛娴嬭瘯 + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/dx/10/', + $imageUrlBuilder->waterImg($url, $image, 100, 'SouthEast', 10) + ); + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + \Qiniu\waterImg($url, $image, 100, 'SouthEast', 'sad') + ); + + // 绾佃酱杈硅窛娴嬭瘯 + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/dx/10/dy/10/', + $imageUrlBuilder->waterImg($url, $image, 100, 'SouthEast', 10, 10) + ); + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + \Qiniu\waterImg($url, $image, 100, 'SouthEast', 'sad', 'asdf') + ); + + // 鑷傚簲鍘熷浘鐨勭煭杈规瘮渚嬫祴璇 + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/dx/10/dy/10/ws/0.5/', + $imageUrlBuilder->waterImg($url, $image, 100, 'SouthEast', 10, 10, 0.5) + ); + $this->assertEquals( + $url . '?watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + \Qiniu\waterImg($url, $image, 100, 'SouthEast', 'sad', 'asdf', 2) + ); + + // 澶氬弬鏁版祴璇 + $this->assertEquals( + $url2 . '|watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + $imageUrlBuilder->waterImg($url2, $image) + ); + $this->assertEquals( + $url2 . '|watermark/1/image/aHR0cDovL2RldmVsb3Blci5xaW5pdS5jb20vcmVzb3VyY2UvbG9nby0yLmpwZw==' + . '/dissolve/100/gravity/SouthEast/', + \Qiniu\waterImg($url2, $image) + ); + } + + /** + * 鏂囧瓧姘村嵃娴嬭瘯 + * + * @test + * @param void + * @return void + * @author Sherlock Ren + */ + public function waterTextTest() + { + $imageUrlBuilder = new \Qiniu\Processing\ImageUrlBuilder(); + $url = 'http://78re52.com1.z0.glb.clouddn.com/resource/gogopher.jpg'; + $url2 = $url . '?imageView2/1/w/200/h/200/format/png/ignore-error/1/'; + $text = '娴嬭瘯涓涓'; + $font = '寰蒋闆呴粦'; + $fontColor = '#FF0000'; + + // 姘村嵃绠鍗曟祴璇 + $this->assertEquals($url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/' + . 'fontsize/500/dissolve/100/gravity/SouthEast/', $imageUrlBuilder->waterText($url, $text, $font, 500)); + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/' + . 'dissolve/100/gravity/SouthEast/', + \Qiniu\waterText($url, $text, $font, 'sdf') + ); + + // 瀛椾綋棰滆壊娴嬭瘯 + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fontsize/500/fill/' + . 'I0ZGMDAwMA==/dissolve/100/gravity/SouthEast/', + $imageUrlBuilder->waterText($url, $text, $font, 500, $fontColor) + ); + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fill/I0ZGMDAwMA==' + . '/dissolve/100/gravity/SouthEast/', + \Qiniu\waterText($url, $text, $font, 'sdf', $fontColor) + ); + + // 閫忔槑搴︽祴璇 + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fontsize/500/fill/I0ZGMDAwMA==' + . '/dissolve/80/gravity/SouthEast/', + $imageUrlBuilder->waterText($url, $text, $font, 500, $fontColor, 80) + ); + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fill/I0ZGMDAwMA==' + . '/gravity/SouthEast/', + \Qiniu\waterText($url, $text, $font, 'sdf', $fontColor, 101) + ); + + // 姘村嵃浣嶇疆娴嬭瘯 + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fontsize/500/fill/I0ZGMDAwMA==' + . '/dissolve/80/gravity/East/', + $imageUrlBuilder->waterText($url, $text, $font, 500, $fontColor, 80, 'East') + ); + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fill/I0ZGMDAwMA==/', + \Qiniu\waterText($url, $text, $font, 'sdf', $fontColor, 101, 'sdfsdf') + ); + + // 妯酱璺濈娴嬭瘯 + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fontsize/500/fill/I0ZGMDAwMA==' + . '/dissolve/80/gravity/East/dx/10/', + $imageUrlBuilder->waterText($url, $text, $font, 500, $fontColor, 80, 'East', 10) + ); + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fill/I0ZGMDAwMA==/', + \Qiniu\waterText($url, $text, $font, 'sdf', $fontColor, 101, 'sdfsdf', 'sdfs') + ); + + // 绾佃酱璺濈娴嬭瘯 + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fontsize/500/fill/I0ZGMDAwMA==' + . '/dissolve/80/gravity/East/dx/10/dy/10/', + $imageUrlBuilder->waterText($url, $text, $font, 500, $fontColor, 80, 'East', 10, 10) + ); + $this->assertEquals( + $url . '?watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/fill/I0ZGMDAwMA==/', + \Qiniu\waterText($url, $text, $font, 'sdf', $fontColor, 101, 'sdfsdf', 'sdfs', 'ssdf') + ); + // 澶氬弬鏁版祴璇 + $this->assertEquals( + $url2 . '|watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/' + . 'fontsize/500/dissolve/100/gravity/SouthEast/', + $imageUrlBuilder->waterText($url2, $text, $font, 500) + ); + $this->assertEquals( + $url2 . '|watermark/2/text/5rWL6K-V5LiA5LiL/font/5b6u6L2v6ZuF6buR/' + . 'fontsize/500/dissolve/100/gravity/SouthEast/', + \Qiniu\waterText($url2, $text, $font, 500) + ); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/PfopTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/PfopTest.php new file mode 100644 index 000000000..211df029e --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/PfopTest.php @@ -0,0 +1,66 @@ +execute($key, $fops); + $this->assertNull($error); + list($status, $error) = PersistentFop::status($id); + $this->assertNotNull($status); + $this->assertNull($error); + } + + + public function testPfops() + { + global $testAuth; + $bucket = 'testres'; + $key = 'sintel_trailer.mp4'; + $fops = array( + 'avthumb/m3u8/segtime/10/vcodec/libx264/s/320x240', + 'vframe/jpg/offset/7/w/480/h/360', + ); + $pfop = new PersistentFop($testAuth, $bucket); + + list($id, $error) = $pfop->execute($key, $fops); + $this->assertNull($error); + + list($status, $error) = PersistentFop::status($id); + $this->assertNotNull($status); + $this->assertNull($error); + } + + public function testMkzip() + { + global $testAuth; + $bucket = 'phpsdk'; + $key = 'php-logo.png'; + $pfop = new PersistentFop($testAuth, $bucket); + + $url1 = 'http://phpsdk.qiniudn.com/php-logo.png'; + $url2 = 'http://phpsdk.qiniudn.com/php-sdk.html'; + $zipKey = 'test.zip'; + + $fops = 'mkzip/2/url/' . \Qiniu\base64_urlSafeEncode($url1); + $fops .= '/url/' . \Qiniu\base64_urlSafeEncode($url2); + $fops .= '|saveas/' . \Qiniu\base64_urlSafeEncode("$bucket:$zipKey"); + + list($id, $error) = $pfop->execute($key, $fops); + $this->assertNull($error); + + list($status, $error) = PersistentFop::status($id); + $this->assertNotNull($status); + $this->assertNull($error); + } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ResumeUpTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ResumeUpTest.php new file mode 100644 index 000000000..ba896dcf6 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ResumeUpTest.php @@ -0,0 +1,58 @@ +bucketName = $bucketName; + + global $testAuth; + $this->auth = $testAuth; + } + + public function test4ML() + { + $key = 'resumePutFile4ML'; + $upManager = new UploadManager(); + $token = $this->auth->uploadToken($this->bucketName, $key); + $tempFile = qiniuTempFile(4*1024*1024+10); + list($ret, $error) = $upManager->putFile($token, $key, $tempFile); + $this->assertNull($error); + $this->assertNotNull($ret['hash']); + unlink($tempFile); + } + + public function test4ML2() + { + $key = 'resumePutFile4ML'; + $zone = new Zone('http://up.fake.qiniu.com', 'http://up.qiniu.com'); + $cfg = new Config($zone); + $upManager = new UploadManager($cfg); + $token = $this->auth->uploadToken($this->bucketName, $key); + $tempFile = qiniuTempFile(4*1024*1024+10); + list($ret, $error) = $upManager->putFile($token, $key, $tempFile); + $this->assertNull($error); + $this->assertNotNull($ret['hash']); + unlink($tempFile); + } + // public function test8M() + // { + // $key = 'resumePutFile8M'; + // $upManager = new UploadManager(); + // $token = $this->auth->uploadToken($this->bucketName, $key); + // $tempFile = qiniuTempFile(8*1024*1024+10); + // list($ret, $error) = $upManager->putFile($token, $key, $tempFile); + // $this->assertNull($error); + // $this->assertNotNull($ret['hash']); + // unlink($tempFile); + // } +} diff --git a/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ZoneTest.php b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ZoneTest.php new file mode 100644 index 000000000..b633b9a3c --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/Qiniu/Tests/ZoneTest.php @@ -0,0 +1,127 @@ +bucketName = $bucketName; + + global $bucketNameBC; + $this->bucketNameBC = $bucketNameBC; + + global $bucketNameNA; + $this->bucketNameNA = $bucketNameNA; + + global $accessKey; + $this->ak = $accessKey; + + $this->zone = new Zone(); + $this->zoneHttps = new Zone('https'); + } + + public function testUpHosts() + { + + // test nb http + list($upHosts, $err) = $this->zone->getUpHosts($this->ak, $this->bucketName); + $this->assertNull($err); + $this->assertEquals('http://up.qiniu.com', $upHosts[0]); + $this->assertEquals('http://upload.qiniu.com', $upHosts[1]); + + // test bc http + list($upHosts, $err) = $this->zone->getUpHosts($this->ak, $this->bucketNameBC); + $this->assertNull($err); + $this->assertEquals('http://up-z1.qiniu.com', $upHosts[0]); + $this->assertEquals('http://upload-z1.qiniu.com', $upHosts[1]); + + // test na http + list($upHosts, $err) = $this->zone->getUpHosts($this->ak, $this->bucketNameNA); + $this->assertNull($err); + $this->assertEquals('http://up-na0.qiniu.com', $upHosts[0]); + + // test nb https + list($upHosts, $err) = $this->zoneHttps->getUpHosts($this->ak, $this->bucketName); + $this->assertNull($err); + $this->assertEquals('https://up.qbox.me', $upHosts[0]); + + // test bc https + list($upHosts, $err) = $this->zoneHttps->getUpHosts($this->ak, $this->bucketNameBC); + $this->assertNull($err); + $this->assertEquals('https://up-z1.qbox.me', $upHosts[0]); + + // test na https + list($upHosts, $err) = $this->zoneHttps->getUpHosts($this->ak, $this->bucketNameNA); + $this->assertNull($err); + $this->assertEquals('https://up-na0.qbox.me', $upHosts[0]); + } + + public function testUpHostByToken() + { + $uptoken_bc = 'QWYn5TFQsLLU1pL5MFEmX3s5DmHdUThav9WyOWOm:bl77a3xPdTyBNYFGVRy + oIQNyp_s=:eyJzY29wZSI6InBocHNkay1iYyIsImRlYWRsaW5lIjoxNDcwNzI1MzE1LCJ1cEhvc + 3RzIjpbImh0dHA6XC9cL3VwLXoxLnFpbml1LmNvbSIsImh0dHA6XC9cL3VwbG9hZC16MS5xaW5p + dS5jb20iLCItSCB1cC16MS5xaW5pdS5jb20gaHR0cDpcL1wvMTA2LjM4LjIyNy4yNyJdfQ=='; + + list($upHost, $err) = $this->zone->getUpHostByToken($uptoken_bc); + $this->assertEquals('http://up-z1.qiniu.com', $upHost); + $this->assertEquals(null, $err); + + list($upHostBackup, $err) = $this->zone->getBackupUpHostByToken($uptoken_bc); + $this->assertEquals('http://upload-z1.qiniu.com', $upHostBackup); + $this->assertEquals(null, $err); + + + $uptoken_bc_https = 'QWYn5TFQsLLU1pL5MFEmX3s5DmHdUThav9WyOWOm:7I47O-vFcN5TKO + 6D7cobHPVkyIA=:eyJzY29wZSI6InBocHNkay1iYyIsImRlYWRsaW5lIjoxNDcwNzIyNzQ1LCJ1c + Ehvc3RzIjpbImh0dHBzOlwvXC91cC16MS5xYm94Lm1lIl19'; + list($upHost, $err) = $this->zoneHttps->getUpHostByToken($uptoken_bc_https); + $this->assertEquals('https://up-z1.qbox.me', $upHost); + $this->assertEquals(null, $err); + + list($upHostBackup, $err) = $this->zoneHttps->getBackupUpHostByToken($uptoken_bc_https); + $this->assertEquals('https://up-z1.qbox.me', $upHostBackup); + $this->assertEquals(null, $err); + } + + public function testIoHosts() + { + + // test nb http + $ioHost = $this->zone->getIoHost($this->ak, $this->bucketName); + $this->assertEquals('http://iovip.qbox.me', $ioHost); + + // test bc http + $ioHost = $this->zone->getIoHost($this->ak, $this->bucketNameBC); + $this->assertEquals('http://iovip-z1.qbox.me', $ioHost); + + // test na http + $ioHost = $this->zone->getIoHost($this->ak, $this->bucketNameNA); + $this->assertEquals('http://iovip-na0.qbox.me', $ioHost); + + // test nb https + $ioHost = $this->zoneHttps->getIoHost($this->ak, $this->bucketName); + $this->assertEquals('https://iovip.qbox.me', $ioHost); + + // test bc https + $ioHost = $this->zoneHttps->getIoHost($this->ak, $this->bucketNameBC); + $this->assertEquals('https://iovip-z1.qbox.me', $ioHost); + + // test na https + $ioHost = $this->zoneHttps->getIoHost($this->ak, $this->bucketNameNA); + $this->assertEquals('https://iovip-na0.qbox.me', $ioHost); + } +} diff --git a/vendor/qiniu/php-sdk/tests/bootstrap.php b/vendor/qiniu/php-sdk/tests/bootstrap.php new file mode 100644 index 000000000..329f99495 --- /dev/null +++ b/vendor/qiniu/php-sdk/tests/bootstrap.php @@ -0,0 +1,40 @@ + 0) { + fseek($file, $size-1); + fwrite($file, ' '); + } + fclose($file); + return $fileName; +} diff --git a/vendor/topthink/think-captcha/.gitignore b/vendor/topthink/think-captcha/.gitignore new file mode 100644 index 000000000..85d49cb86 --- /dev/null +++ b/vendor/topthink/think-captcha/.gitignore @@ -0,0 +1,3 @@ +/vendor/ +/composer.lock +.idea \ No newline at end of file diff --git a/vendor/topthink/think-captcha/LICENSE b/vendor/topthink/think-captcha/LICENSE new file mode 100644 index 000000000..835ce60c8 --- /dev/null +++ b/vendor/topthink/think-captcha/LICENSE @@ -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/vendor/topthink/think-captcha/README.md b/vendor/topthink/think-captcha/README.md new file mode 100644 index 000000000..3d338e888 --- /dev/null +++ b/vendor/topthink/think-captcha/README.md @@ -0,0 +1,33 @@ +# think-captcha +thinkphp5 楠岃瘉鐮佺被搴 + +## 瀹夎 +> composer require topthink/think-captcha + + +##浣跨敤 + +###妯℃澘閲岃緭鍑洪獙璇佺爜 + +~~~ +
    {:captcha_img()}
    +~~~ +鎴栬 +~~~ +
    captcha
    +~~~ +> 涓婇潰涓ょ鐨勬渶缁堟晥鏋滄槸涓鏍风殑 + +### 鎺у埗鍣ㄩ噷楠岃瘉 +浣跨敤TP5鐨勫唴缃獙璇佸姛鑳藉嵆鍙 +~~~ +$this->validate($data,[ + 'captcha|楠岃瘉鐮'=>'required|captcha' +]); +~~~ +鎴栬呮墜鍔ㄩ獙璇 +~~~ +if(!captcha_check($captcha)){ + //楠岃瘉澶辫触 +}; +~~~ \ No newline at end of file diff --git a/vendor/topthink/think-captcha/assets/bgs/1.jpg b/vendor/topthink/think-captcha/assets/bgs/1.jpg new file mode 100644 index 000000000..d417136bb Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/1.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/2.jpg b/vendor/topthink/think-captcha/assets/bgs/2.jpg new file mode 100644 index 000000000..56640bde4 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/2.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/3.jpg b/vendor/topthink/think-captcha/assets/bgs/3.jpg new file mode 100644 index 000000000..83e5bd902 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/3.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/4.jpg b/vendor/topthink/think-captcha/assets/bgs/4.jpg new file mode 100644 index 000000000..97a3721bc Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/4.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/5.jpg b/vendor/topthink/think-captcha/assets/bgs/5.jpg new file mode 100644 index 000000000..220a17a22 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/5.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/6.jpg b/vendor/topthink/think-captcha/assets/bgs/6.jpg new file mode 100644 index 000000000..be53ea0a2 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/6.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/7.jpg b/vendor/topthink/think-captcha/assets/bgs/7.jpg new file mode 100644 index 000000000..fbf537fab Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/7.jpg differ diff --git a/vendor/topthink/think-captcha/assets/bgs/8.jpg b/vendor/topthink/think-captcha/assets/bgs/8.jpg new file mode 100644 index 000000000..e10cf2811 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/bgs/8.jpg differ diff --git a/vendor/topthink/think-captcha/assets/ttfs/1.ttf b/vendor/topthink/think-captcha/assets/ttfs/1.ttf new file mode 100644 index 000000000..d4ee15587 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/ttfs/1.ttf differ diff --git a/vendor/topthink/think-captcha/assets/ttfs/2.ttf b/vendor/topthink/think-captcha/assets/ttfs/2.ttf new file mode 100644 index 000000000..3a452b68f Binary files /dev/null and b/vendor/topthink/think-captcha/assets/ttfs/2.ttf differ diff --git a/vendor/topthink/think-captcha/assets/ttfs/3.ttf b/vendor/topthink/think-captcha/assets/ttfs/3.ttf new file mode 100644 index 000000000..d07a4d930 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/ttfs/3.ttf differ diff --git a/vendor/topthink/think-captcha/assets/ttfs/4.ttf b/vendor/topthink/think-captcha/assets/ttfs/4.ttf new file mode 100644 index 000000000..54a14ed1c Binary files /dev/null and b/vendor/topthink/think-captcha/assets/ttfs/4.ttf differ diff --git a/vendor/topthink/think-captcha/assets/ttfs/5.ttf b/vendor/topthink/think-captcha/assets/ttfs/5.ttf new file mode 100644 index 000000000..d672876df Binary files /dev/null and b/vendor/topthink/think-captcha/assets/ttfs/5.ttf differ diff --git a/vendor/topthink/think-captcha/assets/ttfs/6.ttf b/vendor/topthink/think-captcha/assets/ttfs/6.ttf new file mode 100644 index 000000000..7f183e208 Binary files /dev/null and b/vendor/topthink/think-captcha/assets/ttfs/6.ttf differ diff --git a/vendor/topthink/think-captcha/assets/zhttfs/1.ttf b/vendor/topthink/think-captcha/assets/zhttfs/1.ttf new file mode 100644 index 000000000..1c14f7fab Binary files /dev/null and b/vendor/topthink/think-captcha/assets/zhttfs/1.ttf differ diff --git a/vendor/topthink/think-captcha/composer.json b/vendor/topthink/think-captcha/composer.json new file mode 100644 index 000000000..272e5163e --- /dev/null +++ b/vendor/topthink/think-captcha/composer.json @@ -0,0 +1,20 @@ +{ + "name": "topthink/think-captcha", + "description": "captcha package for thinkphp5", + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ], + "license": "Apache-2.0", + "require": {}, + "autoload": { + "psr-4": { + "think\\captcha\\": "src/" + }, + "files": [ + "src/helper.php" + ] + } +} diff --git a/vendor/topthink/think-captcha/src/Captcha.php b/vendor/topthink/think-captcha/src/Captcha.php new file mode 100644 index 000000000..a09a86ca6 --- /dev/null +++ b/vendor/topthink/think-captcha/src/Captcha.php @@ -0,0 +1,320 @@ + +// +---------------------------------------------------------------------- + +namespace think\captcha; + +use think\Session; + +class Captcha +{ + protected $config = [ + 'seKey' => 'ThinkPHP.CN', + // 楠岃瘉鐮佸姞瀵嗗瘑閽 + 'codeSet' => '2345678abcdefhijkmnpqrstuvwxyzABCDEFGHJKLMNPQRTUVWXY', + // 楠岃瘉鐮佸瓧绗﹂泦鍚 + 'expire' => 1800, + // 楠岃瘉鐮佽繃鏈熸椂闂达紙s锛 + 'useZh' => false, + // 浣跨敤涓枃楠岃瘉鐮 + 'zhSet' => '浠互鎴戝埌浠栦細浣滄椂瑕佸姩鍥戒骇鐨勪竴鏄伐灏卞勾闃朵箟鍙戞垚閮ㄦ皯鍙嚭鑳芥柟杩涘湪浜嗕笉鍜屾湁澶ц繖涓讳腑浜轰笂涓烘潵鍒嗙敓瀵逛簬瀛︿笅绾у湴涓敤鍚岃闈㈣绉嶈繃鍛藉害闈╄屽瀛愬悗鑷ぞ鍔犲皬鏈轰篃缁忓姏绾挎湰鐢甸珮閲忛暱鍏氬緱瀹炲瀹氭繁娉曡〃鐫姘寸悊鍖栦簤鐜版墍浜岃捣鏀夸笁濂藉崄鎴樻棤鍐滀娇鎬у墠绛夊弽浣撳悎鏂楄矾鍥炬妸缁撶閲屾鏂板紑璁轰箣鐗╀粠褰撲袱浜涜繕澶╄祫浜嬮槦鎵圭偣鑲查噸鍏舵濅笌闂村唴鍘诲洜浠舵棩鍒╃浉鐢卞帇鍛樻皵涓氫唬鍏ㄧ粍鏁版灉鏈熷骞冲悇鍩烘垨鏈堟瘺鐒跺搴斿舰鎯冲埗蹇冩牱骞查兘鍚戝彉鍏抽棶姣斿睍閭e畠鏈鍙婂娌$湅娌绘彁浜旇В绯绘灄鑰呯背缇ゅご鎰忓彧鏄庡洓閬撻┈璁ゆ鏂囬氫絾鏉¤緝鍏嬪張鍏瓟棰嗗啗娴佸叆鎺ュ腑浣嶆儏杩愬櫒骞堕鍘熸补鏀剧珛棰樿川鎸囧缓鍖洪獙娲讳紬寰堟暀鍐崇壒姝ゅ父鐭冲己鏋佸湡灏戝凡鏍瑰叡鐩村洟缁熷紡杞埆閫犲垏涔濅綘鍙栬タ鎸佹绘枡杩炰换蹇楄璋冧竷涔堝北绋嬬櫨鎶ユ洿瑙佸繀鐪熶繚鐑鎵嬫敼绠″宸卞皢淇敮璇嗙梾璞″嚑鍏堣佸厜涓撲粈鍏瀷鍏风ず澶嶅畨甯︽瘡涓滃鍒欏畬椋庡洖鍗楀箍鍔宠疆绉戝寳鎵撶Н杞﹁缁欒妭鍋氬姟琚暣鑱旀绫婚泦鍙峰垪娓╄鍗虫鐭ヨ酱鐮斿崟鑹插潥鎹熼槻鍙叉媺涓栬杈惧皵鍦虹粐鍘嗚姳鍙楁眰浼犲彛鏂喌閲囩簿閲戠晫鍝佸垽鍙傚眰姝㈣竟娓呰嚦涓囩‘绌朵功鏈姸鍘傞』绂诲啀鐩捣浜ゆ潈涓斿効闈掓墠璇佷綆瓒婇檯鍏瘯瑙勬柉杩戞敞鍔炲竷闂ㄩ搧闇璧拌鍘垮叺鍥洪櫎鑸紩榻垮崈鑳滅粏褰辨祹鐧芥牸鏁堢疆鎺ㄧ┖閰嶅垁鍙剁巼杩颁粖閫夊吇寰疯瘽鏌ュ樊鍗婃晫濮嬬墖鏂藉搷鏀跺崕瑙夊鍚嶇孩缁潎鑽爣璁伴毦瀛樻祴澹韩绱ф恫娲惧噯鏂よ闄嶇淮鏉胯鐮磋堪鎶娑堝簳搴婄敯鍔跨鎰熷線绁炰究璐烘潙鏋勭収瀹归潪鎼炰簹纾ㄦ棌鐏绠楅傝鎸夊肩編鎬侀粍鏄撳姜鏈嶆棭鐝害鍓婁俊鎺掑彴澹拌鍑荤礌寮犲瘑瀹充警鑽変綍鏍戣偉缁у彸灞炲競涓ュ緞铻烘宸﹂〉鎶楄嫃鏄捐嫤鑻卞揩绉板潖绉荤害宸存潗鐪侀粦姝﹀煿钁楁渤甯濅粎閽堟庢浜姪鍗囩帇鐪煎ス鎶撳惈鑻楀壇鏉傛櫘璋堝洿椋熷皠婧愪緥鑷撮吀鏃у嵈鍏呰冻鐭垝鍓傚鐜惤棣栧昂娉㈡壙绮夎返搴滈奔闅忚冨埢闈犲婊″か澶卞寘浣忎績鏋濆眬鑿屾潌鍛ㄦ姢宀╁笀涓炬洸鏄ュ厓瓒呰礋鐮傚皝鎹㈠お妯¤传鍑忛槼鎵睙鏋愪憨鏈ㄨ█鐞冩湞鍖绘牎鍙ゅ憿绋诲畫鍚敮杈撴粦绔欏彟鍗瓧榧撳垰鍐欏垬寰暐鑼冧緵闃垮潡鏌愬姛濂楀弸闄愰」浣欏掑嵎鍒涘緥闆ㄨ楠ㄨ繙甯垵鐨挱浼樺崰姝绘瘨鍦堜紵瀛h鎺ф縺鎵惧彨浜戜簰璺熻绮矑姣嶇粌濉為挗椤剁瓥鍙岀暀璇鍚搁樆鏁呭鐩炬櫄涓濆コ鏁g剨鍔熸牚浜查櫌鍐峰交寮归敊鏁e晢瑙嗚壓鐏増鐑堥浂瀹よ交琛鍊嶇己鍘樻车瀵熺粷瀵屽煄鍐插柗澹ょ畝鍚︽煴鏉庢湜鐩樼闆勪技鍥板珐鐩婃床鑴辨姇閫佸ゴ渚ф鼎鐩栨尌璺濊Е鏄熸澗閫佽幏鍏寸嫭瀹樻贩绾緷鏈獊鏋跺鍐珷婀垮亸绾瑰悆鎵ч榾鐭垮璐g啛绋冲ず纭环鍔炕濂囩敳棰勮亴璇勮鑳屽崗鎹熸渚电伆铏界煕鍘氱綏娉ヨ緹鍛婂嵉绠辨帉姘ф仼鐖卞仠鏇炬憾钀ョ粓绾插瓱閽卞緟灏戒縿缂╂矙閫闄堣濂嬫杞借優骞煎摢鍓ヨ揩鏃嬪緛妲藉掓彙鎷呬粛鍛椴滃惂鍗$矖浠嬮捇閫愬急鑴氭曠洂鏈槾涓伴浘鍐犱笝琛楄幈璐濊緪鑲犱粯鍚夋笚鐟炴儕椤挎尋绉掓偓濮嗙儌妫硸鍦e嚬闄惰瘝杩熻殨浜跨煩搴烽伒鐗ч伃骞呭洯鑵旇棣欒倝寮熷眿鏁忔仮蹇樼紪鍗拌渹鎬ユ嬁鎵╀激椋為湶鏍哥紭娓告尟鎿嶅ぎ浼嶅煙鐢氳繀杈夊紓搴忓厤绾稿涔′箙闅剁几澶瑰康鍏版槧娌熶箼鍚楀剴鏉姹界7鑹版櫠鎻掑焹鐕冩閾佽ˉ鍜辫娊姘哥摝鍊鹃樀纰虫紨濞侀檮鐗欒娊姘哥摝鏂滅亴娆х尞椤虹尓娲嬭厫璇烽忓徃鍗辨嫭鑴夊疁绗戣嫢灏炬潫澹毚浼佽彍绌楁姹夋剤缁挎嫋鐗涗唤鏌撴棦绉嬮亶閿荤帀澶忕枟灏栨畺浜曡垂宸炶鍚硅崳閾滄部鏇挎粴瀹㈠彫鏃辨偀鍒鸿剳鎺疮钘忔暍浠ら殭鐐夊3纭叅杩庨摳绮樻帰涓磋杽鏃杽绂忕旱鎷╃ぜ鎰夸紡娈嬮浄寤剁儫鍙ョ函娓愯曡窇娉芥參鏍介瞾璧ょ箒澧冩疆妯帀閿ュ笇姹犺触鑸瑰亣浜皳鎵樹紮鍝叉鍓叉憜璐″憟鍔茶储浠矇鐐奸夯缃鎭溅绌胯揣閿榻愰紶鎶界敾楗查緳搴撳畧绛戞埧姝屽瘨鍠滃摜娲楄殌搴熺撼鑵逛箮褰曢暅濡囨伓鑴傚簞鎿﹂櫓璧為挓鎽囧吀鏌勮京绔硅胺鍗栦贡铏氭ˉ濂ヤ集璧跺瀭閫旈澹佺綉鎴噹閬楅潤璋嬪紕鎸傝闀囧鐩涜愭彺鎵庤檻閿綊绗﹀簡鑱氱粫鎽╁繖鑸為亣绱㈤【鑳剁緤婀栭拤浠侀煶杩圭浼哥伅閬挎硾浜$瓟鍕囬鐨囨煶鍝堟彮鐢樿姒傚娴撳矝琚皝娲阿鐐祰鏂戣鎳傜伒铔嬮棴瀛╅噴涔冲法寰掔閾朵紛鏅潶绱寑闇夋潨涔愬嫆闅斿集缁╂嫑缁嶈儭鍛肩棝宄伴浂鏌寸哀鍗堣烦灞呭皻涓佺Е绋嶈拷姊佹姌鑰楃⒈娈婂矖鎸栨皬鍒冨墽鍫嗚但鑽疯兏琛″嫟鑶滅瘒鐧婚┗妗堝垔绉х紦鍑稿焦鍓窛闆摼娓斿暒鑴告埛娲涘鍕冪洘涔版潹瀹楃劍璧涙棗婊ょ鐐偂鍧愯捀鍑濈珶闄锋灙榛庢晳鍐掓殫娲炵姱绛掓偍瀹嬪姬鐖嗚艾娑傚懗娲ヨ噦闅滆闄嗗晩鍋ュ皧璞嗘嫈鑾姷妗戝潯缂濊鎸戞薄鍐版煬鍢村暐楗瀵勮档鍠婂灚涓规浮鑰冲埁铏庣瑪绋鏄嗘氮钀ㄨ尪婊存祬鎷ョ┐瑕嗕鸡濞樺惃娴歌鐝犻泴濡堢传鎴忓閿ら渿宀佽矊娲佸墫鐗㈤攱鐤戦湼闂煍鐚涜瘔鍒风嫚蹇界伨闂逛箶鍞愭紡闂绘矆鐔旀隘鑽掕寧鐢峰嚒鎶㈠儚娴嗘梺鐜讳害蹇犲敱钂欎簣绾锋崟閿佸挨涔樹箤鏅烘贰鍏佸彌鐣滀繕鎽搁攬鎵瘯鐠冨疂鑺埛閴寸鍑钂嬮挋鑲╄吘鏋姏杞ㄥ爞鎷岀埜寰绁濆姳鑲厭缁崇┓濉樼嚗娉¤鏈楀杺閾濊蒋娓犻鎯锤绮患澧欒秼褰煎眾澧ㄧ鍚嗗嵏鑸。瀛欓緞宀獥浼戝', + // 涓枃楠岃瘉鐮佸瓧绗︿覆 + 'useImgBg' => false, + // 浣跨敤鑳屾櫙鍥剧墖 + 'fontSize' => 25, + // 楠岃瘉鐮佸瓧浣撳ぇ灏(px) + 'useCurve' => true, + // 鏄惁鐢绘贩娣嗘洸绾 + 'useNoise' => true, + // 鏄惁娣诲姞鏉傜偣 + 'imageH' => 0, + // 楠岃瘉鐮佸浘鐗囬珮搴 + 'imageW' => 0, + // 楠岃瘉鐮佸浘鐗囧搴 + 'length' => 5, + // 楠岃瘉鐮佷綅鏁 + 'fontttf' => '', + // 楠岃瘉鐮佸瓧浣擄紝涓嶈缃殢鏈鸿幏鍙 + 'bg' => [243, 251, 254], + // 鑳屾櫙棰滆壊 + 'reset' => true, + // 楠岃瘉鎴愬姛鍚庢槸鍚﹂噸缃 + ]; + + private $_image = null; // 楠岃瘉鐮佸浘鐗囧疄渚 + private $_color = null; // 楠岃瘉鐮佸瓧浣撻鑹 + + /** + * 鏋舵瀯鏂规硶 璁剧疆鍙傛暟 + * @access public + * @param array $config 閰嶇疆鍙傛暟 + */ + public function __construct($config = []) + { + $this->config = array_merge($this->config, $config); + } + + /** + * 浣跨敤 $this->name 鑾峰彇閰嶇疆 + * @access public + * @param string $name 閰嶇疆鍚嶇О + * @return mixed 閰嶇疆鍊 + */ + public function __get($name) + { + return $this->config[$name]; + } + + /** + * 璁剧疆楠岃瘉鐮侀厤缃 + * @access public + * @param string $name 閰嶇疆鍚嶇О + * @param string $value 閰嶇疆鍊 + * @return void + */ + public function __set($name, $value) + { + if (isset($this->config[$name])) { + $this->config[$name] = $value; + } + } + + /** + * 妫鏌ラ厤缃 + * @access public + * @param string $name 閰嶇疆鍚嶇О + * @return bool + */ + public function __isset($name) + { + return isset($this->config[$name]); + } + + /** + * 楠岃瘉楠岃瘉鐮佹槸鍚︽纭 + * @access public + * @param string $code 鐢ㄦ埛楠岃瘉鐮 + * @param string $id 楠岃瘉鐮佹爣璇 + * @return bool 鐢ㄦ埛楠岃瘉鐮佹槸鍚︽纭 + */ + public function check($code, $id = '') + { + $key = $this->authcode($this->seKey) . $id; + // 楠岃瘉鐮佷笉鑳戒负绌 + $secode = Session::get($key, ''); + if (empty($code) || empty($secode)) { + return false; + } + // session 杩囨湡 + if (time() - $secode['verify_time'] > $this->expire) { + Session::delete($key, ''); + return false; + } + + if ($this->authcode(strtoupper($code)) == $secode['verify_code']) { + $this->reset && Session::delete($key, ''); + return true; + } + + return false; + } + + /** + * 杈撳嚭楠岃瘉鐮佸苟鎶婇獙璇佺爜鐨勫间繚瀛樼殑session涓 + * 楠岃瘉鐮佷繚瀛樺埌session鐨勬牸寮忎负锛 array('verify_code' => '楠岃瘉鐮佸', 'verify_time' => '楠岃瘉鐮佸垱寤烘椂闂'); + * @access public + * @param string $id 瑕佺敓鎴愰獙璇佺爜鐨勬爣璇 + * @return \think\Response + */ + public function entry($id = '') + { + // 鍥剧墖瀹(px) + $this->imageW || $this->imageW = $this->length * $this->fontSize * 1.5 + $this->length * $this->fontSize / 2; + // 鍥剧墖楂(px) + $this->imageH || $this->imageH = $this->fontSize * 2.5; + // 寤虹珛涓骞 $this->imageW x $this->imageH 鐨勫浘鍍 + $this->_image = imagecreate($this->imageW, $this->imageH); + // 璁剧疆鑳屾櫙 + imagecolorallocate($this->_image, $this->bg[0], $this->bg[1], $this->bg[2]); + + // 楠岃瘉鐮佸瓧浣撻殢鏈洪鑹 + $this->_color = imagecolorallocate($this->_image, mt_rand(1, 150), mt_rand(1, 150), mt_rand(1, 150)); + // 楠岃瘉鐮佷娇鐢ㄩ殢鏈哄瓧浣 + $ttfPath = __DIR__ . '/../assets/' . ($this->useZh ? 'zhttfs' : 'ttfs') . '/'; + + if (empty($this->fontttf)) { + $dir = dir($ttfPath); + $ttfs = []; + while (false !== ($file = $dir->read())) { + if ('.' != $file[0] && substr($file, -4) == '.ttf') { + $ttfs[] = $file; + } + } + $dir->close(); + $this->fontttf = $ttfs[array_rand($ttfs)]; + } + $this->fontttf = $ttfPath . $this->fontttf; + + if ($this->useImgBg) { + $this->_background(); + } + + if ($this->useNoise) { + // 缁樻潅鐐 + $this->_writeNoise(); + } + if ($this->useCurve) { + // 缁樺共鎵扮嚎 + $this->_writeCurve(); + } + + // 缁橀獙璇佺爜 + $code = []; // 楠岃瘉鐮 + $codeNX = 0; // 楠岃瘉鐮佺N涓瓧绗︾殑宸﹁竟璺 + if ($this->useZh) { + // 涓枃楠岃瘉鐮 + for ($i = 0; $i < $this->length; $i++) { + $code[$i] = iconv_substr($this->zhSet, floor(mt_rand(0, mb_strlen($this->zhSet, 'utf-8') - 1)), 1, 'utf-8'); + imagettftext($this->_image, $this->fontSize, mt_rand(-40, 40), $this->fontSize * ($i + 1) * 1.5, $this->fontSize + mt_rand(10, 20), $this->_color, $this->fontttf, $code[$i]); + } + } else { + for ($i = 0; $i < $this->length; $i++) { + $code[$i] = $this->codeSet[mt_rand(0, strlen($this->codeSet) - 1)]; + $codeNX += mt_rand($this->fontSize * 1.2, $this->fontSize * 1.6); + imagettftext($this->_image, $this->fontSize, mt_rand(-40, 40), $codeNX, $this->fontSize * 1.6, $this->_color, $this->fontttf, $code[$i]); + } + } + + // 淇濆瓨楠岃瘉鐮 + $key = $this->authcode($this->seKey); + $code = $this->authcode(strtoupper(implode('', $code))); + $secode = []; + $secode['verify_code'] = $code; // 鎶婃牎楠岀爜淇濆瓨鍒皊ession + $secode['verify_time'] = time(); // 楠岃瘉鐮佸垱寤烘椂闂 + Session::set($key . $id, $secode, ''); + + ob_start(); + // 杈撳嚭鍥惧儚 + imagepng($this->_image); + $content = ob_get_clean(); + imagedestroy($this->_image); + + return response($content, 200, ['Content-Length' => strlen($content)])->contentType('image/png'); + } + + /** + * 鐢讳竴鏉$敱涓ゆ潯杩炲湪涓璧锋瀯鎴愮殑闅忔満姝e鸡鍑芥暟鏇茬嚎浣滃共鎵扮嚎(浣犲彲浠ユ敼鎴愭洿甯呯殑鏇茬嚎鍑芥暟) + * + * 楂樹腑鐨勬暟瀛﹀叕寮忓拫閮藉繕浜嗘秴锛屽啓鍑烘潵 + * 姝e鸡鍨嬪嚱鏁拌В鏋愬紡锛歽=Asin(蠅x+蠁)+b + * 鍚勫父鏁板煎鍑芥暟鍥惧儚鐨勫奖鍝嶏細 + * A锛氬喅瀹氬嘲鍊硷紙鍗崇旱鍚戞媺浼稿帇缂╃殑鍊嶆暟锛 + * b锛氳〃绀烘尝褰㈠湪Y杞寸殑浣嶇疆鍏崇郴鎴栫旱鍚戠Щ鍔ㄨ窛绂伙紙涓婂姞涓嬪噺锛 + * 蠁锛氬喅瀹氭尝褰笌X杞翠綅缃叧绯绘垨妯悜绉诲姩璺濈锛堝乏鍔犲彸鍑忥級 + * 蠅锛氬喅瀹氬懆鏈燂紙鏈灏忔鍛ㄦ湡T=2蟺/鈭O夆垼锛 + * + */ + private function _writeCurve() + { + $px = $py = 0; + + // 鏇茬嚎鍓嶉儴鍒 + $A = mt_rand(1, $this->imageH / 2); // 鎸箙 + $b = mt_rand(-$this->imageH / 4, $this->imageH / 4); // Y杞存柟鍚戝亸绉婚噺 + $f = mt_rand(-$this->imageH / 4, $this->imageH / 4); // X杞存柟鍚戝亸绉婚噺 + $T = mt_rand($this->imageH, $this->imageW * 2); // 鍛ㄦ湡 + $w = (2 * M_PI) / $T; + + $px1 = 0; // 鏇茬嚎妯潗鏍囪捣濮嬩綅缃 + $px2 = mt_rand($this->imageW / 2, $this->imageW * 0.8); // 鏇茬嚎妯潗鏍囩粨鏉熶綅缃 + + for ($px = $px1; $px <= $px2; $px = $px + 1) { + if (0 != $w) { + $py = $A * sin($w * $px + $f) + $b + $this->imageH / 2; // y = Asin(蠅x+蠁) + b + $i = (int)($this->fontSize / 5); + while ($i > 0) { + imagesetpixel($this->_image, $px + $i, $py + $i, $this->_color); // 杩欓噷(while)寰幆鐢诲儚绱犵偣姣攊magettftext鍜宨magestring鐢ㄥ瓧浣撳ぇ灏忎竴娆$敾鍑猴紙涓嶇敤杩檞hile寰幆锛夋ц兘瑕佸ソ寰堝 + $i--; + } + } + } + + // 鏇茬嚎鍚庨儴鍒 + $A = mt_rand(1, $this->imageH / 2); // 鎸箙 + $f = mt_rand(-$this->imageH / 4, $this->imageH / 4); // X杞存柟鍚戝亸绉婚噺 + $T = mt_rand($this->imageH, $this->imageW * 2); // 鍛ㄦ湡 + $w = (2 * M_PI) / $T; + $b = $py - $A * sin($w * $px + $f) - $this->imageH / 2; + $px1 = $px2; + $px2 = $this->imageW; + + for ($px = $px1; $px <= $px2; $px = $px + 1) { + if (0 != $w) { + $py = $A * sin($w * $px + $f) + $b + $this->imageH / 2; // y = Asin(蠅x+蠁) + b + $i = (int)($this->fontSize / 5); + while ($i > 0) { + imagesetpixel($this->_image, $px + $i, $py + $i, $this->_color); + $i--; + } + } + } + } + + /** + * 鐢绘潅鐐 + * 寰鍥剧墖涓婂啓涓嶅悓棰滆壊鐨勫瓧姣嶆垨鏁板瓧 + */ + private function _writeNoise() + { + $codeSet = '2345678abcdefhijkmnpqrstuvwxyz'; + for ($i = 0; $i < 10; $i++) { + //鏉傜偣棰滆壊 + $noiseColor = imagecolorallocate($this->_image, mt_rand(150, 225), mt_rand(150, 225), mt_rand(150, 225)); + for ($j = 0; $j < 5; $j++) { + // 缁樻潅鐐 + imagestring($this->_image, 5, mt_rand(-10, $this->imageW), mt_rand(-10, $this->imageH), $codeSet[mt_rand(0, 29)], $noiseColor); + } + } + } + + /** + * 缁樺埗鑳屾櫙鍥剧墖 + * 娉細濡傛灉楠岃瘉鐮佽緭鍑哄浘鐗囨瘮杈冨ぇ锛屽皢鍗犵敤姣旇緝澶氱殑绯荤粺璧勬簮 + */ + private function _background() + { + $path = dirname(__FILE__) . '/verify/bgs/'; + $dir = dir($path); + + $bgs = []; + while (false !== ($file = $dir->read())) { + if ('.' != $file[0] && substr($file, -4) == '.jpg') { + $bgs[] = $path . $file; + } + } + $dir->close(); + + $gb = $bgs[array_rand($bgs)]; + + list($width, $height) = @getimagesize($gb); + // Resample + $bgImage = @imagecreatefromjpeg($gb); + @imagecopyresampled($this->_image, $bgImage, 0, 0, 0, 0, $this->imageW, $this->imageH, $width, $height); + @imagedestroy($bgImage); + } + + /* 鍔犲瘑楠岃瘉鐮 */ + private function authcode($str) + { + $key = substr(md5($this->seKey), 5, 8); + $str = substr(md5($str), 8, 10); + return md5($key . $str); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-captcha/src/CaptchaController.php b/vendor/topthink/think-captcha/src/CaptchaController.php new file mode 100644 index 000000000..8a0535d82 --- /dev/null +++ b/vendor/topthink/think-captcha/src/CaptchaController.php @@ -0,0 +1,23 @@ + +// +---------------------------------------------------------------------- + +namespace think\captcha; + +use think\Config; + +class CaptchaController +{ + public function index($id = "") + { + $captcha = new Captcha((array)Config::get('captcha')); + return $captcha->entry($id); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-captcha/src/helper.php b/vendor/topthink/think-captcha/src/helper.php new file mode 100644 index 000000000..ef3970f91 --- /dev/null +++ b/vendor/topthink/think-captcha/src/helper.php @@ -0,0 +1,64 @@ + +// +---------------------------------------------------------------------- + +\think\Route::get('captcha/[:id]', "\\think\\captcha\\CaptchaController@index"); + +\think\Validate::extend('captcha', function ($value, $id = "") { + return captcha_check($value, $id, (array)\think\Config::get('captcha')); +}); + +\think\Validate::setTypeMsg('captcha', '楠岃瘉鐮侀敊璇!'); + + +/** + * @param string $id + * @param array $config + * @return \think\Response + */ +function captcha($id = "", $config = []) +{ + $captcha = new \think\captcha\Captcha($config); + return $captcha->entry($id); +} + + +/** + * @param $id + * @return string + */ +function captcha_src($id = "") +{ + return \think\Url::build('/captcha' . ($id ? "/{$id}" : '')); +} + + +/** + * @param $id + * @return mixed + */ +function captcha_img($id = "") +{ + return 'captcha'; +} + + +/** + * @param $value + * @param string $id + * @param array $config + * @return bool + */ +function captcha_check($value, $id = "", $config = []) +{ + $captcha = new \think\captcha\Captcha($config); + return $captcha->check($value, $id); +} + diff --git a/vendor/topthink/think-helper/.gitignore b/vendor/topthink/think-helper/.gitignore new file mode 100644 index 000000000..e244eda0b --- /dev/null +++ b/vendor/topthink/think-helper/.gitignore @@ -0,0 +1,2 @@ +/vendor/ +/.idea/ \ No newline at end of file diff --git a/vendor/topthink/think-helper/LICENSE b/vendor/topthink/think-helper/LICENSE new file mode 100644 index 000000000..8dada3eda --- /dev/null +++ b/vendor/topthink/think-helper/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/topthink/think-helper/README.md b/vendor/topthink/think-helper/README.md new file mode 100644 index 000000000..27ef67092 --- /dev/null +++ b/vendor/topthink/think-helper/README.md @@ -0,0 +1,92 @@ +# thinkphp5 甯哥敤鐨勪竴浜涙墿灞曠被搴 + +> 鏇存柊瀹屽杽涓 + +> 浠ヤ笅绫诲簱閮藉湪`\\think\\helper`鍛藉悕绌洪棿涓 + +## Str +> 瀛楃涓叉搷浣 + +``` +// 妫鏌ュ瓧绗︿覆涓槸鍚﹀寘鍚煇浜涘瓧绗︿覆 +Str::contains($haystack, $needles) + +// 妫鏌ュ瓧绗︿覆鏄惁浠ユ煇浜涘瓧绗︿覆缁撳熬 +Str::endsWith($haystack, $needles) + +// 鑾峰彇鎸囧畾闀垮害鐨勯殢鏈哄瓧姣嶆暟瀛楃粍鍚堢殑瀛楃涓 +Str::random($length = 16) + +// 瀛楃涓茶浆灏忓啓 +Str::lower($value) + +// 瀛楃涓茶浆澶у啓 +Str::upper($value) + +// 鑾峰彇瀛楃涓茬殑闀垮害 +Str::length($value) + +// 鎴彇瀛楃涓 +Str::substr($string, $start, $length = null) + +``` + +## Hash +> 鍒涘缓瀵嗙爜鐨勫搱甯 + +``` +// 鍒涘缓 +Hash::make($value, $type = null, array $options = []) + +// 妫鏌 +Hash::check($value, $hashedValue, $type = null, array $options = []) + +``` + +## Time +> 鏃堕棿鎴虫搷浣 + +``` +// 浠婃棩寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::today(); + +// 鏄ㄦ棩寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::yesterday(); + +// 鏈懆寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::week(); + +// 涓婂懆寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::lastWeek(); + +// 鏈湀寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::month(); + +// 涓婃湀寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::lastMonth(); + +// 浠婂勾寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::year(); + +// 鍘诲勾寮濮嬪拰缁撴潫鐨勬椂闂存埑 +Time::lastYear(); + +// 鑾峰彇7澶╁墠闆剁偣鍒扮幇鍦ㄧ殑鏃堕棿鎴 +Time::dayToNow(7) + +// 鑾峰彇7澶╁墠闆剁偣鍒版槰鏃ョ粨鏉熺殑鏃堕棿鎴 +Time::dayToNow(7, true) + +// 鑾峰彇7澶╁墠鐨勬椂闂存埑 +Time::daysAgo(7) + +// 鑾峰彇7澶╁悗鐨勬椂闂存埑 +Time::daysAfter(7) + +// 澶╂暟杞崲鎴愮鏁 +Time::daysToSecond(5) + +// 鍛ㄦ暟杞崲鎴愮鏁 +Time::weekToSecond(5) + +``` \ No newline at end of file diff --git a/vendor/topthink/think-helper/composer.json b/vendor/topthink/think-helper/composer.json new file mode 100644 index 000000000..d246fa3c9 --- /dev/null +++ b/vendor/topthink/think-helper/composer.json @@ -0,0 +1,19 @@ +{ + "name": "topthink/think-helper", + "description": "The ThinkPHP5 Helper Package", + "license": "Apache-2.0", + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ], + "autoload": { + "psr-4": { + "think\\helper\\": "src" + }, + "files": [ + "src/helper.php" + ] + } +} diff --git a/vendor/topthink/think-helper/src/Arr.php b/vendor/topthink/think-helper/src/Arr.php new file mode 100644 index 000000000..7350bb245 --- /dev/null +++ b/vendor/topthink/think-helper/src/Arr.php @@ -0,0 +1,41 @@ + +// +---------------------------------------------------------------------- + +namespace think\helper; + + +class Arr +{ + + public static function isAssoc(array $array) + { + $keys = array_keys($array); + + return array_keys($keys) !== $keys; + } + + public static function sortRecursive($array) + { + foreach ($array as &$value) { + if (is_array($value)) { + $value = static::sortRecursive($value); + } + } + + if (static::isAssoc($array)) { + ksort($array); + } else { + sort($array); + } + + return $array; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-helper/src/Hash.php b/vendor/topthink/think-helper/src/Hash.php new file mode 100644 index 000000000..b56543ea0 --- /dev/null +++ b/vendor/topthink/think-helper/src/Hash.php @@ -0,0 +1,48 @@ + +// +---------------------------------------------------------------------- + +namespace think\helper; + + +class Hash +{ + protected static $handle = []; + + public static function make($value, $type = null, array $options = []) + { + return self::handle($type)->make($value, $options); + } + + public static function check($value, $hashedValue, $type = null, array $options = []) + { + return self::handle($type)->check($value, $hashedValue, $options); + } + + public static function handle($type) + { + if (is_null($type)) { + if (PHP_VERSION_ID >= 50500) { + $type = 'bcrypt'; + } else { + $type = 'md5'; + } + } + if (empty(self::$handle[$type])) { + $class = "\\think\\helper\\hash\\" . ucfirst($type); + if (!class_exists($class)) { + throw new \ErrorException("Not found {$type} hash type!"); + } + self::$handle[$type] = new $class(); + } + return self::$handle[$type]; + } + +} \ No newline at end of file diff --git a/vendor/topthink/think-helper/src/Str.php b/vendor/topthink/think-helper/src/Str.php new file mode 100644 index 000000000..ba56cb44a --- /dev/null +++ b/vendor/topthink/think-helper/src/Str.php @@ -0,0 +1,202 @@ + +// +---------------------------------------------------------------------- +namespace think\helper; + +class Str +{ + + protected static $snakeCache = []; + + protected static $camelCache = []; + + protected static $studlyCache = []; + + /** + * 妫鏌ュ瓧绗︿覆涓槸鍚﹀寘鍚煇浜涘瓧绗︿覆 + * @param string $haystack + * @param string|array $needles + * @return bool + */ + public static function contains($haystack, $needles) + { + foreach ((array) $needles as $needle) { + if ($needle != '' && mb_strpos($haystack, $needle) !== false) { + return true; + } + } + + return false; + } + + /** + * 妫鏌ュ瓧绗︿覆鏄惁浠ユ煇浜涘瓧绗︿覆缁撳熬 + * + * @param string $haystack + * @param string|array $needles + * @return bool + */ + public static function endsWith($haystack, $needles) + { + foreach ((array) $needles as $needle) { + if ((string) $needle === static::substr($haystack, -static::length($needle))) { + return true; + } + } + + return false; + } + + /** + * 妫鏌ュ瓧绗︿覆鏄惁浠ユ煇浜涘瓧绗︿覆寮澶 + * + * @param string $haystack + * @param string|array $needles + * @return bool + */ + public static function startsWith($haystack, $needles) + { + foreach ((array) $needles as $needle) { + if ($needle != '' && mb_strpos($haystack, $needle) === 0) { + return true; + } + } + + return false; + } + + /** + * 鑾峰彇鎸囧畾闀垮害鐨勯殢鏈哄瓧姣嶆暟瀛楃粍鍚堢殑瀛楃涓 + * + * @param int $length + * @return string + */ + public static function random($length = 16) + { + $pool = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + + return static::substr(str_shuffle(str_repeat($pool, $length)), 0, $length); + } + + /** + * 瀛楃涓茶浆灏忓啓 + * + * @param string $value + * @return string + */ + public static function lower($value) + { + return mb_strtolower($value, 'UTF-8'); + } + + /** + * 瀛楃涓茶浆澶у啓 + * + * @param string $value + * @return string + */ + public static function upper($value) + { + return mb_strtoupper($value, 'UTF-8'); + } + + /** + * 鑾峰彇瀛楃涓茬殑闀垮害 + * + * @param string $value + * @return int + */ + public static function length($value) + { + return mb_strlen($value); + } + + /** + * 鎴彇瀛楃涓 + * + * @param string $string + * @param int $start + * @param int|null $length + * @return string + */ + public static function substr($string, $start, $length = null) + { + return mb_substr($string, $start, $length, 'UTF-8'); + } + + /** + * 椹煎嘲杞笅鍒掔嚎 + * + * @param string $value + * @param string $delimiter + * @return string + */ + public static function snake($value, $delimiter = '_') + { + $key = $value; + + if (isset(static::$snakeCache[$key][$delimiter])) { + return static::$snakeCache[$key][$delimiter]; + } + + if (!ctype_lower($value)) { + $value = preg_replace('/\s+/u', '', $value); + + $value = static::lower(preg_replace('/(.)(?=[A-Z])/u', '$1' . $delimiter, $value)); + } + + return static::$snakeCache[$key][$delimiter] = $value; + } + + /** + * 涓嬪垝绾胯浆椹煎嘲(棣栧瓧姣嶅皬鍐) + * + * @param string $value + * @return string + */ + public static function camel($value) + { + if (isset(static::$camelCache[$value])) { + return static::$camelCache[$value]; + } + + return static::$camelCache[$value] = lcfirst(static::studly($value)); + } + + /** + * 涓嬪垝绾胯浆椹煎嘲(棣栧瓧姣嶅ぇ鍐) + * + * @param string $value + * @return string + */ + public static function studly($value) + { + $key = $value; + + if (isset(static::$studlyCache[$key])) { + return static::$studlyCache[$key]; + } + + $value = ucwords(str_replace(['-', '_'], ' ', $value)); + + return static::$studlyCache[$key] = str_replace(' ', '', $value); + } + + /** + * 杞负棣栧瓧姣嶅ぇ鍐欑殑鏍囬鏍煎紡 + * + * @param string $value + * @return string + */ + public static function title($value) + { + return mb_convert_case($value, MB_CASE_TITLE, 'UTF-8'); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-helper/src/Time.php b/vendor/topthink/think-helper/src/Time.php new file mode 100644 index 000000000..4284ed46c --- /dev/null +++ b/vendor/topthink/think-helper/src/Time.php @@ -0,0 +1,198 @@ + +// +---------------------------------------------------------------------- +namespace think\helper; + +class Time +{ + /** + * 杩斿洖浠婃棩寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function today() + { + return [ + mktime(0, 0, 0, date('m'), date('d'), date('Y')), + mktime(23, 59, 59, date('m'), date('d'), date('Y')) + ]; + } + + /** + * 杩斿洖鏄ㄦ棩寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function yesterday() + { + $yesterday = date('d') - 1; + return [ + mktime(0, 0, 0, date('m'), $yesterday, date('Y')), + mktime(23, 59, 59, date('m'), $yesterday, date('Y')) + ]; + } + + /** + * 杩斿洖鏈懆寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function week() + { + $timestamp = time(); + return [ + strtotime(date('Y-m-d', strtotime("+0 week Monday", $timestamp))), + strtotime(date('Y-m-d', strtotime("+0 week Sunday", $timestamp))) + 24 * 3600 - 1 + ]; + } + + /** + * 杩斿洖涓婂懆寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function lastWeek() + { + $timestamp = time(); + return [ + strtotime(date('Y-m-d', strtotime("last week Monday", $timestamp))), + strtotime(date('Y-m-d', strtotime("last week Sunday", $timestamp))) + 24 * 3600 - 1 + ]; + } + + /** + * 杩斿洖鏈湀寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function month($everyDay = false) + { + return [ + mktime(0, 0, 0, date('m'), 1, date('Y')), + mktime(23, 59, 59, date('m'), date('t'), date('Y')) + ]; + } + + /** + * 杩斿洖涓婁釜鏈堝紑濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function lastMonth() + { + $begin = mktime(0, 0, 0, date('m') - 1, 1, date('Y')); + $end = mktime(23, 59, 59, date('m') - 1, date('t', $begin), date('Y')); + + return [$begin, $end]; + } + + /** + * 杩斿洖浠婂勾寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function year() + { + return [ + mktime(0, 0, 0, 1, 1, date('Y')), + mktime(23, 59, 59, 12, 31, date('Y')) + ]; + } + + /** + * 杩斿洖鍘诲勾寮濮嬪拰缁撴潫鐨勬椂闂存埑 + * + * @return array + */ + public static function lastYear() + { + $year = date('Y') - 1; + return [ + mktime(0, 0, 0, 1, 1, $year), + mktime(23, 59, 59, 12, 31, $year) + ]; + } + + public static function dayOf() + { + + } + + /** + * 鑾峰彇鍑犲ぉ鍓嶉浂鐐瑰埌鐜板湪/鏄ㄦ棩缁撴潫鐨勬椂闂存埑 + * + * @param int $day 澶╂暟 + * @param bool $now 杩斿洖鐜板湪鎴栬呮槰澶╃粨鏉熸椂闂存埑 + * @return array + */ + public static function dayToNow($day = 1, $now = true) + { + $end = time(); + if (!$now) { + list($foo, $end) = self::yesterday(); + } + + return [ + mktime(0, 0, 0, date('m'), date('d') - $day, date('Y')), + $end + ]; + } + + /** + * 杩斿洖鍑犲ぉ鍓嶇殑鏃堕棿鎴 + * + * @param int $day + * @return int + */ + public static function daysAgo($day = 1) + { + $nowTime = time(); + return $nowTime - self::daysToSecond($day); + } + + /** + * 杩斿洖鍑犲ぉ鍚庣殑鏃堕棿鎴 + * + * @param int $day + * @return int + */ + public static function daysAfter($day = 1) + { + $nowTime = time(); + return $nowTime + self::daysToSecond($day); + } + + /** + * 澶╂暟杞崲鎴愮鏁 + * + * @param int $day + * @return int + */ + public static function daysToSecond($day = 1) + { + return $day * 86400; + } + + /** + * 鍛ㄦ暟杞崲鎴愮鏁 + * + * @param int $week + * @return int + */ + public static function weekToSecond($week = 1) + { + return self::daysToSecond() * 7 * $week; + } + + private static function startTimeToEndTime() + { + + } +} \ No newline at end of file diff --git a/vendor/topthink/think-helper/src/hash/Bcrypt.php b/vendor/topthink/think-helper/src/hash/Bcrypt.php new file mode 100644 index 000000000..8f55a7714 --- /dev/null +++ b/vendor/topthink/think-helper/src/hash/Bcrypt.php @@ -0,0 +1,51 @@ + +// +---------------------------------------------------------------------- +namespace think\helper\hash; + +class Bcrypt +{ + + /** + * Default crypt cost factor. + * + * @var int + */ + protected $rounds = 10; + + public function make($value, array $options = []) + { + $cost = isset($options['rounds']) ? $options['rounds'] : $this->rounds; + + $hash = password_hash($value, PASSWORD_BCRYPT, ['cost' => $cost]); + + if ($hash === false) { + throw new \RuntimeException('Bcrypt hashing not supported.'); + } + + return $hash; + } + + public function check($value, $hashedValue, array $options = []) + { + if (strlen($hashedValue) === 0) { + return false; + } + + return password_verify($value, $hashedValue); + } + + public function setRounds($rounds) + { + $this->rounds = (int)$rounds; + + return $this; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-helper/src/hash/Md5.php b/vendor/topthink/think-helper/src/hash/Md5.php new file mode 100644 index 000000000..5ec07ca0e --- /dev/null +++ b/vendor/topthink/think-helper/src/hash/Md5.php @@ -0,0 +1,42 @@ + +// +---------------------------------------------------------------------- +namespace think\helper\hash; + +class Md5 +{ + + protected $salt = 'think'; + + public function make($value, array $options = []) + { + $salt = isset($options['salt']) ? $options['salt'] : $this->salt; + + return md5(md5($value) . $salt); + } + + public function check($value, $hashedValue, array $options = []) + { + if (strlen($hashedValue) === 0) { + return false; + } + + $salt = isset($options['salt']) ? $options['salt'] : $this->salt; + + return md5(md5($value) . $salt) == $hashedValue; + } + + public function setSalt($salt) + { + $this->salt = (string)$salt; + + return $this; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-helper/src/helper.php b/vendor/topthink/think-helper/src/helper.php new file mode 100644 index 000000000..5d2b63adc --- /dev/null +++ b/vendor/topthink/think-helper/src/helper.php @@ -0,0 +1,67 @@ + +// +---------------------------------------------------------------------- + +if (!function_exists('class_basename')) { + /** + * 鑾峰彇绫诲悕(涓嶅寘鍚懡鍚嶇┖闂) + * + * @param string|object $class + * @return string + */ + function class_basename($class) + { + $class = is_object($class) ? get_class($class) : $class; + + return basename(str_replace('\\', '/', $class)); + } +} + +if (!function_exists('class_uses_recursive')) { + /** + *鑾峰彇涓涓被閲屾墍鏈夌敤鍒扮殑trait锛屽寘鎷埗绫荤殑 + * + * @param $class + * @return array + */ + function class_uses_recursive($class) + { + if (is_object($class)) { + $class = get_class($class); + } + + $results = []; + + foreach (array_merge([$class => $class], class_parents($class)) as $class) { + $results += trait_uses_recursive($class); + } + + return array_unique($results); + } +} + +if (!function_exists('trait_uses_recursive')) { + /** + * 鑾峰彇涓涓猼rait閲屾墍鏈夊紩鐢ㄥ埌鐨則rait + * + * @param string $trait + * @return array + */ + function trait_uses_recursive($trait) + { + $traits = class_uses($trait); + + foreach ($traits as $trait) { + $traits += trait_uses_recursive($trait); + } + + return $traits; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-installer/.gitignore b/vendor/topthink/think-installer/.gitignore new file mode 100644 index 000000000..8f4c02d44 --- /dev/null +++ b/vendor/topthink/think-installer/.gitignore @@ -0,0 +1,3 @@ +/.idea +composer.lock +/vendor \ No newline at end of file diff --git a/vendor/topthink/think-installer/composer.json b/vendor/topthink/think-installer/composer.json new file mode 100644 index 000000000..4005de224 --- /dev/null +++ b/vendor/topthink/think-installer/composer.json @@ -0,0 +1,25 @@ +{ + "name": "topthink/think-installer", + "type": "composer-plugin", + "require": { + "composer-plugin-api": "^1.0" + }, + "require-dev": { + "composer/composer": "1.0.*@dev" + }, + "license": "Apache-2.0", + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ], + "autoload": { + "psr-4": { + "think\\composer\\": "src" + } + }, + "extra": { + "class": "think\\composer\\Plugin" + } +} diff --git a/vendor/topthink/think-installer/src/Plugin.php b/vendor/topthink/think-installer/src/Plugin.php new file mode 100644 index 000000000..757c30fff --- /dev/null +++ b/vendor/topthink/think-installer/src/Plugin.php @@ -0,0 +1,26 @@ +getInstallationManager(); + + //妗嗘灦鏍稿績 + $manager->addInstaller(new ThinkFramework($io, $composer)); + + //鍗曞厓娴嬭瘯 + $manager->addInstaller(new ThinkTesting($io, $composer)); + + //鎵╁睍 + $manager->addInstaller(new ThinkExtend($io, $composer)); + + } +} \ No newline at end of file diff --git a/vendor/topthink/think-installer/src/ThinkExtend.php b/vendor/topthink/think-installer/src/ThinkExtend.php new file mode 100644 index 000000000..d78f11845 --- /dev/null +++ b/vendor/topthink/think-installer/src/ThinkExtend.php @@ -0,0 +1,77 @@ + +// +---------------------------------------------------------------------- + +namespace think\composer; + +use Composer\Installer\LibraryInstaller; +use Composer\Package\PackageInterface; +use Composer\Repository\InstalledRepositoryInterface; + +class ThinkExtend extends LibraryInstaller +{ + + public function install(InstalledRepositoryInterface $repo, PackageInterface $package) + { + parent::install($repo, $package); + $this->copyExtraFiles($package); + } + + public function update(InstalledRepositoryInterface $repo, PackageInterface $initial, PackageInterface $target) + { + parent::update($repo, $initial, $target); + $this->copyExtraFiles($target); + + } + + protected function copyExtraFiles(PackageInterface $package) + { + if ($this->composer->getPackage()->getType() == 'project') { + + $extra = $package->getExtra(); + + if (!empty($extra['think-config'])) { + + $composerExtra = $this->composer->getPackage()->getExtra(); + + $appDir = !empty($composerExtra['app-path']) ? $composerExtra['app-path'] : 'application'; + + if (is_dir($appDir)) { + + $extraDir = $appDir . DIRECTORY_SEPARATOR . 'extra'; + $this->filesystem->ensureDirectoryExists($extraDir); + + //閰嶇疆鏂囦欢 + foreach ((array) $extra['think-config'] as $name => $config) { + $target = $extraDir . DIRECTORY_SEPARATOR . $name . '.php'; + $source = $this->getInstallPath($package) . DIRECTORY_SEPARATOR . $config; + + if (is_file($target)) { + $this->io->write("File {$target} exist!"); + continue; + } + + if (!is_file($source)) { + $this->io->write("File {$target} not exist!"); + continue; + } + + copy($source, $target); + } + } + } + } + } + + public function supports($packageType) + { + return 'think-extend' === $packageType; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-installer/src/ThinkFramework.php b/vendor/topthink/think-installer/src/ThinkFramework.php new file mode 100644 index 000000000..7c3deaa79 --- /dev/null +++ b/vendor/topthink/think-installer/src/ThinkFramework.php @@ -0,0 +1,59 @@ +getPrettyName()) { + throw new \InvalidArgumentException('Unable to install this library!'); + } + + if ($this->composer->getPackage()->getType() !== 'project') { + return parent::getInstallPath($package); + } + + if ($this->composer->getPackage()) { + $extra = $this->composer->getPackage()->getExtra(); + if (!empty($extra['think-path'])) { + return $extra['think-path']; + } + } + + return 'thinkphp'; + } + + public function install(InstalledRepositoryInterface $repo, PackageInterface $package) + { + parent::install($repo, $package); + if ($this->composer->getPackage()->getType() == 'project') { + //remove tests dir + $this->filesystem->removeDirectory($this->getInstallPath($package) . DIRECTORY_SEPARATOR . 'tests'); + } + } + + public function update(InstalledRepositoryInterface $repo, PackageInterface $initial, PackageInterface $target) + { + parent::update($repo, $initial, $target); + if ($this->composer->getPackage()->getType() == 'project') { + //remove tests dir + $this->filesystem->removeDirectory($this->getInstallPath($target) . DIRECTORY_SEPARATOR . 'tests'); + } + } + + /** + * {@inheritDoc} + */ + public function supports($packageType) + { + return 'think-framework' === $packageType; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-installer/src/ThinkTesting.php b/vendor/topthink/think-installer/src/ThinkTesting.php new file mode 100644 index 000000000..bf27f7276 --- /dev/null +++ b/vendor/topthink/think-installer/src/ThinkTesting.php @@ -0,0 +1,68 @@ + +// +---------------------------------------------------------------------- + +namespace think\composer; + + +use Composer\Installer\LibraryInstaller; +use Composer\Package\PackageInterface; +use Composer\Repository\InstalledRepositoryInterface; + +class ThinkTesting extends LibraryInstaller +{ + /** + * {@inheritDoc} + */ + public function getInstallPath(PackageInterface $package) + { + if ('topthink/think-testing' !== $package->getPrettyName()) { + throw new \InvalidArgumentException('Unable to install this library!'); + } + + return parent::getInstallPath($package); + } + + public function install(InstalledRepositoryInterface $repo, PackageInterface $package) + { + parent::install($repo, $package); + + $this->copyTestDir($package); + + + } + + public function update(InstalledRepositoryInterface $repo, PackageInterface $initial, PackageInterface $target) + { + parent::update($repo, $initial, $target); + + $this->copyTestDir($target); + + } + + private function copyTestDir(PackageInterface $package) + { + $appDir = dirname($this->vendorDir); + $source = $this->getInstallPath($package) . DIRECTORY_SEPARATOR . 'example'; + if (!is_file($appDir . DIRECTORY_SEPARATOR . 'phpunit.xml')) { + $this->filesystem->copyThenRemove($source, $appDir); + } else { + $this->filesystem->removeDirectoryPhp($source); + } + } + + /** + * {@inheritDoc} + */ + public function supports($packageType) + { + return 'think-testing' === $packageType; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-mongo/LICENSE b/vendor/topthink/think-mongo/LICENSE new file mode 100644 index 000000000..8dada3eda --- /dev/null +++ b/vendor/topthink/think-mongo/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/topthink/think-mongo/README.md b/vendor/topthink/think-mongo/README.md new file mode 100644 index 000000000..45e4bb1a2 --- /dev/null +++ b/vendor/topthink/think-mongo/README.md @@ -0,0 +1,23 @@ +ThinkPHP 5.0 MongoDb椹卞姩 +=============== + +棣栧厛瀹夎瀹樻柟鐨刴ongodb鎵╁睍锛 + +http://pecl.php.net/package/mongodb + +鐒跺悗锛岄厤缃簲鐢ㄧ殑鏁版嵁搴撻厤缃枃浠禶database.php`鐨刞type`鍙傛暟涓猴細 + +~~~ +'type' => '\think\mongo\Connection', +~~~ + +鍗冲彲姝e父浣跨敤MongoDb锛屼緥濡傦細 +~~~ +Db::name('demo') + ->find(); +Db::name('demo') + ->field('id,name') + ->limit(10) + ->order('id','desc') + ->select(); +~~~ diff --git a/vendor/topthink/think-mongo/composer.json b/vendor/topthink/think-mongo/composer.json new file mode 100644 index 000000000..c6f585fd4 --- /dev/null +++ b/vendor/topthink/think-mongo/composer.json @@ -0,0 +1,19 @@ +{ + "name": "topthink/think-mongo", + "description": "mongodb driver for thinkphp5", + "license": "Apache-2.0", + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "require": {}, + "autoload": { + "psr-4": { + "think\\mongo\\": "src" + }, + "files": [ + ] + } +} \ No newline at end of file diff --git a/vendor/topthink/think-mongo/src/Builder.php b/vendor/topthink/think-mongo/src/Builder.php new file mode 100644 index 000000000..fea35d75d --- /dev/null +++ b/vendor/topthink/think-mongo/src/Builder.php @@ -0,0 +1,487 @@ + +// +---------------------------------------------------------------------- + +namespace think\mongo; + +use MongoDB\BSON\Javascript; +use MongoDB\BSON\ObjectID; +use MongoDB\BSON\Regex; +use MongoDB\Driver\BulkWrite; +use MongoDB\Driver\Command; +use MongoDB\Driver\Query as MongoQuery; +use think\Exception; + +class Builder +{ + // connection瀵硅薄瀹炰緥 + protected $connection; + // 鏌ヨ瀵硅薄瀹炰緥 + protected $query; + // 鏌ヨ鍙傛暟 + protected $options = []; + // 鏈鍚庢彃鍏D + protected $insertId = []; + // 鏌ヨ琛ㄨ揪寮 + protected $exp = ['<>' => 'ne', 'neq' => 'ne', '=' => 'eq', '>' => 'gt', '>=' => 'gte', '<' => 'lt', '<=' => 'lte', 'in' => 'in', 'not in' => 'nin', 'nin' => 'nin', 'mod' => 'mod', 'exists' => 'exists', 'null' => 'null', 'notnull' => 'not null', 'not null' => 'not null', 'regex' => 'regex', 'type' => 'type', 'all' => 'all', '> time' => '> time', '< time' => '< time', 'between' => 'between', 'not between' => 'not between', 'between time' => 'between time', 'not between time' => 'not between time', 'notbetween time' => 'not between time', 'like' => 'like', 'near' => 'near']; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Connection $connection 鏁版嵁搴撹繛鎺ュ璞″疄渚 + * @param Query $query 鏁版嵁搴撴煡璇㈠璞″疄渚 + */ + public function __construct(Connection $connection, Query $query) + { + $this->connection = $connection; + $this->query = $query; + } + + /** + * key鍒嗘瀽 + * @access protected + * @param string $key + * @return string + */ + protected function parseKey($key) + { + if (strpos($key, '.')) { + list($collection, $key) = explode('.', $key); + } + if ('id' == $key && $this->connection->getConfig('pk_convert_id')) { + $key = '_id'; + } + return trim($key); + } + + /** + * value鍒嗘瀽 + * @access protected + * @param mixed $value + * @param string $field + * @return string + */ + protected function parseValue($value, $field = '') + { + if ('_id' == $field && !($value instanceof ObjectID)) { + return new ObjectID($value); + } + return $value; + } + + /** + * insert鏁版嵁鍒嗘瀽 + * @access protected + * @param array $data 鏁版嵁 + * @param array $options 鏌ヨ鍙傛暟 + * @return array + */ + protected function parseData($data, $options) + { + if (empty($data)) { + return []; + } + + $result = []; + foreach ($data as $key => $val) { + $item = $this->parseKey($key); + if (is_object($val)) { + $result[$item] = $val; + } elseif (isset($val[0]) && 'exp' == $val[0]) { + $result[$item] = $val[1]; + } elseif (is_null($val)) { + $result[$item] = 'NULL'; + } else { + $result[$item] = $this->parseValue($val, $key); + } + } + return $result; + } + + /** + * Set鏁版嵁鍒嗘瀽 + * @access protected + * @param array $data 鏁版嵁 + * @param array $options 鏌ヨ鍙傛暟 + * @return array + */ + protected function parseSet($data, $options) + { + if (empty($data)) { + return []; + } + + $result = []; + foreach ($data as $key => $val) { + $item = $this->parseKey($key); + if (is_array($val) && isset($val[0]) && in_array($val[0], ['$inc', '$set', '$unset', '$push', '$pushall', '$addtoset', '$pop', '$pull', '$pullall'])) { + $result[$val[0]][$item] = $this->parseValue($val[1], $key); + } else { + $result['$set'][$item] = $this->parseValue($val, $key); + } + } + return $result; + } + + /** + * 鐢熸垚鏌ヨ杩囨护鏉′欢 + * @access public + * @param mixed $where + * @return array + */ + public function parseWhere($where) + { + if (empty($where)) { + $where = []; + } + + $filter = []; + foreach ($where as $logic => $val) { + foreach ($val as $field => $value) { + if ($value instanceof \Closure) { + // 浣跨敤闂寘鏌ヨ + $query = new Query($this->connection); + call_user_func_array($value, [ & $query]); + $filter[$logic][] = $this->parseWhere($query->getOptions('where')); + } else { + if (strpos($field, '|')) { + // 涓嶅悓瀛楁浣跨敤鐩稿悓鏌ヨ鏉′欢锛圤R锛 + $array = explode('|', $field); + foreach ($array as $k) { + $filter['$or'][] = $this->parseWhereItem($k, $value); + } + } elseif (strpos($field, '&')) { + // 涓嶅悓瀛楁浣跨敤鐩稿悓鏌ヨ鏉′欢锛圓ND锛 + $array = explode('&', $field); + foreach ($array as $k) { + $filter['$and'][] = $this->parseWhereItem($k, $value); + } + } else { + // 瀵瑰瓧娈典娇鐢ㄨ〃杈惧紡鏌ヨ + $field = is_string($field) ? $field : ''; + $filter[$logic][] = $this->parseWhereItem($field, $value); + } + } + } + } + return $filter; + } + + // where瀛愬崟鍏冨垎鏋 + protected function parseWhereItem($field, $val) + { + $key = $field ? $this->parseKey($field) : ''; + // 鏌ヨ瑙勫垯鍜屾潯浠 + if (!is_array($val)) { + $val = ['=', $val]; + } + list($exp, $value) = $val; + + // 瀵逛竴涓瓧娈典娇鐢ㄥ涓煡璇㈡潯浠 + if (is_array($exp)) { + $data = []; + foreach ($val as $value) { + $exp = $value[0]; + $value = $value[1]; + if (!in_array($exp, $this->exp)) { + $exp = strtolower($exp); + if (isset($this->exp[$exp])) { + $exp = $this->exp[$exp]; + } + } + $k = '$' . $exp; + $data[$k] = $value; + } + $query[$key] = $data; + return $query; + } elseif (!in_array($exp, $this->exp)) { + $exp = strtolower($exp); + if (isset($this->exp[$exp])) { + $exp = $this->exp[$exp]; + } else { + throw new Exception('where express error:' . $exp); + } + } + + $query = []; + if ('=' == $exp) { + // 鏅氭煡璇 + $query[$key] = $this->parseValue($value, $key); + } elseif (in_array($exp, ['neq', 'ne', 'gt', 'egt', 'gte', 'lt', 'lte', 'elt', 'mod'])) { + // 姣旇緝杩愮畻 + $k = '$' . $exp; + $query[$key] = [$k => $this->parseValue($value, $key)]; + } elseif ('null' == $exp) { + // NULL 鏌ヨ + $query[$key] = null; + } elseif ('not null' == $exp) { + $query[$key] = ['$ne' => null]; + } elseif ('all' == $exp) { + // 婊¤冻鎵鏈夋寚瀹氭潯浠 + $query[$key] = ['$all', $this->parseValue($value, $key)]; + } elseif ('between' == $exp) { + // 鍖洪棿鏌ヨ + $value = is_array($value) ? $value : explode(',', $value); + $query[$key] = ['$gte' => $this->parseValue($value[0], $key), '$lte' => $this->parseValue($value[1], $key)]; + } elseif ('not between' == $exp) { + // 鑼冨洿鏌ヨ + $value = is_array($value) ? $value : explode(',', $value); + $query[$key] = ['$lt' => $this->parseValue($value[0], $key), '$gt' => $this->parseValue($value[1], $key)]; + } elseif ('exists' == $exp) { + // 瀛楁鏄惁瀛樺湪 + $query[$key] = ['$exists' => (bool) $value]; + } elseif ('type' == $exp) { + // 绫诲瀷鏌ヨ + $query[$key] = ['$type' => intval($value)]; + } elseif ('exp' == $exp) { + // 琛ㄨ揪寮忔煡璇 + $query['$where'] = $value instanceof Javascript ? $value : new Javascript($value); + } elseif ('like' == $exp) { + // 妯$硦鏌ヨ 閲囩敤姝e垯鏂瑰紡 + $query[$key] = $value instanceof Regex ? $value : new Regex($value, 'i'); + } elseif (in_array($exp, ['nin', 'in'])) { + // IN 鏌ヨ + $value = is_array($value) ? $value : explode(',', $value); + foreach ($value as $k => $val) { + $value[$k] = $this->parseValue($val, $key); + } + $query[$key] = ['$' . $exp => $value]; + } elseif ('regex' == $exp) { + $query[$key] = $value instanceof Regex ? $value : new Regex($value, 'i'); + } elseif ('< time' == $exp) { + $query[$key] = ['$lt' => $this->parseDateTime($value, $field)]; + } elseif ('> time' == $exp) { + $query[$key] = ['$gt' => $this->parseDateTime($value, $field)]; + } elseif ('between time' == $exp) { + // 鍖洪棿鏌ヨ + $value = is_array($value) ? $value : explode(',', $value); + $query[$key] = ['$gte' => $this->parseDateTime($value[0], $field), '$lte' => $this->parseDateTime($value[1], $field)]; + } elseif ('not between time' == $exp) { + // 鑼冨洿鏌ヨ + $value = is_array($value) ? $value : explode(',', $value); + $query[$key] = ['$lt' => $this->parseDateTime($value[0], $field), '$gt' => $this->parseDateTime($value[1], $field)]; + } elseif ('near' == $exp) { + // 缁忕含搴︽煡璇 + $query[$key] = ['$near' => $this->parseValue($value, $key)]; + } else { + // 鏅氭煡璇 + $query[$key] = $this->parseValue($value, $key); + } + return $query; + } + + /** + * 鏃ユ湡鏃堕棿鏉′欢瑙f瀽 + * @access protected + * @param string $value + * @param string $key + * @return string + */ + protected function parseDateTime($value, $key) + { + // 鑾峰彇鏃堕棿瀛楁绫诲瀷 + $type = $this->query->getTableInfo('', 'type'); + if (isset($type[$key])) { + $value = strtotime($value) ?: $value; + if (preg_match('/(datetime|timestamp)/is', $type[$key])) { + // 鏃ユ湡鍙婃椂闂存埑绫诲瀷 + $value = date('Y-m-d H:i:s', $value); + } elseif (preg_match('/(date)/is', $type[$key])) { + // 鏃ユ湡鍙婃椂闂存埑绫诲瀷 + $value = date('Y-m-d', $value); + } + } + return $value; + } + + /** + * 鑾峰彇鏈鍚庡啓鍏ョ殑ID 濡傛灉鏄痠nsertAll鏂规硶鐨勮瘽 杩斿洖鎵鏈夊啓鍏ョ殑ID + * @access public + * @return mixed + */ + public function getLastInsID() + { + return $this->insertId; + } + + /** + * 鐢熸垚insert BulkWrite瀵硅薄 + * @access public + * @param array $data 鏁版嵁 + * @param array $options 琛ㄨ揪寮 + * @return BulkWrite + */ + public function insert(array $data, $options = []) + { + // 鍒嗘瀽骞跺鐞嗘暟鎹 + $data = $this->parseData($data, $options); + $bulk = new BulkWrite; + if ($insertId = $bulk->insert($data)) { + $this->insertId = $insertId; + } + $this->log('insert', $data, $options); + return $bulk; + } + + /** + * 鐢熸垚insertall BulkWrite瀵硅薄 + * @access public + * @param array $dataSet 鏁版嵁闆 + * @param array $options 鍙傛暟 + * @return BulkWrite + */ + public function insertAll($dataSet, $options = []) + { + $bulk = new BulkWrite; + foreach ($dataSet as $data) { + // 鍒嗘瀽骞跺鐞嗘暟鎹 + $data = $this->parseData($data, $options); + if ($insertId = $bulk->insert($data)) { + $this->insertId[] = $insertId; + } + } + $this->log('insert', $dataSet, $options); + return $bulk; + } + + /** + * 鐢熸垚update BulkWrite瀵硅薄 + * @access public + * @param array $data 鏁版嵁 + * @param array $options 鍙傛暟 + * @return BulkWrite + */ + public function update($data, $options = []) + { + $data = $this->parseSet($data, $options); + $where = $this->parseWhere($options['where']); + + if (1 == $options['limit']) { + $updateOptions = ['multi' => false]; + } else { + $updateOptions = ['multi' => true]; + } + $bulk = new BulkWrite; + $bulk->update($where, $data, $updateOptions); + $this->log('update', $data, $where); + return $bulk; + } + + /** + * 鐢熸垚delete BulkWrite瀵硅薄 + * @access public + * @param array $options 鍙傛暟 + * @return BulkWrite + */ + public function delete($options) + { + $where = $this->parseWhere($options['where']); + $bulk = new BulkWrite; + if (1 == $options['limit']) { + $deleteOptions = ['limit' => 1]; + } else { + $deleteOptions = ['limit' => 0]; + } + $bulk->delete($where, $deleteOptions); + $this->log('remove', $where, $deleteOptions); + return $bulk; + } + + /** + * 鐢熸垚Mongo鏌ヨ瀵硅薄 + * @access public + * @param array $options 鍙傛暟 + * @return MongoQuery + */ + public function select($options) + { + $where = $this->parseWhere($options['where']); + $query = new MongoQuery($where, $options); + $this->log('find', $where, $options); + return $query; + } + + /** + * 鐢熸垚Count鍛戒护 + * @access public + * @param array $options 鍙傛暟 + * @return Command + */ + public function count($options) + { + $cmd['count'] = $options['table']; + $cmd['query'] = $this->parseWhere($options['where']); + foreach (['hint', 'limit', 'maxTimeMS', 'skip'] as $option) { + if (isset($options[$option])) { + $cmd[$option] = $options[$option]; + } + } + $command = new Command($cmd); + $this->log('cmd', 'count', $cmd); + return $command; + } + + /** + * 鐢熸垚distinct鍛戒护 + * @access public + * @param array $options 鍙傛暟 + * @param string $field 瀛楁鍚 + * @return Command + */ + public function distinct($options, $field) + { + $cmd = [ + 'distinct' => $options['table'], + 'key' => $field, + ]; + + if (!empty($options['where'])) { + $cmd['query'] = $this->parseWhere($options['where']); + } + + if (isset($options['maxTimeMS'])) { + $cmd['maxTimeMS'] = $options['maxTimeMS']; + } + $command = new Command($cmd); + $this->log('cmd', 'distinct', $cmd); + return $command; + } + + /** + * 鏌ヨ鎵鏈夌殑collection + * @access public + * @return Command + */ + public function listcollections() + { + $cmd = ['listCollections' => 1]; + $command = new Command($cmd); + $this->log('cmd', 'listCollections', $cmd); + return $command; + } + + /** + * 鏌ヨ鏁版嵁琛ㄧ殑鐘舵佷俊鎭 + * @access public + * @return Command + */ + public function collStats($options) + { + $cmd = ['collStats' => $options['table']]; + $command = new Command($cmd); + $this->log('cmd', 'collStats', $cmd); + return $command; + } + + protected function log($type, $data, $options = []) + { + if ($this->connection->getConfig('debug')) { + $this->connection->log($type, $data, $options); + } + } +} diff --git a/vendor/topthink/think-mongo/src/Connection.php b/vendor/topthink/think-mongo/src/Connection.php new file mode 100644 index 000000000..2670c2433 --- /dev/null +++ b/vendor/topthink/think-mongo/src/Connection.php @@ -0,0 +1,594 @@ + +// +---------------------------------------------------------------------- + +namespace think\mongo; + +use MongoDB\BSON\ObjectID; +use MongoDB\Driver\BulkWrite; +use MongoDB\Driver\Command; +use MongoDB\Driver\Cursor; +use MongoDB\Driver\Exception\AuthenticationException; +use MongoDB\Driver\Exception\BulkWriteException; +use MongoDB\Driver\Exception\ConnectionException; +use MongoDB\Driver\Exception\InvalidArgumentException; +use MongoDB\Driver\Exception\RuntimeException; +use MongoDB\Driver\Manager; +use MongoDB\Driver\Query as MongoQuery; +use MongoDB\Driver\ReadPreference; +use MongoDB\Driver\WriteConcern; +use think\Collection; +use think\Db; +use think\Debug; +use think\Exception; +use think\Log; + +/** + * Mongo鏁版嵁搴撻┍鍔 + */ +class Connection +{ + protected $dbName = ''; // dbName + /** @var string 褰撳墠SQL鎸囦护 */ + protected $queryStr = ''; + // 鏌ヨ鏁版嵁绫诲瀷 + protected $typeMap = 'array'; + protected $mongo; // MongoDb Object + protected $cursor; // MongoCursor Object + + // 鐩戝惉鍥炶皟 + protected static $event = []; + /** @var PDO[] 鏁版嵁搴撹繛鎺D 鏀寔澶氫釜杩炴帴 */ + protected $links = []; + /** @var PDO 褰撳墠杩炴帴ID */ + protected $linkID; + protected $linkRead; + protected $linkWrite; + + // 杩斿洖鎴栬呭奖鍝嶈褰曟暟 + protected $numRows = 0; + // 閿欒淇℃伅 + protected $error = ''; + // 鏌ヨ瀵硅薄 + protected $query = []; + // 鏌ヨ鍙傛暟 + protected $options = []; + // 鏁版嵁搴撹繛鎺ュ弬鏁伴厤缃 + protected $config = [ + // 鏁版嵁搴撶被鍨 + 'type' => '', + // 鏈嶅姟鍣ㄥ湴鍧 + 'hostname' => '', + // 鏁版嵁搴撳悕 + 'database' => '', + // 鐢ㄦ埛鍚 + 'username' => '', + // 瀵嗙爜 + 'password' => '', + // 绔彛 + 'hostport' => '', + // 杩炴帴dsn + 'dsn' => '', + // 鏁版嵁搴撹繛鎺ュ弬鏁 + 'params' => [], + // 鏁版嵁搴撶紪鐮侀粯璁ら噰鐢╱tf8 + 'charset' => 'utf8', + // 涓婚敭鍚 + 'pk' => '_id', + // 鏁版嵁搴撹〃鍓嶇紑 + 'prefix' => '', + // 鏁版嵁搴撹皟璇曟ā寮 + 'debug' => false, + // 鏁版嵁搴撻儴缃叉柟寮:0 闆嗕腑寮(鍗曚竴鏈嶅姟鍣),1 鍒嗗竷寮(涓讳粠鏈嶅姟鍣) + 'deploy' => 0, + // 鏁版嵁搴撹鍐欐槸鍚﹀垎绂 涓讳粠寮忔湁鏁 + 'rw_separate' => false, + // 璇诲啓鍒嗙鍚 涓绘湇鍔″櫒鏁伴噺 + 'master_num' => 1, + // 鎸囧畾浠庢湇鍔″櫒搴忓彿 + 'slave_no' => '', + // 鏄惁涓ユ牸妫鏌ュ瓧娈垫槸鍚﹀瓨鍦 + 'fields_strict' => true, + // 鏁版嵁闆嗚繑鍥炵被鍨 + 'resultset_type' => 'array', + // 鑷姩鍐欏叆鏃堕棿鎴冲瓧娈 + 'auto_timestamp' => false, + // 鏄惁闇瑕佽繘琛孲QL鎬ц兘鍒嗘瀽 + 'sql_explain' => false, + // 鏄惁_id杞崲涓篿d + 'pk_convert_id' => false, + // typeMap + 'type_map' => ['root' => 'array', 'document' => 'array'], + // Query瀵硅薄 + 'query' => '\\think\\mongo\\Query', + ]; + + /** + * 鏋舵瀯鍑芥暟 璇诲彇鏁版嵁搴撻厤缃俊鎭 + * @access public + * @param array $config 鏁版嵁搴撻厤缃暟缁 + */ + public function __construct(array $config = []) + { + if (!class_exists('\MongoDB\Driver\Manager')) { + throw new Exception('require mongodb > 1.0'); + } + if (!empty($config)) { + $this->config = array_merge($this->config, $config); + } + } + + /** + * 杩炴帴鏁版嵁搴撴柟娉 + * @access public + * @param array $config 杩炴帴鍙傛暟 + * @param integer $linkNum 杩炴帴搴忓彿 + * @throws InvalidArgumentException + * @throws RuntimeException + */ + public function connect(array $config = [], $linkNum = 0) + { + if (!isset($this->links[$linkNum])) { + if (empty($config)) { + $config = $this->config; + } else { + $config = array_merge($this->config, $config); + } + $this->dbName = $config['database']; + $this->typeMap = $config['type_map']; + + if ($config['pk_convert_id'] && '_id' == $config['pk']) { + $this->config['pk'] = 'id'; + } + $host = 'mongodb://' . ($config['username'] ? "{$config['username']}" : '') . ($config['password'] ? ":{$config['password']}@" : '') . $config['hostname'] . ($config['hostport'] ? ":{$config['hostport']}" : '') . '/' . ($config['database'] ? "{$config['database']}" : ''); + if ($config['debug']) { + $startTime = microtime(true); + } + $this->links[$linkNum] = new Manager($host, $this->config['params']); + if ($config['debug']) { + // 璁板綍鏁版嵁搴撹繛鎺ヤ俊鎭 + Log::record('[ DB ] CONNECT:[ UseTime:' . number_format(microtime(true) - $startTime, 6) . 's ] ' . $config['dsn'], 'sql'); + } + } + return $this->links[$linkNum]; + } + + /** + * 鍒涘缓鎸囧畾妯″瀷鐨勬煡璇㈠璞 + * @access public + * @param string $model 妯″瀷绫诲悕绉 + * @param string $queryClass 鏌ヨ瀵硅薄绫诲悕 + * @return Query + */ + public function getQuery($model = 'db', $queryClass = '') + { + if (!isset($this->query[$model])) { + $class = $queryClass ?: $this->config['query']; + $this->query[$model] = new $class($this, 'db' == $model ? '' : $model); + } + return $this->query[$model]; + } + + /** + * 璋冪敤Query绫荤殑鏌ヨ鏂规硶 + * @access public + * @param string $method 鏂规硶鍚嶇О + * @param array $args 璋冪敤鍙傛暟 + * @return mixed + */ + public function __call($method, $args) + { + return call_user_func_array([$this->getQuery(), $method], $args); + } + + /** + * 鑾峰彇鏁版嵁搴撶殑閰嶇疆鍙傛暟 + * @access public + * @param string $config 閰嶇疆鍚嶇О + * @return mixed + */ + public function getConfig($config = '') + { + return $config ? $this->config[$config] : $this->config; + } + + /** + * 璁剧疆鏁版嵁搴撶殑閰嶇疆鍙傛暟 + * @access public + * @param string $config 閰嶇疆鍚嶇О + * @param mixed $value 閰嶇疆鍊 + * @return void + */ + public function setConfig($config, $value) + { + $this->config[$config] = $value; + } + + /** + * 鑾峰彇Mongo Manager瀵硅薄 + * @access public + * @return Manager|null + */ + public function getMongo() + { + if (!$this->mongo) { + return; + } else { + return $this->mongo; + } + } + + /** + * 璁剧疆/鑾峰彇褰撳墠鎿嶄綔鐨刣atabase + * @access public + * @param string $db db + * @throws Exception + */ + public function db($db = null) + { + if (is_null($db)) { + return $this->dbName; + } else { + $this->dbName = $db; + } + } + + /** + * 鎵ц鏌ヨ + * @access public + * @param string $namespace 褰撳墠鏌ヨ鐨刢ollection + * @param MongoQuery $query 鏌ヨ瀵硅薄 + * @param ReadPreference $readPreference readPreference + * @param string|bool $class 杩斿洖鐨勬暟鎹泦绫诲瀷 + * @param string|array $typeMap 鎸囧畾杩斿洖鐨則ypeMap + * @return mixed + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function query($namespace, MongoQuery $query, ReadPreference $readPreference = null, $class = false, $typeMap = null) + { + $this->initConnect(false); + Db::$queryTimes++; + + if (false === strpos($namespace, '.')) { + $namespace = $this->dbName . '.' . $namespace; + } + if ($this->config['debug'] && !empty($this->queryStr)) { + // 璁板綍鎵ц鎸囦护 + $this->queryStr = 'db' . strstr($namespace, '.') . '.' . $this->queryStr; + } + $this->debug(true); + $this->cursor = $this->mongo->executeQuery($namespace, $query, $readPreference); + $this->debug(false); + return $this->getResult($class, $typeMap); + } + + /** + * 鎵ц鎸囦护 + * @access public + * @param Command $command 鎸囦护 + * @param string $dbName 褰撳墠鏁版嵁搴撳悕 + * @param ReadPreference $readPreference readPreference + * @param string|bool $class 杩斿洖鐨勬暟鎹泦绫诲瀷 + * @param string|array $typeMap 鎸囧畾杩斿洖鐨則ypeMap + * @return mixed + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function command(Command $command, $dbName = '', ReadPreference $readPreference = null, $class = false, $typeMap) + { + $this->initConnect(false); + Db::$queryTimes++; + + $this->debug(true); + $dbName = $dbName ?: $this->dbName; + if ($this->config['debug'] && !empty($this->queryStr)) { + $this->queryStr = 'db.' . $dbName . '.' . $this->queryStr; + } + $this->cursor = $this->mongo->executeCommand($dbName, $command, $readPreference); + $this->debug(false); + return $this->getResult($class, $typeMap); + + } + + /** + * 鑾峰緱鏁版嵁闆 + * @access protected + * @param bool|string $class true 杩斿洖Mongo cursor瀵硅薄 瀛楃涓茬敤浜庢寚瀹氳繑鍥炵殑绫诲悕 + * @param string|array $typeMap 鎸囧畾杩斿洖鐨則ypeMap + * @return mixed + */ + protected function getResult($class = '', $typeMap = null) + { + if (true === $class) { + return $this->cursor; + } + // 璁剧疆缁撴灉鏁版嵁绫诲瀷 + if (is_null($typeMap)) { + $typeMap = $this->typeMap; + } + $typeMap = is_string($typeMap) ? ['root' => $typeMap] : $typeMap; + $this->cursor->setTypeMap($typeMap); + + // 鑾峰彇鏁版嵁闆 + $result = $this->cursor->toArray(); + if ($this->getConfig('pk_convert_id')) { + // 杞崲ObjectID 瀛楁 + foreach ($result as &$data) { + $this->convertObjectID($data); + } + } + $this->numRows = count($result); + + return $result; + } + + /** + * ObjectID澶勭悊 + * @access public + * @param array $data + * @return void + */ + private function convertObjectID(&$data) + { + if (isset($data['_id'])) { + $data['id'] = $data['_id']->__toString(); + unset($data['_id']); + } + } + + /** + * 鎵ц鍐欐搷浣 + * @access public + * @param string $namespace + * @param BulkWrite $bulk + * @param WriteConcern $writeConcern + * + * @return WriteResult + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function execute($namespace, BulkWrite $bulk, WriteConcern $writeConcern = null) + { + $this->initConnect(true); + Db::$executeTimes++; + if (false === strpos($namespace, '.')) { + $namespace = $this->dbName . '.' . $namespace; + } + if ($this->config['debug'] && !empty($this->queryStr)) { + // 璁板綍鎵ц鎸囦护 + $this->queryStr = 'db' . strstr($namespace, '.') . '.' . $this->queryStr; + } + $this->debug(true); + $writeResult = $this->mongo->executeBulkWrite($namespace, $bulk, $writeConcern); + $this->debug(false); + $this->numRows = $writeResult->getMatchedCount(); + return $writeResult; + } + + /** + * 鏁版嵁搴撴棩蹇楄褰曪紙浠呬緵鍙傝冿級 + * @access public + * @param string $type 绫诲瀷 + * @param mixed $data 鏁版嵁 + * @param array $options 鍙傛暟 + * @return void + */ + public function log($type, $data, $options = []) + { + if (!$this->config['debug']) { + return; + } + if (is_array($data)) { + array_walk_recursive($data, function (&$value) { + if ($value instanceof ObjectID) { + $value = $value->__toString(); + } + }); + } + switch (strtolower($type)) { + case 'find': + $this->queryStr = $type . '(' . ($data ? json_encode($data) : '') . ')'; + if (isset($options['sort'])) { + $this->queryStr .= '.sort(' . json_encode($options['sort']) . ')'; + } + if (isset($options['limit'])) { + $this->queryStr .= '.limit(' . $options['limit'] . ')'; + } + $this->queryStr .= ';'; + break; + case 'insert': + case 'remove': + $this->queryStr = $type . '(' . ($data ? json_encode($data) : '') . ');'; + break; + case 'update': + $this->queryStr = $type . '(' . json_encode($options) . ',' . json_encode($data) . ');'; + break; + case 'cmd': + $this->queryStr = $data . '(' . json_encode($options) . ');'; + break; + } + $this->options = $options; + } + + /** + * 鑾峰彇鎵ц鐨勬寚浠 + * @access public + * @return string + */ + public function getQueryStr() + { + return $this->queryStr; + } + + /** + * 鐩戝惉SQL鎵ц + * @access public + * @param callable $callback 鍥炶皟鏂规硶 + * @return void + */ + public function listen($callback) + { + self::$event[] = $callback; + } + + /** + * 瑙﹀彂SQL浜嬩欢 + * @access protected + * @param string $sql 璇彞 + * @param float $runtime 杩愯鏃堕棿 + * @param array $options 鍙傛暟 + * @return bool + */ + protected function trigger($sql, $runtime, $options = []) + { + if (!empty(self::$event)) { + foreach (self::$event as $callback) { + if (is_callable($callback)) { + call_user_func_array($callback, [$sql, $runtime, $options]); + } + } + } else { + // 鏈敞鍐岀洃鍚垯璁板綍鍒版棩蹇椾腑 + Log::record('[ Mongo ] ' . $sql . ' [ RunTime:' . $runtime . 's ]', 'sql'); + } + } + + /** + * 鏁版嵁搴撹皟璇 璁板綍褰撳墠SQL鍙婂垎鏋愭ц兘 + * @access protected + * @param boolean $start 璋冭瘯寮濮嬫爣璁 true 寮濮 false 缁撴潫 + * @param string $sql 鎵ц鐨凷QL璇彞 鐣欑┖鑷姩鑾峰彇 + * @return void + */ + protected function debug($start, $sql = '') + { + if (!empty($this->config['debug'])) { + // 寮鍚暟鎹簱璋冭瘯妯″紡 + if ($start) { + Debug::remark('queryStartTime', 'time'); + } else { + // 璁板綍鎿嶄綔缁撴潫鏃堕棿 + Debug::remark('queryEndTime', 'time'); + $runtime = Debug::getRangeTime('queryStartTime', 'queryEndTime'); + $sql = $sql ?: $this->queryStr; + // SQL鐩戝惉 + $this->trigger($sql, $runtime, $this->options); + } + } + } + + /** + * 閲婃斁鏌ヨ缁撴灉 + * @access public + */ + public function free() + { + $this->cursor = null; + } + + /** + * 鍏抽棴鏁版嵁搴 + * @access public + */ + public function close() + { + $this->mongo = null; + $this->cursor = null; + $this->linkRead = null; + $this->linkWrite = null; + $this->links = []; + } + + /** + * 鍒濆鍖栨暟鎹簱杩炴帴 + * @access protected + * @param boolean $master 鏄惁涓绘湇鍔″櫒 + * @return void + */ + protected function initConnect($master = true) + { + if (!empty($this->config['deploy'])) { + // 閲囩敤鍒嗗竷寮忔暟鎹簱 + if ($master) { + if (!$this->linkWrite) { + $this->linkWrite = $this->multiConnect(true); + } + $this->mongo = $this->linkWrite; + } else { + if (!$this->linkRead) { + $this->linkRead = $this->multiConnect(false); + } + $this->mongo = $this->linkRead; + } + } elseif (!$this->mongo) { + // 榛樿鍗曟暟鎹簱 + $this->mongo = $this->connect(); + } + } + + /** + * 杩炴帴鍒嗗竷寮忔湇鍔″櫒 + * @access protected + * @param boolean $master 涓绘湇鍔″櫒 + * @return PDO + */ + protected function multiConnect($master = false) + { + $_config = []; + // 鍒嗗竷寮忔暟鎹簱閰嶇疆瑙f瀽 + foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) { + $_config[$name] = explode(',', $this->config[$name]); + } + + // 涓绘湇鍔″櫒搴忓彿 + $m = floor(mt_rand(0, $this->config['master_num'] - 1)); + + if ($this->config['rw_separate']) { + // 涓讳粠寮忛噰鐢ㄨ鍐欏垎绂 + if ($master) // 涓绘湇鍔″櫒鍐欏叆 + { + $r = $m; + } elseif (is_numeric($this->config['slave_no'])) { + // 鎸囧畾鏈嶅姟鍣ㄨ + $r = $this->config['slave_no']; + } else { + // 璇绘搷浣滆繛鎺ヤ粠鏈嶅姟鍣 姣忔闅忔満杩炴帴鐨勬暟鎹簱 + $r = floor(mt_rand($this->config['master_num'], count($_config['hostname']) - 1)); + } + } else { + // 璇诲啓鎿嶄綔涓嶅尯鍒嗘湇鍔″櫒 姣忔闅忔満杩炴帴鐨勬暟鎹簱 + $r = floor(mt_rand(0, count($_config['hostname']) - 1)); + } + $dbConfig = []; + foreach (['username', 'password', 'hostname', 'hostport', 'database', 'dsn', 'charset'] as $name) { + $dbConfig[$name] = isset($_config[$name][$r]) ? $_config[$name][$r] : $_config[$name][0]; + } + return $this->connect($dbConfig, $r); + } + + /** + * 鏋愭瀯鏂规硶 + * @access public + */ + public function __destruct() + { + // 閲婃斁鏌ヨ + $this->free(); + + // 鍏抽棴杩炴帴 + $this->close(); + } +} diff --git a/vendor/topthink/think-mongo/src/Query.php b/vendor/topthink/think-mongo/src/Query.php new file mode 100644 index 000000000..114bf609c --- /dev/null +++ b/vendor/topthink/think-mongo/src/Query.php @@ -0,0 +1,1993 @@ + +// +---------------------------------------------------------------------- + +namespace think\mongo; + +use MongoDB\BSON\ObjectID; +use MongoDB\Driver\BulkWrite; +use MongoDB\Driver\Command; +use MongoDB\Driver\Cursor; +use MongoDB\Driver\Exception\AuthenticationException; +use MongoDB\Driver\Exception\BulkWriteException; +use MongoDB\Driver\Exception\ConnectionException; +use MongoDB\Driver\Exception\InvalidArgumentException; +use MongoDB\Driver\Exception\RuntimeException; +use MongoDB\Driver\Query as MongoQuery; +use MongoDB\Driver\ReadPreference; +use MongoDB\Driver\WriteConcern; +use think\Cache; +use think\Collection; +use think\Config; +use think\Db; +use think\db\exception\DataNotFoundException; +use think\db\exception\ModelNotFoundException; +use think\Exception; +use think\exception\DbException; +use think\Loader; +use think\Model; +use think\Paginator; + +class Query +{ + // 鏁版嵁搴揅onnection瀵硅薄瀹炰緥 + protected $connection; + // 鏁版嵁搴揃uilder瀵硅薄瀹炰緥 + protected $builder; + // 褰撳墠妯″瀷绫诲悕绉 + protected $model; + // 褰撳墠鏁版嵁琛ㄥ悕绉帮紙鍚墠缂锛 + protected $table = ''; + // 褰撳墠鏁版嵁琛ㄥ悕绉帮紙涓嶅惈鍓嶇紑锛 + protected $name = ''; + // 褰撳墠鏁版嵁琛ㄤ富閿 + protected $pk; + // 褰撳墠鏁版嵁琛ㄥ墠缂 + protected $prefix = ''; + // 鏌ヨ鍙傛暟 + protected $options = []; + // 鏁版嵁琛ㄤ俊鎭 + protected static $info = []; + // 鍥炶皟浜嬩欢 + private static $event = []; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + * @param Connection $connection 鏁版嵁搴撳璞″疄渚 + * @param string $model 妯″瀷鍚 + */ + public function __construct(Connection $connection = null, $model = '') + { + $this->connection = $connection ?: Db::connect([], true); + $this->prefix = $this->connection->getConfig('prefix'); + $this->model = $model; + // 璁剧疆褰撳墠杩炴帴鐨凚uilder瀵硅薄 + $this->setBuilder(); + } + + /** + * 鍒╃敤__call鏂规硶瀹炵幇涓浜涚壒娈婄殑Model鏂规硶 + * @access public + * @param string $method 鏂规硶鍚嶇О + * @param array $args 璋冪敤鍙傛暟 + * @return mixed + * @throws DbException + * @throws Exception + */ + public function __call($method, $args) + { + if (strtolower(substr($method, 0, 5)) == 'getby') { + // 鏍规嵁鏌愪釜瀛楁鑾峰彇璁板綍 + $field = Loader::parseName(substr($method, 5)); + $where[$field] = $args[0]; + return $this->where($where)->find(); + } elseif (strtolower(substr($method, 0, 10)) == 'getfieldby') { + // 鏍规嵁鏌愪釜瀛楁鑾峰彇璁板綍鐨勬煇涓 + $name = Loader::parseName(substr($method, 10)); + $where[$name] = $args[0]; + return $this->where($where)->value($args[1]); + } else { + throw new Exception('method not exists:' . __CLASS__ . '->' . $method); + } + } + + /** + * 鑾峰彇褰撳墠鐨勬暟鎹簱Connection瀵硅薄 + * @access public + * @return Connection + */ + public function getConnection() + { + return $this->connection; + } + + /** + * 鍒囨崲褰撳墠鐨勬暟鎹簱杩炴帴 + * @access public + * @param mixed $config + * @return $this + */ + public function connect($config) + { + $this->connection = Db::connect($config); + $this->setBuilder(); + return $this; + } + + /** + * 璁剧疆褰撳墠鐨勬暟鎹簱Builder瀵硅薄 + * @access protected + * @return void + */ + protected function setBuilder() + { + $this->builder = new Builder($this->connection, $this); + } + + /** + * 鎸囧畾榛樿鐨勬暟鎹〃鍚嶏紙涓嶅惈鍓嶇紑锛 + * @access public + * @param string $name + * @return $this + */ + public function name($name) + { + $this->name = $name; + return $this; + } + + /** + * 鎸囧畾榛樿鏁版嵁琛ㄥ悕锛堝惈鍓嶇紑锛 + * @access public + * @param string $table 琛ㄥ悕 + * @return $this + */ + public function setTable($table) + { + $this->table = $table; + return $this; + } + + /** + * 寰楀埌褰撳墠鎴栬呮寚瀹氬悕绉扮殑鏁版嵁琛 + * @access public + * @param string $name + * @return string + */ + public function getTable($name = '') + { + if ($name || empty($this->table)) { + $name = $name ?: $this->name; + $tableName = $this->prefix; + if ($name) { + $tableName .= Loader::parseName($name); + } + } else { + $tableName = $this->table; + } + return $tableName; + } + + /** + * 鎸囧畾鏁版嵁琛ㄤ富閿 + * @access public + * @param string $pk 涓婚敭 + * @return $this + */ + public function pk($pk) + { + $this->pk = $pk; + return $this; + } + + /** + * 鍘婚櫎鏌愪釜鏌ヨ鏉′欢 + * @access public + * @param string $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function removeWhereField($field, $logic = 'and') + { + $logic = '$' . strtoupper($logic); + if (isset($this->options['where'][$logic][$field])) { + unset($this->options['where'][$logic][$field]); + } + return $this; + } + + /** + * 鍘婚櫎鏌ヨ鍙傛暟 + * @access public + * @param string|bool $option 鍙傛暟鍚 true 琛ㄧず鍘婚櫎鎵鏈夊弬鏁 + * @return $this + */ + public function removeOption($option = true) + { + if (true === $option) { + $this->options = []; + } elseif (is_string($option) && isset($this->options[$option])) { + unset($this->options[$option]); + } + return $this; + } + + /** + * 灏哠QL璇彞涓殑__TABLE_NAME__瀛楃涓叉浛鎹㈡垚甯﹀墠缂鐨勮〃鍚嶏紙灏忓啓锛 + * @access public + * @param string $sql sql璇彞 + * @return string + */ + public function parseSqlTable($sql) + { + if (false !== strpos($sql, '__')) { + $prefix = $this->prefix; + $sql = preg_replace_callback("/__([A-Z0-9_-]+)__/sU", function ($match) use ($prefix) { + return $prefix . strtolower($match[1]); + }, $sql); + } + return $sql; + } + + /** + * 鎵ц鏌ヨ 杩斿洖鏁版嵁闆 + * @access public + * @param string $namespace + * @param MongoQuery $query 鏌ヨ瀵硅薄 + * @param ReadPreference $readPreference readPreference + * @param bool|string $class 鎸囧畾杩斿洖鐨勬暟鎹泦瀵硅薄 + * @param string|array $typeMap 鎸囧畾杩斿洖鐨則ypeMap + * @return mixed + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function query($namespace, MongoQuery $query, ReadPreference $readPreference = null, $class = false, $typeMap = null) + { + return $this->connection->query($namespace, $query, $readPreference, $class, $typeMap); + } + + /** + * 鎵ц鎸囦护 杩斿洖鏁版嵁闆 + * @access public + * @param Command $command 鎸囦护 + * @param string $dbName + * @param ReadPreference $readPreference readPreference + * @param bool|string $class 鎸囧畾杩斿洖鐨勬暟鎹泦瀵硅薄 + * @param string|array $typeMap 鎸囧畾杩斿洖鐨則ypeMap + * @return mixed + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function command(Command $command, $dbName = '', ReadPreference $readPreference = null, $class = false, $typeMap = null) + { + return $this->connection->command($command, $dbName, $readPreference, $class, $typeMap); + } + + /** + * 鎵ц璇彞 + * @access public + * @param string $namespace + * @param BulkWrite $bulk + * @param WriteConcern $writeConcern + * @return int + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function execute($namespace, BulkWrite $bulk, WriteConcern $writeConcern = null) + { + return $this->connection->execute($namespace, $bulk, $writeConcern); + } + + /** + * 鑾峰彇鏈杩戞彃鍏ョ殑ID + * @access public + * @return string + */ + public function getLastInsID() + { + $id = $this->builder->getLastInsID(); + if ($id instanceof ObjectID) { + $id = $id->__toString(); + } + return $id; + } + + /** + * 鑾峰彇鏈杩戜竴娆℃墽琛岀殑鎸囦护 + * @access public + * @return string + */ + public function getLastSql() + { + return $this->connection->getQueryStr(); + } + + /** + * 鑾峰彇鏁版嵁搴撶殑閰嶇疆鍙傛暟 + * @access public + * @param string $name 鍙傛暟鍚嶇О + * @return boolean + */ + public function getConfig($name = '') + { + return $this->connection->getConfig($name); + } + + /** + * 寰楀埌鏌愪釜瀛楁鐨勫 + * @access public + * @param string $field 瀛楁鍚 + * @param mixed $default 榛樿鍊 + * @return mixed + */ + public function value($field, $default = null) + { + $result = null; + if (!empty($this->options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $this->options['cache']; + if (empty($this->options['table'])) { + $this->options['table'] = $this->getTable(); + } + $key = is_string($cache['key']) ? $cache['key'] : md5($field . serialize($this->options)); + $result = Cache::get($key); + } + if (!$result) { + if (isset($this->options['field'])) { + unset($this->options['field']); + } + $cursor = $this->field($field)->fetchCursor(true)->find(); + $cursor->setTypeMap(['root' => 'array']); + $resultSet = $cursor->toArray(); + $data = isset($resultSet[0]) ? $resultSet[0] : null; + $result = $data[$field]; + if (isset($cache)) { + // 缂撳瓨鏁版嵁 + $this->cacheData($key, $result, $cache); + } + } else { + // 娓呯┖鏌ヨ鏉′欢 + $this->options = []; + } + return !is_null($result) ? $result : $default; + } + + /** + * 寰楀埌鏌愪釜鍒楃殑鏁扮粍 + * @access public + * @param string $field 瀛楁鍚 澶氫釜瀛楁鐢ㄩ楀彿鍒嗛殧 + * @param string $key 绱㈠紩 + * @return array + */ + public function column($field, $key = '') + { + $result = false; + if (!empty($this->options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $this->options['cache']; + if (empty($this->options['table'])) { + $this->options['table'] = $this->getTable(); + } + $guid = is_string($cache['key']) ? $cache['key'] : md5($field . serialize($this->options)); + $result = Cache::get($guid); + } + if (!$result) { + if (isset($this->options['field'])) { + unset($this->options['field']); + } + if ($key && '*' != $field) { + $field = $key . ',' . $field; + } + $cursor = $this->field($field)->fetchCursor(true)->select(); + $cursor->setTypeMap(['root' => 'array']); + $resultSet = $cursor->toArray(); + if ($resultSet) { + $fields = array_keys($resultSet[0]); + $count = count($fields); + $key1 = array_shift($fields); + $key2 = $fields ? array_shift($fields) : ''; + $key = $key ?: $key1; + foreach ($resultSet as $val) { + $name = $val[$key]; + if ($name instanceof ObjectID) { + $name = $name->__toString(); + } + if (2 == $count) { + $result[$name] = $val[$key2]; + } elseif (1 == $count) { + $result[$name] = $val[$key1]; + } else { + $result[$name] = $val; + } + } + } else { + $result = []; + } + + if (isset($cache) && isset($guid)) { + // 缂撳瓨鏁版嵁 + $this->cacheData($guid, $result, $cache); + } + } else { + // 娓呯┖鏌ヨ鏉′欢 + $this->options = []; + } + return $result; + } + + /** + * 鎵цcommand + * @access public + * @param string|array|object $command 鎸囦护 + * @param mixed $extra 棰濆鍙傛暟 + * @param string $db 鏁版嵁搴撳悕 + * @return array + */ + public function cmd($command, $extra = null, $db = null) + { + if (is_array($command) || is_object($command)) { + if ($this->connection->getConfig('debug')) { + $this->connection->log('cmd', 'cmd', $command); + } + // 鐩存帴鍒涘缓Command瀵硅薄 + $command = new Command($command); + } else { + // 璋冪敤Builder灏佽鐨凜ommand瀵硅薄 + $options = $this->parseExpress(); + $command = $this->builder->$command($options, $extra); + } + return $this->command($command, $db); + } + + /** + * 鎸囧畾distinct鏌ヨ + * @access public + * @param string $field 瀛楁鍚 + * @return array + */ + public function distinct($field) + { + $result = $this->cmd('distinct', $field); + return $result[0]['values']; + } + + /** + * 鑾峰彇鏁版嵁搴撶殑鎵鏈塩ollection + * @access public + * @param string $db 鏁版嵁搴撳悕绉 鐣欑┖涓哄綋鍓嶆暟鎹簱 + * @throws Exception + */ + public function listCollections($db = '') + { + $cursor = $this->cmd('listCollections', null, $db); + $result = []; + foreach ($cursor as $collection) { + $result[] = $collection['name']; + } + return $result; + } + + /** + * COUNT鏌ヨ + * @access public + * @return integer + */ + public function count() + { + $result = $this->cmd('count'); + return $result[0]['n']; + } + + /** + * 璁剧疆璁板綍鐨勬煇涓瓧娈靛 + * 鏀寔浣跨敤鏁版嵁搴撳瓧娈靛拰鏂规硶 + * @access public + * @param string|array $field 瀛楁鍚 + * @param mixed $value 瀛楁鍊 + * @return integer + */ + public function setField($field, $value = '') + { + if (is_array($field)) { + $data = $field; + } else { + $data[$field] = $value; + } + return $this->update($data); + } + + /** + * 瀛楁鍊(寤惰繜)澧為暱 + * @access public + * @param string $field 瀛楁鍚 + * @param integer $step 澧為暱鍊 + * @param integer $lazyTime 寤舵椂鏃堕棿(s) + * @return integer|true + * @throws Exception + */ + public function setInc($field, $step = 1, $lazyTime = 0) + { + $condition = !empty($this->options['where']) ? $this->options['where'] : []; + if (empty($condition)) { + // 娌℃湁鏉′欢涓嶅仛浠讳綍鏇存柊 + throw new Exception('no data to update'); + } + if ($lazyTime > 0) { + // 寤惰繜鍐欏叆 + $guid = md5($this->getTable() . '_' . $field . '_' . serialize($condition)); + $step = $this->lazyWrite($guid, $step, $lazyTime); + if (empty($step)) { + return true; // 绛夊緟涓嬫鍐欏叆 + } + } + return $this->setField($field, ['$inc', $step]); + } + + /** + * 瀛楁鍊硷紙寤惰繜锛夊噺灏 + * @access public + * @param string $field 瀛楁鍚 + * @param integer $step 鍑忓皯鍊 + * @param integer $lazyTime 寤舵椂鏃堕棿(s) + * @return integer|true + * @throws Exception + */ + public function setDec($field, $step = 1, $lazyTime = 0) + { + $condition = !empty($this->options['where']) ? $this->options['where'] : []; + if (empty($condition)) { + // 娌℃湁鏉′欢涓嶅仛浠讳綍鏇存柊 + throw new Exception('no data to update'); + } + if ($lazyTime > 0) { + // 寤惰繜鍐欏叆 + $guid = md5($this->getTable() . '_' . $field . '_' . serialize($condition)); + $step = $this->lazyWrite($guid, -$step, $lazyTime); + if (empty($step)) { + return true; // 绛夊緟涓嬫鍐欏叆 + } + } + return $this->setField($field, ['$inc', -1 * $step]); + } + + /** + * 寤舵椂鏇存柊妫鏌 杩斿洖false琛ㄧず闇瑕佸欢鏃 + * 鍚﹀垯杩斿洖瀹為檯鍐欏叆鐨勬暟鍊 + * @access public + * @param string $guid 鍐欏叆鏍囪瘑 + * @param integer $step 鍐欏叆姝ヨ繘鍊 + * @param integer $lazyTime 寤舵椂鏃堕棿(s) + * @return false|integer + */ + protected function lazyWrite($guid, $step, $lazyTime) + { + if (false !== ($value = Cache::get($guid))) { + // 瀛樺湪缂撳瓨鍐欏叆鏁版嵁 + if ($_SERVER['REQUEST_TIME'] > Cache::get($guid . '_time') + $lazyTime) { + // 寤舵椂鏇存柊鏃堕棿鍒颁簡锛屽垹闄ょ紦瀛樻暟鎹 骞跺疄闄呭啓鍏ユ暟鎹簱 + Cache::rm($guid); + Cache::rm($guid . '_time'); + return $value + $step; + } else { + // 杩藉姞鏁版嵁鍒扮紦瀛 + Cache::set($guid, $value + $step, 0); + return false; + } + } else { + // 娌℃湁缂撳瓨鏁版嵁 + Cache::set($guid, $step, 0); + // 璁℃椂寮濮 + Cache::set($guid . '_time', $_SERVER['REQUEST_TIME'], 0); + return false; + } + } + + /** + * 鎸囧畾AND鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @return $this + */ + public function where($field, $op = null, $condition = null) + { + $param = func_get_args(); + array_shift($param); + $this->parseWhereExp('and', $field, $op, $condition, $param); + return $this; + } + + /** + * 鎸囧畾OR鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @return $this + */ + public function whereOr($field, $op = null, $condition = null) + { + $param = func_get_args(); + array_shift($param); + $this->parseWhereExp('or', $field, $op, $condition, $param); + return $this; + } + + /** + * 鎸囧畾NOR鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @return $this + */ + public function whereNor($field, $op = null, $condition = null) + { + $param = func_get_args(); + array_shift($param); + $this->parseWhereExp('nor', $field, $op, $condition, $param); + return $this; + } + + /** + * 鎸囧畾Null鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNull($field, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'null', null); + return $this; + } + + /** + * 鎸囧畾NotNull鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotNull($field, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'notnull', null); + return $this; + } + + /** + * 鎸囧畾In鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereIn($field, $condition, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'in', $condition); + return $this; + } + + /** + * 鎸囧畾NotIn鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotIn($field, $condition, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'not in', $condition); + return $this; + } + + /** + * 鎸囧畾Like鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereLike($field, $condition, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'like', $condition); + return $this; + } + + /** + * 鎸囧畾Between鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereBetween($field, $condition, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'between', $condition); + return $this; + } + + /** + * 鎸囧畾NotBetween鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereNotBetween($field, $condition, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'not between', $condition); + return $this; + } + + /** + * 鎸囧畾Exp鏌ヨ鏉′欢 + * @access public + * @param mixed $field 鏌ヨ瀛楁 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @return $this + */ + public function whereExp($field, $condition, $logic = 'and') + { + $this->parseWhereExp($logic, $field, 'exp', $condition); + return $this; + } + + /** + * 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + * @access public + * @param string $logic 鏌ヨ閫昏緫 and or xor + * @param string|array|\Closure $field 鏌ヨ瀛楁 + * @param mixed $op 鏌ヨ琛ㄨ揪寮 + * @param mixed $condition 鏌ヨ鏉′欢 + * @param array $param 鏌ヨ鍙傛暟 + * @return void + */ + protected function parseWhereExp($logic, $field, $op, $condition, $param = []) + { + $logic = '$' . strtolower($logic); + if ($field instanceof \Closure) { + $this->options['where'][$logic][] = is_string($op) ? [$op, $field] : $field; + return; + } + $where = []; + if (is_null($op) && is_null($condition)) { + if (is_array($field)) { + // 鏁扮粍鎵归噺鏌ヨ + $where = $field; + } elseif ($field) { + // 瀛楃涓叉煡璇 + $where[] = ['exp', $field]; + } else { + $where = ''; + } + } elseif (is_array($op)) { + $where[$field] = $param; + } elseif (in_array(strtolower($op), ['null', 'notnull', 'not null'])) { + // null鏌ヨ + $where[$field] = [$op, '']; + } elseif (is_null($condition)) { + // 瀛楁鐩哥瓑鏌ヨ + $where[$field] = ['=', $op]; + } else { + $where[$field] = [$op, $condition]; + } + + if (!empty($where)) { + if (!isset($this->options['where'][$logic])) { + $this->options['where'][$logic] = []; + } + $this->options['where'][$logic] = array_merge($this->options['where'][$logic], $where); + } + } + + /** + * 鏌ヨ鏃ユ湡鎴栬呮椂闂 + * @access public + * @param string $field 鏃ユ湡瀛楁鍚 + * @param string $op 姣旇緝杩愮畻绗︽垨鑰呰〃杈惧紡 + * @param string|array $range 姣旇緝鑼冨洿 + * @return $this + */ + public function whereTime($field, $op, $range = null) + { + if (is_null($range)) { + // 浣跨敤鏃ユ湡琛ㄨ揪寮 + $date = getdate(); + switch (strtolower($op)) { + case 'today': + case 'd': + $range = ['today', 'tomorrow']; + break; + case 'week': + case 'w': + $range = 'this week 00:00:00'; + break; + case 'month': + case 'm': + $range = mktime(0, 0, 0, $date['mon'], 1, $date['year']); + break; + case 'year': + case 'y': + $range = mktime(0, 0, 0, 1, 1, $date['year']); + break; + case 'yesterday': + $range = ['yesterday', 'today']; + break; + case 'last week': + $range = ['last week 00:00:00', 'this week 00:00:00']; + break; + case 'last month': + $range = [date('y-m-01', strtotime('-1 month')), mktime(0, 0, 0, $date['mon'], 1, $date['year'])]; + break; + case 'last year': + $range = [mktime(0, 0, 0, 1, 1, $date['year'] - 1), mktime(0, 0, 0, 1, 1, $date['year'])]; + break; + default: + $range = $op; + } + $op = is_array($range) ? 'between' : '>'; + } + $this->where($field, strtolower($op) . ' time', $range); + return $this; + } + + /** + * 鍒嗛〉鏌ヨ + * @param int|null $listRows 姣忛〉鏁伴噺 + * @param bool $simple 绠娲佹ā寮 + * @param array $config 閰嶇疆鍙傛暟 + * page:褰撳墠椤, + * path:url璺緞, + * query:url棰濆鍙傛暟, + * fragment:url閿氱偣, + * var_page:鍒嗛〉鍙橀噺, + * list_rows:姣忛〉鏁伴噺 + * type:鍒嗛〉绫诲悕, + * namespace:鍒嗛〉绫诲懡鍚嶇┖闂 + * @return \think\Paginator + * @throws DbException + */ + public function paginate($listRows = null, $simple = false, $config = []) + { + $config = array_merge(Config::get('paginate'), $config); + $listRows = $listRows ?: $config['list_rows']; + $class = strpos($config['type'], '\\') ? $config['type'] : '\\think\\paginator\\driver\\' . ucwords($config['type']); + $page = isset($config['page']) ? (int) $config['page'] : call_user_func([ + $class, + 'getCurrentPage', + ], $config['var_page']); + + $page = $page < 1 ? 1 : $page; + + $config['path'] = isset($config['path']) ? $config['path'] : call_user_func([$class, 'getCurrentPath']); + + /** @var Paginator $paginator */ + if (!$simple) { + $options = $this->getOptions(); + $total = $this->count(); + $results = $this->options($options)->page($page, $listRows)->select(); + } else { + $results = $this->limit(($page - 1) * $listRows, $listRows + 1)->select(); + $total = null; + } + return $class::make($results, $listRows, $page, $total, $simple, $config); + } + + /** + * 鎸囧畾褰撳墠鎿嶄綔鐨勬暟鎹〃 + * @access public + * @param string $table 琛ㄥ悕 + * @return $this + */ + public function table($table) + { + $this->options['table'] = $table; + return $this; + } + + /** + * 鏌ヨ缂撳瓨 + * @access public + * @param mixed $key 缂撳瓨key + * @param integer $expire 缂撳瓨鏈夋晥鏈 + * @param string $tag 缂撳瓨鏍囩 + * @return $this + */ + public function cache($key = true, $expire = null, $tag = null) + { + // 澧炲姞蹇嵎璋冪敤鏂瑰紡 cache(10) 绛夊悓浜 cache(true, 10) + if (is_numeric($key) && is_null($expire)) { + $expire = $key; + $key = true; + } + if (false !== $key) { + $this->options['cache'] = ['key' => $key, 'expire' => $expire, 'tag' => $tag]; + } + return $this; + } + + /** + * 涓嶄富鍔ㄨ幏鍙栨暟鎹泦 + * @access public + * @param bool $cursor 鏄惁杩斿洖 Cursor 瀵硅薄 + * @return $this + */ + public function fetchCursor($cursor = true) + { + $this->options['fetch_cursor'] = $cursor; + return $this; + } + + /** + * 璁剧疆typeMap + * @access public + * @param string|array $typeMap + * @return $this + */ + public function typeMap($typeMap) + { + $this->options['typeMap'] = $typeMap; + return $this; + } + + /** + * 璁剧疆浠庝富鏈嶅姟鍣ㄨ鍙栨暟鎹 + * @access public + * @return $this + */ + public function master() + { + $this->options['master'] = true; + return $this; + } + + /** + * 璁剧疆鏌ヨ鏁版嵁涓嶅瓨鍦ㄦ槸鍚︽姏鍑哄紓甯 + * @access public + * @param bool $fail 鏄惁涓ユ牸妫鏌ュ瓧娈 + * @return $this + */ + public function failException($fail = true) + { + $this->options['fail'] = $fail; + return $this; + } + + /** + * awaitData + * @access public + * @param bool $awaitData + * @return $this + */ + public function awaitData($awaitData) + { + $this->options['awaitData'] = $awaitData; + return $this; + } + + /** + * batchSize + * @access public + * @param integer $batchSize + * @return $this + */ + public function batchSize($batchSize) + { + $this->options['batchSize'] = $batchSize; + return $this; + } + + /** + * exhaust + * @access public + * @param bool $exhaust + * @return $this + */ + public function exhaust($exhaust) + { + $this->options['exhaust'] = $exhaust; + return $this; + } + + /** + * 璁剧疆modifiers + * @access public + * @param array $modifiers + * @return $this + */ + public function modifiers($modifiers) + { + $this->options['modifiers'] = $modifiers; + return $this; + } + + /** + * 璁剧疆noCursorTimeout + * @access public + * @param bool $noCursorTimeout + * @return $this + */ + public function noCursorTimeout($noCursorTimeout) + { + $this->options['noCursorTimeout'] = $noCursorTimeout; + return $this; + } + + /** + * 璁剧疆oplogReplay + * @access public + * @param bool $oplogReplay + * @return $this + */ + public function oplogReplay($oplogReplay) + { + $this->options['oplogReplay'] = $oplogReplay; + return $this; + } + + /** + * 璁剧疆partial + * @access public + * @param bool $partial + * @return $this + */ + public function partial($partial) + { + $this->options['partial'] = $partial; + return $this; + } + + /** + * 鏌ヨ娉ㄩ噴 + * @access public + * @param string $comment 娉ㄩ噴 + * @return $this + */ + public function comment($comment) + { + $this->options['comment'] = $comment; + return $this; + } + + /** + * maxTimeMS + * @access public + * @param string $maxTimeMS + * @return $this + */ + public function maxTimeMS($maxTimeMS) + { + $this->options['maxTimeMS'] = $maxTimeMS; + return $this; + } + + /** + * 璁剧疆杩斿洖瀛楁 + * @access public + * @param array $field + * @param boolean $except 鏄惁鎺掗櫎 + * @return $this + */ + public function field($field, $except = false) + { + if (is_string($field)) { + $field = array_map('trim', explode(',', $field)); + } + $projection = []; + foreach ($field as $key => $val) { + if (is_numeric($key)) { + $projection[$val] = $except ? 0 : 1; + } else { + $projection[$key] = $val; + } + } + $this->options['projection'] = $projection; + return $this; + } + + /** + * 璁剧疆skip + * @access public + * @param integer $skip + * @return $this + */ + public function skip($skip) + { + $this->options['skip'] = $skip; + return $this; + } + + /** + * 璁剧疆slaveOk + * @access public + * @param bool $slaveOk + * @return $this + */ + public function slaveOk($slaveOk) + { + $this->options['slaveOk'] = $slaveOk; + return $this; + } + + /** + * 鍏宠仈棰勮浇鍏ユ煡璇 + * @access public + * @param mixed $with + * @return $this + */ + public function with($with) + { + $this->options['with'] = $with; + return $this; + } + + /** + * 鍏宠仈缁熻 + * @access public + * @param string|array $relation 鍏宠仈鏂规硶鍚 + * @return $this + */ + public function withCount($relation) + { + $this->options['with_count'] = $relation; + return $this; + } + + /** + * 鎸囧畾鏌ヨ鏁伴噺 + * @access public + * @param mixed $offset 璧峰浣嶇疆 + * @param mixed $length 鏌ヨ鏁伴噺 + * @return $this + */ + public function limit($offset, $length = null) + { + if (is_null($length)) { + if (is_numeric($offset)) { + $length = $offset; + $offset = 0; + } else { + list($offset, $length) = explode(',', $offset); + } + } + $this->options['skip'] = intval($offset); + $this->options['limit'] = intval($length); + + return $this; + } + + /** + * 鎸囧畾鍒嗛〉 + * @access public + * @param mixed $page 椤垫暟 + * @param mixed $listRows 姣忛〉鏁伴噺 + * @return $this + */ + public function page($page, $listRows = null) + { + if (is_null($listRows) && strpos($page, ',')) { + list($page, $listRows) = explode(',', $page); + } + $this->options['page'] = [intval($page), intval($listRows)]; + return $this; + } + + /** + * 璁剧疆sort + * @access public + * @param array|string|object $field + * @param string $order + * @return $this + */ + public function order($field, $order = '') + { + if (is_array($field)) { + $this->options['sort'] = $field; + } else { + $this->options['sort'][$field] = 'asc' == strtolower($order) ? 1 : -1; + } + return $this; + } + + /** + * 璁剧疆tailable + * @access public + * @param bool $tailable + * @return $this + */ + public function tailable($tailable) + { + $this->options['tailable'] = $tailable; + return $this; + } + + /** + * 璁剧疆writeConcern瀵硅薄 + * @access public + * @param WriteConcern $writeConcern + * @return $this + */ + public function writeConcern($writeConcern) + { + $this->options['writeConcern'] = $writeConcern; + return $this; + } + + /** + * 鑾峰彇褰撳墠鏁版嵁琛ㄧ殑涓婚敭 + * @access public + * @return string|array + */ + public function getPk() + { + return !empty($this->pk) ? $this->pk : $this->getConfig('pk'); + } + + /** + * 鏌ヨ鍙傛暟璧嬪 + * @access protected + * @param array $options 琛ㄨ揪寮忓弬鏁 + * @return $this + */ + protected function options(array $options) + { + $this->options = $options; + return $this; + } + + /** + * 鑾峰彇褰撳墠鐨勬煡璇㈠弬鏁 + * @access public + * @param string $name 鍙傛暟鍚 + * @return mixed + */ + public function getOptions($name = '') + { + return isset($this->options[$name]) ? $this->options[$name] : $this->options; + } + + /** + * 璁剧疆鍏宠仈鏌ヨ + * @access public + * @param string $relation 鍏宠仈鍚嶇О + * @return $this + */ + public function relation($relation) + { + $this->options['relation'] = $relation; + return $this; + } + + /** + * 鎶婁富閿艰浆鎹负鏌ヨ鏉′欢 鏀寔澶嶅悎涓婚敭 + * @access public + * @param array|string $data 涓婚敭鏁版嵁 + * @param mixed $options 琛ㄨ揪寮忓弬鏁 + * @return void + * @throws Exception + */ + protected function parsePkWhere($data, &$options) + { + $pk = $this->getPk(); + + if (is_string($pk)) { + // 鏍规嵁涓婚敭鏌ヨ + if (is_array($data)) { + $where[$pk] = isset($data[$pk]) ? $data[$pk] : ['in', $data]; + } else { + $where[$pk] = strpos($data, ',') ? ['in', $data] : $data; + } + } + + if (!empty($where)) { + if (isset($options['where']['$and'])) { + $options['where']['$and'] = array_merge($options['where']['$and'], $where); + } else { + $options['where']['$and'] = $where; + } + } + return; + } + + /** + * 鎻掑叆璁板綍 + * @access public + * @param mixed $data 鏁版嵁 + * @param boolean $replace 鏄惁replace锛堢洰鍓嶆棤鏁堬級 + * @param boolean $getLastInsID 杩斿洖鑷涓婚敭 + * @return WriteResult + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function insert(array $data, $replace = null, $getLastInsID = false) + { + if (empty($data)) { + throw new Exception('miss data to insert'); + } + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + $data = array_merge($options['data'], $data); + // 鐢熸垚bulk瀵硅薄 + $bulk = $this->builder->insert($data, $options); + $writeConcern = isset($options['writeConcern']) ? $options['writeConcern'] : null; + $writeResult = $this->execute($options['table'], $bulk, $writeConcern); + $result = $writeResult->getInsertedCount(); + if ($result) { + $lastInsId = $this->getLastInsID(); + if ($lastInsId) { + $pk = $this->getPk(); + $data[$pk] = $lastInsId; + } + $options['data'] = $data; + $this->trigger('after_insert', $options); + + if ($getLastInsID) { + return $lastInsId; + } + } + return $result; + } + + /** + * 鎻掑叆璁板綍骞惰幏鍙栬嚜澧濱D + * @access public + * @param mixed $data 鏁版嵁 + * @return integer + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function insertGetId(array $data) + { + return $this->insert($data, null, true); + } + + /** + * 鎵归噺鎻掑叆璁板綍 + * @access public + * @param mixed $dataSet 鏁版嵁闆 + * @return integer + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function insertAll(array $dataSet) + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + if (!is_array(reset($dataSet))) { + return false; + } + + // 鐢熸垚bulkWrite瀵硅薄 + $bulk = $this->builder->insertAll($dataSet, $options); + $writeConcern = isset($options['writeConcern']) ? $options['writeConcern'] : null; + $writeResult = $this->execute($options['table'], $bulk, $writeConcern); + return $writeResult->getInsertedCount(); + } + + /** + * 鏇存柊璁板綍 + * @access public + * @param mixed $data 鏁版嵁 + * @return int + * @throws Exception + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function update(array $data) + { + $options = $this->parseExpress(); + $data = array_merge($options['data'], $data); + if (isset($options['cache']) && is_string($options['cache']['key'])) { + $key = $options['cache']['key']; + } + $pk = $this->getPk(); + if (empty($options['where'])) { + // 濡傛灉瀛樺湪涓婚敭鏁版嵁 鍒欒嚜鍔ㄤ綔涓烘洿鏂版潯浠 + if (is_string($pk) && isset($data[$pk])) { + $where[$pk] = $data[$pk]; + $key = 'mongo:' . $options['table'] . '|' . $data[$pk]; + unset($data[$pk]); + } elseif (is_array($pk)) { + // 澧炲姞澶嶅悎涓婚敭鏀寔 + foreach ($pk as $field) { + if (isset($data[$field])) { + $where[$field] = $data[$field]; + } else { + // 濡傛灉缂哄皯澶嶅悎涓婚敭鏁版嵁鍒欎笉鎵ц + throw new Exception('miss complex primary data'); + } + unset($data[$field]); + } + } + if (!isset($where)) { + // 濡傛灉娌℃湁浠讳綍鏇存柊鏉′欢鍒欎笉鎵ц + throw new Exception('miss update condition'); + } else { + $options['where']['$and'] = $where; + } + } elseif (!isset($key) && is_string($pk) && isset($options['where']['$and'][$pk])) { + $key = $this->getCacheKey($options['where']['$and'][$pk], $options); + } + + // 鐢熸垚bulkWrite瀵硅薄 + $bulk = $this->builder->update($data, $options); + $writeConcern = isset($options['writeConcern']) ? $options['writeConcern'] : null; + $writeResult = $this->execute($options['table'], $bulk, $writeConcern); + // 妫娴嬬紦瀛 + if (isset($key) && Cache::get($key)) { + // 鍒犻櫎缂撳瓨 + Cache::rm($key); + } + $result = $writeResult->getModifiedCount(); + if ($result) { + if (isset($where[$pk])) { + $data[$pk] = $where[$pk]; + } elseif (is_string($pk) && isset($key) && strpos($key, '|')) { + list($a, $val) = explode('|', $key); + $data[$pk] = $val; + } + $options['data'] = $data; + $this->trigger('after_update', $options); + } + return $result; + } + + /** + * 鍒犻櫎璁板綍 + * @access public + * @param array $data 琛ㄨ揪寮 true 琛ㄧず寮哄埗鍒犻櫎 + * @return int + * @throws Exception + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + * @throws BulkWriteException + */ + public function delete($data = null) + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + $pk = $this->getPk(); + if (!is_null($data) && true !== $data) { + if (!is_array($data)) { + // 缂撳瓨鏍囪瘑 + $key = 'mongo:' . $options['table'] . '|' . $data; + } + // AR妯″紡鍒嗘瀽涓婚敭鏉′欢 + $this->parsePkWhere($data, $options); + } elseif (!isset($key) && is_string($pk) && isset($options['where']['$and'][$pk])) { + $key = $this->getCacheKey($options['where']['$and'][$pk], $options); + } + + if (true !== $data && empty($options['where'])) { + // 濡傛灉涓嶆槸寮哄埗鍒犻櫎涓旀潯浠朵负绌 涓嶈繘琛屽垹闄ゆ搷浣 + throw new Exception('delete without condition'); + } + + // 鐢熸垚bulkWrite瀵硅薄 + $bulk = $this->builder->delete($options); + $writeConcern = isset($options['writeConcern']) ? $options['writeConcern'] : null; + // 鎵ц鎿嶄綔 + $writeResult = $this->execute($options['table'], $bulk, $writeConcern); + // 妫娴嬬紦瀛 + if (isset($key) && Cache::get($key)) { + // 鍒犻櫎缂撳瓨 + Cache::rm($key); + } + $result = $writeResult->getDeletedCount(); + if ($result) { + if (!is_array($data) && is_string($pk) && isset($key) && strpos($key, '|')) { + list($a, $val) = explode('|', $key); + $item[$pk] = $val; + $data = $item; + } + $options['data'] = $data; + $this->trigger('after_delete', $options); + } + return $result; + } + + /** + * 鎵ц鏌ヨ浣嗗彧杩斿洖Cursor瀵硅薄 + * @access public + * @return Cursor + */ + public function getCursor() + { + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + // 鐢熸垚MongoQuery瀵硅薄 + $query = $this->builder->select($options); + // 鎵ц鏌ヨ鎿嶄綔 + $readPreference = isset($options['readPreference']) ? $options['readPreference'] : null; + return $this->query($options['table'], $query, $readPreference, true, $options['typeMap']); + } + + /** + * 鏌ユ壘璁板綍 + * @access public + * @param array|string|Query|\Closure $data + * @return Collection|false|Cursor|string + * @throws ModelNotFoundException + * @throws DataNotFoundException + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function select($data = null) + { + if ($data instanceof Query) { + return $data->select(); + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [ & $this]); + $data = null; + } + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + + if (!is_null($data)) { + // 涓婚敭鏉′欢鍒嗘瀽 + $this->parsePkWhere($data, $options); + } + + $resultSet = false; + if (!empty($options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $options['cache']; + $key = is_string($cache['key']) ? $cache['key'] : md5(serialize($options)); + $resultSet = Cache::get($key); + } + if (!$resultSet) { + // 鐢熸垚MongoQuery瀵硅薄 + $query = $this->builder->select($options); + + $options['data'] = $data; + if ($resultSet = $this->trigger('before_select', $options)) { + } else { + // 鎵ц鏌ヨ鎿嶄綔 + $readPreference = isset($options['readPreference']) ? $options['readPreference'] : null; + $resultSet = $this->query($options['table'], $query, $readPreference, $options['fetch_cursor'], $options['typeMap']); + + if ($resultSet instanceof Cursor) { + // 杩斿洖MongoDB\Driver\Cursor瀵硅薄 + return $resultSet; + } + } + if (isset($cache)) { + // 缂撳瓨鏁版嵁闆 + $this->cacheData($key, $resultSet, $cache); + } + } + + // 鏁版嵁鍒楄〃璇诲彇鍚庣殑澶勭悊 + if (!empty($this->model)) { + // 鐢熸垚妯″瀷瀵硅薄 + $modelName = $this->model; + if (count($resultSet) > 0) { + foreach ($resultSet as $key => $result) { + /** @var Model $result */ + $model = new $modelName($result); + $model->isUpdate(true); + + // 鍏宠仈鏌ヨ + if (!empty($options['relation'])) { + $model->relationQuery($options['relation']); + } + // 鍏宠仈缁熻 + if (!empty($options['with_count'])) { + $model->relationCount($model, $options['with_count']); + } + $resultSet[$key] = $model; + } + if (!empty($options['with'])) { + // 棰勮浇鍏 + $model->eagerlyResultSet($resultSet, $options['with']); + } + // 妯″瀷鏁版嵁闆嗚浆鎹 + $resultSet = $model->toCollection($resultSet); + } else { + $resultSet = (new $modelName)->toCollection($resultSet); + } + } elseif ('collection' == $this->connection->getConfig('resultset_type')) { + // 杩斿洖Collection瀵硅薄 + $resultSet = new Collection($resultSet); + } + if (!empty($options['fail']) && count($resultSet) == 0) { + $this->throwNotFound($options); + } + return $resultSet; + } + + /** + * 缂撳瓨鏁版嵁 + * @access public + * @param string $key 缂撳瓨鏍囪瘑 + * @param mixed $data 缂撳瓨鏁版嵁 + * @param array $config 缂撳瓨鍙傛暟 + */ + protected function cacheData($key, $data, $config = []) + { + if (isset($config['tag'])) { + Cache::tag($config['tag'])->set($key, $data, $config['expire']); + } else { + Cache::set($key, $data, $config['expire']); + } + } + + /** + * 鐢熸垚缂撳瓨鏍囪瘑 + * @access public + * @param mixed $value 缂撳瓨鏁版嵁 + * @param array $options 缂撳瓨鍙傛暟 + */ + protected function getCacheKey($value, $options) + { + if (is_scalar($value)) { + $data = $value; + } elseif (is_array($value) && '=' == $value[0]) { + $data = $value[1]; + } + if (isset($data)) { + return 'mongo:' . $options['table'] . '|' . $data; + } + } + + /** + * 鏌ユ壘鍗曟潯璁板綍 + * @access public + * @param array|string|Query|\Closure $data + * @return array|false|Cursor|string|Model + * @throws ModelNotFoundException + * @throws DataNotFoundException + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function find($data = null) + { + if ($data instanceof Query) { + return $data->find(); + } elseif ($data instanceof \Closure) { + call_user_func_array($data, [ & $this]); + $data = null; + } + // 鍒嗘瀽鏌ヨ琛ㄨ揪寮 + $options = $this->parseExpress(); + $pk = $this->getPk(); + if (!is_null($data)) { + // AR妯″紡鍒嗘瀽涓婚敭鏉′欢 + $this->parsePkWhere($data, $options); + } elseif (!empty($options['cache']) && true === $options['cache']['key'] && is_string($pk) && isset($options['where']['$and'][$pk])) { + $key = $this->getCacheKey($options['where']['$and'][$pk], $options); + } + + $options['limit'] = 1; + $result = false; + if (!empty($options['cache'])) { + // 鍒ゆ柇鏌ヨ缂撳瓨 + $cache = $options['cache']; + if (true === $cache['key'] && !is_null($data) && !is_array($data)) { + $key = 'mongo:' . $options['table'] . '|' . $data; + } elseif (!isset($key)) { + $key = is_string($cache['key']) ? $cache['key'] : md5(serialize($options)); + } + $result = Cache::get($key); + } + if (false === $result) { + // 鐢熸垚鏌ヨSQL + $query = $this->builder->select($options); + if (is_string($pk)) { + if (!is_array($data)) { + if (isset($key) && strpos($key, '|')) { + list($a, $val) = explode('|', $key); + $item[$pk] = $val; + } else { + $item[$pk] = $data; + } + $data = $item; + } + } + $options['data'] = $data; + // 浜嬩欢鍥炶皟 + if ($result = $this->trigger('before_find', $options)) { + } else { + // 鎵ц鏌ヨ + $readPreference = isset($options['readPreference']) ? $options['readPreference'] : null; + $resultSet = $this->query($options['table'], $query, $readPreference, $options['fetch_cursor'], $options['typeMap']); + + if ($resultSet instanceof Cursor) { + // 杩斿洖MongoDB\Driver\Cursor瀵硅薄 + return $resultSet; + } + $result = isset($resultSet[0]) ? $resultSet[0] : null; + } + if (isset($cache)) { + // 缂撳瓨鏁版嵁 + $this->cacheData($key, $result, $cache); + } + } + + // 鏁版嵁澶勭悊 + if (!empty($result)) { + if (!empty($this->model)) { + // 杩斿洖妯″瀷瀵硅薄 + $model = $this->model; + $result = new $model($result); + $result->isUpdate(true, isset($options['where']['$and']) ? $options['where']['$and'] : null); + // 鍏宠仈鏌ヨ + if (!empty($options['relation'])) { + $result->relationQuery($options['relation']); + } + if (!empty($options['with'])) { + // 棰勮浇鍏 + $result->eagerlyResult($result, $options['with']); + } + // 鍏宠仈缁熻 + if (!empty($options['with_count'])) { + $result->relationCount($result, $options['with_count']); + } + } + } elseif (!empty($options['fail'])) { + $this->throwNotFound($options); + } + return $result; + } + + /** + * 鏌ヨ澶辫触 鎶涘嚭寮傚父 + * @access public + * @param array $options 鏌ヨ鍙傛暟 + * @throws ModelNotFoundException + * @throws DataNotFoundException + */ + protected function throwNotFound($options = []) + { + if (!empty($this->model)) { + throw new ModelNotFoundException('model data Not Found:' . $this->model, $this->model, $options); + } else { + throw new DataNotFoundException('table data not Found:' . $options['table'], $options['table'], $options); + } + } + + /** + * 鏌ユ壘澶氭潯璁板綍 濡傛灉涓嶅瓨鍦ㄥ垯鎶涘嚭寮傚父 + * @access public + * @param array|string|Query|\Closure $data + * @return array|\PDOStatement|string|Model + * @throws ModelNotFoundException + * @throws DataNotFoundException + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function selectOrFail($data = null) + { + return $this->failException(true)->select($data); + } + + /** + * 鏌ユ壘鍗曟潯璁板綍 濡傛灉涓嶅瓨鍦ㄥ垯鎶涘嚭寮傚父 + * @access public + * @param array|string|Query|\Closure $data + * @return array|\PDOStatement|string|Model + * @throws ModelNotFoundException + * @throws DataNotFoundException + * @throws AuthenticationException + * @throws InvalidArgumentException + * @throws ConnectionException + * @throws RuntimeException + */ + public function findOrFail($data = null) + { + return $this->failException(true)->find($data); + } + + /** + * 鍒嗘壒鏁版嵁杩斿洖澶勭悊 + * @access public + * @param integer $count 姣忔澶勭悊鐨勬暟鎹暟閲 + * @param callable $callback 澶勭悊鍥炶皟鏂规硶 + * @param string $column 鍒嗘壒澶勭悊鐨勫瓧娈靛悕 + * @return boolean + */ + public function chunk($count, $callback, $column = null) + { + $column = $column ?: $this->getPk(); + $options = $this->getOptions(); + $resultSet = $this->limit($count)->order($column, 'asc')->select(); + + while (!empty($resultSet)) { + if (false === call_user_func($callback, $resultSet)) { + return false; + } + $end = end($resultSet); + $lastId = is_array($end) ? $end[$column] : $end->$column; + $resultSet = $this->options($options) + ->limit($count) + ->where($column, '>', $lastId) + ->order($column, 'asc') + ->select(); + } + return true; + } + + /** + * 鑾峰彇鏁版嵁琛ㄤ俊鎭 + * @access public + * @param string $tableName 鏁版嵁琛ㄥ悕 鐣欑┖鑷姩鑾峰彇 + * @param string $fetch 鑾峰彇淇℃伅绫诲瀷 鍖呮嫭 fields type pk + * @return mixed + */ + public function getTableInfo($tableName = '', $fetch = '') + { + if (!$tableName) { + $tableName = $this->getTable(); + } + if (is_array($tableName)) { + $tableName = key($tableName) ?: current($tableName); + } + + if (strpos($tableName, ',')) { + // 澶氳〃涓嶈幏鍙栧瓧娈典俊鎭 + return false; + } else { + $tableName = $this->parseSqlTable($tableName); + } + + $guid = md5($tableName); + if (!isset(self::$info[$guid])) { + $result = $this->table($tableName)->find(); + if ($result instanceof Model) { + $result = $result->toArray(); + } + $fields = array_keys($result); + $type = []; + foreach ($result as $key => $val) { + // 璁板綍瀛楁绫诲瀷 + $type[$key] = getType($val); + if ('_id' == $key) { + $pk = $key; + } + } + if (!isset($pk)) { + // 璁剧疆涓婚敭 + $pk = null; + } + $result = ['fields' => $fields, 'type' => $type, 'pk' => $pk]; + self::$info[$guid] = $result; + } + return $fetch ? self::$info[$guid][$fetch] : self::$info[$guid]; + } + + /** + * 鍒嗘瀽琛ㄨ揪寮忥紙鍙敤浜庢煡璇㈡垨鑰呭啓鍏ユ搷浣滐級 + * @access protected + * @return array + */ + protected function parseExpress() + { + $options = $this->options; + + // 鑾峰彇鏁版嵁琛 + if (empty($options['table'])) { + $options['table'] = $this->getTable(); + } + + foreach (['where', 'data'] as $name) { + if (!isset($options[$name])) { + $options[$name] = []; + } + } + + $modifiers = empty($options['modifiers']) ? [] : $options['modifiers']; + if (isset($options['comment'])) { + $modifiers['$comment'] = $options['comment']; + } + + if (isset($options['maxTimeMS'])) { + $modifiers['$maxTimeMS'] = $options['maxTimeMS']; + } + + if (!empty($modifiers)) { + $options['modifiers'] = $modifiers; + } + + if (!isset($options['projection']) || '*' == $options['projection']) { + $options['projection'] = []; + } + + if (!isset($options['typeMap'])) { + $options['typeMap'] = $this->getConfig('type_map'); + } + + if (!isset($options['limit'])) { + $options['limit'] = 0; + } + + foreach (['master', 'fetch_cursor'] as $name) { + if (!isset($options[$name])) { + $options[$name] = false; + } + } + + if (isset($options['page'])) { + // 鏍规嵁椤垫暟璁$畻limit + list($page, $listRows) = $options['page']; + $page = $page > 0 ? $page : 1; + $listRows = $listRows > 0 ? $listRows : (is_numeric($options['limit']) ? $options['limit'] : 20); + $offset = $listRows * ($page - 1); + $options['skip'] = intval($offset); + $options['limit'] = intval($listRows); + } + + $this->options = []; + return $options; + } + + /** + * 娉ㄥ唽鍥炶皟鏂规硶 + * @access public + * @param string $event 浜嬩欢鍚 + * @param callable $callback 鍥炶皟鏂规硶 + * @return void + */ + public static function event($event, $callback) + { + self::$event[$event] = $callback; + } + + /** + * 瑙﹀彂浜嬩欢 + * @access protected + * @param string $event 浜嬩欢鍚 + * @param mixed $params 棰濆鍙傛暟 + * @return bool + */ + protected function trigger($event, $params = []) + { + $result = false; + if (isset(self::$event[$event])) { + $callback = self::$event[$event]; + $result = call_user_func_array($callback, [$params, $this]); + } + return $result; + } + +} diff --git a/vendor/topthink/think-queue/.gitignore b/vendor/topthink/think-queue/.gitignore new file mode 100644 index 000000000..4aec78217 --- /dev/null +++ b/vendor/topthink/think-queue/.gitignore @@ -0,0 +1,4 @@ +/vendor/ +/.idea/ +/composer.lock +/thinkphp/ diff --git a/vendor/topthink/think-queue/LICENSE b/vendor/topthink/think-queue/LICENSE new file mode 100644 index 000000000..8dada3eda --- /dev/null +++ b/vendor/topthink/think-queue/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/topthink/think-queue/README.md b/vendor/topthink/think-queue/README.md new file mode 100644 index 000000000..14ad46166 --- /dev/null +++ b/vendor/topthink/think-queue/README.md @@ -0,0 +1,132 @@ +# think-queue + +## 瀹夎 +> composer require topthink/think-queue + +## 閰嶇疆 +> 閰嶇疆鏂囦欢浣嶄簬 `application/extra/queue.php` +### 鍏叡閰嶇疆 + +``` +[ + 'connector'=>'sync' //椹卞姩绫诲瀷锛屽彲閫夋嫨 sync(榛樿):鍚屾鎵ц锛宒atabase:鏁版嵁搴撻┍鍔,redis:Redis椹卞姩,topthink:Topthink椹卞姩 + //鎴栧叾浠栬嚜瀹氫箟鐨勫畬鏁寸殑绫诲悕 +] +``` + +### 椹卞姩閰嶇疆 +> 鍚勪釜椹卞姩鐨勫叿浣撳彲鐢ㄩ厤缃」鍦╜think\queue\connector`鐩綍涓嬪悇涓┍鍔ㄧ被閲岀殑`options`灞炴т腑锛屽啓鍦ㄤ笂闈㈢殑`queue`閰嶇疆閲屽嵆鍙鐩 + + +## 浣跨敤 Database +> 鍒涘缓濡備笅鏁版嵁琛 + +``` +CREATE TABLE `prefix_jobs` ( + `id` int(11) NOT NULL AUTO_INCREMENT, + `queue` varchar(255) NOT NULL, + `payload` longtext NOT NULL, + `attempts` tinyint(3) unsigned NOT NULL, + `reserved` tinyint(3) unsigned NOT NULL, + `reserved_at` int(10) unsigned DEFAULT NULL, + `available_at` int(10) unsigned NOT NULL, + `created_at` int(10) unsigned NOT NULL, + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8; +``` + +## 鍒涘缓浠诲姟绫 +> 鍗曟ā鍧楅」鐩帹鑽愪娇鐢 `app\job` 浣滀负浠诲姟绫荤殑鍛藉悕绌洪棿 +> 澶氭ā鍧楅」鐩彲鐢ㄤ娇鐢 `app\module\job` 浣滀负浠诲姟绫荤殑鍛藉悕绌洪棿 +> 涔熷彲浠ユ斁鍦ㄤ换鎰忓彲浠ヨ嚜鍔ㄥ姞杞藉埌鐨勫湴鏂 + +浠诲姟绫讳笉闇缁ф壙浠讳綍绫伙紝濡傛灉杩欎釜绫诲彧鏈変竴涓换鍔★紝閭d箞灏卞彧闇瑕佹彁渚涗竴涓猔fire`鏂规硶灏卞彲浠ヤ簡锛屽鏋滄湁澶氫釜灏忎换鍔★紝灏卞啓澶氫釜鏂规硶锛屼笅闈㈠彂甯冧换鍔$殑鏃跺欎細鏈夊尯鍒 +姣忎釜鏂规硶浼氫紶鍏ヤ袱涓弬鏁 `think\queue\Job $job`锛堝綋鍓嶇殑浠诲姟瀵硅薄锛 鍜 `$data`锛堝彂甯冧换鍔℃椂鑷畾涔夌殑鏁版嵁锛 + +杩樻湁涓彲閫夌殑浠诲姟澶辫触鎵ц鐨勬柟娉 `failed` 浼犲叆鐨勫弬鏁颁负`$data`锛堝彂甯冧换鍔℃椂鑷畾涔夌殑鏁版嵁锛 + +### 涓嬮潰鍐欎袱涓緥瀛 + +``` +namespace app\job; + +use think\queue\Job; + +class Job1{ + + public function fire(Job $job, $data){ + + //....杩欓噷鎵ц鍏蜂綋鐨勪换鍔 + + if ($job->attempts() > 3) { + //閫氳繃杩欎釜鏂规硶鍙互妫鏌ヨ繖涓换鍔″凡缁忛噸璇曚簡鍑犳浜 + } + + + //濡傛灉浠诲姟鎵ц鎴愬姛鍚 璁板緱鍒犻櫎浠诲姟锛屼笉鐒惰繖涓换鍔′細閲嶅鎵ц锛岀洿鍒拌揪鍒版渶澶ч噸璇曟鏁板悗澶辫触鍚庯紝鎵цfailed鏂规硶 + $job->delete(); + + // 涔熷彲浠ラ噸鏂板彂甯冭繖涓换鍔 + $job->release($delay); //$delay涓哄欢杩熸椂闂 + + } + + public function failed($data){ + + // ...浠诲姟杈惧埌鏈澶ч噸璇曟鏁板悗锛屽け璐ヤ簡 + } + +} + +``` + +``` + +namespace app\lib\job; + +use think\queue\Job; + +class Job2{ + + public function task1(Job $job, $data){ + + + } + + public function task2(Job $job, $data){ + + + } + + public function failed($data){ + + + } + +} + +``` + + +## 鍙戝竷浠诲姟 +> `think\Queue:push($job, $data = '', $queue = null)` 鍜 `think\Queue::later($delay, $job, $data = '', $queue = null)` 涓や釜鏂规硶锛屽墠鑰呮槸绔嬪嵆鎵ц锛屽悗鑰呮槸鍦╜$delay`绉掑悗鎵ц + +`$job` 鏄换鍔″悕 +鍗曟ā鍧楃殑锛屼笖鍛藉悕绌洪棿鏄痐app\job`鐨勶紝姣斿涓婇潰鐨勪緥瀛愪竴,鍐檂Job1`绫诲悕鍗冲彲 +澶氭ā鍧楃殑锛屼笖鍛藉悕绌洪棿鏄痐app\module\job`鐨勶紝鍐檂model/Job1`鍗冲彲 +鍏朵粬鐨勯渶瑕佷簺瀹屾暣鐨勭被鍚嶏紝姣斿涓婇潰鐨勪緥瀛愪簩锛岄渶瑕佸啓瀹屾暣鐨勭被鍚峘app\lib\job\Job2` +濡傛灉涓涓换鍔$被閲屾湁澶氫釜灏忎换鍔$殑璇濓紝濡備笂闈㈢殑渚嬪瓙浜岋紝闇瑕佺敤@+鏂规硶鍚峘app\lib\job\Job2@task1`銆乣app\lib\job\Job2@task2` + +`$data` 鏄綘瑕佷紶鍒颁换鍔¢噷鐨勫弬鏁 + +`$queue` 闃熷垪鍚嶏紝鎸囧畾杩欎釜浠诲姟鏄湪鍝釜闃熷垪涓婃墽琛岋紝鍚屼笅闈㈢洃鎺ч槦鍒楃殑鏃跺欐寚瀹氱殑闃熷垪鍚,鍙笉濉 + +## 鐩戝惉浠诲姟骞舵墽琛 + +> php think queue:listen + +> php think queue:work --daemon锛堜笉鍔--daemon涓烘墽琛屽崟涓换鍔★級 + +涓ょ锛屽叿浣撶殑鍙夊弬鏁板彲浠ヨ緭鍏ュ懡浠ゅ姞 --help 鏌ョ湅 + +>鍙厤鍚坰upervisor浣跨敤锛屼繚璇佽繘绋嬪父椹 \ No newline at end of file diff --git a/vendor/topthink/think-queue/composer.json b/vendor/topthink/think-queue/composer.json new file mode 100644 index 000000000..dd16bc396 --- /dev/null +++ b/vendor/topthink/think-queue/composer.json @@ -0,0 +1,29 @@ +{ + "name": "topthink/think-queue", + "description": "The ThinkPHP5 Queue Package", + "type": "think-extend", + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ], + "license": "Apache-2.0", + "autoload": { + "psr-4": { + "think\\": "src" + }, + "files": [ + "src/common.php" + ] + }, + "require": { + "topthink/think-helper": ">=1.0.4", + "topthink/think-installer": ">=1.0.10" + }, + "extra": { + "think-config": { + "queue": "src/config.php" + } + } +} diff --git a/vendor/topthink/think-queue/src/Queue.php b/vendor/topthink/think-queue/src/Queue.php new file mode 100644 index 000000000..10ff604bd --- /dev/null +++ b/vendor/topthink/think-queue/src/Queue.php @@ -0,0 +1,49 @@ + +// +---------------------------------------------------------------------- + +namespace think; + +use think\helper\Str; +use think\queue\Connector; + +/** + * Class Queue + * @package think\queue + * + * @method static push($job, $data = '', $queue = null) + * @method static later($delay, $job, $data = '', $queue = null) + * @method static pop($queue = null) + * @method static marshal() + */ +class Queue +{ + /** @var Connector */ + protected static $connector; + + private static function buildConnector() + { + $options = Config::get('queue'); + $type = !empty($options['connector']) ? $options['connector'] : 'Sync'; + + if (!isset(self::$connector)) { + + $class = false !== strpos($type, '\\') ? $type : '\\think\\queue\\connector\\' . Str::studly($type); + + self::$connector = new $class($options); + } + return self::$connector; + } + + public static function __callStatic($name, $arguments) + { + return call_user_func_array([self::buildConnector(), $name], $arguments); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/common.php b/vendor/topthink/think-queue/src/common.php new file mode 100644 index 000000000..6504ec917 --- /dev/null +++ b/vendor/topthink/think-queue/src/common.php @@ -0,0 +1,36 @@ + +// +---------------------------------------------------------------------- + +\think\Console::addDefaultCommands([ + "think\\queue\\command\\Work", + "think\\queue\\command\\Restart", + "think\\queue\\command\\Listen", + "think\\queue\\command\\Subscribe" +]); + +if (!function_exists('queue')) { + + /** + * 娣诲姞鍒伴槦鍒 + * @param $job + * @param string $data + * @param int $delay + * @param null $queue + */ + function queue($job, $data = '', $delay = 0, $queue = null) + { + if ($delay > 0) { + \think\Queue::later($delay, $job, $data, $queue); + } else { + \think\Queue::push($job, $data, $queue); + } + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/config.php b/vendor/topthink/think-queue/src/config.php new file mode 100644 index 000000000..41fd544f4 --- /dev/null +++ b/vendor/topthink/think-queue/src/config.php @@ -0,0 +1,14 @@ + +// +---------------------------------------------------------------------- + +return [ + 'connector' => 'Sync' +]; \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/CallQueuedHandler.php b/vendor/topthink/think-queue/src/queue/CallQueuedHandler.php new file mode 100644 index 000000000..101e3bdfb --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/CallQueuedHandler.php @@ -0,0 +1,36 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + +class CallQueuedHandler +{ + + public function call(Job $job, array $data) + { + $command = unserialize($data['command']); + + call_user_func([$command, 'handle']); + + if (!$job->isDeletedOrReleased()) { + $job->delete(); + } + } + + public function failed(array $data, $e) + { + $command = unserialize($data['command']); + + if (method_exists($command, 'failed')) { + $command->failed($e); + } + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/Connector.php b/vendor/topthink/think-queue/src/queue/Connector.php new file mode 100644 index 000000000..429575899 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/Connector.php @@ -0,0 +1,69 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + +use InvalidArgumentException; + +abstract class Connector +{ + protected $options = []; + + abstract function push($job, $data = '', $queue = null); + + abstract function later($delay, $job, $data = '', $queue = null); + + abstract public function pop($queue = null); + + public function marshal() + { + throw new \RuntimeException('pop queues not support for this type'); + } + + protected function createPayload($job, $data = '', $queue = null) + { + if (is_object($job)) { + $payload = json_encode([ + 'job' => 'think\queue\CallQueuedHandler@call', + 'data' => [ + 'commandName' => get_class($job), + 'command' => serialize(clone $job), + ], + ]); + } else { + $payload = json_encode($this->createPlainPayload($job, $data)); + } + + if (JSON_ERROR_NONE !== json_last_error()) { + throw new InvalidArgumentException('Unable to create payload: ' . json_last_error_msg()); + } + + return $payload; + } + + protected function createPlainPayload($job, $data) + { + return ['job' => $job, 'data' => $data]; + } + + protected function setMeta($payload, $key, $value) + { + $payload = json_decode($payload, true); + $payload[$key] = $value; + $payload = json_encode($payload); + + if (JSON_ERROR_NONE !== json_last_error()) { + throw new InvalidArgumentException('Unable to create payload: ' . json_last_error_msg()); + } + + return $payload; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/Job.php b/vendor/topthink/think-queue/src/queue/Job.php new file mode 100644 index 000000000..9d15a635c --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/Job.php @@ -0,0 +1,213 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + +use DateTime; +use think\Config; + +abstract class Job +{ + + /** + * The job handler instance. + * @var mixed + */ + protected $instance; + + /** + * The name of the queue the job belongs to. + * @var string + */ + protected $queue; + + /** + * Indicates if the job has been deleted. + * @var bool + */ + protected $deleted = false; + + /** + * Indicates if the job has been released. + * @var bool + */ + protected $released = false; + + /** + * Fire the job. + * @return void + */ + abstract public function fire(); + + /** + * Delete the job from the queue. + * @return void + */ + public function delete() + { + $this->deleted = true; + } + + /** + * Determine if the job has been deleted. + * @return bool + */ + public function isDeleted() + { + return $this->deleted; + } + + /** + * Release the job back into the queue. + * @param int $delay + * @return void + */ + public function release($delay = 0) + { + $this->released = true; + } + + /** + * Determine if the job was released back into the queue. + * @return bool + */ + public function isReleased() + { + return $this->released; + } + + /** + * Determine if the job has been deleted or released. + * @return bool + */ + public function isDeletedOrReleased() + { + return $this->isDeleted() || $this->isReleased(); + } + + /** + * Get the number of times the job has been attempted. + * @return int + */ + abstract public function attempts(); + + /** + * Get the raw body string for the job. + * @return string + */ + abstract public function getRawBody(); + + /** + * Resolve and fire the job handler method. + * @param array $payload + * @return void + */ + protected function resolveAndFire(array $payload) + { + list($class, $method) = $this->parseJob($payload['job']); + + $this->instance = $this->resolve($class); + if ($this->instance) { + $this->instance->{$method}($this, $payload['data']); + } + } + + /** + * Parse the job declaration into class and method. + * @param string $job + * @return array + */ + protected function parseJob($job) + { + $segments = explode('@', $job); + + return count($segments) > 1 ? $segments : [$segments[0], 'fire']; + } + + /** + * Resolve the given job handler. + * @param string $name + * @return mixed + */ + protected function resolve($name) + { + if (strpos($name, '\\') === false) { + + if (strpos($name, '/') === false) { + $module = ''; + } else { + list($module, $name) = explode('/', $name, 2); + } + + $name = Config::get('app_namespace') . ($module ? '\\' . strtolower($module) : '') . '\\job\\' . $name; + } + if (class_exists($name)) { + return new $name(); + } + } + + /** + * Call the failed method on the job instance. + * @return void + */ + public function failed() + { + $payload = json_decode($this->getRawBody(), true); + + list($class, $method) = $this->parseJob($payload['job']); + + $this->instance = $this->resolve($class); + if ($this->instance && method_exists($this->instance, 'failed')) { + $this->instance->failed($payload['data']); + } + } + + /** + * Calculate the number of seconds with the given delay. + * @param \DateTime|int $delay + * @return int + */ + protected function getSeconds($delay) + { + if ($delay instanceof DateTime) { + return max(0, $delay->getTimestamp() - $this->getTime()); + } + + return (int) $delay; + } + + /** + * Get the current system time. + * @return int + */ + protected function getTime() + { + return time(); + } + + /** + * Get the name of the queued job class. + * @return string + */ + public function getName() + { + return json_decode($this->getRawBody(), true)['job']; + } + + /** + * Get the name of the queue the job belongs to. + * @return string + */ + public function getQueue() + { + return $this->queue; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/Listener.php b/vendor/topthink/think-queue/src/queue/Listener.php new file mode 100644 index 000000000..5917dd52d --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/Listener.php @@ -0,0 +1,166 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + + +use Closure; +use think\Process; + +class Listener +{ + + /** + * @var string + */ + protected $commandPath; + + /** + * @var int + */ + protected $sleep = 3; + + /** + * @var int + */ + protected $maxTries = 0; + + /** + * @var string + */ + protected $workerCommand; + + /** + * @var \Closure|null + */ + protected $outputHandler; + + /** + * @param string $commandPath + */ + public function __construct($commandPath) + { + $this->commandPath = $commandPath; + $this->workerCommand = + '"' . PHP_BINARY . '" think queue:work --queue="%s" --delay=%s --memory=%s --sleep=%s --tries=%s'; + } + + /** + * @param string $queue + * @param string $delay + * @param string $memory + * @param int $timeout + * @return void + */ + public function listen($queue, $delay, $memory, $timeout = 60) + { + $process = $this->makeProcess($queue, $delay, $memory, $timeout); + + while (true) { + $this->runProcess($process, $memory); + } + } + + /** + * @param \Think\Process $process + * @param int $memory + */ + public function runProcess(Process $process, $memory) + { + $process->run(function ($type, $line) { + $this->handleWorkerOutput($type, $line); + }); + + + if ($this->memoryExceeded($memory)) { + $this->stop(); + } + } + + /** + * @param string $queue + * @param int $delay + * @param int $memory + * @param int $timeout + * @return \think\Process + */ + public function makeProcess($queue, $delay, $memory, $timeout) + { + $string = $this->workerCommand; + $command = sprintf($string, $queue, $delay, $memory, $this->sleep, $this->maxTries); + + return new Process($command, $this->commandPath, null, null, $timeout); + } + + /** + * @param int $type + * @param string $line + * @return void + */ + protected function handleWorkerOutput($type, $line) + { + if (isset($this->outputHandler)) { + call_user_func($this->outputHandler, $type, $line); + } + } + + /** + * @param int $memoryLimit + * @return bool + */ + public function memoryExceeded($memoryLimit) + { + return (memory_get_usage() / 1024 / 1024) >= $memoryLimit; + } + + /** + * @return void + */ + public function stop() + { + die; + } + + /** + * @param \Closure $outputHandler + * @return void + */ + public function setOutputHandler(Closure $outputHandler) + { + $this->outputHandler = $outputHandler; + } + + /** + * @return int + */ + public function getSleep() + { + return $this->sleep; + } + + /** + * @param int $sleep + * @return void + */ + public function setSleep($sleep) + { + $this->sleep = $sleep; + } + + /** + * @param int $tries + * @return void + */ + public function setMaxTries($tries) + { + $this->maxTries = $tries; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/Queueable.php b/vendor/topthink/think-queue/src/queue/Queueable.php new file mode 100644 index 000000000..ef31608ae --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/Queueable.php @@ -0,0 +1,46 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + +trait Queueable +{ + + /** @var string 闃熷垪鍚嶇О */ + public $queue; + + /** @var integer 寤惰繜鏃堕棿 */ + public $delay; + + /** + * 璁剧疆闃熷垪鍚 + * @param $queue + * @return $this + */ + public function queue($queue) + { + $this->queue = $queue; + + return $this; + } + + /** + * 璁剧疆寤惰繜鏃堕棿 + * @param $delay + * @return $this + */ + public function delay($delay) + { + $this->delay = $delay; + + return $this; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/ShouldQueue.php b/vendor/topthink/think-queue/src/queue/ShouldQueue.php new file mode 100644 index 000000000..d32a5b633 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/ShouldQueue.php @@ -0,0 +1,17 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + +interface ShouldQueue +{ + +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/Worker.php b/vendor/topthink/think-queue/src/queue/Worker.php new file mode 100644 index 000000000..c3efd70c6 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/Worker.php @@ -0,0 +1,113 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue; + +use Exception; +use think\Hook; +use think\Queue; + +class Worker +{ + + /** + * 鎵ц涓嬩釜浠诲姟 + * @param string $queue + * @param int $delay + * @param int $sleep + * @param int $maxTries + * @return array + */ + public function pop($queue = null, $delay = 0, $sleep = 3, $maxTries = 0) + { + + $job = $this->getNextJob($queue); + + if (!is_null($job)) { + return $this->process($job, $maxTries, $delay); + } + + $this->sleep($sleep); + + return ['job' => null, 'failed' => false]; + } + + /** + * 鑾峰彇涓嬩釜浠诲姟 + * @param string $queue + * @return Job + */ + protected function getNextJob($queue) + { + if (is_null($queue)) { + return Queue::pop(); + } + + foreach (explode(',', $queue) as $queue) { + if (!is_null($job = Queue::pop($queue))) { + return $job; + } + } + } + + /** + * Process a given job from the queue. + * @param \think\queue\Job $job + * @param int $maxTries + * @param int $delay + * @return array + * @throws Exception + */ + public function process(Job $job, $maxTries = 0, $delay = 0) + { + if ($maxTries > 0 && $job->attempts() > $maxTries) { + return $this->logFailedJob($job); + } + + try { + $job->fire(); + + return ['job' => $job, 'failed' => false]; + } catch (Exception $e) { + if (!$job->isDeleted()) { + $job->release($delay); + } + + throw $e; + } + } + + /** + * Log a failed job into storage. + * @param \Think\Queue\Job $job + * @return array + */ + protected function logFailedJob(Job $job) + { + if (Hook::listen('queue.failed', $job, null, true)) { + $job->delete(); + $job->failed(); + } + + return ['job' => $job, 'failed' => true]; + } + + /** + * Sleep the script for a given number of seconds. + * @param int $seconds + * @return void + */ + public function sleep($seconds) + { + sleep($seconds); + } + +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/command/Listen.php b/vendor/topthink/think-queue/src/queue/command/Listen.php new file mode 100644 index 000000000..6332a4c10 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/command/Listen.php @@ -0,0 +1,61 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\command; + + +use think\console\Command; +use think\console\Input; +use think\console\input\Option; +use think\console\Output; +use think\queue\Listener; + +class Listen extends Command +{ + /** @var Listener */ + protected $listener; + + public function configure() + { + $this->setName('queue:listen') + ->addOption('queue', null, Option::VALUE_OPTIONAL, 'The queue to listen on', null) + ->addOption('delay', null, Option::VALUE_OPTIONAL, 'Amount of time to delay failed jobs', 0) + ->addOption('memory', null, Option::VALUE_OPTIONAL, 'The memory limit in megabytes', 128) + ->addOption('timeout', null, Option::VALUE_OPTIONAL, 'Seconds a job may run before timing out', 60) + ->addOption('sleep', null, Option::VALUE_OPTIONAL, 'Seconds to wait before checking queue for jobs', 3) + ->addOption('tries', null, Option::VALUE_OPTIONAL, 'Number of times to attempt a job before logging it failed', 0) + ->setDescription('Listen to a given queue'); + } + + public function initialize(Input $input, Output $output) + { + $this->listener = new Listener(getcwd()); + $this->listener->setSleep($input->getOption('sleep')); + $this->listener->setMaxTries($input->getOption('tries')); + + $this->listener->setOutputHandler(function ($type, $line) use ($output) { + $output->write($line); + }); + } + + public function execute(Input $input, Output $output) + { + $delay = $input->getOption('delay'); + + $memory = $input->getOption('memory'); + + $timeout = $input->getOption('timeout'); + + $queue = $input->getOption('queue') ?: 'default'; + + $this->listener->listen($queue, $delay, $memory, $timeout); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/command/Restart.php b/vendor/topthink/think-queue/src/queue/command/Restart.php new file mode 100644 index 000000000..e84cef71d --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/command/Restart.php @@ -0,0 +1,32 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\command; + + +use think\Cache; +use think\console\Command; +use think\console\Input; +use think\console\Output; + +class Restart extends Command +{ + public function configure() + { + $this->setName('queue:restart')->setDescription('Restart queue worker daemons after their current job'); + } + + public function execute(Input $input, Output $output) + { + Cache::set('think:queue:restart', time()); + $output->writeln("Broadcasting queue restart signal."); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/command/Subscribe.php b/vendor/topthink/think-queue/src/queue/command/Subscribe.php new file mode 100644 index 000000000..11dbeb189 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/command/Subscribe.php @@ -0,0 +1,46 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\command; + +use think\console\Command; +use think\console\Input; +use think\console\input\Argument; +use think\console\input\Option; +use think\console\Output; +use think\Queue; +use think\Url; + +class Subscribe extends Command +{ + public function configure() + { + $this->setName('queue:subscribe') + ->setDescription('Subscribe a URL to an push queue') + ->addArgument('name', Argument::REQUIRED, 'name') + ->addArgument('url', Argument::REQUIRED, 'The URL to be subscribed.') + ->addArgument('queue', Argument::OPTIONAL, 'The URL to be subscribed.') + ->addOption('option', null, Option::VALUE_IS_ARRAY | Option::VALUE_OPTIONAL, 'the options'); + } + + public function execute(Input $input, Output $output) + { + + $url = $input->getArgument('url'); + if (!preg_match('/^https?:\/\//', $url)) { + $url = Url::build($url); + } + + Queue::subscribe($input->getArgument('name'), $url, $input->getArgument('queue'), $input->getOption('option')); + + $output->write('Queue subscriber added: ' . $input->getArgument('url') . ''); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/command/Work.php b/vendor/topthink/think-queue/src/queue/command/Work.php new file mode 100644 index 000000000..53e23de64 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/command/Work.php @@ -0,0 +1,202 @@ + +// +---------------------------------------------------------------------- +namespace think\queue\command; + +use think\Config; +use think\console\Command; +use think\console\Input; +use think\console\input\Option; +use think\console\Output; +use think\queue\Job; +use think\queue\Worker; +use Exception; +use Throwable; +use think\Cache; +use think\exception\Handle; +use think\exception\ThrowableError; + +class Work extends Command +{ + + /** + * The queue worker instance. + * @var \think\queue\Worker + */ + protected $worker; + + protected function initialize(Input $input, Output $output) + { + $this->worker = new Worker(); + } + + protected function configure() + { + $this->setName('queue:work') + ->addOption('queue', null, Option::VALUE_OPTIONAL, 'The queue to listen on') + ->addOption('daemon', null, Option::VALUE_NONE, 'Run the worker in daemon mode') + ->addOption('delay', null, Option::VALUE_OPTIONAL, 'Amount of time to delay failed jobs', 0) + ->addOption('force', null, Option::VALUE_NONE, 'Force the worker to run even in maintenance mode') + ->addOption('memory', null, Option::VALUE_OPTIONAL, 'The memory limit in megabytes', 128) + ->addOption('sleep', null, Option::VALUE_OPTIONAL, 'Number of seconds to sleep when no job is available', 3) + ->addOption('tries', null, Option::VALUE_OPTIONAL, 'Number of times to attempt a job before logging it failed', 0) + ->setDescription('Process the next job on a queue'); + } + + /** + * Execute the console command. + * @param Input $input + * @param Output $output + * @return int|null|void + */ + public function execute(Input $input, Output $output) + { + $queue = $input->getOption('queue'); + + $delay = $input->getOption('delay'); + + $memory = $input->getOption('memory'); + + if ($input->getOption('daemon')) { + $this->daemon( + $queue, $delay, $memory, + $input->getOption('sleep'), $input->getOption('tries') + ); + } else { + $response = $this->worker->pop($queue, $delay, $input->getOption('sleep'), $input->getOption('tries')); + $this->output($response); + } + } + + protected function output($response) + { + if (!is_null($response['job'])) { + /** @var Job $job */ + $job = $response['job']; + if ($response['failed']) { + $this->output->writeln('Failed: ' . $job->getName()); + } else { + $this->output->writeln('Processed: ' . $job->getName()); + } + } + } + + /** + * 鍚姩涓涓畧鎶よ繘绋嬫墽琛屼换鍔. + * + * @param string $queue + * @param int $delay + * @param int $memory + * @param int $sleep + * @param int $maxTries + * @return array + */ + protected function daemon($queue = null, $delay = 0, $memory = 128, $sleep = 3, $maxTries = 0) + { + $lastRestart = $this->getTimestampOfLastQueueRestart(); + + while (true) { + $this->runNextJobForDaemon( + $queue, $delay, $sleep, $maxTries + ); + + if ($this->memoryExceeded($memory) || $this->queueShouldRestart($lastRestart)) { + $this->stop(); + } + } + } + + /** + * 浠ュ畧鎶よ繘绋嬬殑鏂瑰紡鎵ц涓嬩釜浠诲姟. + * + * @param string $queue + * @param int $delay + * @param int $sleep + * @param int $maxTries + * @return void + */ + protected function runNextJobForDaemon($queue, $delay, $sleep, $maxTries) + { + try { + $response = $this->worker->pop($queue, $delay, $sleep, $maxTries); + + $this->output($response); + } catch (Exception $e) { + $this->getExceptionHandler()->report($e); + } catch (Throwable $e) { + $this->getExceptionHandler()->report(new ThrowableError($e)); + } + } + + /** + * 鑾峰彇涓婃閲嶅惎瀹堟姢杩涚▼鐨勬椂闂 + * + * @return int|null + */ + protected function getTimestampOfLastQueueRestart() + { + return Cache::get('think:queue:restart'); + } + + /** + * 妫鏌ユ槸鍚﹁閲嶅惎瀹堟姢杩涚▼ + * + * @param int|null $lastRestart + * @return bool + */ + protected function queueShouldRestart($lastRestart) + { + return $this->getTimestampOfLastQueueRestart() != $lastRestart; + } + + /** + * 妫鏌ュ唴瀛樻槸鍚﹁秴鍑 + * @param int $memoryLimit + * @return bool + */ + protected function memoryExceeded($memoryLimit) + { + return (memory_get_usage() / 1024 / 1024) >= $memoryLimit; + } + + /** + * 鑾峰彇寮傚父澶勭悊瀹炰緥 + * + * @return \think\exception\Handle + */ + protected function getExceptionHandler() + { + static $handle; + + if (!$handle) { + + if ($class = Config::get('exception_handle')) { + if (class_exists($class) && is_subclass_of($class, "\\think\\exception\\Handle")) { + $handle = new $class; + } + } + if (!$handle) { + $handle = new Handle(); + } + } + + return $handle; + } + + /** + * 鍋滄鎵ц浠诲姟鐨勫畧鎶よ繘绋. + * @return void + */ + public function stop() + { + die; + } + +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/connector/Database.php b/vendor/topthink/think-queue/src/queue/connector/Database.php new file mode 100644 index 000000000..28fb96320 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/connector/Database.php @@ -0,0 +1,171 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\connector; + +use think\Db; +use think\queue\Connector; +use think\queue\job\Database as DatabaseJob; + +class Database extends Connector +{ + protected $db; + + protected $options = [ + 'expire' => 60, + 'default' => 'default', + 'table' => 'jobs', + 'dsn' => [] + ]; + + public function __construct($options) + { + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + + $this->db = Db::connect($this->options['dsn']); + } + + public function push($job, $data = '', $queue = null) + { + return $this->pushToDatabase(0, $queue, $this->createPayload($job, $data)); + } + + public function later($delay, $job, $data = '', $queue = null) + { + return $this->pushToDatabase($delay, $queue, $this->createPayload($job, $data)); + } + + public function pop($queue = null) + { + $queue = $this->getQueue($queue); + + if (!is_null($this->options['expire'])) { + $this->releaseJobsThatHaveBeenReservedTooLong($queue); + } + + if ($job = $this->getNextAvailableJob($queue)) { + $this->markJobAsReserved($job->id); + + $this->db->commit(); + + return new DatabaseJob($this, $job, $queue); + } + + $this->db->commit(); + } + + /** + * 閲嶆柊鍙戝竷浠诲姟 + * @param string $queue + * @param \StdClass $job + * @param int $delay + * @return mixed + */ + public function release($queue, $job, $delay) + { + return $this->pushToDatabase($delay, $queue, $job->payload, $job->attempts); + } + + /** + * Push a raw payload to the database with a given delay. + * + * @param \DateTime|int $delay + * @param string|null $queue + * @param string $payload + * @param int $attempts + * @return mixed + */ + protected function pushToDatabase($delay, $queue, $payload, $attempts = 0) + { + return $this->db->name($this->options['table'])->insert([ + 'queue' => $this->getQueue($queue), + 'payload' => $payload, + 'attempts' => $attempts, + 'reserved' => 0, + 'reserved_at' => null, + 'available_at' => time() + $delay, + 'created_at' => time() + ]); + } + + /** + * 鑾峰彇涓嬩釜鏈夋晥浠诲姟 + * + * @param string|null $queue + * @return \StdClass|null + */ + protected function getNextAvailableJob($queue) + { + $this->db->startTrans(); + + $job = $this->db->name($this->options['table']) + ->lock(true) + ->where('queue', $this->getQueue($queue)) + ->where('reserved', 0) + ->where('available_at', '<=', time()) + ->order('id', 'asc') + ->find(); + + return $job ? (object)$job : null; + } + + /** + * 鏍囪浠诲姟姝e湪鎵ц. + * + * @param string $id + * @return void + */ + protected function markJobAsReserved($id) + { + $this->db->name($this->options['table'])->where('id', $id)->update([ + 'reserved' => 1, + 'reserved_at' => time() + ]); + } + + /** + * 閲嶆柊鍙戝竷瓒呮椂鐨勪换鍔 + * + * @param string $queue + * @return void + */ + protected function releaseJobsThatHaveBeenReservedTooLong($queue) + { + $expired = time() - $this->options['expire']; + + $this->db->name($this->options['table']) + ->where('queue', $this->getQueue($queue)) + ->where('reserved', 1) + ->where('reserved_at', '<=', $expired) + ->update([ + 'reserved' => 0, + 'reserved_at' => null, + 'attempts' => ['exp', 'attempts + 1'] + ]); + } + + /** + * 鍒犻櫎浠诲姟 + * @param string $id + * @return void + */ + public function deleteReserved($id) + { + $this->db->name($this->options['table'])->delete($id); + } + + protected function getQueue($queue) + { + return $queue ?: $this->options['default']; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/connector/Redis.php b/vendor/topthink/think-queue/src/queue/connector/Redis.php new file mode 100644 index 000000000..2c5a2a131 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/connector/Redis.php @@ -0,0 +1,247 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\connector; + +use Exception; +use think\helper\Str; +use think\queue\Connector; +use think\queue\job\Redis as RedisJob; + +class Redis extends Connector +{ + /** @var \Redis */ + protected $redis; + + protected $options = [ + 'expire' => 60, + 'default' => 'default', + 'host' => '127.0.0.1', + 'port' => 6379, + 'password' => '', + 'select' => 0, + 'timeout' => 0, + 'persistent' => false + ]; + + public function __construct($options) + { + if (!extension_loaded('redis')) { + throw new Exception('redis鎵╁睍鏈畨瑁'); + } + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + + $func = $this->options['persistent'] ? 'pconnect' : 'connect'; + $this->redis = new \Redis; + $this->redis->$func($this->options['host'], $this->options['port'], $this->options['timeout']); + + if ('' != $this->options['password']) { + $this->redis->auth($this->options['password']); + } + + if (0 != $this->options['select']) { + $this->redis->select($this->options['select']); + } + } + + public function push($job, $data = '', $queue = null) + { + return $this->pushRaw($this->createPayload($job, $data), $queue); + } + + public function later($delay, $job, $data = '', $queue = null) + { + $payload = $this->createPayload($job, $data); + + $this->redis->zAdd($this->getQueue($queue) . ':delayed', time() + $delay, $payload); + } + + public function pop($queue = null) + { + $original = $queue ?: $this->options['default']; + + $queue = $this->getQueue($queue); + + if (!is_null($this->options['expire'])) { + $this->migrateAllExpiredJobs($queue); + } + + $job = $this->redis->lPop($queue); + + if ($job !== false) { + $this->redis->zAdd($queue . ':reserved', time() + $this->options['expire'], $job); + + return new RedisJob($this, $job, $original); + } + } + + /** + * 閲嶆柊鍙戝竷浠诲姟 + * + * @param string $queue + * @param string $payload + * @param int $delay + * @param int $attempts + * @return void + */ + public function release($queue, $payload, $delay, $attempts) + { + $payload = $this->setMeta($payload, 'attempts', $attempts); + + $this->redis->zAdd($this->getQueue($queue) . ':delayed', time() + $delay, $payload); + } + + public function pushRaw($payload, $queue = null) + { + $this->redis->rPush($this->getQueue($queue), $payload); + + return json_decode($payload, true)['id']; + } + + protected function createPayload($job, $data = '', $queue = null) + { + $payload = $this->setMeta( + parent::createPayload($job, $data), 'id', $this->getRandomId() + ); + + return $this->setMeta($payload, 'attempts', 1); + } + + /** + * 鍒犻櫎浠诲姟 + * + * @param string $queue + * @param string $job + * @return void + */ + public function deleteReserved($queue, $job) + { + $this->redis->zRem($this->getQueue($queue) . ':reserved', $job); + } + + /** + * 绉诲姩鎵鏈変换鍔 + * + * @param string $queue + * @return void + */ + protected function migrateAllExpiredJobs($queue) + { + $this->migrateExpiredJobs($queue . ':delayed', $queue, false); + + $this->migrateExpiredJobs($queue . ':reserved', $queue); + } + + /** + * 绉诲姩寤惰繜浠诲姟 + * + * @param string $from + * @param string $to + * @param bool $attempt + */ + public function migrateExpiredJobs($from, $to, $attempt = true) + { + $this->redis->watch($from); + + $jobs = $this->getExpiredJobs( + $from, $time = time() + ); + if (count($jobs) > 0) { + $this->transaction(function () use ($from, $to, $time, $jobs, $attempt) { + $this->removeExpiredJobs($from, $time); + $this->pushExpiredJobsOntoNewQueue($to, $jobs, $attempt); + }); + } + $this->redis->unwatch(); + } + + /** + * redis浜嬪姟 + * @param \Closure $closure + */ + protected function transaction(\Closure $closure) + { + $this->redis->multi(); + try { + call_user_func($closure); + if (!$this->redis->exec()) { + $this->redis->discard(); + } + } catch (Exception $e) { + $this->redis->discard(); + } + } + + /** + * 鑾峰彇鎵鏈夊埌鏈熶换鍔 + * + * @param string $from + * @param int $time + * @return array + */ + protected function getExpiredJobs($from, $time) + { + return $this->redis->zRangeByScore($from, '-inf', $time); + } + + /** + * 鍒犻櫎杩囨湡浠诲姟 + * + * @param string $from + * @param int $time + * @return void + */ + protected function removeExpiredJobs($from, $time) + { + $this->redis->zRemRangeByScore($from, '-inf', $time); + } + + /** + * 閲嶆柊鍙戝竷鍒版湡浠诲姟 + * + * @param string $to + * @param array $jobs + * @param boolean $attempt + */ + protected function pushExpiredJobsOntoNewQueue($to, $jobs, $attempt = true) + { + if ($attempt) { + foreach ($jobs as &$job) { + $attempts = json_decode($job, true)['attempts']; + $job = $this->setMeta($job, 'attempts', $attempts + 1); + } + } + call_user_func_array([$this->redis, 'rPush'], array_merge([$to], $jobs)); + } + + /** + * 闅忔満id + * + * @return string + */ + protected function getRandomId() + { + return Str::random(32); + } + + /** + * 鑾峰彇闃熷垪鍚 + * + * @param string|null $queue + * @return string + */ + protected function getQueue($queue) + { + return 'queues:' . ($queue ?: $this->options['default']); + } +} diff --git a/vendor/topthink/think-queue/src/queue/connector/Sync.php b/vendor/topthink/think-queue/src/queue/connector/Sync.php new file mode 100644 index 000000000..6db4c2c45 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/connector/Sync.php @@ -0,0 +1,57 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\connector; + +use Exception; +use think\queue\Connector; +use think\queue\job\Sync as SyncJob; +use Throwable; + +class Sync extends Connector +{ + + public function push($job, $data = '', $queue = null) + { + $queueJob = $this->resolveJob($this->createPayload($job, $data, $queue)); + + try { + set_time_limit(0); + $queueJob->fire(); + } catch (Exception $e) { + $queueJob->failed(); + + throw $e; + } catch (Throwable $e) { + $queueJob->failed(); + + throw $e; + } + + return 0; + } + + public function later($delay, $job, $data = '', $queue = null) + { + return $this->push($job, $data, $queue); + } + + public function pop($queue = null) + { + + } + + protected function resolveJob($payload) + { + return new SyncJob($payload); + } + +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/connector/Topthink.php b/vendor/topthink/think-queue/src/queue/connector/Topthink.php new file mode 100644 index 000000000..b4a1c4c8d --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/connector/Topthink.php @@ -0,0 +1,226 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\connector; + +use think\exception\HttpException; +use think\queue\Connector; +use think\Request; +use think\queue\job\Topthink as TopthinkJob; +use think\Response; + +class Topthink extends Connector +{ + protected $options = [ + 'token' => '', + 'project_id' => '', + 'protocol' => 'https', + 'host' => 'qns.topthink.com', + 'port' => 443, + 'api_version' => 1, + 'max_retries' => 3, + 'default' => 'default' + ]; + + /** @var Request */ + protected $request; + + protected $url; + + protected $curl = null; + + protected $last_status; + + protected $headers = []; + + public function __construct($options) + { + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + + $this->url = "{$this->options['protocol']}://{$this->options['host']}:{$this->options['port']}/v{$this->options['api_version']}/"; + + $this->headers['Authorization'] = "Bearer {$this->options['token']}"; + + $this->request = Request::instance(); + } + + public function push($job, $data = '', $queue = null) + { + return $this->pushRaw(0, $queue, $this->createPayload($job, $data)); + } + + public function later($delay, $job, $data = '', $queue = null) + { + return $this->pushRaw($delay, $queue, $this->createPayload($job, $data)); + } + + public function release($queue, $job, $delay) + { + return $this->pushRaw($delay, $queue, $job->payload, $job->attempts); + } + + public function marshal() + { + $job = new TopthinkJob($this, $this->marshalPushedJob(), $this->request->header('topthink-message-queue')); + if ($this->request->header('topthink-message-status') == 'success') { + $job->fire(); + } else { + $job->failed(); + } + return new Response('OK'); + } + + public function pushRaw($delay, $queue, $payload, $attempts = 0) + { + $queue_name = $this->getQueue($queue); + $queue = rawurlencode($queue_name); + $url = "project/{$this->options['project_id']}/queue/{$queue}/message"; + $message = [ + 'payload' => $payload, + 'attempts' => $attempts, + 'delay' => $delay + ]; + + return $this->apiCall('POST', $url, $message)->id; + } + + public function deleteMessage($queue, $id) + { + $queue = rawurlencode($queue); + $url = "project/{$this->options['project_id']}/queue/{$queue}/message/{$id}"; + return $this->apiCall('DELETE', $url); + } + + protected function apiCall($type, $url, $params = []) + { + $url = "{$this->url}$url"; + + if ($this->curl == null) { + $this->curl = curl_init(); + } + + switch ($type = strtoupper($type)) { + case 'DELETE': + curl_setopt($this->curl, CURLOPT_URL, $url); + curl_setopt($this->curl, CURLOPT_CUSTOMREQUEST, $type); + curl_setopt($this->curl, CURLOPT_POSTFIELDS, json_encode($params)); + break; + case 'PUT': + curl_setopt($this->curl, CURLOPT_URL, $url); + curl_setopt($this->curl, CURLOPT_CUSTOMREQUEST, $type); + curl_setopt($this->curl, CURLOPT_POSTFIELDS, json_encode($params)); + break; + case 'POST': + curl_setopt($this->curl, CURLOPT_URL, $url); + curl_setopt($this->curl, CURLOPT_CUSTOMREQUEST, $type); + curl_setopt($this->curl, CURLOPT_POST, true); + curl_setopt($this->curl, CURLOPT_POSTFIELDS, $params); + break; + case 'GET': + curl_setopt($this->curl, CURLOPT_POSTFIELDS, null); + curl_setopt($this->curl, CURLOPT_CUSTOMREQUEST, $type); + curl_setopt($this->curl, CURLOPT_HTTPGET, true); + $url .= '?' . http_build_query($params); + curl_setopt($this->curl, CURLOPT_URL, $url); + break; + } + + curl_setopt($this->curl, CURLOPT_SSL_VERIFYPEER, false); + curl_setopt($this->curl, CURLOPT_RETURNTRANSFER, true); + + $headers = []; + foreach ($this->headers as $k => $v) { + if ($k == 'Connection') { + $v = 'Close'; + } + $headers[] = "$k: $v"; + } + + curl_setopt($this->curl, CURLOPT_HTTPHEADER, $headers); + curl_setopt($this->curl, CURLOPT_CONNECTTIMEOUT, 10); + + return $this->callWithRetries(); + } + + protected function callWithRetries() + { + for ($retry = 0; $retry < $this->options['max_retries']; $retry++) { + $out = curl_exec($this->curl); + if ($out === false) { + $this->reportHttpError(0, curl_error($this->curl)); + } + $this->last_status = curl_getinfo($this->curl, CURLINFO_HTTP_CODE); + + if ($this->last_status >= 200 && $this->last_status < 300) { + return self::jsonDecode($out); + } elseif ($this->last_status >= 500) { + self::waitRandomInterval($retry); + } else { + $this->reportHttpError($this->last_status, $out); + } + } + $this->reportHttpError($this->last_status, "Service unavailable"); + return null; + } + + protected static function jsonDecode($response) + { + $data = json_decode($response); + + $json_error = json_last_error(); + if ($json_error != JSON_ERROR_NONE) { + throw new \RuntimeException($json_error); + } + + return $data; + } + + protected static function waitRandomInterval($retry) + { + $max_delay = pow(4, $retry) * 100 * 1000; + usleep(rand(0, $max_delay)); + } + + protected function reportHttpError($status, $text) + { + throw new HttpException($status, "http error: {$status} | {$text}"); + } + + /** + * Marshal out the pushed job and payload. + * + * @return object + */ + protected function marshalPushedJob() + { + return (object) [ + 'id' => $this->request->header('topthink-message-id'), + 'payload' => $this->request->getContent(), + 'attempts' => $this->request->header('topthink-message-attempts') + ]; + } + + + public function __destruct() + { + if ($this->curl != null) { + curl_close($this->curl); + $this->curl = null; + } + } + + public function pop($queue = null) + { + throw new \RuntimeException('pop queues not support for this type'); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/job/Database.php b/vendor/topthink/think-queue/src/queue/job/Database.php new file mode 100644 index 000000000..ab6107d87 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/job/Database.php @@ -0,0 +1,88 @@ + +// +---------------------------------------------------------------------- +namespace think\queue\job; + +use think\queue\Job; +use think\queue\connector\Database as DatabaseQueue; + +class Database extends Job +{ + /** + * The database queue instance. + * @var DatabaseQueue + */ + protected $database; + + /** + * The database job payload. + * @var Object + */ + protected $job; + + public function __construct(DatabaseQueue $database, $job, $queue) + { + $this->job = $job; + $this->queue = $queue; + $this->database = $database; + $this->job->attempts = $this->job->attempts + 1; + } + + /** + * 鎵ц浠诲姟 + * @return void + */ + public function fire() + { + $this->resolveAndFire(json_decode($this->job->payload, true)); + } + + /** + * 鍒犻櫎浠诲姟 + * @return void + */ + public function delete() + { + parent::delete(); + $this->database->deleteReserved($this->job->id); + } + + /** + * 閲嶆柊鍙戝竷浠诲姟 + * @param int $delay + * @return void + */ + public function release($delay = 0) + { + parent::release($delay); + + $this->delete(); + + $this->database->release($this->queue, $this->job, $delay); + } + + /** + * 鑾峰彇褰撳墠浠诲姟灏濊瘯娆℃暟 + * @return int + */ + public function attempts() + { + return (int)$this->job->attempts; + } + + /** + * Get the raw body string for the job. + * @return string + */ + public function getRawBody() + { + return $this->job->payload; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/job/Redis.php b/vendor/topthink/think-queue/src/queue/job/Redis.php new file mode 100644 index 000000000..d56d2b661 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/job/Redis.php @@ -0,0 +1,95 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\job; + + +use think\queue\Job; +use think\queue\connector\Redis as RedisQueue; + +class Redis extends Job +{ + + + /** + * The redis queue instance. + * @var RedisQueue + */ + protected $redis; + + /** + * The database job payload. + * @var Object + */ + protected $job; + + public function __construct(RedisQueue $redis, $job, $queue) + { + $this->job = $job; + $this->queue = $queue; + $this->redis = $redis; + } + + /** + * Fire the job. + * @return void + */ + public function fire() + { + $this->resolveAndFire(json_decode($this->getRawBody(), true)); + } + + /** + * Get the number of times the job has been attempted. + * @return int + */ + public function attempts() + { + return json_decode($this->job, true)['attempts']; + } + + /** + * Get the raw body string for the job. + * @return string + */ + public function getRawBody() + { + return $this->job; + } + + + /** + * 鍒犻櫎浠诲姟 + * + * @return void + */ + public function delete() + { + parent::delete(); + + $this->redis->deleteReserved($this->queue, $this->job); + } + + /** + * 閲嶆柊鍙戝竷浠诲姟 + * + * @param int $delay + * @return void + */ + public function release($delay = 0) + { + parent::release($delay); + + $this->delete(); + + $this->redis->release($this->queue, $this->job, $delay, $this->attempts() + 1); + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/job/Sync.php b/vendor/topthink/think-queue/src/queue/job/Sync.php new file mode 100644 index 000000000..d724f7e19 --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/job/Sync.php @@ -0,0 +1,57 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\job; + + +use think\queue\Job; + +class Sync extends Job +{ + /** + * The queue message data. + * + * @var string + */ + protected $payload; + + public function __construct($payload) + { + $this->payload = $payload; + } + + /** + * Fire the job. + * @return void + */ + public function fire() + { + $this->resolveAndFire(json_decode($this->payload, true)); + } + + /** + * Get the number of times the job has been attempted. + * @return int + */ + public function attempts() + { + return 1; + } + + /** + * Get the raw body string for the job. + * @return string + */ + public function getRawBody() + { + return $this->payload; + } +} \ No newline at end of file diff --git a/vendor/topthink/think-queue/src/queue/job/Topthink.php b/vendor/topthink/think-queue/src/queue/job/Topthink.php new file mode 100644 index 000000000..bab4cf1ba --- /dev/null +++ b/vendor/topthink/think-queue/src/queue/job/Topthink.php @@ -0,0 +1,86 @@ + +// +---------------------------------------------------------------------- + +namespace think\queue\job; + + +use think\queue\Job; +use think\queue\connector\Topthink as TopthinkQueue; + +class Topthink extends Job +{ + + /** + * The Iron queue instance. + * + * @var TopthinkQueue + */ + protected $topthink; + + /** + * The IronMQ message instance. + * + * @var object + */ + protected $job; + + public function __construct(TopthinkQueue $topthink, $job, $queue) + { + $this->topthink = $topthink; + $this->job = $job; + $this->queue = $queue; + $this->job->attempts = $this->job->attempts + 1; + } + + /** + * Fire the job. + * @return void + */ + public function fire() + { + $this->resolveAndFire(json_decode($this->job->payload, true)); + } + + /** + * Get the number of times the job has been attempted. + * @return int + */ + public function attempts() + { + return (int)$this->job->attempts; + } + + public function delete() + { + parent::delete(); + + $this->topthink->deleteMessage($this->queue, $this->job->id); + } + + public function release($delay = 0) + { + parent::release($delay); + + $this->delete(); + + $this->topthink->release($this->queue, $this->job, $delay); + } + + /** + * Get the raw body string for the job. + * @return string + */ + public function getRawBody() + { + return $this->job->payload; + } + +} \ No newline at end of file diff --git a/vendor/topthink/think-worker/LICENSE b/vendor/topthink/think-worker/LICENSE new file mode 100644 index 000000000..8dada3eda --- /dev/null +++ b/vendor/topthink/think-worker/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/topthink/think-worker/README.md b/vendor/topthink/think-worker/README.md new file mode 100644 index 000000000..4ac581604 --- /dev/null +++ b/vendor/topthink/think-worker/README.md @@ -0,0 +1,53 @@ +ThinkPHP 5.0 Workerman 鎵╁睍 +=============== + +## 瀹夎 +composer require topthink/think-worker + +## 浣跨敤鏂规硶 +棣栧厛鍒涘缓鎺у埗鍣ㄧ被骞剁户鎵 think\worker\Server锛岀劧鍚庤缃睘鎬у拰娣诲姞鍥炶皟鏂规硶 + +~~~ +namespace app\index\controller; + +use think\worker\Server; + +class Worker extends Server +{ + protected $socket = 'http://0.0.0.0:2346'; + + public function onMessage($connection,$data) + { + $connection->send(json_encode($data)); + } +} +~~~ +鏀寔workerman鎵鏈夌殑鍥炶皟鏂规硶瀹氫箟锛堝洖璋冩柟娉曞繀椤绘槸public绫诲瀷锛 + + +鍦ㄥ簲鐢ㄦ牴鐩綍澧炲姞鍏ュ彛鏂囦欢 server.php + +~~~ +#!/usr/bin/env php + +// +---------------------------------------------------------------------- + +namespace think\worker; + +use Workerman\Worker; + +/** + * Worker鎺у埗鍣ㄦ墿灞曠被 + */ +abstract class Server +{ + protected $worker; + protected $socket = ''; + protected $protocol = 'http'; + protected $host = '0.0.0.0'; + protected $port = '2346'; + protected $processes = 4; + + /** + * 鏋舵瀯鍑芥暟 + * @access public + */ + public function __construct() + { + // 瀹炰緥鍖 Websocket 鏈嶅姟 + $this->worker = new Worker($this->socket ?: $this->protocol . '://' . $this->host . ':' . $this->port); + // 璁剧疆杩涚▼鏁 + $this->worker->count = $this->processes; + // 鍒濆鍖 + $this->init(); + + // 璁剧疆鍥炶皟 + foreach (['onWorkerStart', 'onConnect', 'onMessage', 'onClose', 'onError', 'onBufferFull', 'onBufferDrain', 'onWorkerStop', 'onWorkerReload'] as $event) { + if (method_exists($this, $event)) { + $this->worker->$event = [$this, $event]; + } + } + // Run worker + Worker::runAll(); + } + + protected function init() + { + } + +} diff --git a/vendor/workerman/workerman/.gitignore b/vendor/workerman/workerman/.gitignore new file mode 100644 index 000000000..9b668caef --- /dev/null +++ b/vendor/workerman/workerman/.gitignore @@ -0,0 +1,5 @@ +logs +.buildpath +.project +.settings +.idea \ No newline at end of file diff --git a/vendor/workerman/workerman/Autoloader.php b/vendor/workerman/workerman/Autoloader.php new file mode 100644 index 000000000..45773c914 --- /dev/null +++ b/vendor/workerman/workerman/Autoloader.php @@ -0,0 +1,69 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman; + +/** + * Autoload. + */ +class Autoloader +{ + /** + * Autoload root path. + * + * @var string + */ + protected static $_autoloadRootPath = ''; + + /** + * Set autoload root path. + * + * @param string $root_path + * @return void + */ + public static function setRootPath($root_path) + { + self::$_autoloadRootPath = $root_path; + } + + /** + * Load files by namespace. + * + * @param string $name + * @return boolean + */ + public static function loadByNamespace($name) + { + $class_path = str_replace('\\', DIRECTORY_SEPARATOR, $name); + if (strpos($name, 'Workerman\\') === 0) { + $class_file = __DIR__ . substr($class_path, strlen('Workerman')) . '.php'; + } else { + if (self::$_autoloadRootPath) { + $class_file = self::$_autoloadRootPath . DIRECTORY_SEPARATOR . $class_path . '.php'; + } + if (empty($class_file) || !is_file($class_file)) { + $class_file = __DIR__ . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . "$class_path.php"; + } + } + + if (is_file($class_file)) { + require_once($class_file); + if (class_exists($name, false)) { + return true; + } + } + return false; + } +} + +spl_autoload_register('\Workerman\Autoloader::loadByNamespace'); \ No newline at end of file diff --git a/vendor/workerman/workerman/Connection/AsyncTcpConnection.php b/vendor/workerman/workerman/Connection/AsyncTcpConnection.php new file mode 100644 index 000000000..116aa6924 --- /dev/null +++ b/vendor/workerman/workerman/Connection/AsyncTcpConnection.php @@ -0,0 +1,316 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Connection; + +use Workerman\Events\EventInterface; +use Workerman\Lib\Timer; +use Workerman\Worker; +use Exception; + +/** + * AsyncTcpConnection. + */ +class AsyncTcpConnection extends TcpConnection +{ + /** + * Emitted when socket connection is successfully established. + * + * @var callback + */ + public $onConnect = null; + + /** + * Transport layer protocol. + * + * @var string + */ + public $transport = 'tcp'; + + /** + * Status. + * + * @var int + */ + protected $_status = self::STATUS_INITIAL; + + /** + * Remote host. + * + * @var string + */ + protected $_remoteHost = ''; + + /** + * Connect start time. + * + * @var string + */ + protected $_connectStartTime = 0; + + /** + * Remote URI. + * + * @var string + */ + protected $_remoteURI = ''; + + /** + * Context option. + * + * @var resource + */ + protected $_contextOption = null; + + /** + * Reconnect timer. + * + * @var int + */ + protected $_reconnectTimer = null; + + + /** + * PHP built-in protocols. + * + * @var array + */ + protected static $_builtinTransports = array( + 'tcp' => 'tcp', + 'udp' => 'udp', + 'unix' => 'unix', + 'ssl' => 'ssl', + 'sslv2' => 'sslv2', + 'sslv3' => 'sslv3', + 'tls' => 'tls' + ); + + /** + * Construct. + * + * @param string $remote_address + * @param array $context_option + * @throws Exception + */ + public function __construct($remote_address, $context_option = null) + { + $address_info = parse_url($remote_address); + if (!$address_info) { + echo new \Exception('bad remote_address'); + $this->_remoteAddress = $remote_address; + } else { + if (!isset($address_info['port'])) { + $address_info['port'] = 80; + } + if (!isset($address_info['path'])) { + $address_info['path'] = '/'; + } + if (!isset($address_info['query'])) { + $address_info['query'] = ''; + } else { + $address_info['query'] = '?' . $address_info['query']; + } + $this->_remoteAddress = "{$address_info['host']}:{$address_info['port']}"; + $this->_remoteHost = $address_info['host']; + $this->_remoteURI = "{$address_info['path']}{$address_info['query']}"; + $scheme = isset($address_info['scheme']) ? $address_info['scheme'] : 'tcp'; + } + + $this->id = self::$_idRecorder++; + // Check application layer protocol class. + if (!isset(self::$_builtinTransports[$scheme])) { + $scheme = ucfirst($scheme); + $this->protocol = '\\Protocols\\' . $scheme; + if (!class_exists($this->protocol)) { + $this->protocol = "\\Workerman\\Protocols\\$scheme"; + if (!class_exists($this->protocol)) { + throw new Exception("class \\Protocols\\$scheme not exist"); + } + } + } else { + $this->transport = self::$_builtinTransports[$scheme]; + } + + // For statistics. + self::$statistics['connection_count']++; + $this->maxSendBufferSize = self::$defaultMaxSendBufferSize; + $this->_contextOption = $context_option; + } + + /** + * Do connect. + * + * @return void + */ + public function connect() + { + if ($this->_status !== self::STATUS_INITIAL && $this->_status !== self::STATUS_CLOSING && + $this->_status !== self::STATUS_CLOSED) { + return; + } + $this->_status = self::STATUS_CONNECTING; + $this->_connectStartTime = microtime(true); + // Open socket connection asynchronously. + if ($this->_contextOption) { + $context = stream_context_create($this->_contextOption); + $this->_socket = stream_socket_client("{$this->transport}://{$this->_remoteAddress}", $errno, $errstr, 0, + STREAM_CLIENT_ASYNC_CONNECT, $context); + } else { + $this->_socket = stream_socket_client("{$this->transport}://{$this->_remoteAddress}", $errno, $errstr, 0, + STREAM_CLIENT_ASYNC_CONNECT); + } + // If failed attempt to emit onError callback. + if (!$this->_socket) { + $this->emitError(WORKERMAN_CONNECT_FAIL, $errstr); + if ($this->_status === self::STATUS_CLOSING) { + $this->destroy(); + } + if ($this->_status === self::STATUS_CLOSED) { + $this->onConnect = null; + } + return; + } + // Add socket to global event loop waiting connection is successfully established or faild. + Worker::$globalEvent->add($this->_socket, EventInterface::EV_WRITE, array($this, 'checkConnection')); + } + + /** + * Reconnect. + * + * @param int $after + * @return void + */ + public function reConnect($after = 0) { + $this->_status = self::STATUS_INITIAL; + if ($this->_reconnectTimer) { + Timer::del($this->_reconnectTimer); + } + if ($after > 0) { + $this->_reconnectTimer = Timer::add($after, array($this, 'connect'), null, false); + return; + } + return $this->connect(); + } + + /** + * Get remote address. + * + * @return string + */ + public function getRemoteHost() + { + return $this->_remoteHost; + } + + /** + * Get remote URI. + * + * @return string + */ + public function getRemoteURI() + { + return $this->_remoteURI; + } + + /** + * Try to emit onError callback. + * + * @param int $code + * @param string $msg + * @return void + */ + protected function emitError($code, $msg) + { + $this->_status = self::STATUS_CLOSING; + if ($this->onError) { + try { + call_user_func($this->onError, $this, $code, $msg); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + } + + /** + * Check connection is successfully established or faild. + * + * @param resource $socket + * @return void + */ + public function checkConnection($socket) + { + // Check socket state. + if ($address = stream_socket_get_name($socket, true)) { + // Remove write listener. + Worker::$globalEvent->del($socket, EventInterface::EV_WRITE); + // Nonblocking. + stream_set_blocking($socket, 0); + // Compatible with hhvm + if (function_exists('stream_set_read_buffer')) { + stream_set_read_buffer($socket, 0); + } + // Try to open keepalive for tcp and disable Nagle algorithm. + if (function_exists('socket_import_stream') && $this->transport === 'tcp') { + $raw_socket = socket_import_stream($socket); + socket_set_option($raw_socket, SOL_SOCKET, SO_KEEPALIVE, 1); + socket_set_option($raw_socket, SOL_TCP, TCP_NODELAY, 1); + } + // Register a listener waiting read event. + Worker::$globalEvent->add($socket, EventInterface::EV_READ, array($this, 'baseRead')); + // There are some data waiting to send. + if ($this->_sendBuffer) { + Worker::$globalEvent->add($socket, EventInterface::EV_WRITE, array($this, 'baseWrite')); + } + $this->_status = self::STATUS_ESTABLISH; + $this->_remoteAddress = $address; + + // Try to emit onConnect callback. + if ($this->onConnect) { + try { + call_user_func($this->onConnect, $this); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + // Try to emit protocol::onConnect + if (method_exists($this->protocol, 'onConnect')) { + try { + call_user_func(array($this->protocol, 'onConnect'), $this); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + } else { + // Connection failed. + $this->emitError(WORKERMAN_CONNECT_FAIL, 'connect ' . $this->_remoteAddress . ' fail after ' . round(microtime(true) - $this->_connectStartTime, 4) . ' seconds'); + if ($this->_status === self::STATUS_CLOSING) { + $this->destroy(); + } + if ($this->_status === self::STATUS_CLOSED) { + $this->onConnect = null; + } + } + } +} diff --git a/vendor/workerman/workerman/Connection/ConnectionInterface.php b/vendor/workerman/workerman/Connection/ConnectionInterface.php new file mode 100644 index 000000000..b39e5ae9b --- /dev/null +++ b/vendor/workerman/workerman/Connection/ConnectionInterface.php @@ -0,0 +1,83 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Connection; + +/** + * ConnectionInterface. + */ +abstract class ConnectionInterface +{ + /** + * Statistics for status command. + * + * @var array + */ + public static $statistics = array( + 'connection_count' => 0, + 'total_request' => 0, + 'throw_exception' => 0, + 'send_fail' => 0, + ); + + /** + * Emitted when data is received. + * + * @var callback + */ + public $onMessage = null; + + /** + * Emitted when the other end of the socket sends a FIN packet. + * + * @var callback + */ + public $onClose = null; + + /** + * Emitted when an error occurs with connection. + * + * @var callback + */ + public $onError = null; + + /** + * Sends data on the connection. + * + * @param string $send_buffer + * @return void|boolean + */ + abstract public function send($send_buffer); + + /** + * Get remote IP. + * + * @return string + */ + abstract public function getRemoteIp(); + + /** + * Get remote port. + * + * @return int + */ + abstract public function getRemotePort(); + + /** + * Close connection. + * + * @param $data + * @return void + */ + abstract public function close($data = null); +} diff --git a/vendor/workerman/workerman/Connection/TcpConnection.php b/vendor/workerman/workerman/Connection/TcpConnection.php new file mode 100644 index 000000000..0083e4690 --- /dev/null +++ b/vendor/workerman/workerman/Connection/TcpConnection.php @@ -0,0 +1,719 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Connection; + +use Workerman\Events\EventInterface; +use Workerman\Worker; +use Exception; + +/** + * TcpConnection. + */ +class TcpConnection extends ConnectionInterface +{ + /** + * Read buffer size. + * + * @var int + */ + const READ_BUFFER_SIZE = 65535; + + /** + * Status initial. + * + * @var int + */ + const STATUS_INITIAL = 0; + + /** + * Status connecting. + * + * @var int + */ + const STATUS_CONNECTING = 1; + + /** + * Status connection established. + * + * @var int + */ + const STATUS_ESTABLISH = 2; + + /** + * Status closing. + * + * @var int + */ + const STATUS_CLOSING = 4; + + /** + * Status closed. + * + * @var int + */ + const STATUS_CLOSED = 8; + + /** + * Emitted when data is received. + * + * @var callback + */ + public $onMessage = null; + + /** + * Emitted when the other end of the socket sends a FIN packet. + * + * @var callback + */ + public $onClose = null; + + /** + * Emitted when an error occurs with connection. + * + * @var callback + */ + public $onError = null; + + /** + * Emitted when the send buffer becomes full. + * + * @var callback + */ + public $onBufferFull = null; + + /** + * Emitted when the send buffer becomes empty. + * + * @var callback + */ + public $onBufferDrain = null; + + /** + * Application layer protocol. + * The format is like this Workerman\\Protocols\\Http. + * + * @var \Workerman\Protocols\ProtocolInterface + */ + public $protocol = null; + + /** + * Transport (tcp/udp/unix/ssl). + * + * @var string + */ + public $transport = 'tcp'; + + /** + * Which worker belong to. + * + * @var Worker + */ + public $worker = null; + + /** + * Connection->id. + * + * @var int + */ + public $id = 0; + + /** + * A copy of $worker->id which used to clean up the connection in worker->connections + * + * @var int + */ + protected $_id = 0; + + /** + * Sets the maximum send buffer size for the current connection. + * OnBufferFull callback will be emited When the send buffer is full. + * + * @var int + */ + public $maxSendBufferSize = 1048576; + + /** + * Default send buffer size. + * + * @var int + */ + public static $defaultMaxSendBufferSize = 1048576; + + /** + * Maximum acceptable packet size. + * + * @var int + */ + public static $maxPackageSize = 10485760; + + /** + * Id recorder. + * + * @var int + */ + protected static $_idRecorder = 1; + + /** + * Socket + * + * @var resource + */ + protected $_socket = null; + + /** + * Send buffer. + * + * @var string + */ + protected $_sendBuffer = ''; + + /** + * Receive buffer. + * + * @var string + */ + protected $_recvBuffer = ''; + + /** + * Current package length. + * + * @var int + */ + protected $_currentPackageLength = 0; + + /** + * Connection status. + * + * @var int + */ + protected $_status = self::STATUS_ESTABLISH; + + /** + * Remote address. + * + * @var string + */ + protected $_remoteAddress = ''; + + /** + * Is paused. + * + * @var bool + */ + protected $_isPaused = false; + + /** + * SSL handshake completed or not + * + * @var bool + */ + protected $_sslHandshakeCompleted = false; + + /** + * Construct. + * + * @param resource $socket + * @param string $remote_address + */ + public function __construct($socket, $remote_address = '') + { + self::$statistics['connection_count']++; + $this->id = $this->_id = self::$_idRecorder++; + $this->_socket = $socket; + stream_set_blocking($this->_socket, 0); + // Compatible with hhvm + if (function_exists('stream_set_read_buffer')) { + stream_set_read_buffer($this->_socket, 0); + } + Worker::$globalEvent->add($this->_socket, EventInterface::EV_READ, array($this, 'baseRead')); + $this->maxSendBufferSize = self::$defaultMaxSendBufferSize; + $this->_remoteAddress = $remote_address; + } + + /** + * Sends data on the connection. + * + * @param string $send_buffer + * @param bool $raw + * @return void|bool|null + */ + public function send($send_buffer, $raw = false) + { + if ($this->_status === self::STATUS_CLOSING || $this->_status === self::STATUS_CLOSED) { + return false; + } + + // Try to call protocol::encode($send_buffer) before sending. + if (false === $raw && $this->protocol) { + $parser = $this->protocol; + $send_buffer = $parser::encode($send_buffer, $this); + if ($send_buffer === '') { + return null; + } + } + + if ($this->_status !== self::STATUS_ESTABLISH || + ($this->transport === 'ssl' && $this->_sslHandshakeCompleted !== true) + ) { + if ($this->_sendBuffer) { + if ($this->bufferIsFull()) { + self::$statistics['send_fail']++; + return false; + } + } + $this->_sendBuffer .= $send_buffer; + $this->checkBufferWillFull(); + return null; + } + + + // Attempt to send data directly. + if ($this->_sendBuffer === '') { + $len = @fwrite($this->_socket, $send_buffer); + // send successful. + if ($len === strlen($send_buffer)) { + return true; + } + // Send only part of the data. + if ($len > 0) { + $this->_sendBuffer = substr($send_buffer, $len); + } else { + // Connection closed? + if (!is_resource($this->_socket) || feof($this->_socket)) { + self::$statistics['send_fail']++; + if ($this->onError) { + try { + call_user_func($this->onError, $this, WORKERMAN_SEND_FAIL, 'client closed'); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + $this->destroy(); + return false; + } + $this->_sendBuffer = $send_buffer; + } + Worker::$globalEvent->add($this->_socket, EventInterface::EV_WRITE, array($this, 'baseWrite')); + // Check if the send buffer will be full. + $this->checkBufferWillFull(); + return null; + } else { + if ($this->bufferIsFull()) { + self::$statistics['send_fail']++; + return false; + } + + $this->_sendBuffer .= $send_buffer; + // Check if the send buffer is full. + $this->checkBufferWillFull(); + } + } + + /** + * Get remote IP. + * + * @return string + */ + public function getRemoteIp() + { + $pos = strrpos($this->_remoteAddress, ':'); + if ($pos) { + return trim(substr($this->_remoteAddress, 0, $pos), '[]'); + } + return ''; + } + + /** + * Get remote port. + * + * @return int + */ + public function getRemotePort() + { + if ($this->_remoteAddress) { + return (int)substr(strrchr($this->_remoteAddress, ':'), 1); + } + return 0; + } + + /** + * Pauses the reading of data. That is onMessage will not be emitted. Useful to throttle back an upload. + * + * @return void + */ + public function pauseRecv() + { + Worker::$globalEvent->del($this->_socket, EventInterface::EV_READ); + $this->_isPaused = true; + } + + /** + * Resumes reading after a call to pauseRecv. + * + * @return void + */ + public function resumeRecv() + { + if ($this->_isPaused === true) { + Worker::$globalEvent->add($this->_socket, EventInterface::EV_READ, array($this, 'baseRead')); + $this->_isPaused = false; + $this->baseRead($this->_socket, false); + } + } + + /** + * Base read handler. + * + * @param resource $socket + * @param bool $check_eof + * @return void + */ + public function baseRead($socket, $check_eof = true) + { + // SSL handshake. + if ($this->transport === 'ssl' && $this->_sslHandshakeCompleted !== true) { + stream_set_blocking($socket, true); + stream_set_timeout($socket, 1); + $ret = stream_socket_enable_crypto($socket, true, STREAM_CRYPTO_METHOD_SSLv23_SERVER); + if(!$ret) { + echo new \Exception('ssl handshake fail, stream_socket_enable_crypto return ' . var_export($ret, true)); + return $this->destroy(); + } + if (isset($this->onSslHandshake)) { + try { + call_user_func($this->onSslHandshake, $this); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + $this->_sslHandshakeCompleted = true; + if ($this->_sendBuffer) { + Worker::$globalEvent->add($socket, EventInterface::EV_WRITE, array($this, 'baseWrite')); + } + return; + } + + $buffer = fread($socket, self::READ_BUFFER_SIZE); + + // Check connection closed. + if ($buffer === '' || $buffer === false) { + if ($check_eof && (feof($socket) || !is_resource($socket) || $buffer === false)) { + $this->destroy(); + return; + } + } else { + $this->_recvBuffer .= $buffer; + } + + // If the application layer protocol has been set up. + if ($this->protocol) { + $parser = $this->protocol; + while ($this->_recvBuffer !== '' && !$this->_isPaused) { + // The current packet length is known. + if ($this->_currentPackageLength) { + // Data is not enough for a package. + if ($this->_currentPackageLength > strlen($this->_recvBuffer)) { + break; + } + } else { + // Get current package length. + $this->_currentPackageLength = $parser::input($this->_recvBuffer, $this); + // The packet length is unknown. + if ($this->_currentPackageLength === 0) { + break; + } elseif ($this->_currentPackageLength > 0 && $this->_currentPackageLength <= self::$maxPackageSize) { + // Data is not enough for a package. + if ($this->_currentPackageLength > strlen($this->_recvBuffer)) { + break; + } + } // Wrong package. + else { + echo 'error package. package_length=' . var_export($this->_currentPackageLength, true); + $this->destroy(); + return; + } + } + + // The data is enough for a packet. + self::$statistics['total_request']++; + // The current packet length is equal to the length of the buffer. + if (strlen($this->_recvBuffer) === $this->_currentPackageLength) { + $one_request_buffer = $this->_recvBuffer; + $this->_recvBuffer = ''; + } else { + // Get a full package from the buffer. + $one_request_buffer = substr($this->_recvBuffer, 0, $this->_currentPackageLength); + // Remove the current package from the receive buffer. + $this->_recvBuffer = substr($this->_recvBuffer, $this->_currentPackageLength); + } + // Reset the current packet length to 0. + $this->_currentPackageLength = 0; + if (!$this->onMessage) { + continue; + } + try { + // Decode request buffer before Emitting onMessage callback. + call_user_func($this->onMessage, $this, $parser::decode($one_request_buffer, $this)); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + return; + } + + if ($this->_recvBuffer === '' || $this->_isPaused) { + return; + } + + // Applications protocol is not set. + self::$statistics['total_request']++; + if (!$this->onMessage) { + $this->_recvBuffer = ''; + return; + } + try { + call_user_func($this->onMessage, $this, $this->_recvBuffer); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + // Clean receive buffer. + $this->_recvBuffer = ''; + } + + /** + * Base write handler. + * + * @return void|bool + */ + public function baseWrite() + { + $len = @fwrite($this->_socket, $this->_sendBuffer); + if ($len === strlen($this->_sendBuffer)) { + Worker::$globalEvent->del($this->_socket, EventInterface::EV_WRITE); + $this->_sendBuffer = ''; + // Try to emit onBufferDrain callback when the send buffer becomes empty. + if ($this->onBufferDrain) { + try { + call_user_func($this->onBufferDrain, $this); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + if ($this->_status === self::STATUS_CLOSING) { + $this->destroy(); + } + return true; + } + if ($len > 0) { + $this->_sendBuffer = substr($this->_sendBuffer, $len); + } else { + self::$statistics['send_fail']++; + $this->destroy(); + } + } + + /** + * This method pulls all the data out of a readable stream, and writes it to the supplied destination. + * + * @param TcpConnection $dest + * @return void + */ + public function pipe($dest) + { + $source = $this; + $this->onMessage = function ($source, $data) use ($dest) { + $dest->send($data); + }; + $this->onClose = function ($source) use ($dest) { + $dest->destroy(); + }; + $dest->onBufferFull = function ($dest) use ($source) { + $source->pauseRecv(); + }; + $dest->onBufferDrain = function ($dest) use ($source) { + $source->resumeRecv(); + }; + } + + /** + * Remove $length of data from receive buffer. + * + * @param int $length + * @return void + */ + public function consumeRecvBuffer($length) + { + $this->_recvBuffer = substr($this->_recvBuffer, $length); + } + + /** + * Close connection. + * + * @param mixed $data + * @param bool $raw + * @return void + */ + public function close($data = null, $raw = false) + { + if ($this->_status === self::STATUS_CLOSING || $this->_status === self::STATUS_CLOSED) { + return; + } else { + if ($data !== null) { + $this->send($data, $raw); + } + $this->_status = self::STATUS_CLOSING; + } + if ($this->_sendBuffer === '') { + $this->destroy(); + } + } + + /** + * Get the real socket. + * + * @return resource + */ + public function getSocket() + { + return $this->_socket; + } + + /** + * Check whether the send buffer will be full. + * + * @return void + */ + protected function checkBufferWillFull() + { + if ($this->maxSendBufferSize <= strlen($this->_sendBuffer)) { + if ($this->onBufferFull) { + try { + call_user_func($this->onBufferFull, $this); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + } + } + + /** + * Whether send buffer is full. + * + * @return bool + */ + protected function bufferIsFull() + { + // Buffer has been marked as full but still has data to send then the packet is discarded. + if ($this->maxSendBufferSize <= strlen($this->_sendBuffer)) { + if ($this->onError) { + try { + call_user_func($this->onError, $this, WORKERMAN_SEND_FAIL, 'send buffer full and drop package'); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + return true; + } + return false; + } + + /** + * Destroy connection. + * + * @return void + */ + public function destroy() + { + // Avoid repeated calls. + if ($this->_status === self::STATUS_CLOSED) { + return; + } + // Remove event listener. + Worker::$globalEvent->del($this->_socket, EventInterface::EV_READ); + Worker::$globalEvent->del($this->_socket, EventInterface::EV_WRITE); + // Close socket. + @fclose($this->_socket); + // Remove from worker->connections. + if ($this->worker) { + unset($this->worker->connections[$this->_id]); + } + $this->_status = self::STATUS_CLOSED; + // Try to emit onClose callback. + if ($this->onClose) { + try { + call_user_func($this->onClose, $this); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + // Try to emit protocol::onClose + if (method_exists($this->protocol, 'onClose')) { + try { + call_user_func(array($this->protocol, 'onClose'), $this); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + if ($this->_status === self::STATUS_CLOSED) { + // Cleaning up the callback to avoid memory leaks. + $this->onMessage = $this->onClose = $this->onError = $this->onBufferFull = $this->onBufferDrain = null; + } + } + + /** + * Destruct. + * + * @return void + */ + public function __destruct() + { + self::$statistics['connection_count']--; + } +} diff --git a/vendor/workerman/workerman/Connection/UdpConnection.php b/vendor/workerman/workerman/Connection/UdpConnection.php new file mode 100644 index 000000000..e4cd2fee4 --- /dev/null +++ b/vendor/workerman/workerman/Connection/UdpConnection.php @@ -0,0 +1,115 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Connection; + +/** + * UdpConnection. + */ +class UdpConnection extends ConnectionInterface +{ + /** + * Application layer protocol. + * The format is like this Workerman\\Protocols\\Http. + * + * @var \Workerman\Protocols\ProtocolInterface + */ + public $protocol = null; + + /** + * Udp socket. + * + * @var resource + */ + protected $_socket = null; + + /** + * Remote address. + * + * @var string + */ + protected $_remoteAddress = ''; + + /** + * Construct. + * + * @param resource $socket + * @param string $remote_address + */ + public function __construct($socket, $remote_address) + { + $this->_socket = $socket; + $this->_remoteAddress = $remote_address; + } + + /** + * Sends data on the connection. + * + * @param string $send_buffer + * @param bool $raw + * @return void|boolean + */ + public function send($send_buffer, $raw = false) + { + if (false === $raw && $this->protocol) { + $parser = $this->protocol; + $send_buffer = $parser::encode($send_buffer, $this); + if ($send_buffer === '') { + return null; + } + } + return strlen($send_buffer) === stream_socket_sendto($this->_socket, $send_buffer, 0, $this->_remoteAddress); + } + + /** + * Get remote IP. + * + * @return string + */ + public function getRemoteIp() + { + $pos = strrpos($this->_remoteAddress, ':'); + if ($pos) { + return trim(substr($this->_remoteAddress, 0, $pos), '[]'); + } + return ''; + } + + /** + * Get remote port. + * + * @return int + */ + public function getRemotePort() + { + if ($this->_remoteAddress) { + return (int)substr(strrchr($this->_remoteAddress, ':'), 1); + } + return 0; + } + + /** + * Close connection. + * + * @param mixed $data + * @param bool $raw + * @return bool + */ + public function close($data = null, $raw = false) + { + if ($data !== null) { + $this->send($data, $raw); + } + return true; + } +} diff --git a/vendor/workerman/workerman/Events/Ev.php b/vendor/workerman/workerman/Events/Ev.php new file mode 100644 index 000000000..acc1a3613 --- /dev/null +++ b/vendor/workerman/workerman/Events/Ev.php @@ -0,0 +1,173 @@ + + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events; + +use Workerman\Worker; + +/** + * ev eventloop + */ +class Ev implements EventInterface +{ + /** + * All listeners for read/write event. + * + * @var array + */ + protected $_allEvents = array(); + + /** + * Event listeners of signal. + * + * @var array + */ + protected $_eventSignal = array(); + + /** + * All timer event listeners. + * [func, args, event, flag, time_interval] + * + * @var array + */ + protected $_eventTimer = array(); + + /** + * Timer id. + * + * @var int + */ + protected static $_timerId = 1; + + /** + * Add a timer. + * {@inheritdoc} + */ + public function add($fd, $flag, $func, $args = null) + { + $callback = function ($event, $socket) use ($fd, $func) { + try { + call_user_func($func, $fd); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + }; + + switch ($flag) { + case self::EV_SIGNAL: + $event = new \EvSignal($fd, $callback); + $this->_eventSignal[$fd] = $event; + return true; + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + $repeat = $flag == self::EV_TIMER_ONCE ? 0 : $fd; + $param = array($func, (array)$args, $flag, $fd, self::$_timerId); + $event = new \EvTimer($fd, $repeat, array($this, 'timerCallback'), $param); + $this->_eventTimer[self::$_timerId] = $event; + return self::$_timerId++; + default : + $fd_key = (int)$fd; + $real_flag = $flag === self::EV_READ ? \Ev::READ : \Ev::WRITE; + $event = new \EvIo($fd, $real_flag, $callback); + $this->_allEvents[$fd_key][$flag] = $event; + return true; + } + + } + + /** + * Remove a timer. + * {@inheritdoc} + */ + public function del($fd, $flag) + { + switch ($flag) { + case self::EV_READ: + case self::EV_WRITE: + $fd_key = (int)$fd; + if (isset($this->_allEvents[$fd_key][$flag])) { + $this->_allEvents[$fd_key][$flag]->stop(); + unset($this->_allEvents[$fd_key][$flag]); + } + if (empty($this->_allEvents[$fd_key])) { + unset($this->_allEvents[$fd_key]); + } + break; + case self::EV_SIGNAL: + $fd_key = (int)$fd; + if (isset($this->_eventSignal[$fd_key])) { + $this->_allEvents[$fd_key][$flag]->stop(); + unset($this->_eventSignal[$fd_key]); + } + break; + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + if (isset($this->_eventTimer[$fd])) { + $this->_eventTimer[$fd]->stop(); + unset($this->_eventTimer[$fd]); + } + break; + } + return true; + } + + /** + * Timer callback. + * + * @param \EvWatcher $event + */ + public function timerCallback($event) + { + $param = $event->data; + $timer_id = $param[4]; + if ($param[2] === self::EV_TIMER_ONCE) { + $this->_eventTimer[$timer_id]->stop(); + unset($this->_eventTimer[$timer_id]); + } + try { + call_user_func_array($param[0], $param[1]); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + + /** + * Remove all timers. + * + * @return void + */ + public function clearAllTimer() + { + foreach ($this->_eventTimer as $event) { + $event->stop(); + } + $this->_eventTimer = array(); + } + + /** + * Main loop. + * + * @see EventInterface::loop() + */ + public function loop() + { + \Ev::run(); + } +} diff --git a/vendor/workerman/workerman/Events/Event.php b/vendor/workerman/workerman/Events/Event.php new file mode 100644 index 000000000..2d85aafee --- /dev/null +++ b/vendor/workerman/workerman/Events/Event.php @@ -0,0 +1,188 @@ + + * @copyright 鏈変釜楝<42765633@qq.com> + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events; + +use Workerman\Worker; + +/** + * libevent eventloop + */ +class Event implements EventInterface +{ + /** + * Event base. + * @var object + */ + protected $_eventBase = null; + + /** + * All listeners for read/write event. + * @var array + */ + protected $_allEvents = array(); + + /** + * Event listeners of signal. + * @var array + */ + protected $_eventSignal = array(); + + /** + * All timer event listeners. + * [func, args, event, flag, time_interval] + * @var array + */ + protected $_eventTimer = array(); + + /** + * Timer id. + * @var int + */ + protected static $_timerId = 1; + + /** + * construct + * @return void + */ + public function __construct() + { + $this->_eventBase = new \EventBase(); + } + + /** + * @see EventInterface::add() + */ + public function add($fd, $flag, $func, $args=array()) + { + switch ($flag) { + case self::EV_SIGNAL: + + $fd_key = (int)$fd; + $event = \Event::signal($this->_eventBase, $fd, $func); + if (!$event||!$event->add()) { + return false; + } + $this->_eventSignal[$fd_key] = $event; + return true; + + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + + $param = array($func, (array)$args, $flag, $fd, self::$_timerId); + $event = new \Event($this->_eventBase, -1, \Event::TIMEOUT|\Event::PERSIST, array($this, "timerCallback"), $param); + if (!$event||!$event->addTimer($fd)) { + return false; + } + $this->_eventTimer[self::$_timerId] = $event; + return self::$_timerId++; + + default : + $fd_key = (int)$fd; + $real_flag = $flag === self::EV_READ ? \Event::READ | \Event::PERSIST : \Event::WRITE | \Event::PERSIST; + $event = new \Event($this->_eventBase, $fd, $real_flag, $func, $fd); + if (!$event||!$event->add()) { + return false; + } + $this->_allEvents[$fd_key][$flag] = $event; + return true; + } + } + + /** + * @see Events\EventInterface::del() + */ + public function del($fd, $flag) + { + switch ($flag) { + + case self::EV_READ: + case self::EV_WRITE: + + $fd_key = (int)$fd; + if (isset($this->_allEvents[$fd_key][$flag])) { + $this->_allEvents[$fd_key][$flag]->del(); + unset($this->_allEvents[$fd_key][$flag]); + } + if (empty($this->_allEvents[$fd_key])) { + unset($this->_allEvents[$fd_key]); + } + break; + + case self::EV_SIGNAL: + + $fd_key = (int)$fd; + if (isset($this->_eventSignal[$fd_key])) { + $this->_allEvents[$fd_key][$flag]->del(); + unset($this->_eventSignal[$fd_key]); + } + break; + + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + if (isset($this->_eventTimer[$fd])) { + $this->_eventTimer[$fd]->del(); + unset($this->_eventTimer[$fd]); + } + break; + } + return true; + } + + /** + * Timer callback. + * @param null $fd + * @param int $what + * @param int $timer_id + */ + public function timerCallback($fd, $what, $param) + { + $timer_id = $param[4]; + + if ($param[2] === self::EV_TIMER_ONCE) { + $this->_eventTimer[$timer_id]->del(); + unset($this->_eventTimer[$timer_id]); + } + + try { + call_user_func_array($param[0], $param[1]); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + + /** + * @see Events\EventInterface::clearAllTimer() + * @return void + */ + public function clearAllTimer() + { + foreach ($this->_eventTimer as $event) { + $event->del(); + } + $this->_eventTimer = array(); + } + + + /** + * @see EventInterface::loop() + */ + public function loop() + { + $this->_eventBase->loop(); + } +} diff --git a/vendor/workerman/workerman/Events/EventInterface.php b/vendor/workerman/workerman/Events/EventInterface.php new file mode 100644 index 000000000..d71b5d438 --- /dev/null +++ b/vendor/workerman/workerman/Events/EventInterface.php @@ -0,0 +1,86 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events; + +interface EventInterface +{ + /** + * Read event. + * + * @var int + */ + const EV_READ = 1; + + /** + * Write event. + * + * @var int + */ + const EV_WRITE = 2; + + /** + * Signal event. + * + * @var int + */ + const EV_SIGNAL = 4; + + /** + * Timer event. + * + * @var int + */ + const EV_TIMER = 8; + + /** + * Timer once event. + * + * @var int + */ + const EV_TIMER_ONCE = 16; + + /** + * Add event listener to event loop. + * + * @param mixed $fd + * @param int $flag + * @param callable $func + * @param mixed $args + * @return bool + */ + public function add($fd, $flag, $func, $args = null); + + /** + * Remove event listener from event loop. + * + * @param mixed $fd + * @param int $flag + * @return bool + */ + public function del($fd, $flag); + + /** + * Remove all timers. + * + * @return void + */ + public function clearAllTimer(); + + /** + * Main loop. + * + * @return void + */ + public function loop(); +} diff --git a/vendor/workerman/workerman/Events/Libevent.php b/vendor/workerman/workerman/Events/Libevent.php new file mode 100644 index 000000000..5644bf1e7 --- /dev/null +++ b/vendor/workerman/workerman/Events/Libevent.php @@ -0,0 +1,205 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events; + +use Workerman\Worker; + +/** + * libevent eventloop + */ +class Libevent implements EventInterface +{ + /** + * Event base. + * + * @var resource + */ + protected $_eventBase = null; + + /** + * All listeners for read/write event. + * + * @var array + */ + protected $_allEvents = array(); + + /** + * Event listeners of signal. + * + * @var array + */ + protected $_eventSignal = array(); + + /** + * All timer event listeners. + * [func, args, event, flag, time_interval] + * + * @var array + */ + protected $_eventTimer = array(); + + /** + * construct + */ + public function __construct() + { + $this->_eventBase = event_base_new(); + } + + /** + * {@inheritdoc} + */ + public function add($fd, $flag, $func, $args = array()) + { + switch ($flag) { + case self::EV_SIGNAL: + $fd_key = (int)$fd; + $real_flag = EV_SIGNAL | EV_PERSIST; + $this->_eventSignal[$fd_key] = event_new(); + if (!event_set($this->_eventSignal[$fd_key], $fd, $real_flag, $func, null)) { + return false; + } + if (!event_base_set($this->_eventSignal[$fd_key], $this->_eventBase)) { + return false; + } + if (!event_add($this->_eventSignal[$fd_key])) { + return false; + } + return true; + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + $event = event_new(); + $timer_id = (int)$event; + if (!event_set($event, 0, EV_TIMEOUT, array($this, 'timerCallback'), $timer_id)) { + return false; + } + + if (!event_base_set($event, $this->_eventBase)) { + return false; + } + + $time_interval = $fd * 1000000; + if (!event_add($event, $time_interval)) { + return false; + } + $this->_eventTimer[$timer_id] = array($func, (array)$args, $event, $flag, $time_interval); + return $timer_id; + + default : + $fd_key = (int)$fd; + $real_flag = $flag === self::EV_READ ? EV_READ | EV_PERSIST : EV_WRITE | EV_PERSIST; + + $event = event_new(); + + if (!event_set($event, $fd, $real_flag, $func, null)) { + return false; + } + + if (!event_base_set($event, $this->_eventBase)) { + return false; + } + + if (!event_add($event)) { + return false; + } + + $this->_allEvents[$fd_key][$flag] = $event; + + return true; + } + + } + + /** + * {@inheritdoc} + */ + public function del($fd, $flag) + { + switch ($flag) { + case self::EV_READ: + case self::EV_WRITE: + $fd_key = (int)$fd; + if (isset($this->_allEvents[$fd_key][$flag])) { + event_del($this->_allEvents[$fd_key][$flag]); + unset($this->_allEvents[$fd_key][$flag]); + } + if (empty($this->_allEvents[$fd_key])) { + unset($this->_allEvents[$fd_key]); + } + break; + case self::EV_SIGNAL: + $fd_key = (int)$fd; + if (isset($this->_eventSignal[$fd_key])) { + event_del($this->_eventSignal[$fd_key]); + unset($this->_eventSignal[$fd_key]); + } + break; + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + // 杩欓噷 fd 涓簍imerid + if (isset($this->_eventTimer[$fd])) { + event_del($this->_eventTimer[$fd][2]); + unset($this->_eventTimer[$fd]); + } + break; + } + return true; + } + + /** + * Timer callback. + * + * @param mixed $_null1 + * @param int $_null2 + * @param mixed $timer_id + */ + protected function timerCallback($_null1, $_null2, $timer_id) + { + if ($this->_eventTimer[$timer_id][3] === self::EV_TIMER) { + event_add($this->_eventTimer[$timer_id][2], $this->_eventTimer[$timer_id][4]); + } + try { + call_user_func_array($this->_eventTimer[$timer_id][0], $this->_eventTimer[$timer_id][1]); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + if (isset($this->_eventTimer[$timer_id]) && $this->_eventTimer[$timer_id][3] === self::EV_TIMER_ONCE) { + $this->del($timer_id, self::EV_TIMER_ONCE); + } + } + + /** + * {@inheritdoc} + */ + public function clearAllTimer() + { + foreach ($this->_eventTimer as $task_data) { + event_del($task_data[2]); + } + $this->_eventTimer = array(); + } + + /** + * {@inheritdoc} + */ + public function loop() + { + event_base_loop($this->_eventBase); + } +} + diff --git a/vendor/workerman/workerman/Events/React.php b/vendor/workerman/workerman/Events/React.php new file mode 100644 index 000000000..be8c87fb7 --- /dev/null +++ b/vendor/workerman/workerman/Events/React.php @@ -0,0 +1,248 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events; +use React\EventLoop\LoopInterface; +use React\EventLoop\Timer\TimerInterface; + +/** + * select eventloop + */ +class React implements LoopInterface +{ + /** + * @var React\EventLoop\LoopInterface + */ + protected $_loop = null; + + /** + * React constructor. + */ + public function __construct() { + if (function_exists('event_base_new')) { + $this->_loop = new \Workerman\Events\React\LibEventLoop(); + } elseif (class_exists('EventBase', false)) { + $this->_loop = new \Workerman\Events\React\ExtEventLoop(); + } else { + $this->_loop = new \Workerman\Events\React\StreamSelectLoop(); + } + } + + /** + * Add event listener to event loop. + * + * @param $fd + * @param $flag + * @param $func + * @param array $args + * @return bool + */ + public function add($fd, $flag, $func, $args = array()) + { + $args = (array)$args; + switch ($flag) { + case EventInterface::EV_READ: + return $this->_loop->addReadStream($fd, $func); + case EventInterface::EV_WRITE: + return $this->_loop->addWriteStream($fd, $func); + case EventInterface::EV_SIGNAL: + return $this->_loop->addSignal($fd, $func); + case EventInterface::EV_TIMER: + return $this->_loop->addPeriodicTimer($fd, function() use ($func, $args) { + call_user_func_array($func, $args); + }); + case EventInterface::EV_TIMER_ONCE: + return $this->_loop->addTimer($fd, function() use ($func, $args) { + call_user_func_array($func, $args); + }); + } + return false; + } + + /** + * Remove event listener from event loop. + * + * @param mixed $fd + * @param int $flag + * @return bool + */ + public function del($fd, $flag) + { + switch ($flag) { + case EventInterface::EV_READ: + return $this->_loop->removeReadStream($fd); + case EventInterface::EV_WRITE: + return $this->_loop->removeWriteStream($fd); + case EventInterface::EV_SIGNAL: + return $this->_loop->removeSignal($fd); + case EventInterface::EV_TIMER: + case EventInterface::EV_TIMER_ONCE; + if ($fd !== null){ + return $this->_loop->cancelTimer($fd); + } + } + return false; + } + + + /** + * Main loop. + * + * @return void + */ + public function loop() + { + $this->_loop->run(); + } + + /** + * Register a listener to be notified when a stream is ready to read. + * + * @param resource $stream The PHP stream resource to check. + * @param callable $listener Invoked when the stream is ready. + */ + public function addReadStream($stream, callable $listener) { + return call_user_func(array($this->_loop, 'addReadStream'), $stream, $listener); + } + + /** + * Register a listener to be notified when a stream is ready to write. + * + * @param resource $stream The PHP stream resource to check. + * @param callable $listener Invoked when the stream is ready. + */ + public function addWriteStream($stream, callable $listener) { + return call_user_func(array($this->_loop, 'addWriteStream'), $stream, $listener); + } + + /** + * Remove the read event listener for the given stream. + * + * @param resource $stream The PHP stream resource. + */ + public function removeReadStream($stream) { + return call_user_func(array($this->_loop, 'removeReadStream'), $stream); + } + + /** + * Remove the write event listener for the given stream. + * + * @param resource $stream The PHP stream resource. + */ + public function removeWriteStream($stream) { + return call_user_func(array($this->_loop, 'removeWriteStream'), $stream); + } + + /** + * Remove all listeners for the given stream. + * + * @param resource $stream The PHP stream resource. + */ + public function removeStream($stream) { + return call_user_func(array($this->_loop, 'removeStream'), $stream); + } + + /** + * Enqueue a callback to be invoked once after the given interval. + * + * The execution order of timers scheduled to execute at the same time is + * not guaranteed. + * + * @param int|float $interval The number of seconds to wait before execution. + * @param callable $callback The callback to invoke. + * + * @return TimerInterface + */ + public function addTimer($interval, callable $callback) { + return call_user_func(array($this->_loop, 'addTimer'), $interval, $callback); + } + + /** + * Enqueue a callback to be invoked repeatedly after the given interval. + * + * The execution order of timers scheduled to execute at the same time is + * not guaranteed. + * + * @param int|float $interval The number of seconds to wait before execution. + * @param callable $callback The callback to invoke. + * + * @return TimerInterface + */ + public function addPeriodicTimer($interval, callable $callback) { + return call_user_func(array($this->_loop, 'addPeriodicTimer'), $interval, $callback); + } + + /** + * Cancel a pending timer. + * + * @param TimerInterface $timer The timer to cancel. + */ + public function cancelTimer(TimerInterface $timer) { + return call_user_func(array($this->_loop, 'cancelTimer'), $timer); + } + + /** + * Check if a given timer is active. + * + * @param TimerInterface $timer The timer to check. + * + * @return boolean True if the timer is still enqueued for execution. + */ + public function isTimerActive(TimerInterface $timer) { + return call_user_func(array($this->_loop, 'isTimerActive'), $timer); + } + + /** + * Schedule a callback to be invoked on the next tick of the event loop. + * + * Callbacks are guaranteed to be executed in the order they are enqueued, + * before any timer or stream events. + * + * @param callable $listener The callback to invoke. + */ + public function nextTick(callable $listener) { + return call_user_func(array($this->_loop, 'nextTick'), $listener); + } + + /** + * Schedule a callback to be invoked on a future tick of the event loop. + * + * Callbacks are guaranteed to be executed in the order they are enqueued. + * + * @param callable $listener The callback to invoke. + */ + public function futureTick(callable $listener) { + return call_user_func(array($this->_loop, 'futureTick'), $listener); + } + + /** + * Perform a single iteration of the event loop. + */ + public function tick() { + return call_user_func(array($this->_loop, 'tick')); + } + + /** + * Run the event loop until there are no more tasks to perform. + */ + public function run() { + return call_user_func(array($this->_loop, 'run')); + } + + /** + * Instruct a running event loop to stop. + */ + public function stop() { + return call_user_func(array($this->_loop, 'stop')); + } +} diff --git a/vendor/workerman/workerman/Events/React/ExtEventLoop.php b/vendor/workerman/workerman/Events/React/ExtEventLoop.php new file mode 100644 index 000000000..ccb504182 --- /dev/null +++ b/vendor/workerman/workerman/Events/React/ExtEventLoop.php @@ -0,0 +1,76 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events\React; + +/** + * Class ExtEventLoop + * @package Workerman\Events\React + */ +class ExtEventLoop extends \React\EventLoop\ExtEventLoop +{ + /** + * Event base. + * + * @var EventBase + */ + protected $_eventBase = null; + + /** + * All signal Event instances. + * + * @var array + */ + protected $_signalEvents = array(); + + /** + * Construct + */ + public function __construct() + { + parent::__construct(); + $class = new \ReflectionClass('\React\EventLoop\ExtEventLoop'); + $property = $class->getProperty('eventBase'); + $property->setAccessible(true); + $this->_eventBase = $property->getValue($this); + } + + /** + * Add signal handler. + * + * @param $signal + * @param $callback + * @return bool + */ + public function addSignal($signal, $callback) + { + $event = \Event::signal($this->_eventBase, $signal, $callback); + if (!$event||!$event->add()) { + return false; + } + $this->_signalEvents[$signal] = $event; + } + + /** + * Remove signal handler. + * + * @param $signal + */ + public function removeSignal($signal) + { + if (isset($this->_signalEvents[$signal])) { + $this->_signalEvents[$signal]->del(); + unset($this->_signalEvents[$signal]); + } + } +} diff --git a/vendor/workerman/workerman/Events/React/LibEventLoop.php b/vendor/workerman/workerman/Events/React/LibEventLoop.php new file mode 100644 index 000000000..1dbf3900d --- /dev/null +++ b/vendor/workerman/workerman/Events/React/LibEventLoop.php @@ -0,0 +1,77 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events\React; + +/** + * Class LibEventLoop + * @package Workerman\Events\React + */ +class LibEventLoop extends \React\EventLoop\LibEventLoop +{ + /** + * Event base. + * + * @var event_base resource + */ + protected $_eventBase = null; + + /** + * All signal Event instances. + * + * @var array + */ + protected $_signalEvents = array(); + + /** + * Construct. + */ + public function __construct() + { + parent::__construct(); + $class = new \ReflectionClass('\React\EventLoop\LibEventLoop'); + $property = $class->getProperty('eventBase'); + $property->setAccessible(true); + $this->_eventBase = $property->getValue($this); + } + + /** + * Add signal handler. + * + * @param $signal + * @param $callback + * @return bool + */ + public function addSignal($signal, $callback) + { + $event = event_new(); + $this->_signalEvents[$signal] = $event; + event_set($event, $signal, EV_SIGNAL | EV_PERSIST, $callback); + event_base_set($event, $this->_eventBase); + event_add($event); + } + + /** + * Remove signal handler. + * + * @param $signal + */ + public function removeSignal($signal) + { + if (isset($this->_signalEvents[$signal])) { + $event = $this->_signalEvents[$signal]; + event_del($event); + unset($this->_signalEvents[$signal]); + } + } +} diff --git a/vendor/workerman/workerman/Events/React/StreamSelectLoop.php b/vendor/workerman/workerman/Events/React/StreamSelectLoop.php new file mode 100644 index 000000000..5ab8bc7e8 --- /dev/null +++ b/vendor/workerman/workerman/Events/React/StreamSelectLoop.php @@ -0,0 +1,77 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events\React; + +/** + * Class StreamSelectLoop + * @package Workerman\Events\React + */ +class StreamSelectLoop extends \React\EventLoop\StreamSelectLoop +{ + /** + * Add signal handler. + * + * @param $signal + * @param $callback + * @return bool + */ + public function addSignal($signal, $callback) + { + if(PHP_EOL !== "\r\n") { + pcntl_signal($signal, $callback); + } + } + + /** + * Remove signal handler. + * + * @param $signal + */ + public function removeSignal($signal) + { + if(PHP_EOL !== "\r\n") { + pcntl_signal($signal, SIG_IGN); + } + } + + /** + * Emulate a stream_select() implementation that does not break when passed + * empty stream arrays. + * + * @param array &$read An array of read streams to select upon. + * @param array &$write An array of write streams to select upon. + * @param integer|null $timeout Activity timeout in microseconds, or null to wait forever. + * + * @return integer|false The total number of streams that are ready for read/write. + * Can return false if stream_select() is interrupted by a signal. + */ + protected function streamSelect(array &$read, array &$write, $timeout) + { + if ($read || $write) { + $except = null; + // Calls signal handlers for pending signals + pcntl_signal_dispatch(); + // suppress warnings that occur, when stream_select is interrupted by a signal + return @stream_select($read, $write, $except, $timeout === null ? null : 0, $timeout); + } + + // Calls signal handlers for pending signals + if(PHP_EOL !== "\r\n") { + pcntl_signal_dispatch(); + } + $timeout && usleep($timeout); + + return 0; + } +} diff --git a/vendor/workerman/workerman/Events/Select.php b/vendor/workerman/workerman/Events/Select.php new file mode 100644 index 000000000..1b9d16f16 --- /dev/null +++ b/vendor/workerman/workerman/Events/Select.php @@ -0,0 +1,263 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Events; + +/** + * select eventloop + */ +class Select implements EventInterface +{ + /** + * All listeners for read/write event. + * + * @var array + */ + public $_allEvents = array(); + + /** + * Event listeners of signal. + * + * @var array + */ + public $_signalEvents = array(); + + /** + * Fds waiting for read event. + * + * @var array + */ + protected $_readFds = array(); + + /** + * Fds waiting for write event. + * + * @var array + */ + protected $_writeFds = array(); + + /** + * Timer scheduler. + * {['data':timer_id, 'priority':run_timestamp], ..} + * + * @var \SplPriorityQueue + */ + protected $_scheduler = null; + + /** + * All timer event listeners. + * [[func, args, flag, timer_interval], ..] + * + * @var array + */ + protected $_task = array(); + + /** + * Timer id. + * + * @var int + */ + protected $_timerId = 1; + + /** + * Select timeout. + * + * @var int + */ + protected $_selectTimeout = 100000000; + + /** + * Paired socket channels + * + * @var array + */ + protected $channel = array(); + + /** + * Construct. + */ + public function __construct() + { + // Create a pipeline and put into the collection of the read to read the descriptor to avoid empty polling. + $this->channel = stream_socket_pair(STREAM_PF_UNIX, STREAM_SOCK_STREAM, STREAM_IPPROTO_IP); + if ($this->channel) { + stream_set_blocking($this->channel[0], 0); + $this->_readFds[0] = $this->channel[0]; + } + // Init SplPriorityQueue. + $this->_scheduler = new \SplPriorityQueue(); + $this->_scheduler->setExtractFlags(\SplPriorityQueue::EXTR_BOTH); + } + + /** + * {@inheritdoc} + */ + public function add($fd, $flag, $func, $args = array()) + { + switch ($flag) { + case self::EV_READ: + $fd_key = (int)$fd; + $this->_allEvents[$fd_key][$flag] = array($func, $fd); + $this->_readFds[$fd_key] = $fd; + break; + case self::EV_WRITE: + $fd_key = (int)$fd; + $this->_allEvents[$fd_key][$flag] = array($func, $fd); + $this->_writeFds[$fd_key] = $fd; + break; + case self::EV_SIGNAL: + $fd_key = (int)$fd; + $this->_signalEvents[$fd_key][$flag] = array($func, $fd); + pcntl_signal($fd, array($this, 'signalHandler')); + break; + case self::EV_TIMER: + case self::EV_TIMER_ONCE: + $run_time = microtime(true) + $fd; + $this->_scheduler->insert($this->_timerId, -$run_time); + $this->_task[$this->_timerId] = array($func, (array)$args, $flag, $fd); + $this->tick(); + return $this->_timerId++; + } + + return true; + } + + /** + * Signal handler. + * + * @param int $signal + */ + public function signalHandler($signal) + { + call_user_func_array($this->_signalEvents[$signal][self::EV_SIGNAL][0], array($signal)); + } + + /** + * {@inheritdoc} + */ + public function del($fd, $flag) + { + $fd_key = (int)$fd; + switch ($flag) { + case self::EV_READ: + unset($this->_allEvents[$fd_key][$flag], $this->_readFds[$fd_key]); + if (empty($this->_allEvents[$fd_key])) { + unset($this->_allEvents[$fd_key]); + } + return true; + case self::EV_WRITE: + unset($this->_allEvents[$fd_key][$flag], $this->_writeFds[$fd_key]); + if (empty($this->_allEvents[$fd_key])) { + unset($this->_allEvents[$fd_key]); + } + return true; + case self::EV_SIGNAL: + unset($this->_signalEvents[$fd_key]); + pcntl_signal($fd, SIG_IGN); + break; + case self::EV_TIMER: + case self::EV_TIMER_ONCE; + unset($this->_task[$fd_key]); + return true; + } + return false; + } + + /** + * Tick for timer. + * + * @return void + */ + protected function tick() + { + while (!$this->_scheduler->isEmpty()) { + $scheduler_data = $this->_scheduler->top(); + $timer_id = $scheduler_data['data']; + $next_run_time = -$scheduler_data['priority']; + $time_now = microtime(true); + $this->_selectTimeout = ($next_run_time - $time_now) * 1000000; + if ($this->_selectTimeout <= 0) { + $this->_scheduler->extract(); + + if (!isset($this->_task[$timer_id])) { + continue; + } + + // [func, args, flag, timer_interval] + $task_data = $this->_task[$timer_id]; + if ($task_data[2] === self::EV_TIMER) { + $next_run_time = $time_now + $task_data[3]; + $this->_scheduler->insert($timer_id, -$next_run_time); + } + call_user_func_array($task_data[0], $task_data[1]); + if (isset($this->_task[$timer_id]) && $task_data[2] === self::EV_TIMER_ONCE) { + $this->del($timer_id, self::EV_TIMER_ONCE); + } + continue; + } + return; + } + $this->_selectTimeout = 100000000; + } + + /** + * {@inheritdoc} + */ + public function clearAllTimer() + { + $this->_scheduler = new \SplPriorityQueue(); + $this->_scheduler->setExtractFlags(\SplPriorityQueue::EXTR_BOTH); + $this->_task = array(); + } + + /** + * {@inheritdoc} + */ + public function loop() + { + $e = null; + while (1) { + // Calls signal handlers for pending signals + pcntl_signal_dispatch(); + + $read = $this->_readFds; + $write = $this->_writeFds; + // Waiting read/write/signal/timeout events. + $ret = @stream_select($read, $write, $e, 0, $this->_selectTimeout); + + if (!$this->_scheduler->isEmpty()) { + $this->tick(); + } + + if (!$ret) { + continue; + } + + foreach ($read as $fd) { + $fd_key = (int)$fd; + if (isset($this->_allEvents[$fd_key][self::EV_READ])) { + call_user_func_array($this->_allEvents[$fd_key][self::EV_READ][0], + array($this->_allEvents[$fd_key][self::EV_READ][1])); + } + } + + foreach ($write as $fd) { + $fd_key = (int)$fd; + if (isset($this->_allEvents[$fd_key][self::EV_WRITE])) { + call_user_func_array($this->_allEvents[$fd_key][self::EV_WRITE][0], + array($this->_allEvents[$fd_key][self::EV_WRITE][1])); + } + } + } + } +} diff --git a/vendor/workerman/workerman/Lib/Constants.php b/vendor/workerman/workerman/Lib/Constants.php new file mode 100644 index 000000000..81440747a --- /dev/null +++ b/vendor/workerman/workerman/Lib/Constants.php @@ -0,0 +1,35 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +// Date.timezone +if (!ini_get('date.timezone')) { + date_default_timezone_set('Asia/Shanghai'); +} +// Display errors. +ini_set('display_errors', 'on'); +// Reporting all. +error_reporting(E_ALL); + +// For onError callback. +define('WORKERMAN_CONNECT_FAIL', 1); +// For onError callback. +define('WORKERMAN_SEND_FAIL', 2); + +// Compatible with php7 +if(!class_exists('Error')) +{ + class Error extends Exception + { + } +} \ No newline at end of file diff --git a/vendor/workerman/workerman/Lib/Timer.php b/vendor/workerman/workerman/Lib/Timer.php new file mode 100644 index 000000000..b382cbb4b --- /dev/null +++ b/vendor/workerman/workerman/Lib/Timer.php @@ -0,0 +1,176 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Lib; + +use Workerman\Events\EventInterface; +use Exception; + +/** + * Timer. + * + * example: + * Workerman\Lib\Timer::add($time_interval, callback, array($arg1, $arg2..)); + */ +class Timer +{ + /** + * Tasks that based on ALARM signal. + * [ + * run_time => [[$func, $args, $persistent, time_interval],[$func, $args, $persistent, time_interval],..]], + * run_time => [[$func, $args, $persistent, time_interval],[$func, $args, $persistent, time_interval],..]], + * .. + * ] + * + * @var array + */ + protected static $_tasks = array(); + + /** + * event + * + * @var \Workerman\Events\EventInterface + */ + protected static $_event = null; + + /** + * Init. + * + * @param \Workerman\Events\EventInterface $event + * @return void + */ + public static function init($event = null) + { + if ($event) { + self::$_event = $event; + } else { + pcntl_signal(SIGALRM, array('\Workerman\Lib\Timer', 'signalHandle'), false); + } + } + + /** + * ALARM signal handler. + * + * @return void + */ + public static function signalHandle() + { + if (!self::$_event) { + pcntl_alarm(1); + self::tick(); + } + } + + /** + * Add a timer. + * + * @param int $time_interval + * @param callback $func + * @param mixed $args + * @param bool $persistent + * @return bool + */ + public static function add($time_interval, $func, $args = array(), $persistent = true) + { + if ($time_interval <= 0) { + echo new Exception("bad time_interval"); + return false; + } + + if (self::$_event) { + return self::$_event->add($time_interval, + $persistent ? EventInterface::EV_TIMER : EventInterface::EV_TIMER_ONCE, $func, $args); + } + + if (!is_callable($func)) { + echo new Exception("not callable"); + return false; + } + + if (empty(self::$_tasks)) { + pcntl_alarm(1); + } + + $time_now = time(); + $run_time = $time_now + $time_interval; + if (!isset(self::$_tasks[$run_time])) { + self::$_tasks[$run_time] = array(); + } + self::$_tasks[$run_time][] = array($func, (array)$args, $persistent, $time_interval); + return true; + } + + + /** + * Tick. + * + * @return void + */ + public static function tick() + { + if (empty(self::$_tasks)) { + pcntl_alarm(0); + return; + } + + $time_now = time(); + foreach (self::$_tasks as $run_time => $task_data) { + if ($time_now >= $run_time) { + foreach ($task_data as $index => $one_task) { + $task_func = $one_task[0]; + $task_args = $one_task[1]; + $persistent = $one_task[2]; + $time_interval = $one_task[3]; + try { + call_user_func_array($task_func, $task_args); + } catch (\Exception $e) { + echo $e; + } + if ($persistent) { + self::add($time_interval, $task_func, $task_args); + } + } + unset(self::$_tasks[$run_time]); + } + } + } + + /** + * Remove a timer. + * + * @param mixed $timer_id + * @return bool + */ + public static function del($timer_id) + { + if (self::$_event) { + return self::$_event->del($timer_id, EventInterface::EV_TIMER); + } + + return false; + } + + /** + * Remove all timers. + * + * @return void + */ + public static function delAll() + { + self::$_tasks = array(); + pcntl_alarm(0); + if (self::$_event) { + self::$_event->clearAllTimer(); + } + } +} diff --git a/vendor/workerman/workerman/MIT-LICENSE.txt b/vendor/workerman/workerman/MIT-LICENSE.txt new file mode 100644 index 000000000..fd6b1c83f --- /dev/null +++ b/vendor/workerman/workerman/MIT-LICENSE.txt @@ -0,0 +1,21 @@ +The MIT License + +Copyright (c) 2009-2015 walkor and contributors (see https://github.com/walkor/workerman/contributors) + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/vendor/workerman/workerman/Protocols/Frame.php b/vendor/workerman/workerman/Protocols/Frame.php new file mode 100644 index 000000000..4a6b13ecb --- /dev/null +++ b/vendor/workerman/workerman/Protocols/Frame.php @@ -0,0 +1,61 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Protocols; + +use Workerman\Connection\TcpConnection; + +/** + * Frame Protocol. + */ +class Frame +{ + /** + * Check the integrity of the package. + * + * @param string $buffer + * @param TcpConnection $connection + * @return int + */ + public static function input($buffer, TcpConnection $connection) + { + if (strlen($buffer) < 4) { + return 0; + } + $unpack_data = unpack('Ntotal_length', $buffer); + return $unpack_data['total_length']; + } + + /** + * Decode. + * + * @param string $buffer + * @return string + */ + public static function decode($buffer) + { + return substr($buffer, 4); + } + + /** + * Encode. + * + * @param string $buffer + * @return string + */ + public static function encode($buffer) + { + $total_length = 4 + strlen($buffer); + return pack('N', $total_length) . $buffer; + } +} diff --git a/vendor/workerman/workerman/Protocols/Http.php b/vendor/workerman/workerman/Protocols/Http.php new file mode 100644 index 000000000..30c3ba918 --- /dev/null +++ b/vendor/workerman/workerman/Protocols/Http.php @@ -0,0 +1,577 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Protocols; + +use Workerman\Connection\TcpConnection; +use Workerman\Worker; + +/** + * http protocol + */ +class Http +{ + /** + * The supported HTTP methods + * @var array + */ + public static $methods = array('GET', 'POST', 'PUT', 'DELETE', 'HEAD', 'OPTIONS'); + + /** + * Check the integrity of the package. + * + * @param string $recv_buffer + * @param TcpConnection $connection + * @return int + */ + public static function input($recv_buffer, TcpConnection $connection) + { + if (!strpos($recv_buffer, "\r\n\r\n")) { + // Judge whether the package length exceeds the limit. + if (strlen($recv_buffer) >= TcpConnection::$maxPackageSize) { + $connection->close(); + return 0; + } + return 0; + } + + list($header,) = explode("\r\n\r\n", $recv_buffer, 2); + $method = substr($header, 0, strpos($header, ' ')); + + if(in_array($method, static::$methods)) { + return static::getRequestSize($header, $method); + }else{ + $connection->send("HTTP/1.1 400 Bad Request\r\n\r\n", true); + return 0; + } + } + + /** + * Get whole size of the request + * includes the request headers and request body. + * @param string $header The request headers + * @param string $method The request method + * @return integer + */ + protected static function getRequestSize($header, $method) + { + if($method=='GET') { + return strlen($header) + 4; + } + $match = array(); + if (preg_match("/\r\nContent-Length: ?(\d+)/i", $header, $match)) { + $content_length = isset($match[1]) ? $match[1] : 0; + return $content_length + strlen($header) + 4; + } + return 0; + } + + /** + * Parse $_POST銆$_GET銆$_COOKIE. + * + * @param string $recv_buffer + * @param TcpConnection $connection + * @return array + */ + public static function decode($recv_buffer, TcpConnection $connection) + { + // Init. + $_POST = $_GET = $_COOKIE = $_REQUEST = $_SESSION = $_FILES = array(); + $GLOBALS['HTTP_RAW_POST_DATA'] = ''; + // Clear cache. + HttpCache::$header = array('Connection' => 'Connection: keep-alive'); + HttpCache::$instance = new HttpCache(); + // $_SERVER + $_SERVER = array( + 'QUERY_STRING' => '', + 'REQUEST_METHOD' => '', + 'REQUEST_URI' => '', + 'SERVER_PROTOCOL' => '', + 'SERVER_SOFTWARE' => 'workerman/'.Worker::VERSION, + 'SERVER_NAME' => '', + 'HTTP_HOST' => '', + 'HTTP_USER_AGENT' => '', + 'HTTP_ACCEPT' => '', + 'HTTP_ACCEPT_LANGUAGE' => '', + 'HTTP_ACCEPT_ENCODING' => '', + 'HTTP_COOKIE' => '', + 'HTTP_CONNECTION' => '', + 'REMOTE_ADDR' => '', + 'REMOTE_PORT' => '0', + ); + + // Parse headers. + list($http_header, $http_body) = explode("\r\n\r\n", $recv_buffer, 2); + $header_data = explode("\r\n", $http_header); + + list($_SERVER['REQUEST_METHOD'], $_SERVER['REQUEST_URI'], $_SERVER['SERVER_PROTOCOL']) = explode(' ', + $header_data[0]); + + $http_post_boundary = ''; + unset($header_data[0]); + foreach ($header_data as $content) { + // \r\n\r\n + if (empty($content)) { + continue; + } + list($key, $value) = explode(':', $content, 2); + $key = str_replace('-', '_', strtoupper($key)); + $value = trim($value); + $_SERVER['HTTP_' . $key] = $value; + switch ($key) { + // HTTP_HOST + case 'HOST': + $tmp = explode(':', $value); + $_SERVER['SERVER_NAME'] = $tmp[0]; + if (isset($tmp[1])) { + $_SERVER['SERVER_PORT'] = $tmp[1]; + } + break; + // cookie + case 'COOKIE': + parse_str(str_replace('; ', '&', $_SERVER['HTTP_COOKIE']), $_COOKIE); + break; + // content-type + case 'CONTENT_TYPE': + if (!preg_match('/boundary="?(\S+)"?/', $value, $match)) { + $_SERVER['CONTENT_TYPE'] = $value; + } else { + $_SERVER['CONTENT_TYPE'] = 'multipart/form-data'; + $http_post_boundary = '--' . $match[1]; + } + break; + case 'CONTENT_LENGTH': + $_SERVER['CONTENT_LENGTH'] = $value; + break; + } + } + + // Parse $_POST. + if ($_SERVER['REQUEST_METHOD'] === 'POST') { + if (isset($_SERVER['CONTENT_TYPE']) && $_SERVER['CONTENT_TYPE'] === 'multipart/form-data') { + self::parseUploadFiles($http_body, $http_post_boundary); + } else { + parse_str($http_body, $_POST); + // $GLOBALS['HTTP_RAW_POST_DATA'] + $GLOBALS['HTTP_RAW_REQUEST_DATA'] = $GLOBALS['HTTP_RAW_POST_DATA'] = $http_body; + } + } + + if ($_SERVER['REQUEST_METHOD'] === 'PUT') { + $GLOBALS['HTTP_RAW_REQUEST_DATA'] = $http_body; + } + + if ($_SERVER['REQUEST_METHOD'] === 'DELETE') { + $GLOBALS['HTTP_RAW_REQUEST_DATA'] = $http_body; + } + + // QUERY_STRING + $_SERVER['QUERY_STRING'] = parse_url($_SERVER['REQUEST_URI'], PHP_URL_QUERY); + if ($_SERVER['QUERY_STRING']) { + // $GET + parse_str($_SERVER['QUERY_STRING'], $_GET); + } else { + $_SERVER['QUERY_STRING'] = ''; + } + + // REQUEST + $_REQUEST = array_merge($_GET, $_POST); + + // REMOTE_ADDR REMOTE_PORT + $_SERVER['REMOTE_ADDR'] = $connection->getRemoteIp(); + $_SERVER['REMOTE_PORT'] = $connection->getRemotePort(); + + return array('get' => $_GET, 'post' => $_POST, 'cookie' => $_COOKIE, 'server' => $_SERVER, 'files' => $_FILES); + } + + /** + * Http encode. + * + * @param string $content + * @param TcpConnection $connection + * @return string + */ + public static function encode($content, TcpConnection $connection) + { + // Default http-code. + if (!isset(HttpCache::$header['Http-Code'])) { + $header = "HTTP/1.1 200 OK\r\n"; + } else { + $header = HttpCache::$header['Http-Code'] . "\r\n"; + unset(HttpCache::$header['Http-Code']); + } + + // Content-Type + if (!isset(HttpCache::$header['Content-Type'])) { + $header .= "Content-Type: text/html;charset=utf-8\r\n"; + } + + // other headers + foreach (HttpCache::$header as $key => $item) { + if ('Set-Cookie' === $key && is_array($item)) { + foreach ($item as $it) { + $header .= $it . "\r\n"; + } + } else { + $header .= $item . "\r\n"; + } + } + + // header + $header .= "Server: workerman/" . Worker::VERSION . "\r\nContent-Length: " . strlen($content) . "\r\n\r\n"; + + // save session + self::sessionWriteClose(); + + // the whole http package + return $header . $content; + } + + /** + * 璁剧疆http澶 + * + * @return bool|void + */ + public static function header($content, $replace = true, $http_response_code = 0) + { + if (PHP_SAPI != 'cli') { + return $http_response_code ? header($content, $replace, $http_response_code) : header($content, $replace); + } + if (strpos($content, 'HTTP') === 0) { + $key = 'Http-Code'; + } else { + $key = strstr($content, ":", true); + if (empty($key)) { + return false; + } + } + + if ('location' === strtolower($key) && !$http_response_code) { + return self::header($content, true, 302); + } + + if (isset(HttpCache::$codes[$http_response_code])) { + HttpCache::$header['Http-Code'] = "HTTP/1.1 $http_response_code " . HttpCache::$codes[$http_response_code]; + if ($key === 'Http-Code') { + return true; + } + } + + if ($key === 'Set-Cookie') { + HttpCache::$header[$key][] = $content; + } else { + HttpCache::$header[$key] = $content; + } + + return true; + } + + /** + * Remove header. + * + * @param string $name + * @return void + */ + public static function headerRemove($name) + { + if (PHP_SAPI != 'cli') { + header_remove($name); + return; + } + unset(HttpCache::$header[$name]); + } + + /** + * Set cookie. + * + * @param string $name + * @param string $value + * @param integer $maxage + * @param string $path + * @param string $domain + * @param bool $secure + * @param bool $HTTPOnly + * @return bool|void + */ + public static function setcookie( + $name, + $value = '', + $maxage = 0, + $path = '', + $domain = '', + $secure = false, + $HTTPOnly = false + ) { + if (PHP_SAPI != 'cli') { + return setcookie($name, $value, $maxage, $path, $domain, $secure, $HTTPOnly); + } + return self::header( + 'Set-Cookie: ' . $name . '=' . rawurlencode($value) + . (empty($domain) ? '' : '; Domain=' . $domain) + . (empty($maxage) ? '' : '; Max-Age=' . $maxage) + . (empty($path) ? '' : '; Path=' . $path) + . (!$secure ? '' : '; Secure') + . (!$HTTPOnly ? '' : '; HttpOnly'), false); + } + + /** + * sessionStart + * + * @return bool + */ + public static function sessionStart() + { + if (PHP_SAPI != 'cli') { + return session_start(); + } + + self::tryGcSessions(); + + if (HttpCache::$instance->sessionStarted) { + echo "already sessionStarted\n"; + return true; + } + HttpCache::$instance->sessionStarted = true; + // Generate a SID. + if (!isset($_COOKIE[HttpCache::$sessionName]) || !is_file(HttpCache::$sessionPath . '/ses' . $_COOKIE[HttpCache::$sessionName])) { + $file_name = tempnam(HttpCache::$sessionPath, 'ses'); + if (!$file_name) { + return false; + } + HttpCache::$instance->sessionFile = $file_name; + $session_id = substr(basename($file_name), strlen('ses')); + return self::setcookie( + HttpCache::$sessionName + , $session_id + , ini_get('session.cookie_lifetime') + , ini_get('session.cookie_path') + , ini_get('session.cookie_domain') + , ini_get('session.cookie_secure') + , ini_get('session.cookie_httponly') + ); + } + if (!HttpCache::$instance->sessionFile) { + HttpCache::$instance->sessionFile = HttpCache::$sessionPath . '/ses' . $_COOKIE[HttpCache::$sessionName]; + } + // Read session from session file. + if (HttpCache::$instance->sessionFile) { + $raw = file_get_contents(HttpCache::$instance->sessionFile); + if ($raw) { + session_decode($raw); + } + } + return true; + } + + /** + * Save session. + * + * @return bool + */ + public static function sessionWriteClose() + { + if (PHP_SAPI != 'cli') { + return session_write_close(); + } + if (!empty(HttpCache::$instance->sessionStarted) && !empty($_SESSION)) { + $session_str = session_encode(); + if ($session_str && HttpCache::$instance->sessionFile) { + return file_put_contents(HttpCache::$instance->sessionFile, $session_str); + } + } + return empty($_SESSION); + } + + /** + * End, like call exit in php-fpm. + * + * @param string $msg + * @throws \Exception + */ + public static function end($msg = '') + { + if (PHP_SAPI != 'cli') { + exit($msg); + } + if ($msg) { + echo $msg; + } + throw new \Exception('jump_exit'); + } + + /** + * Get mime types. + * + * @return string + */ + public static function getMimeTypesFile() + { + return __DIR__ . '/Http/mime.types'; + } + + /** + * Parse $_FILES. + * + * @param string $http_body + * @param string $http_post_boundary + * @return void + */ + protected static function parseUploadFiles($http_body, $http_post_boundary) + { + $http_body = substr($http_body, 0, strlen($http_body) - (strlen($http_post_boundary) + 4)); + $boundary_data_array = explode($http_post_boundary . "\r\n", $http_body); + if ($boundary_data_array[0] === '') { + unset($boundary_data_array[0]); + } + foreach ($boundary_data_array as $boundary_data_buffer) { + list($boundary_header_buffer, $boundary_value) = explode("\r\n\r\n", $boundary_data_buffer, 2); + // Remove \r\n from the end of buffer. + $boundary_value = substr($boundary_value, 0, -2); + foreach (explode("\r\n", $boundary_header_buffer) as $item) { + list($header_key, $header_value) = explode(": ", $item); + $header_key = strtolower($header_key); + switch ($header_key) { + case "content-disposition": + // Is file data. + if (preg_match('/name=".*?"; filename="(.*?)"$/', $header_value, $match)) { + // Parse $_FILES. + $_FILES[] = array( + 'file_name' => $match[1], + 'file_data' => $boundary_value, + 'file_size' => strlen($boundary_value), + ); + continue; + } // Is post field. + else { + // Parse $_POST. + if (preg_match('/name="(.*?)"$/', $header_value, $match)) { + $_POST[$match[1]] = $boundary_value; + } + } + break; + } + } + } + } + + /** + * Try GC sessions. + * + * @return void + */ + public static function tryGcSessions() + { + if (HttpCache::$sessionGcProbability <= 0 || + HttpCache::$sessionGcDivisor <= 0 || + rand(1, HttpCache::$sessionGcDivisor) > HttpCache::$sessionGcProbability) { + return; + } + + $time_now = time(); + foreach(glob(HttpCache::$sessionPath.'/ses*') as $file) { + if(is_file($file) && $time_now - filemtime($file) > HttpCache::$sessionGcMaxLifeTime) { + unlink($file); + } + } + } +} + +/** + * Http cache for the current http response. + */ +class HttpCache +{ + public static $codes = array( + 100 => 'Continue', + 101 => 'Switching Protocols', + 200 => 'OK', + 201 => 'Created', + 202 => 'Accepted', + 203 => 'Non-Authoritative Information', + 204 => 'No Content', + 205 => 'Reset Content', + 206 => 'Partial Content', + 300 => 'Multiple Choices', + 301 => 'Moved Permanently', + 302 => 'Found', + 303 => 'See Other', + 304 => 'Not Modified', + 305 => 'Use Proxy', + 306 => '(Unused)', + 307 => 'Temporary Redirect', + 400 => 'Bad Request', + 401 => 'Unauthorized', + 402 => 'Payment Required', + 403 => 'Forbidden', + 404 => 'Not Found', + 405 => 'Method Not Allowed', + 406 => 'Not Acceptable', + 407 => 'Proxy Authentication Required', + 408 => 'Request Timeout', + 409 => 'Conflict', + 410 => 'Gone', + 411 => 'Length Required', + 412 => 'Precondition Failed', + 413 => 'Request Entity Too Large', + 414 => 'Request-URI Too Long', + 415 => 'Unsupported Media Type', + 416 => 'Requested Range Not Satisfiable', + 417 => 'Expectation Failed', + 422 => 'Unprocessable Entity', + 423 => 'Locked', + 500 => 'Internal Server Error', + 501 => 'Not Implemented', + 502 => 'Bad Gateway', + 503 => 'Service Unavailable', + 504 => 'Gateway Timeout', + 505 => 'HTTP Version Not Supported', + ); + + /** + * @var HttpCache + */ + public static $instance = null; + public static $header = array(); + public static $sessionPath = ''; + public static $sessionName = ''; + public static $sessionGcProbability = 1; + public static $sessionGcDivisor = 1000; + public static $sessionGcMaxLifeTime = 1440; + public $sessionStarted = false; + public $sessionFile = ''; + + public static function init() + { + self::$sessionName = ini_get('session.name'); + self::$sessionPath = session_save_path(); + if (!self::$sessionPath || strpos(self::$sessionPath, 'tcp://') === 0) { + self::$sessionPath = sys_get_temp_dir(); + } + + if ($gc_probability = ini_get('session.gc_probability')) { + self::$sessionGcProbability = $gc_probability; + } + + if ($gc_divisor = ini_get('session.gc_divisor')) { + self::$sessionGcDivisor = $gc_divisor; + } + + if ($gc_max_life_time = ini_get('session.gc_maxlifetime')) { + self::$sessionGcMaxLifeTime = $gc_max_life_time; + } + + @\session_start(); + } +} + +HttpCache::init(); diff --git a/vendor/workerman/workerman/Protocols/Http/mime.types b/vendor/workerman/workerman/Protocols/Http/mime.types new file mode 100644 index 000000000..8a218b22a --- /dev/null +++ b/vendor/workerman/workerman/Protocols/Http/mime.types @@ -0,0 +1,80 @@ + +types { + text/html html htm shtml; + text/css css; + text/xml xml; + image/gif gif; + image/jpeg jpeg jpg; + application/x-javascript js; + application/atom+xml atom; + application/rss+xml rss; + + text/mathml mml; + text/plain txt; + text/vnd.sun.j2me.app-descriptor jad; + text/vnd.wap.wml wml; + text/x-component htc; + + image/png png; + image/tiff tif tiff; + image/vnd.wap.wbmp wbmp; + image/x-icon ico; + image/x-jng jng; + image/x-ms-bmp bmp; + image/svg+xml svg svgz; + image/webp webp; + + application/java-archive jar war ear; + application/mac-binhex40 hqx; + application/msword doc; + application/pdf pdf; + application/postscript ps eps ai; + application/rtf rtf; + application/vnd.ms-excel xls; + application/vnd.ms-powerpoint ppt; + application/vnd.wap.wmlc wmlc; + application/vnd.google-earth.kml+xml kml; + application/vnd.google-earth.kmz kmz; + application/x-7z-compressed 7z; + application/x-cocoa cco; + application/x-java-archive-diff jardiff; + application/x-java-jnlp-file jnlp; + application/x-makeself run; + application/x-perl pl pm; + application/x-pilot prc pdb; + application/x-rar-compressed rar; + application/x-redhat-package-manager rpm; + application/x-sea sea; + application/x-shockwave-flash swf; + application/x-stuffit sit; + application/x-tcl tcl tk; + application/x-x509-ca-cert der pem crt; + application/x-xpinstall xpi; + application/xhtml+xml xhtml; + application/zip zip; + + application/octet-stream bin exe dll; + application/octet-stream deb; + application/octet-stream dmg; + application/octet-stream eot; + application/octet-stream iso img; + application/octet-stream msi msp msm; + + audio/midi mid midi kar; + audio/mpeg mp3; + audio/ogg ogg; + audio/x-m4a m4a; + audio/x-realaudio ra; + + video/3gpp 3gpp 3gp; + video/mp4 mp4; + video/mpeg mpeg mpg; + video/quicktime mov; + video/webm webm; + video/x-flv flv; + video/x-m4v m4v; + video/x-mng mng; + video/x-ms-asf asx asf; + video/x-ms-wmv wmv; + video/x-msvideo avi; +} diff --git a/vendor/workerman/workerman/Protocols/ProtocolInterface.php b/vendor/workerman/workerman/Protocols/ProtocolInterface.php new file mode 100644 index 000000000..9afe98498 --- /dev/null +++ b/vendor/workerman/workerman/Protocols/ProtocolInterface.php @@ -0,0 +1,52 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Protocols; + +use Workerman\Connection\ConnectionInterface; + +/** + * Protocol interface + */ +interface ProtocolInterface +{ + /** + * Check the integrity of the package. + * Please return the length of package. + * If length is unknow please return 0 that mean wating more data. + * If the package has something wrong please return false the connection will be closed. + * + * @param ConnectionInterface $connection + * @param string $recv_buffer + * @return int|false + */ + public static function input($recv_buffer, ConnectionInterface $connection); + + /** + * Decode package and emit onMessage($message) callback, $message is the result that decode returned. + * + * @param ConnectionInterface $connection + * @param string $recv_buffer + * @return mixed + */ + public static function decode($recv_buffer, ConnectionInterface $connection); + + /** + * Encode package brefore sending to client. + * + * @param ConnectionInterface $connection + * @param mixed $data + * @return string + */ + public static function encode($data, ConnectionInterface $connection); +} diff --git a/vendor/workerman/workerman/Protocols/Text.php b/vendor/workerman/workerman/Protocols/Text.php new file mode 100644 index 000000000..189baf416 --- /dev/null +++ b/vendor/workerman/workerman/Protocols/Text.php @@ -0,0 +1,70 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Protocols; + +use Workerman\Connection\TcpConnection; + +/** + * Text Protocol. + */ +class Text +{ + /** + * Check the integrity of the package. + * + * @param string $buffer + * @param TcpConnection $connection + * @return int + */ + public static function input($buffer, TcpConnection $connection) + { + // Judge whether the package length exceeds the limit. + if (strlen($buffer) >= TcpConnection::$maxPackageSize) { + $connection->close(); + return 0; + } + // Find the position of "\n". + $pos = strpos($buffer, "\n"); + // No "\n", packet length is unknown, continue to wait for the data so return 0. + if ($pos === false) { + return 0; + } + // Return the current package length. + return $pos + 1; + } + + /** + * Encode. + * + * @param string $buffer + * @return string + */ + public static function encode($buffer) + { + // Add "\n" + return $buffer . "\n"; + } + + /** + * Decode. + * + * @param string $buffer + * @return string + */ + public static function decode($buffer) + { + // Remove "\n" + return trim($buffer); + } +} diff --git a/vendor/workerman/workerman/Protocols/Websocket.php b/vendor/workerman/workerman/Protocols/Websocket.php new file mode 100644 index 000000000..2ab5635da --- /dev/null +++ b/vendor/workerman/workerman/Protocols/Websocket.php @@ -0,0 +1,473 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Protocols; + +use Workerman\Connection\ConnectionInterface; +use Workerman\Connection\TcpConnection; +use Workerman\Worker; + +/** + * WebSocket protocol. + */ +class Websocket implements \Workerman\Protocols\ProtocolInterface +{ + /** + * Websocket blob type. + * + * @var string + */ + const BINARY_TYPE_BLOB = "\x81"; + + /** + * Websocket arraybuffer type. + * + * @var string + */ + const BINARY_TYPE_ARRAYBUFFER = "\x82"; + + /** + * Check the integrity of the package. + * + * @param string $buffer + * @param ConnectionInterface $connection + * @return int + */ + public static function input($buffer, ConnectionInterface $connection) + { + // Receive length. + $recv_len = strlen($buffer); + // We need more data. + if ($recv_len < 2) { + return 0; + } + + // Has not yet completed the handshake. + if (empty($connection->websocketHandshake)) { + return static::dealHandshake($buffer, $connection); + } + + // Buffer websocket frame data. + if ($connection->websocketCurrentFrameLength) { + // We need more frame data. + if ($connection->websocketCurrentFrameLength > $recv_len) { + // Return 0, because it is not clear the full packet length, waiting for the frame of fin=1. + return 0; + } + } else { + $firstbyte = ord($buffer[0]); + $secondbyte = ord($buffer[1]); + $data_len = $secondbyte & 127; + $is_fin_frame = $firstbyte >> 7; + $masked = $secondbyte >> 7; + $opcode = $firstbyte & 0xf; + switch ($opcode) { + case 0x0: + break; + // Blob type. + case 0x1: + break; + // Arraybuffer type. + case 0x2: + break; + // Close package. + case 0x8: + // Try to emit onWebSocketClose callback. + if (isset($connection->onWebSocketClose)) { + try { + call_user_func($connection->onWebSocketClose, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } // Close connection. + else { + $connection->close(); + } + return 0; + // Ping package. + case 0x9: + // Try to emit onWebSocketPing callback. + if (isset($connection->onWebSocketPing)) { + try { + call_user_func($connection->onWebSocketPing, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } // Send pong package to client. + else { + $connection->send(pack('H*', '8a00'), true); + } + + // Consume data from receive buffer. + if (!$data_len) { + $head_len = $masked ? 6 : 2; + $connection->consumeRecvBuffer($head_len); + if ($recv_len > $head_len) { + return static::input(substr($buffer, $head_len), $connection); + } + return 0; + } + break; + // Pong package. + case 0xa: + // Try to emit onWebSocketPong callback. + if (isset($connection->onWebSocketPong)) { + try { + call_user_func($connection->onWebSocketPong, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + // Consume data from receive buffer. + if (!$data_len) { + $head_len = $masked ? 6 : 2; + $connection->consumeRecvBuffer($head_len); + if ($recv_len > $head_len) { + return static::input(substr($buffer, $head_len), $connection); + } + return 0; + } + break; + // Wrong opcode. + default : + echo "error opcode $opcode and close websocket connection. Buffer:" . bin2hex($buffer) . "\n"; + $connection->close(); + return 0; + } + + // Calculate packet length. + $head_len = 6; + if ($data_len === 126) { + $head_len = 8; + if ($head_len > $recv_len) { + return 0; + } + $pack = unpack('nn/ntotal_len', $buffer); + $data_len = $pack['total_len']; + } else { + if ($data_len === 127) { + $head_len = 14; + if ($head_len > $recv_len) { + return 0; + } + $arr = unpack('n/N2c', $buffer); + $data_len = $arr['c1']*4294967296 + $arr['c2']; + } + } + $current_frame_length = $head_len + $data_len; + + $total_package_size = strlen($connection->websocketDataBuffer) + $current_frame_length; + if ($total_package_size > TcpConnection::$maxPackageSize) { + echo "error package. package_length=$total_package_size\n"; + $connection->close(); + return 0; + } + + if ($is_fin_frame) { + return $current_frame_length; + } else { + $connection->websocketCurrentFrameLength = $current_frame_length; + } + } + + // Received just a frame length data. + if ($connection->websocketCurrentFrameLength === $recv_len) { + static::decode($buffer, $connection); + $connection->consumeRecvBuffer($connection->websocketCurrentFrameLength); + $connection->websocketCurrentFrameLength = 0; + return 0; + } // The length of the received data is greater than the length of a frame. + elseif ($connection->websocketCurrentFrameLength < $recv_len) { + static::decode(substr($buffer, 0, $connection->websocketCurrentFrameLength), $connection); + $connection->consumeRecvBuffer($connection->websocketCurrentFrameLength); + $current_frame_length = $connection->websocketCurrentFrameLength; + $connection->websocketCurrentFrameLength = 0; + // Continue to read next frame. + return static::input(substr($buffer, $current_frame_length), $connection); + } // The length of the received data is less than the length of a frame. + else { + return 0; + } + } + + /** + * Websocket encode. + * + * @param string $buffer + * @param ConnectionInterface $connection + * @return string + */ + public static function encode($buffer, ConnectionInterface $connection) + { + if (!is_scalar($buffer)) { + throw new \Exception("You can't send(" . gettype($buffer) . ") to client, you need to convert it to a string. "); + } + $len = strlen($buffer); + if (empty($connection->websocketType)) { + $connection->websocketType = static::BINARY_TYPE_BLOB; + } + + $first_byte = $connection->websocketType; + + if ($len <= 125) { + $encode_buffer = $first_byte . chr($len) . $buffer; + } else { + if ($len <= 65535) { + $encode_buffer = $first_byte . chr(126) . pack("n", $len) . $buffer; + } else { + $encode_buffer = $first_byte . chr(127) . pack("xxxxN", $len) . $buffer; + } + } + + // Handshake not completed so temporary buffer websocket data waiting for send. + if (empty($connection->websocketHandshake)) { + if (empty($connection->tmpWebsocketData)) { + $connection->tmpWebsocketData = ''; + } + // If buffer has already full then discard the current package. + if (strlen($connection->tmpWebsocketData) > $connection->maxSendBufferSize) { + if ($connection->onError) { + try { + call_user_func($connection->onError, $connection, WORKERMAN_SEND_FAIL, 'send buffer full and drop package'); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + return ''; + } + $connection->tmpWebsocketData .= $encode_buffer; + // Check buffer is full. + if ($connection->maxSendBufferSize <= strlen($connection->tmpWebsocketData)) { + if ($connection->onBufferFull) { + try { + call_user_func($connection->onBufferFull, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + } + + // Return empty string. + return ''; + } + + return $encode_buffer; + } + + /** + * Websocket decode. + * + * @param string $buffer + * @param ConnectionInterface $connection + * @return string + */ + public static function decode($buffer, ConnectionInterface $connection) + { + $masks = $data = $decoded = null; + $len = ord($buffer[1]) & 127; + if ($len === 126) { + $masks = substr($buffer, 4, 4); + $data = substr($buffer, 8); + } else { + if ($len === 127) { + $masks = substr($buffer, 10, 4); + $data = substr($buffer, 14); + } else { + $masks = substr($buffer, 2, 4); + $data = substr($buffer, 6); + } + } + for ($index = 0; $index < strlen($data); $index++) { + $decoded .= $data[$index] ^ $masks[$index % 4]; + } + if ($connection->websocketCurrentFrameLength) { + $connection->websocketDataBuffer .= $decoded; + return $connection->websocketDataBuffer; + } else { + if ($connection->websocketDataBuffer !== '') { + $decoded = $connection->websocketDataBuffer . $decoded; + $connection->websocketDataBuffer = ''; + } + return $decoded; + } + } + + /** + * Websocket handshake. + * + * @param string $buffer + * @param \Workerman\Connection\TcpConnection $connection + * @return int + */ + protected static function dealHandshake($buffer, $connection) + { + // HTTP protocol. + if (0 === strpos($buffer, 'GET')) { + // Find \r\n\r\n. + $heder_end_pos = strpos($buffer, "\r\n\r\n"); + if (!$heder_end_pos) { + return 0; + } + $header_length = $heder_end_pos + 4; + + // Get Sec-WebSocket-Key. + $Sec_WebSocket_Key = ''; + if (preg_match("/Sec-WebSocket-Key: *(.*?)\r\n/i", $buffer, $match)) { + $Sec_WebSocket_Key = $match[1]; + } else { + $connection->send("HTTP/1.1 400 Bad Request\r\n\r\n400 Bad Request
    Sec-WebSocket-Key not found.
    This is a WebSocket service and can not be accessed via HTTP.
    See http://wiki.workerman.net/Error1 for detail.", + true); + $connection->close(); + return 0; + } + // Calculation websocket key. + $new_key = base64_encode(sha1($Sec_WebSocket_Key . "258EAFA5-E914-47DA-95CA-C5AB0DC85B11", true)); + // Handshake response data. + $handshake_message = "HTTP/1.1 101 Switching Protocols\r\n"; + $handshake_message .= "Upgrade: websocket\r\n"; + $handshake_message .= "Sec-WebSocket-Version: 13\r\n"; + $handshake_message .= "Connection: Upgrade\r\n"; + $handshake_message .= "Server: workerman/".Worker::VERSION."\r\n"; + $handshake_message .= "Sec-WebSocket-Accept: " . $new_key . "\r\n\r\n"; + // Mark handshake complete.. + $connection->websocketHandshake = true; + // Websocket data buffer. + $connection->websocketDataBuffer = ''; + // Current websocket frame length. + $connection->websocketCurrentFrameLength = 0; + // Current websocket frame data. + $connection->websocketCurrentFrameBuffer = ''; + // Consume handshake data. + $connection->consumeRecvBuffer($header_length); + // Send handshake response. + $connection->send($handshake_message, true); + + // There are data waiting to be sent. + if (!empty($connection->tmpWebsocketData)) { + $connection->send($connection->tmpWebsocketData, true); + $connection->tmpWebsocketData = ''; + } + // blob or arraybuffer + if (empty($connection->websocketType)) { + $connection->websocketType = static::BINARY_TYPE_BLOB; + } + // Try to emit onWebSocketConnect callback. + if (isset($connection->onWebSocketConnect)) { + static::parseHttpHeader($buffer); + try { + call_user_func($connection->onWebSocketConnect, $connection, $buffer); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + if (!empty($_SESSION) && class_exists('\GatewayWorker\Lib\Context')) { + $connection->session = \GatewayWorker\Lib\Context::sessionEncode($_SESSION); + } + $_GET = $_SERVER = $_SESSION = $_COOKIE = array(); + } + if (strlen($buffer) > $header_length) { + return static::input(substr($buffer, $header_length), $connection); + } + return 0; + } // Is flash policy-file-request. + elseif (0 === strpos($buffer, 'send($policy_xml, true); + $connection->consumeRecvBuffer(strlen($buffer)); + return 0; + } + // Bad websocket handshake request. + $connection->send("HTTP/1.1 400 Bad Request\r\n\r\n400 Bad Request
    Invalid handshake data for websocket.
    See http://wiki.workerman.net/Error1 for detail.", + true); + $connection->close(); + return 0; + } + + /** + * Parse http header. + * + * @param string $buffer + * @return void + */ + protected static function parseHttpHeader($buffer) + { + // Parse headers. + list($http_header, ) = explode("\r\n\r\n", $buffer, 2); + $header_data = explode("\r\n", $http_header); + + if ($_SERVER) { + $_SERVER = array(); + } + + list($_SERVER['REQUEST_METHOD'], $_SERVER['REQUEST_URI'], $_SERVER['SERVER_PROTOCOL']) = explode(' ', + $header_data[0]); + + unset($header_data[0]); + foreach ($header_data as $content) { + // \r\n\r\n + if (empty($content)) { + continue; + } + list($key, $value) = explode(':', $content, 2); + $key = str_replace('-', '_', strtoupper($key)); + $value = trim($value); + $_SERVER['HTTP_' . $key] = $value; + switch ($key) { + // HTTP_HOST + case 'HOST': + $tmp = explode(':', $value); + $_SERVER['SERVER_NAME'] = $tmp[0]; + if (isset($tmp[1])) { + $_SERVER['SERVER_PORT'] = $tmp[1]; + } + break; + // cookie + case 'COOKIE': + parse_str(str_replace('; ', '&', $_SERVER['HTTP_COOKIE']), $_COOKIE); + break; + } + } + + // QUERY_STRING + $_SERVER['QUERY_STRING'] = parse_url($_SERVER['REQUEST_URI'], PHP_URL_QUERY); + if ($_SERVER['QUERY_STRING']) { + // $GET + parse_str($_SERVER['QUERY_STRING'], $_GET); + } else { + $_SERVER['QUERY_STRING'] = ''; + } + } +} diff --git a/vendor/workerman/workerman/Protocols/Ws.php b/vendor/workerman/workerman/Protocols/Ws.php new file mode 100644 index 000000000..da1451f40 --- /dev/null +++ b/vendor/workerman/workerman/Protocols/Ws.php @@ -0,0 +1,433 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman\Protocols; + +use Workerman\Worker; +use Workerman\Lib\Timer; +use Workerman\Connection\TcpConnection; + +/** + * Websocket protocol for client. + */ +class Ws +{ + /** + * Websocket blob type. + * + * @var string + */ + const BINARY_TYPE_BLOB = "\x81"; + + /** + * Websocket arraybuffer type. + * + * @var string + */ + const BINARY_TYPE_ARRAYBUFFER = "\x82"; + + /** + * Check the integrity of the package. + * + * @param string $buffer + * @param ConnectionInterface $connection + * @return int + */ + public static function input($buffer, $connection) + { + if (empty($connection->handshakeStep)) { + echo "recv data before handshake. Buffer:" . bin2hex($buffer) . "\n"; + return false; + } + // Recv handshake response + if ($connection->handshakeStep === 1) { + return self::dealHandshake($buffer, $connection); + } + $recv_len = strlen($buffer); + if ($recv_len < 2) { + return 0; + } + // Buffer websocket frame data. + if ($connection->websocketCurrentFrameLength) { + // We need more frame data. + if ($connection->websocketCurrentFrameLength > $recv_len) { + // Return 0, because it is not clear the full packet length, waiting for the frame of fin=1. + return 0; + } + } else { + + $firstbyte = ord($buffer[0]); + $secondbyte = ord($buffer[1]); + $data_len = $secondbyte & 127; + $is_fin_frame = $firstbyte >> 7; + $masked = $secondbyte >> 7; + $opcode = $firstbyte & 0xf; + + switch ($opcode) { + case 0x0: + break; + // Blob type. + case 0x1: + break; + // Arraybuffer type. + case 0x2: + break; + // Close package. + case 0x8: + // Try to emit onWebSocketClose callback. + if (isset($connection->onWebSocketClose)) { + try { + call_user_func($connection->onWebSocketClose, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } // Close connection. + else { + $connection->close(); + } + return 0; + // Ping package. + case 0x9: + // Try to emit onWebSocketPing callback. + if (isset($connection->onWebSocketPing)) { + try { + call_user_func($connection->onWebSocketPing, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } // Send pong package to client. + else { + $connection->send(pack('H*', '8a00'), true); + } + // Consume data from receive buffer. + if (!$data_len) { + $head_len = $masked ? 6 : 2; + $connection->consumeRecvBuffer($head_len); + if ($recv_len > $head_len) { + return self::input(substr($buffer, $head_len), $connection); + } + return 0; + } + break; + // Pong package. + case 0xa: + // Try to emit onWebSocketPong callback. + if (isset($connection->onWebSocketPong)) { + try { + call_user_func($connection->onWebSocketPong, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + // Consume data from receive buffer. + if (!$data_len) { + $head_len = $masked ? 6 : 2; + $connection->consumeRecvBuffer($head_len); + if ($recv_len > $head_len) { + return self::input(substr($buffer, $head_len), $connection); + } + return 0; + } + break; + // Wrong opcode. + default : + echo "error opcode $opcode and close websocket connection. Buffer:" . $buffer . "\n"; + $connection->close(); + return 0; + } + // Calculate packet length. + if ($data_len === 126) { + if (strlen($buffer) < 6) { + return 0; + } + $pack = unpack('nn/ntotal_len', $buffer); + $current_frame_length = $pack['total_len'] + 4; + } else if ($data_len === 127) { + if (strlen($buffer) < 10) { + return 0; + } + $arr = unpack('n/N2c', $buffer); + $current_frame_length = $arr['c1']*4294967296 + $arr['c2'] + 10; + } else { + $current_frame_length = $data_len + 2; + } + + $total_package_size = strlen($connection->websocketDataBuffer) + $current_frame_length; + if ($total_package_size > TcpConnection::$maxPackageSize) { + echo "error package. package_length=$total_package_size\n"; + $connection->close(); + return 0; + } + + if ($is_fin_frame) { + return $current_frame_length; + } else { + $connection->websocketCurrentFrameLength = $current_frame_length; + } + } + // Received just a frame length data. + if ($connection->websocketCurrentFrameLength === $recv_len) { + self::decode($buffer, $connection); + $connection->consumeRecvBuffer($connection->websocketCurrentFrameLength); + $connection->websocketCurrentFrameLength = 0; + return 0; + } // The length of the received data is greater than the length of a frame. + elseif ($connection->websocketCurrentFrameLength < $recv_len) { + self::decode(substr($buffer, 0, $connection->websocketCurrentFrameLength), $connection); + $connection->consumeRecvBuffer($connection->websocketCurrentFrameLength); + $current_frame_length = $connection->websocketCurrentFrameLength; + $connection->websocketCurrentFrameLength = 0; + // Continue to read next frame. + return self::input(substr($buffer, $current_frame_length), $connection); + } // The length of the received data is less than the length of a frame. + else { + return 0; + } + } + + /** + * Websocket encode. + * + * @param string $buffer + * @param ConnectionInterface $connection + * @return string + */ + public static function encode($payload, $connection) + { + if (empty($connection->websocketType)) { + $connection->websocketType = self::BINARY_TYPE_BLOB; + } + $payload = (string)$payload; + if (empty($connection->handshakeStep)) { + self::sendHandshake($connection); + } + $mask = 1; + $mask_key = "\x00\x00\x00\x00"; + + $pack = ''; + $length = $length_flag = strlen($payload); + if (65535 < $length) { + $pack = pack('NN', ($length & 0xFFFFFFFF00000000) >> 32, $length & 0x00000000FFFFFFFF); + $length_flag = 127; + } else if (125 < $length) { + $pack = pack('n*', $length); + $length_flag = 126; + } + + $head = ($mask << 7) | $length_flag; + $head = $connection->websocketType . chr($head) . $pack; + + $frame = $head . $mask_key; + // append payload to frame: + for ($i = 0; $i < $length; $i++) { + $frame .= $payload[$i] ^ $mask_key[$i % 4]; + } + if ($connection->handshakeStep === 1) { + // If buffer has already full then discard the current package. + if (strlen($connection->tmpWebsocketData) > $connection->maxSendBufferSize) { + if ($connection->onError) { + try { + call_user_func($connection->onError, $connection, WORKERMAN_SEND_FAIL, 'send buffer full and drop package'); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + return ''; + } + $connection->tmpWebsocketData = $connection->tmpWebsocketData . $frame; + // Check buffer is full. + if ($connection->maxSendBufferSize <= strlen($connection->tmpWebsocketData)) { + if ($connection->onBufferFull) { + try { + call_user_func($connection->onBufferFull, $connection); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + } + return ''; + } + return $frame; + } + + /** + * Websocket decode. + * + * @param string $buffer + * @param ConnectionInterface $connection + * @return string + */ + public static function decode($bytes, $connection) + { + $masked = $bytes[1] >> 7; + $data_length = $masked ? ord($bytes[1]) & 127 : ord($bytes[1]); + $decoded_data = ''; + if ($masked === true) { + if ($data_length === 126) { + $mask = substr($bytes, 4, 4); + $coded_data = substr($bytes, 8); + } else if ($data_length === 127) { + $mask = substr($bytes, 10, 4); + $coded_data = substr($bytes, 14); + } else { + $mask = substr($bytes, 2, 4); + $coded_data = substr($bytes, 6); + } + for ($i = 0; $i < strlen($coded_data); $i++) { + $decoded_data .= $coded_data[$i] ^ $mask[$i % 4]; + } + } else { + if ($data_length === 126) { + $decoded_data = substr($bytes, 4); + } else if ($data_length === 127) { + $decoded_data = substr($bytes, 10); + } else { + $decoded_data = substr($bytes, 2); + } + } + if ($connection->websocketCurrentFrameLength) { + $connection->websocketDataBuffer .= $decoded_data; + return $connection->websocketDataBuffer; + } else { + if ($connection->websocketDataBuffer !== '') { + $decoded_data = $connection->websocketDataBuffer . $decoded_data; + $connection->websocketDataBuffer = ''; + } + return $decoded_data; + } + } + + /** + * Send websocket handshake data. + * + * @return void + */ + public static function onConnect($connection) + { + self::sendHandshake($connection); + } + + /** + * Clean + * + * @param $connection + */ + public static function onClose($connection) + { + $connection->handshakeStep = null; + $connection->websocketCurrentFrameLength = 0; + $connection->tmpWebsocketData = ''; + $connection->websocketDataBuffer = ''; + if (!empty($connection->websocketPingTimer)) { + Timer::del($connection->websocketPingTimer); + $connection->websocketPingTimer = null; + } + } + + /** + * Send websocket handshake. + * + * @param \Workerman\Connection\TcpConnection $connection + * @return void + */ + public static function sendHandshake($connection) + { + if (!empty($connection->handshakeStep)) { + return; + } + // Get Host. + $port = $connection->getRemotePort(); + $host = $port === 80 ? $connection->getRemoteHost() : $connection->getRemoteHost() . ':' . $port; + // Handshake header. + $header = 'GET ' . $connection->getRemoteURI() . " HTTP/1.1\r\n". + "Host: $host\r\n". + "Connection: Upgrade\r\n". + "Upgrade: websocket\r\n". + "Origin: ". (isset($connection->websocketOrigin) ? $connection->websocketOrigin : '*') ."\r\n". + "Sec-WebSocket-Version: 13\r\n". + "Sec-WebSocket-Key: ".base64_encode(sha1(uniqid(mt_rand(), true), true))."\r\n\r\n"; + $connection->send($header, true); + $connection->handshakeStep = 1; + $connection->websocketCurrentFrameLength = 0; + $connection->websocketDataBuffer = ''; + $connection->tmpWebsocketData = ''; + } + + /** + * Websocket handshake. + * + * @param string $buffer + * @param \Workerman\Connection\TcpConnection $connection + * @return int + */ + public static function dealHandshake($buffer, $connection) + { + $pos = strpos($buffer, "\r\n\r\n"); + if ($pos) { + // handshake complete + $connection->handshakeStep = 2; + $handshake_response_length = $pos + 4; + // Try to emit onWebSocketConnect callback. + if (isset($connection->onWebSocketConnect)) { + try { + call_user_func($connection->onWebSocketConnect, $connection, substr($buffer, 0, $handshake_response_length)); + } catch (\Exception $e) { + Worker::log($e); + exit(250); + } catch (\Error $e) { + Worker::log($e); + exit(250); + } + } + // Headbeat. + if (!empty($connection->websocketPingInterval)) { + $connection->websocketPingTimer = Timer::add($connection->websocketPingInterval, function() use ($connection){ + if (false === $connection->send(pack('H*', '8900'), true)) { + Timer::del($connection->websocketPingTimer); + $connection->websocketPingTimer = null; + } + }); + } + + $connection->consumeRecvBuffer($handshake_response_length); + if (!empty($connection->tmpWebsocketData)) { + $connection->send($connection->tmpWebsocketData, true); + $connection->tmpWebsocketData = ''; + } + if (strlen($buffer) > $handshake_response_length) { + return self::input(substr($buffer, $handshake_response_length), $connection); + } + } + return 0; + } +} diff --git a/vendor/workerman/workerman/README.md b/vendor/workerman/workerman/README.md new file mode 100644 index 000000000..17b687dec --- /dev/null +++ b/vendor/workerman/workerman/README.md @@ -0,0 +1,636 @@ +# Workerman +[![Gitter](https://badges.gitter.im/walkor/Workerman.svg)](https://gitter.im/walkor/Workerman?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=body_badge) + +## What is it +Workerman is an asynchronous event driven PHP framework with high performance for easily building fast, scalable network applications.Supports HTTP, Websocket and other custom protocols. Supports libevent, HHVM, [ReactPHP](https://github.com/reactphp/react). + +## Requires + +PHP 5.3 or Higher +A POSIX compatible operating system (Linux, OSX, BSD) +POSIX and PCNTL extensions for PHP + +## Installation + +``` +composer require workerman/workerman +``` + +## Basic Usage + +### A websocket server +```php +count = 4; + +// Emitted when new connection come +$ws_worker->onConnect = function($connection) +{ + echo "New connection\n"; + }; + +// Emitted when data received +$ws_worker->onMessage = function($connection, $data) +{ + // Send hello $data + $connection->send('hello ' . $data); +}; + +// Emitted when connection closed +$ws_worker->onClose = function($connection) +{ + echo "Connection closed\n"; +}; + +// Run worker +Worker::runAll(); +``` + +### An http server +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\Worker; + +// #### http worker #### +$http_worker = new Worker("http://0.0.0.0:2345"); + +// 4 processes +$http_worker->count = 4; + +// Emitted when data received +$http_worker->onMessage = function($connection, $data) +{ + // $_GET, $_POST, $_COOKIE, $_SESSION, $_SERVER, $_FILES are available + var_dump($_GET, $_POST, $_COOKIE, $_SESSION, $_SERVER, $_FILES); + // send data to client + $connection->send("hello world \n"); +}; + +// run all workers +Worker::runAll(); +``` + +### A WebServer +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\WebServer; +use Workerman\Worker; + +// WebServer +$web = new WebServer("http://0.0.0.0:80"); + +// 4 processes +$web->count = 4; + +// Set the root of domains +$web->addRoot('www.your_domain.com', '/your/path/Web'); +$web->addRoot('www.another_domain.com', '/another/path/Web'); +// run all workers +Worker::runAll(); +``` + +### A tcp server +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\Worker; + +// #### create socket and listen 1234 port #### +$tcp_worker = new Worker("tcp://0.0.0.0:1234"); + +// 4 processes +$tcp_worker->count = 4; + +// Emitted when new connection come +$tcp_worker->onConnect = function($connection) +{ + echo "New Connection\n"; +}; + +// Emitted when data received +$tcp_worker->onMessage = function($connection, $data) +{ + // send data to client + $connection->send("hello $data \n"); +}; + +// Emitted when new connection come +$tcp_worker->onClose = function($connection) +{ + echo "Connection closed\n"; +}; + +Worker::runAll(); +``` + +### Custom protocol +Protocols/MyTextProtocol.php +```php +namespace Protocols; +/** + * User defined protocol + * Format Text+"\n" + */ +class MyTextProtocol +{ + public static function input($recv_buffer) + { + // Find the position of the first occurrence of "\n" + $pos = strpos($recv_buffer, "\n"); + // Not a complete package. Return 0 because the length of package can not be calculated + if($pos === false) + { + return 0; + } + // Return length of the package + return $pos+1; + } + + public static function decode($recv_buffer) + { + return trim($recv_buffer); + } + + public static function encode($data) + { + return $data."\n"; + } +} +``` + +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\Worker; + +// #### MyTextProtocol worker #### +$text_worker = new Worker("MyTextProtocol://0.0.0.0:5678"); + +$text_worker->onConnect = function($connection) +{ + echo "New connection\n"; +}; + +$text_worker->onMessage = function($connection, $data) +{ + // send data to client + $connection->send("hello world \n"); +}; + +$text_worker->onClose = function($connection) +{ + echo "Connection closed\n"; +}; + +// run all workers +Worker::runAll(); +``` + +### Timer +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\Worker; +use Workerman\Lib\Timer; + +$task = new Worker(); +$task->onWorkerStart = function($task) +{ + // 2.5 seconds + $time_interval = 2.5; + $timer_id = Timer::add($time_interval, + function() + { + echo "Timer run\n"; + } + ); +}; + +// run all workers +Worker::runAll(); +``` + +### AsyncTcpConnection (tcp/ws/text/frame etc...) +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\Worker; +use Workerman\Connection\AsyncTcpConnection; + +$worker = new Worker(); +$worker->onWorkerStart = function() +{ + // Websocket protocol for client. + $ws_connection = new AsyncTcpConnection("ws://echo.websocket.org:80"); + $ws_connection->onConnect = function($connection){ + $connection->send('hello'); + }; + $ws_connection->onMessage = function($connection, $data){ + echo "recv: $data\n"; + }; + $ws_connection->onError = function($connection, $code, $msg){ + echo "error: $msg\n"; + }; + $ws_connection->onClose = function($connection){ + echo "connection closed\n"; + }; + $ws_connection->connect(); +}; +Worker::runAll(); +``` + +### Async Mysql of ReactPHP +``` +composer require react/mysql +``` + +```php +onWorkerStart = function() { + global $mysql; + $loop = Worker::getEventLoop(); + $mysql = new React\MySQL\Connection($loop, array( + 'host' => '127.0.0.1', + 'dbname' => 'dbname', + 'user' => 'user', + 'passwd' => 'passwd', + )); + $mysql->on('error', function($e){ + echo $e; + }); + $mysql->connect(function ($e) { + if($e) { + echo $e; + } else { + echo "connect success\n"; + } + }); +}; +$worker->onMessage = function($connection, $data) { + global $mysql; + $mysql->query('show databases' /*trim($data)*/, function ($command, $mysql) use ($connection) { + if ($command->hasError()) { + $error = $command->getError(); + } else { + $results = $command->resultRows; + $fields = $command->resultFields; + $connection->send(json_encode($results)); + } + }); +}; +Worker::runAll(); +``` + +### Async Redis of ReactPHP +``` +composer require clue/redis-react +``` + +```php +onWorkerStart = function() { + global $factory; + $loop = Worker::getEventLoop(); + $factory = new Factory($loop); +}; + +$worker->onMessage = function($connection, $data) { + global $factory; + $factory->createClient('localhost:6379')->then(function (Client $client) use ($connection) { + $client->set('greeting', 'Hello world'); + $client->append('greeting', '!'); + + $client->get('greeting')->then(function ($greeting) use ($connection){ + // Hello world! + echo $greeting . PHP_EOL; + $connection->send($greeting); + }); + + $client->incr('invocation')->then(function ($n) use ($connection){ + echo 'This is invocation #' . $n . PHP_EOL; + $connection->send($n); + }); + }); +}; + +Worker::runAll(); +``` + +### Aysnc dns of ReactPHP +``` +composer require react/dns +``` + +```php +require_once __DIR__ . '/vendor/autoload.php'; +use Workerman\Worker; +$worker = new Worker('tcp://0.0.0.0:6161'); +$worker->onWorkerStart = function() { + global $dns; + // Get event-loop. + $loop = Worker::getEventLoop(); + $factory = new React\Dns\Resolver\Factory(); + $dns = $factory->create('8.8.8.8', $loop); +}; +$worker->onMessage = function($connection, $host) { + global $dns; + $host = trim($host); + $dns->resolve($host)->then(function($ip) use($host, $connection) { + $connection->send("$host: $ip"); + },function($e) use($host, $connection){ + $connection->send("$host: {$e->getMessage()}"); + }); +}; + +Worker::runAll(); +``` + +### Http client of ReactPHP +``` +composer require react/http-client +``` + +```php +onWorkerStart = function() { + global $client; + $loop = Worker::getEventLoop(); + $factory = new React\Dns\Resolver\Factory(); + $dns = $factory->createCached('8.8.8.8', $loop); + $factory = new React\HttpClient\Factory(); + $client = $factory->create($loop, $dns); +}; + +$worker->onMessage = function($connection, $host) { + global $client; + $request = $client->request('GET', trim($host)); + $request->on('error', function(Exception $e) use ($connection) { + $connection->send($e); + }); + $request->on('response', function ($response) use ($connection) { + $response->on('data', function ($data, $response) use ($connection) { + $connection->send($data); + }); + }); + $request->end(); +}; + +Worker::runAll(); +``` + +### ZMQ of ReactPHP +``` +composer require react/zmq +``` + +```php +onWorkerStart = function() { + global $pull; + $loop = Worker::getEventLoop(); + $context = new React\ZMQ\Context($loop); + $pull = $context->getSocket(ZMQ::SOCKET_PULL); + $pull->bind('tcp://127.0.0.1:5555'); + + $pull->on('error', function ($e) { + var_dump($e->getMessage()); + }); + + $pull->on('message', function ($msg) { + echo "Received: $msg\n"; + }); +}; + +Worker::runAll(); +``` + +### STOMP of ReactPHP +``` +composer require react/stomp +``` + +```php +onWorkerStart = function() { + global $client; + $loop = Worker::getEventLoop(); + $factory = new React\Stomp\Factory($loop); + $client = $factory->createClient(array('vhost' => '/', 'login' => 'guest', 'passcode' => 'guest')); + + $client + ->connect() + ->then(function ($client) use ($loop) { + $client->subscribe('/topic/foo', function ($frame) { + echo "Message received: {$frame->body}\n"; + }); + }); +}; + +Worker::runAll(); +``` + + + +## Available commands +```php test.php start ``` +```php test.php start -d ``` +![workerman start](http://www.workerman.net/img/workerman-start.png) +```php test.php status ``` +![workerman satus](http://www.workerman.net/img/workerman-status.png?a=123) +```php test.php stop ``` +```php test.php restart ``` +```php test.php reload ``` + +## Documentation + +涓枃涓婚〉:[http://www.workerman.net](http://www.workerman.net) + +涓枃鏂囨。: [http://doc3.workerman.net](http://doc3.workerman.net) + +Documentation:[https://github.com/walkor/workerman-manual](https://github.com/walkor/workerman-manual/blob/master/english/src/SUMMARY.md) + +# Benchmarks +``` +CPU: Intel(R) Core(TM) i3-3220 CPU @ 3.30GHz and 4 processors totally +Memory: 8G +OS: Ubuntu 14.04 LTS +Software: ab +PHP: 5.5.9 +``` + +**Codes** +```php +count=3; +$worker->onMessage = function($connection, $data) +{ + $connection->send("HTTP/1.1 200 OK\r\nConnection: keep-alive\r\nServer: workerman\r\nContent-Length: 5\r\n\r\nhello"); +}; +Worker::runAll(); +``` +**Result** + +```shell +ab -n1000000 -c100 -k http://127.0.0.1:1234/ +This is ApacheBench, Version 2.3 <$Revision: 1528965 $> +Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/ +Licensed to The Apache Software Foundation, http://www.apache.org/ + +Benchmarking 127.0.0.1 (be patient) +Completed 100000 requests +Completed 200000 requests +Completed 300000 requests +Completed 400000 requests +Completed 500000 requests +Completed 600000 requests +Completed 700000 requests +Completed 800000 requests +Completed 900000 requests +Completed 1000000 requests +Finished 1000000 requests + + +Server Software: workerman/3.1.4 +Server Hostname: 127.0.0.1 +Server Port: 1234 + +Document Path: / +Document Length: 5 bytes + +Concurrency Level: 100 +Time taken for tests: 7.240 seconds +Complete requests: 1000000 +Failed requests: 0 +Keep-Alive requests: 1000000 +Total transferred: 73000000 bytes +HTML transferred: 5000000 bytes +Requests per second: 138124.14 [#/sec] (mean) +Time per request: 0.724 [ms] (mean) +Time per request: 0.007 [ms] (mean, across all concurrent requests) +Transfer rate: 9846.74 [Kbytes/sec] received + +Connection Times (ms) + min mean[+/-sd] median max +Connect: 0 0 0.0 0 5 +Processing: 0 1 0.2 1 9 +Waiting: 0 1 0.2 1 9 +Total: 0 1 0.2 1 9 + +Percentage of the requests served within a certain time (ms) + 50% 1 + 66% 1 + 75% 1 + 80% 1 + 90% 1 + 95% 1 + 98% 1 + 99% 1 + 100% 9 (longest request) + +``` + + +## Other links with workerman + +## [PHPSocket.IO](https://github.com/walkor/phpsocket.io) +[Live demo](http://www.workerman.net/demos/phpsocketio-chat/) +[Source code](https://github.com/walkor/phpsocket.io) +![phpsocket.io](http://www.workerman.net/img/socket.io.png) + +## [tadpole](http://kedou.workerman.net/) +[Live demo](http://kedou.workerman.net/) +[Source code](https://github.com/walkor/workerman) +![workerman todpole](http://www.workerman.net/img/workerman-todpole.png) + +## [BrowserQuest](http://www.workerman.net/demos/browserquest/) +[Live demo](http://www.workerman.net/demos/browserquest/) +[Source code](https://github.com/walkor/BrowserQuest-PHP) +![BrowserQuest width workerman](http://www.workerman.net/img/browserquest.jpg) + +## [web vmstat](http://www.workerman.net/demos/vmstat/) +[Live demo](http://www.workerman.net/demos/vmstat/) +[Source code](https://github.com/walkor/workerman-vmstat) +![web vmstat](http://www.workerman.net/img/workerman-vmstat.png) + +## [live-ascii-camera](https://github.com/walkor/live-ascii-camera) +[Live demo camera page](http://www.workerman.net/demos/live-ascii-camera/camera.html) +[Live demo receive page](http://www.workerman.net/demos/live-ascii-camera/) +[Source code](https://github.com/walkor/live-ascii-camera) +![live-ascii-camera](http://www.workerman.net/img/live-ascii-camera.png) + +## [live-camera](https://github.com/walkor/live-camera) +[Live demo camera page](http://www.workerman.net/demos/live-camera/camera.html) +[Live demo receive page](http://www.workerman.net/demos/live-camera/) +[Source code](https://github.com/walkor/live-camera) +![live-camera](http://www.workerman.net/img/live-camera.jpg) + +## [chat room](http://chat.workerman.net/) +[Live demo](http://chat.workerman.net/) +[Source code](https://github.com/walkor/workerman-chat) +![workerman-chat](http://www.workerman.net/img/workerman-chat.png) + +## [statistics](http://www.workerman.net:55757/) +[Live demo](http://www.workerman.net:55757/) +[Source code](https://github.com/walkor/workerman-statistics) +![workerman-statistics](http://www.workerman.net/img/workerman-statistics.png) + +## [flappybird](http://workerman.net/demos/flappy-bird/) +[Live demo](http://workerman.net/demos/flappy-bird/) +[Source code](https://github.com/walkor/workerman-flappy-bird) +![workerman-statistics](http://www.workerman.net/img/workerman-flappy-bird.png) + +## [jsonRpc](https://github.com/walkor/workerman-JsonRpc) +[Source code](https://github.com/walkor/workerman-JsonRpc) +![workerman-jsonRpc](http://www.workerman.net/img/workerman-json-rpc.png) + +## [thriftRpc](https://github.com/walkor/workerman-thrift) +[Source code](https://github.com/walkor/workerman-thrift) +![workerman-thriftRpc](http://www.workerman.net/img/workerman-thrift.png) + +## [web-msg-sender](https://github.com/walkor/web-msg-sender) +[Live demo send page](http://workerman.net:3333/) +[Live demo receive page](http://workerman.net/web-msg-sender.html) +[Source code](https://github.com/walkor/web-msg-sender) +![web-msg-sender](http://www.workerman.net/img/web-msg-sender.png) + +## [shadowsocks-php](https://github.com/walkor/shadowsocks-php) +[Source code](https://github.com/walkor/shadowsocks-php) +![shadowsocks-php](http://www.workerman.net/img/shadowsocks-php.png) + +## [queue](https://github.com/walkor/workerman-queue) +[Source code](https://github.com/walkor/workerman-queue) + +## LICENSE + +Workerman is released under the [MIT license](https://github.com/walkor/workerman/blob/master/MIT-LICENSE.txt). diff --git a/vendor/workerman/workerman/WebServer.php b/vendor/workerman/workerman/WebServer.php new file mode 100644 index 000000000..235246d96 --- /dev/null +++ b/vendor/workerman/workerman/WebServer.php @@ -0,0 +1,301 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman; + +use Workerman\Protocols\Http; +use Workerman\Protocols\HttpCache; + +/** + * WebServer. + */ +class WebServer extends Worker +{ + /** + * Virtual host to path mapping. + * + * @var array ['workerman.net'=>'/home', 'www.workerman.net'=>'home/www'] + */ + protected $serverRoot = array(); + + /** + * Mime mapping. + * + * @var array + */ + protected static $mimeTypeMap = array(); + + + /** + * Used to save user OnWorkerStart callback settings. + * + * @var callback + */ + protected $_onWorkerStart = null; + + /** + * Add virtual host. + * + * @param string $domain + * @param string $root_path + * @return void + */ + public function addRoot($domain, $root_path) + { + $this->serverRoot[$domain] = $root_path; + } + + /** + * Construct. + * + * @param string $socket_name + * @param array $context_option + */ + public function __construct($socket_name, $context_option = array()) + { + list(, $address) = explode(':', $socket_name, 2); + parent::__construct('http:' . $address, $context_option); + $this->name = 'WebServer'; + } + + /** + * Run webserver instance. + * + * @see Workerman.Worker::run() + */ + public function run() + { + $this->_onWorkerStart = $this->onWorkerStart; + $this->onWorkerStart = array($this, 'onWorkerStart'); + $this->onMessage = array($this, 'onMessage'); + parent::run(); + } + + /** + * Emit when process start. + * + * @throws \Exception + */ + public function onWorkerStart() + { + if (empty($this->serverRoot)) { + echo new \Exception('server root not set, please use WebServer::addRoot($domain, $root_path) to set server root path'); + exit(250); + } + + // Init mimeMap. + $this->initMimeTypeMap(); + + // Try to emit onWorkerStart callback. + if ($this->_onWorkerStart) { + try { + call_user_func($this->_onWorkerStart, $this); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + } + + /** + * Init mime map. + * + * @return void + */ + public function initMimeTypeMap() + { + $mime_file = Http::getMimeTypesFile(); + if (!is_file($mime_file)) { + $this->log("$mime_file mime.type file not fond"); + return; + } + $items = file($mime_file, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); + if (!is_array($items)) { + $this->log("get $mime_file mime.type content fail"); + return; + } + foreach ($items as $content) { + if (preg_match("/\s*(\S+)\s+(\S.+)/", $content, $match)) { + $mime_type = $match[1]; + $workerman_file_extension_var = $match[2]; + $workerman_file_extension_array = explode(' ', substr($workerman_file_extension_var, 0, -1)); + foreach ($workerman_file_extension_array as $workerman_file_extension) { + self::$mimeTypeMap[$workerman_file_extension] = $mime_type; + } + } + } + } + + /** + * Emit when http message coming. + * + * @param Connection\TcpConnection $connection + * @return void + */ + public function onMessage($connection) + { + // REQUEST_URI. + $workerman_url_info = parse_url($_SERVER['REQUEST_URI']); + if (!$workerman_url_info) { + Http::header('HTTP/1.1 400 Bad Request'); + $connection->close('

    400 Bad Request

    '); + return; + } + + $workerman_path = isset($workerman_url_info['path']) ? $workerman_url_info['path'] : '/'; + + $workerman_path_info = pathinfo($workerman_path); + $workerman_file_extension = isset($workerman_path_info['extension']) ? $workerman_path_info['extension'] : ''; + if ($workerman_file_extension === '') { + $workerman_path = ($len = strlen($workerman_path)) && $workerman_path[$len - 1] === '/' ? $workerman_path . 'index.php' : $workerman_path . '/index.php'; + $workerman_file_extension = 'php'; + } + + $workerman_root_dir = isset($this->serverRoot[$_SERVER['SERVER_NAME']]) ? $this->serverRoot[$_SERVER['SERVER_NAME']] : current($this->serverRoot); + + $workerman_file = "$workerman_root_dir/$workerman_path"; + + if ($workerman_file_extension === 'php' && !is_file($workerman_file)) { + $workerman_file = "$workerman_root_dir/index.php"; + if (!is_file($workerman_file)) { + $workerman_file = "$workerman_root_dir/index.html"; + $workerman_file_extension = 'html'; + } + } + + // File exsits. + if (is_file($workerman_file)) { + // Security check. + if ((!($workerman_request_realpath = realpath($workerman_file)) || !($workerman_root_dir_realpath = realpath($workerman_root_dir))) || 0 !== strpos($workerman_request_realpath, + $workerman_root_dir_realpath) + ) { + Http::header('HTTP/1.1 400 Bad Request'); + $connection->close('

    400 Bad Request

    '); + return; + } + + $workerman_file = realpath($workerman_file); + + // Request php file. + if ($workerman_file_extension === 'php') { + $workerman_cwd = getcwd(); + chdir($workerman_root_dir); + ini_set('display_errors', 'off'); + ob_start(); + // Try to include php file. + try { + // $_SERVER. + $_SERVER['REMOTE_ADDR'] = $connection->getRemoteIp(); + $_SERVER['REMOTE_PORT'] = $connection->getRemotePort(); + include $workerman_file; + } catch (\Exception $e) { + // Jump_exit? + if ($e->getMessage() != 'jump_exit') { + echo $e; + } + } + $content = ob_get_clean(); + ini_set('display_errors', 'on'); + if (strtolower($_SERVER['HTTP_CONNECTION']) === "keep-alive") { + $connection->send($content); + } else { + $connection->close($content); + } + chdir($workerman_cwd); + return; + } + + // Send file to client. + return self::sendFile($connection, $workerman_file); + } else { + // 404 + Http::header("HTTP/1.1 404 Not Found"); + $connection->close('404 File not found

    404 Not Found

    '); + return; + } + } + + public static function sendFile($connection, $file_path) + { + // Check 304. + $info = stat($file_path); + $modified_time = $info ? date('D, d M Y H:i:s', $info['mtime']) . ' GMT' : ''; + if (!empty($_SERVER['HTTP_IF_MODIFIED_SINCE']) && $info) { + // Http 304. + if ($modified_time === $_SERVER['HTTP_IF_MODIFIED_SINCE']) { + // 304 + Http::header('HTTP/1.1 304 Not Modified'); + // Send nothing but http headers.. + $connection->close(''); + return; + } + } + + // Http header. + if ($modified_time) { + $modified_time = "Last-Modified: $modified_time\r\n"; + } + $file_size = filesize($file_path); + $file_info = pathinfo($file_path); + $extension = isset($file_info['extension']) ? $file_info['extension'] : ''; + $file_name = isset($file_info['filename']) ? $file_info['filename'] : ''; + $header = "HTTP/1.1 200 OK\r\n"; + if (isset(self::$mimeTypeMap[$extension])) { + $header .= "Content-Type: " . self::$mimeTypeMap[$extension] . "\r\n"; + } else { + $header .= "Content-Type: application/octet-stream\r\n"; + $header .= "Content-Disposition: attachment; filename=\"$file_name\"\r\n"; + } + $header .= "Connection: keep-alive\r\n"; + $header .= $modified_time; + $header .= "Content-Length: $file_size\r\n\r\n"; + $trunk_limit_size = 1024*1024; + if ($file_size < $trunk_limit_size) { + return $connection->send($header.file_get_contents($file_path), true); + } + $connection->send($header, true); + + // Read file content from disk piece by piece and send to client. + $connection->fileHandler = fopen($file_path, 'r'); + $do_write = function()use($connection) + { + // Send buffer not full. + while(empty($connection->bufferFull)) + { + // Read from disk. + $buffer = fread($connection->fileHandler, 8192); + // Read eof. + if($buffer === '' || $buffer === false) + { + return; + } + $connection->send($buffer, true); + } + }; + // Send buffer full. + $connection->onBufferFull = function($connection) + { + $connection->bufferFull = true; + }; + // Send buffer drain. + $connection->onBufferDrain = function($connection)use($do_write) + { + $connection->bufferFull = false; + $do_write(); + }; + $do_write(); + } +} diff --git a/vendor/workerman/workerman/Worker.php b/vendor/workerman/workerman/Worker.php new file mode 100644 index 000000000..08e7dd6a8 --- /dev/null +++ b/vendor/workerman/workerman/Worker.php @@ -0,0 +1,1639 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ +namespace Workerman; + +require_once __DIR__ . '/Lib/Constants.php'; + +use Workerman\Events\EventInterface; +use Workerman\Connection\ConnectionInterface; +use Workerman\Connection\TcpConnection; +use Workerman\Connection\UdpConnection; +use Workerman\Lib\Timer; +use Exception; + +/** + * Worker class + * A container for listening ports + */ +class Worker +{ + /** + * Version. + * + * @var string + */ + const VERSION = '3.3.7'; + + /** + * Status starting. + * + * @var int + */ + const STATUS_STARTING = 1; + + /** + * Status running. + * + * @var int + */ + const STATUS_RUNNING = 2; + + /** + * Status shutdown. + * + * @var int + */ + const STATUS_SHUTDOWN = 4; + + /** + * Status reloading. + * + * @var int + */ + const STATUS_RELOADING = 8; + + /** + * After sending the restart command to the child process KILL_WORKER_TIMER_TIME seconds, + * if the process is still living then forced to kill. + * + * @var int + */ + const KILL_WORKER_TIMER_TIME = 2; + + /** + * Default backlog. Backlog is the maximum length of the queue of pending connections. + * + * @var int + */ + const DEFAULT_BACKLOG = 102400; + /** + * Max udp package size. + * + * @var int + */ + const MAX_UDP_PACKAGE_SIZE = 65535; + + /** + * Worker id. + * + * @var int + */ + public $id = 0; + + /** + * Name of the worker processes. + * + * @var string + */ + public $name = 'none'; + + /** + * Number of worker processes. + * + * @var int + */ + public $count = 1; + + /** + * Unix user of processes, needs appropriate privileges (usually root). + * + * @var string + */ + public $user = ''; + + /** + * Unix group of processes, needs appropriate privileges (usually root). + * + * @var string + */ + public $group = ''; + + /** + * reloadable. + * + * @var bool + */ + public $reloadable = true; + + /** + * reuse port. + * + * @var bool + */ + public $reusePort = false; + + /** + * Emitted when worker processes start. + * + * @var callback + */ + public $onWorkerStart = null; + + /** + * Emitted when a socket connection is successfully established. + * + * @var callback + */ + public $onConnect = null; + + /** + * Emitted when data is received. + * + * @var callback + */ + public $onMessage = null; + + /** + * Emitted when the other end of the socket sends a FIN packet. + * + * @var callback + */ + public $onClose = null; + + /** + * Emitted when an error occurs with connection. + * + * @var callback + */ + public $onError = null; + + /** + * Emitted when the send buffer becomes full. + * + * @var callback + */ + public $onBufferFull = null; + + /** + * Emitted when the send buffer becomes empty. + * + * @var callback + */ + public $onBufferDrain = null; + + /** + * Emitted when worker processes stoped. + * + * @var callback + */ + public $onWorkerStop = null; + + /** + * Emitted when worker processes get reload signal. + * + * @var callback + */ + public $onWorkerReload = null; + + /** + * Transport layer protocol. + * + * @var string + */ + public $transport = 'tcp'; + + /** + * Store all connections of clients. + * + * @var array + */ + public $connections = array(); + + /** + * Application layer protocol. + * + * @var Protocols\ProtocolInterface + */ + public $protocol = ''; + + /** + * Root path for autoload. + * + * @var string + */ + protected $_autoloadRootPath = ''; + + /** + * Daemonize. + * + * @var bool + */ + public static $daemonize = false; + + /** + * Stdout file. + * + * @var string + */ + public static $stdoutFile = '/dev/null'; + + /** + * The file to store master process PID. + * + * @var string + */ + public static $pidFile = ''; + + /** + * Log file. + * + * @var mixed + */ + public static $logFile = ''; + + /** + * Global event loop. + * + * @var Events\EventInterface + */ + public static $globalEvent = null; + + /** + * Emitted when the master process get reload signal. + * + * @var callback + */ + public static $onMasterReload = null; + + /** + * Emitted when the master process terminated. + * + * @var callback + */ + public static $onMasterStop = null; + + /** + * The PID of master process. + * + * @var int + */ + protected static $_masterPid = 0; + + /** + * Listening socket. + * + * @var resource + */ + protected $_mainSocket = null; + + /** + * Socket name. The format is like this http://0.0.0.0:80 . + * + * @var string + */ + protected $_socketName = ''; + + /** + * Context of socket. + * + * @var resource + */ + protected $_context = null; + + /** + * All worker instances. + * + * @var array + */ + protected static $_workers = array(); + + /** + * All worker porcesses pid. + * The format is like this [worker_id=>[pid=>pid, pid=>pid, ..], ..] + * + * @var array + */ + protected static $_pidMap = array(); + + /** + * All worker processes waiting for restart. + * The format is like this [pid=>pid, pid=>pid]. + * + * @var array + */ + protected static $_pidsToRestart = array(); + + /** + * Mapping from PID to worker process ID. + * The format is like this [worker_id=>[0=>$pid, 1=>$pid, ..], ..]. + * + * @var array + */ + protected static $_idMap = array(); + + /** + * Current status. + * + * @var int + */ + protected static $_status = self::STATUS_STARTING; + + /** + * Maximum length of the worker names. + * + * @var int + */ + protected static $_maxWorkerNameLength = 12; + + /** + * Maximum length of the socket names. + * + * @var int + */ + protected static $_maxSocketNameLength = 12; + + /** + * Maximum length of the process user names. + * + * @var int + */ + protected static $_maxUserNameLength = 12; + + /** + * The file to store status info of current worker process. + * + * @var string + */ + protected static $_statisticsFile = ''; + + /** + * Start file. + * + * @var string + */ + protected static $_startFile = ''; + + /** + * Status info of current worker process. + * + * @var array + */ + protected static $_globalStatistics = array( + 'start_timestamp' => 0, + 'worker_exit_info' => array() + ); + + /** + * Available event loops. + * + * @var array + */ + protected static $_availableEventLoops = array( + 'libevent', + 'event', + 'ev' + ); + + /** + * Current eventLoop name. + * + * @var string + */ + protected static $_eventLoopName = 'select'; + + /** + * PHP built-in protocols. + * + * @var array + */ + protected static $_builtinTransports = array( + 'tcp' => 'tcp', + 'udp' => 'udp', + 'unix' => 'unix', + 'ssl' => 'tcp' + ); + + /** + * Run all worker instances. + * + * @return void + */ + public static function runAll() + { + self::checkSapiEnv(); + self::init(); + self::parseCommand(); + self::daemonize(); + self::initWorkers(); + self::installSignal(); + self::saveMasterPid(); + self::forkWorkers(); + self::displayUI(); + self::resetStd(); + self::monitorWorkers(); + } + + /** + * Check sapi. + * + * @return void + */ + protected static function checkSapiEnv() + { + // Only for cli. + if (php_sapi_name() != "cli") { + exit("only run in command line mode \n"); + } + } + + /** + * Init. + * + * @return void + */ + protected static function init() + { + // Start file. + $backtrace = debug_backtrace(); + self::$_startFile = $backtrace[count($backtrace) - 1]['file']; + + // Pid file. + if (empty(self::$pidFile)) { + self::$pidFile = __DIR__ . "/../" . str_replace('/', '_', self::$_startFile) . ".pid"; + } + + // Log file. + if (empty(self::$logFile)) { + self::$logFile = __DIR__ . '/../workerman.log'; + } + $log_file = (string)self::$logFile; + touch($log_file); + chmod($log_file, 0622); + + // State. + self::$_status = self::STATUS_STARTING; + + // For statistics. + self::$_globalStatistics['start_timestamp'] = time(); + self::$_statisticsFile = sys_get_temp_dir() . '/workerman.status'; + + // Process title. + self::setProcessTitle('WorkerMan: master process start_file=' . self::$_startFile); + + // Init data for worker id. + self::initId(); + + // Timer init. + Timer::init(); + } + + /** + * Init All worker instances. + * + * @return void + */ + protected static function initWorkers() + { + foreach (self::$_workers as $worker) { + // Worker name. + if (empty($worker->name)) { + $worker->name = 'none'; + } + + // Get maximum length of worker name. + $worker_name_length = strlen($worker->name); + if (self::$_maxWorkerNameLength < $worker_name_length) { + self::$_maxWorkerNameLength = $worker_name_length; + } + + // Get maximum length of socket name. + $socket_name_length = strlen($worker->getSocketName()); + if (self::$_maxSocketNameLength < $socket_name_length) { + self::$_maxSocketNameLength = $socket_name_length; + } + + // Get unix user of the worker process. + if (empty($worker->user)) { + $worker->user = self::getCurrentUser(); + } else { + if (posix_getuid() !== 0 && $worker->user != self::getCurrentUser()) { + self::log('Warning: You must have the root privileges to change uid and gid.'); + } + } + + // Get maximum length of unix user name. + $user_name_length = strlen($worker->user); + if (self::$_maxUserNameLength < $user_name_length) { + self::$_maxUserNameLength = $user_name_length; + } + + // Listen. + if (!$worker->reusePort) { + $worker->listen(); + } + } + } + + /** + * Get all worker instances. + * + * @return array + */ + public static function getAllWorkers() + { + return self::$_workers; + } + + /** + * Get global event-loop instance. + * + * @return EventInterface + */ + public static function getEventLoop() + { + return self::$globalEvent; + } + + /** + * Init idMap. + * return void + */ + protected static function initId() + { + foreach (self::$_workers as $worker_id => $worker) { + $new_id_map = array(); + for($key = 0; $key < $worker->count; $key++) { + $new_id_map[$key] = isset(self::$_idMap[$worker_id][$key]) ? self::$_idMap[$worker_id][$key] : 0; + } + self::$_idMap[$worker_id] = $new_id_map; + } + } + + /** + * Get unix user of current porcess. + * + * @return string + */ + protected static function getCurrentUser() + { + $user_info = posix_getpwuid(posix_getuid()); + return $user_info['name']; + } + + /** + * Display staring UI. + * + * @return void + */ + protected static function displayUI() + { + self::safeEcho("\033[1A\n\033[K-----------------------\033[47;30m WORKERMAN \033[0m-----------------------------\n\033[0m"); + self::safeEcho('Workerman version:'. Worker::VERSION. " PHP version:". PHP_VERSION. "\n"); + self::safeEcho("------------------------\033[47;30m WORKERS \033[0m-------------------------------\n"); + self::safeEcho("\033[47;30muser\033[0m". str_pad('', + self::$_maxUserNameLength + 2 - strlen('user')). "\033[47;30mworker\033[0m". str_pad('', + self::$_maxWorkerNameLength + 2 - strlen('worker')). "\033[47;30mlisten\033[0m". str_pad('', + self::$_maxSocketNameLength + 2 - strlen('listen')). "\033[47;30mprocesses\033[0m \033[47;30m". "status\033[0m\n"); + + foreach (self::$_workers as $worker) { + self::safeEcho(str_pad($worker->user, self::$_maxUserNameLength + 2). str_pad($worker->name, + self::$_maxWorkerNameLength + 2). str_pad($worker->getSocketName(), + self::$_maxSocketNameLength + 2). str_pad(' ' . $worker->count, 9). " \033[32;40m [OK] \033[0m\n"); + } + self::safeEcho("----------------------------------------------------------------\n"); + if (self::$daemonize) { + global $argv; + $start_file = $argv[0]; + self::safeEcho("Input \"php $start_file stop\" to quit. Start success.\n"); + } else { + self::safeEcho("Press Ctrl-C to quit. Start success.\n"); + } + } + + /** + * Parse command. + * php yourfile.php start | stop | restart | reload | status + * + * @return void + */ + protected static function parseCommand() + { + global $argv; + // Check argv; + $start_file = $argv[0]; + if (!isset($argv[1])) { + exit("Usage: php yourfile.php {start|stop|restart|reload|status}\n"); + } + + // Get command. + $command = trim($argv[1]); + $command2 = isset($argv[2]) ? $argv[2] : ''; + + // Start command. + $mode = ''; + if ($command === 'start') { + if ($command2 === '-d' || Worker::$daemonize) { + $mode = 'in DAEMON mode'; + } else { + $mode = 'in DEBUG mode'; + } + } + self::log("Workerman[$start_file] $command $mode"); + + // Get master process PID. + $master_pid = @file_get_contents(self::$pidFile); + $master_is_alive = $master_pid && @posix_kill($master_pid, 0); + // Master is still alive? + if ($master_is_alive) { + if ($command === 'start' && posix_getpid() != $master_pid) { + self::log("Workerman[$start_file] already running"); + exit; + } + } elseif ($command !== 'start' && $command !== 'restart') { + self::log("Workerman[$start_file] not run"); + exit; + } + + // execute command. + switch ($command) { + case 'start': + if ($command2 === '-d') { + Worker::$daemonize = true; + } + break; + case 'status': + if (is_file(self::$_statisticsFile)) { + @unlink(self::$_statisticsFile); + } + // Master process will send status signal to all child processes. + posix_kill($master_pid, SIGUSR2); + // Waiting amoment. + usleep(500000); + // Display statisitcs data from a disk file. + @readfile(self::$_statisticsFile); + exit(0); + case 'restart': + case 'stop': + self::log("Workerman[$start_file] is stoping ..."); + // Send stop signal to master process. + $master_pid && posix_kill($master_pid, SIGINT); + // Timeout. + $timeout = 5; + $start_time = time(); + // Check master process is still alive? + while (1) { + $master_is_alive = $master_pid && posix_kill($master_pid, 0); + if ($master_is_alive) { + // Timeout? + if (time() - $start_time >= $timeout) { + self::log("Workerman[$start_file] stop fail"); + exit; + } + // Waiting amoment. + usleep(10000); + continue; + } + // Stop success. + self::log("Workerman[$start_file] stop success"); + if ($command === 'stop') { + exit(0); + } + if ($command2 === '-d') { + Worker::$daemonize = true; + } + break; + } + break; + case 'reload': + posix_kill($master_pid, SIGUSR1); + self::log("Workerman[$start_file] reload"); + exit; + default : + exit("Usage: php yourfile.php {start|stop|restart|reload|status}\n"); + } + } + + /** + * Install signal handler. + * + * @return void + */ + protected static function installSignal() + { + // stop + pcntl_signal(SIGINT, array('\Workerman\Worker', 'signalHandler'), false); + // reload + pcntl_signal(SIGUSR1, array('\Workerman\Worker', 'signalHandler'), false); + // status + pcntl_signal(SIGUSR2, array('\Workerman\Worker', 'signalHandler'), false); + // ignore + pcntl_signal(SIGPIPE, SIG_IGN, false); + } + + /** + * Reinstall signal handler. + * + * @return void + */ + protected static function reinstallSignal() + { + // uninstall stop signal handler + pcntl_signal(SIGINT, SIG_IGN, false); + // uninstall reload signal handler + pcntl_signal(SIGUSR1, SIG_IGN, false); + // uninstall status signal handler + pcntl_signal(SIGUSR2, SIG_IGN, false); + // reinstall stop signal handler + self::$globalEvent->add(SIGINT, EventInterface::EV_SIGNAL, array('\Workerman\Worker', 'signalHandler')); + // reinstall reload signal handler + self::$globalEvent->add(SIGUSR1, EventInterface::EV_SIGNAL, array('\Workerman\Worker', 'signalHandler')); + // reinstall status signal handler + self::$globalEvent->add(SIGUSR2, EventInterface::EV_SIGNAL, array('\Workerman\Worker', 'signalHandler')); + } + + /** + * Signal handler. + * + * @param int $signal + */ + public static function signalHandler($signal) + { + switch ($signal) { + // Stop. + case SIGINT: + self::stopAll(); + break; + // Reload. + case SIGUSR1: + self::$_pidsToRestart = self::getAllWorkerPids(); + self::reload(); + break; + // Show status. + case SIGUSR2: + self::writeStatisticsToStatusFile(); + break; + } + } + + /** + * Run as deamon mode. + * + * @throws Exception + */ + protected static function daemonize() + { + if (!self::$daemonize) { + return; + } + umask(0); + $pid = pcntl_fork(); + if (-1 === $pid) { + throw new Exception('fork fail'); + } elseif ($pid > 0) { + exit(0); + } + if (-1 === posix_setsid()) { + throw new Exception("setsid fail"); + } + // Fork again avoid SVR4 system regain the control of terminal. + $pid = pcntl_fork(); + if (-1 === $pid) { + throw new Exception("fork fail"); + } elseif (0 !== $pid) { + exit(0); + } + } + + /** + * Redirect standard input and output. + * + * @throws Exception + */ + protected static function resetStd() + { + if (!self::$daemonize) { + return; + } + global $STDOUT, $STDERR; + $handle = fopen(self::$stdoutFile, "a"); + if ($handle) { + unset($handle); + @fclose(STDOUT); + @fclose(STDERR); + $STDOUT = fopen(self::$stdoutFile, "a"); + $STDERR = fopen(self::$stdoutFile, "a"); + } else { + throw new Exception('can not open stdoutFile ' . self::$stdoutFile); + } + } + + /** + * Save pid. + * + * @throws Exception + */ + protected static function saveMasterPid() + { + self::$_masterPid = posix_getpid(); + if (false === @file_put_contents(self::$pidFile, self::$_masterPid)) { + throw new Exception('can not save pid to ' . self::$pidFile); + } + } + + /** + * Get event loop name. + * + * @return string + */ + protected static function getEventLoopName() + { + if (interface_exists('\React\EventLoop\LoopInterface')) { + return 'React'; + } + foreach (self::$_availableEventLoops as $name) { + if (extension_loaded($name)) { + self::$_eventLoopName = $name; + break; + } + } + return self::$_eventLoopName; + } + + /** + * Get all pids of worker processes. + * + * @return array + */ + protected static function getAllWorkerPids() + { + $pid_array = array(); + foreach (self::$_pidMap as $worker_pid_array) { + foreach ($worker_pid_array as $worker_pid) { + $pid_array[$worker_pid] = $worker_pid; + } + } + return $pid_array; + } + + /** + * Fork some worker processes. + * + * @return void + */ + protected static function forkWorkers() + { + foreach (self::$_workers as $worker) { + if (self::$_status === self::STATUS_STARTING) { + if (empty($worker->name)) { + $worker->name = $worker->getSocketName(); + } + $worker_name_length = strlen($worker->name); + if (self::$_maxWorkerNameLength < $worker_name_length) { + self::$_maxWorkerNameLength = $worker_name_length; + } + } + + $worker->count = $worker->count <= 0 ? 1 : $worker->count; + while (count(self::$_pidMap[$worker->workerId]) < $worker->count) { + static::forkOneWorker($worker); + } + } + } + + /** + * Fork one worker process. + * + * @param Worker $worker + * @throws Exception + */ + protected static function forkOneWorker($worker) + { + // Get available worker id. + $id = self::getId($worker->workerId, 0); + if ($id === false) { + return; + } + $pid = pcntl_fork(); + // For master process. + if ($pid > 0) { + self::$_pidMap[$worker->workerId][$pid] = $pid; + self::$_idMap[$worker->workerId][$id] = $pid; + } // For child processes. + elseif (0 === $pid) { + if ($worker->reusePort) { + $worker->listen(); + } + if (self::$_status === self::STATUS_STARTING) { + self::resetStd(); + } + self::$_pidMap = array(); + self::$_workers = array($worker->workerId => $worker); + Timer::delAll(); + self::setProcessTitle('WorkerMan: worker process ' . $worker->name . ' ' . $worker->getSocketName()); + $worker->setUserAndGroup(); + $worker->id = $id; + $worker->run(); + exit(250); + } else { + throw new Exception("forkOneWorker fail"); + } + } + + /** + * Get worker id. + * + * @param int $worker_id + * @param int $pid + */ + protected static function getId($worker_id, $pid) + { + return array_search($pid, self::$_idMap[$worker_id]); + } + + /** + * Set unix user and group for current process. + * + * @return void + */ + public function setUserAndGroup() + { + // Get uid. + $user_info = posix_getpwnam($this->user); + if (!$user_info) { + self::log("Warning: User {$this->user} not exsits"); + return; + } + $uid = $user_info['uid']; + // Get gid. + if ($this->group) { + $group_info = posix_getgrnam($this->group); + if (!$group_info) { + self::log("Warning: Group {$this->group} not exsits"); + return; + } + $gid = $group_info['gid']; + } else { + $gid = $user_info['gid']; + } + + // Set uid and gid. + if ($uid != posix_getuid() || $gid != posix_getgid()) { + if (!posix_setgid($gid) || !posix_initgroups($user_info['name'], $gid) || !posix_setuid($uid)) { + self::log("Warning: change gid or uid fail."); + } + } + } + + /** + * Set process name. + * + * @param string $title + * @return void + */ + protected static function setProcessTitle($title) + { + // >=php 5.5 + if (function_exists('cli_set_process_title')) { + @cli_set_process_title($title); + } // Need proctitle when php<=5.5 . + elseif (extension_loaded('proctitle') && function_exists('setproctitle')) { + @setproctitle($title); + } + } + + /** + * Monitor all child processes. + * + * @return void + */ + protected static function monitorWorkers() + { + self::$_status = self::STATUS_RUNNING; + while (1) { + // Calls signal handlers for pending signals. + pcntl_signal_dispatch(); + // Suspends execution of the current process until a child has exited, or until a signal is delivered + $status = 0; + $pid = pcntl_wait($status, WUNTRACED); + // Calls signal handlers for pending signals again. + pcntl_signal_dispatch(); + // If a child has already exited. + if ($pid > 0) { + // Find out witch worker process exited. + foreach (self::$_pidMap as $worker_id => $worker_pid_array) { + if (isset($worker_pid_array[$pid])) { + $worker = self::$_workers[$worker_id]; + // Exit status. + if ($status !== 0) { + self::log("worker[" . $worker->name . ":$pid] exit with status $status"); + } + + // For Statistics. + if (!isset(self::$_globalStatistics['worker_exit_info'][$worker_id][$status])) { + self::$_globalStatistics['worker_exit_info'][$worker_id][$status] = 0; + } + self::$_globalStatistics['worker_exit_info'][$worker_id][$status]++; + + // Clear process data. + unset(self::$_pidMap[$worker_id][$pid]); + + // Mark id is available. + $id = self::getId($worker_id, $pid); + self::$_idMap[$worker_id][$id] = 0; + + break; + } + } + // Is still running state then fork a new worker process. + if (self::$_status !== self::STATUS_SHUTDOWN) { + self::forkWorkers(); + // If reloading continue. + if (isset(self::$_pidsToRestart[$pid])) { + unset(self::$_pidsToRestart[$pid]); + self::reload(); + } + } else { + // If shutdown state and all child processes exited then master process exit. + if (!self::getAllWorkerPids()) { + self::exitAndClearAll(); + } + } + } else { + // If shutdown state and all child processes exited then master process exit. + if (self::$_status === self::STATUS_SHUTDOWN && !self::getAllWorkerPids()) { + self::exitAndClearAll(); + } + } + } + } + + /** + * Exit current process. + * + * @return void + */ + protected static function exitAndClearAll() + { + foreach (self::$_workers as $worker) { + $socket_name = $worker->getSocketName(); + if ($worker->transport === 'unix' && $socket_name) { + list(, $address) = explode(':', $socket_name, 2); + @unlink($address); + } + } + @unlink(self::$pidFile); + self::log("Workerman[" . basename(self::$_startFile) . "] has been stopped"); + if (self::$onMasterStop) { + call_user_func(self::$onMasterStop); + } + exit(0); + } + + /** + * Execute reload. + * + * @return void + */ + protected static function reload() + { + // For master process. + if (self::$_masterPid === posix_getpid()) { + // Set reloading state. + if (self::$_status !== self::STATUS_RELOADING && self::$_status !== self::STATUS_SHUTDOWN) { + self::log("Workerman[" . basename(self::$_startFile) . "] reloading"); + self::$_status = self::STATUS_RELOADING; + // Try to emit onMasterReload callback. + if (self::$onMasterReload) { + try { + call_user_func(self::$onMasterReload); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + self::initId(); + } + } + + // Send reload signal to all child processes. + $reloadable_pid_array = array(); + foreach (self::$_pidMap as $worker_id => $worker_pid_array) { + $worker = self::$_workers[$worker_id]; + if ($worker->reloadable) { + foreach ($worker_pid_array as $pid) { + $reloadable_pid_array[$pid] = $pid; + } + } else { + foreach ($worker_pid_array as $pid) { + // Send reload signal to a worker process which reloadable is false. + posix_kill($pid, SIGUSR1); + } + } + } + + // Get all pids that are waiting reload. + self::$_pidsToRestart = array_intersect(self::$_pidsToRestart, $reloadable_pid_array); + + // Reload complete. + if (empty(self::$_pidsToRestart)) { + if (self::$_status !== self::STATUS_SHUTDOWN) { + self::$_status = self::STATUS_RUNNING; + } + return; + } + // Continue reload. + $one_worker_pid = current(self::$_pidsToRestart); + // Send reload signal to a worker process. + posix_kill($one_worker_pid, SIGUSR1); + // If the process does not exit after self::KILL_WORKER_TIMER_TIME seconds try to kill it. + Timer::add(self::KILL_WORKER_TIMER_TIME, 'posix_kill', array($one_worker_pid, SIGKILL), false); + } // For child processes. + else { + $worker = current(self::$_workers); + // Try to emit onWorkerReload callback. + if ($worker->onWorkerReload) { + try { + call_user_func($worker->onWorkerReload, $worker); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + + if ($worker->reloadable) { + self::stopAll(); + } + } + } + + /** + * Stop. + * + * @return void + */ + public static function stopAll() + { + self::$_status = self::STATUS_SHUTDOWN; + // For master process. + if (self::$_masterPid === posix_getpid()) { + self::log("Workerman[" . basename(self::$_startFile) . "] Stopping ..."); + $worker_pid_array = self::getAllWorkerPids(); + // Send stop signal to all child processes. + foreach ($worker_pid_array as $worker_pid) { + posix_kill($worker_pid, SIGINT); + Timer::add(self::KILL_WORKER_TIMER_TIME, 'posix_kill', array($worker_pid, SIGKILL), false); + } + } // For child processes. + else { + // Execute exit. + foreach (self::$_workers as $worker) { + $worker->stop(); + } + exit(0); + } + } + + /** + * Write statistics data to disk. + * + * @return void + */ + protected static function writeStatisticsToStatusFile() + { + // For master process. + if (self::$_masterPid === posix_getpid()) { + $loadavg = sys_getloadavg(); + file_put_contents(self::$_statisticsFile, + "---------------------------------------GLOBAL STATUS--------------------------------------------\n"); + file_put_contents(self::$_statisticsFile, + 'Workerman version:' . Worker::VERSION . " PHP version:" . PHP_VERSION . "\n", FILE_APPEND); + file_put_contents(self::$_statisticsFile, 'start time:' . date('Y-m-d H:i:s', + self::$_globalStatistics['start_timestamp']) . ' run ' . floor((time() - self::$_globalStatistics['start_timestamp']) / (24 * 60 * 60)) . ' days ' . floor(((time() - self::$_globalStatistics['start_timestamp']) % (24 * 60 * 60)) / (60 * 60)) . " hours \n", + FILE_APPEND); + $load_str = 'load average: ' . implode(", ", $loadavg); + file_put_contents(self::$_statisticsFile, + str_pad($load_str, 33) . 'event-loop:' . self::getEventLoopName() . "\n", FILE_APPEND); + file_put_contents(self::$_statisticsFile, + count(self::$_pidMap) . ' workers ' . count(self::getAllWorkerPids()) . " processes\n", + FILE_APPEND); + file_put_contents(self::$_statisticsFile, + str_pad('worker_name', self::$_maxWorkerNameLength) . " exit_status exit_count\n", FILE_APPEND); + foreach (self::$_pidMap as $worker_id => $worker_pid_array) { + $worker = self::$_workers[$worker_id]; + if (isset(self::$_globalStatistics['worker_exit_info'][$worker_id])) { + foreach (self::$_globalStatistics['worker_exit_info'][$worker_id] as $worker_exit_status => $worker_exit_count) { + file_put_contents(self::$_statisticsFile, + str_pad($worker->name, self::$_maxWorkerNameLength) . " " . str_pad($worker_exit_status, + 16) . " $worker_exit_count\n", FILE_APPEND); + } + } else { + file_put_contents(self::$_statisticsFile, + str_pad($worker->name, self::$_maxWorkerNameLength) . " " . str_pad(0, 16) . " 0\n", + FILE_APPEND); + } + } + file_put_contents(self::$_statisticsFile, + "---------------------------------------PROCESS STATUS-------------------------------------------\n", + FILE_APPEND); + file_put_contents(self::$_statisticsFile, + "pid\tmemory " . str_pad('listening', self::$_maxSocketNameLength) . " " . str_pad('worker_name', + self::$_maxWorkerNameLength) . " connections " . str_pad('total_request', + 13) . " " . str_pad('send_fail', 9) . " " . str_pad('throw_exception', 15) . "\n", FILE_APPEND); + + chmod(self::$_statisticsFile, 0722); + + foreach (self::getAllWorkerPids() as $worker_pid) { + posix_kill($worker_pid, SIGUSR2); + } + return; + } + + // For child processes. + /** @var Worker $worker */ + $worker = current(self::$_workers); + $worker_status_str = posix_getpid() . "\t" . str_pad(round(memory_get_usage(true) / (1024 * 1024), 2) . "M", + 7) . " " . str_pad($worker->getSocketName(), + self::$_maxSocketNameLength) . " " . str_pad(($worker->name === $worker->getSocketName() ? 'none' : $worker->name), + self::$_maxWorkerNameLength) . " "; + $worker_status_str .= str_pad(ConnectionInterface::$statistics['connection_count'], + 11) . " " . str_pad(ConnectionInterface::$statistics['total_request'], + 14) . " " . str_pad(ConnectionInterface::$statistics['send_fail'], + 9) . " " . str_pad(ConnectionInterface::$statistics['throw_exception'], 15) . "\n"; + file_put_contents(self::$_statisticsFile, $worker_status_str, FILE_APPEND); + } + + /** + * Check errors when current process exited. + * + * @return void + */ + public static function checkErrors() + { + if (self::STATUS_SHUTDOWN != self::$_status) { + $error_msg = "WORKER EXIT UNEXPECTED "; + $errors = error_get_last(); + if ($errors && ($errors['type'] === E_ERROR || + $errors['type'] === E_PARSE || + $errors['type'] === E_CORE_ERROR || + $errors['type'] === E_COMPILE_ERROR || + $errors['type'] === E_RECOVERABLE_ERROR) + ) { + $error_msg .= self::getErrorType($errors['type']) . " {$errors['message']} in {$errors['file']} on line {$errors['line']}"; + } + self::log($error_msg); + } + } + + /** + * Get error message by error code. + * + * @param integer $type + * @return string + */ + protected static function getErrorType($type) + { + switch ($type) { + case E_ERROR: // 1 // + return 'E_ERROR'; + case E_WARNING: // 2 // + return 'E_WARNING'; + case E_PARSE: // 4 // + return 'E_PARSE'; + case E_NOTICE: // 8 // + return 'E_NOTICE'; + case E_CORE_ERROR: // 16 // + return 'E_CORE_ERROR'; + case E_CORE_WARNING: // 32 // + return 'E_CORE_WARNING'; + case E_COMPILE_ERROR: // 64 // + return 'E_COMPILE_ERROR'; + case E_COMPILE_WARNING: // 128 // + return 'E_COMPILE_WARNING'; + case E_USER_ERROR: // 256 // + return 'E_USER_ERROR'; + case E_USER_WARNING: // 512 // + return 'E_USER_WARNING'; + case E_USER_NOTICE: // 1024 // + return 'E_USER_NOTICE'; + case E_STRICT: // 2048 // + return 'E_STRICT'; + case E_RECOVERABLE_ERROR: // 4096 // + return 'E_RECOVERABLE_ERROR'; + case E_DEPRECATED: // 8192 // + return 'E_DEPRECATED'; + case E_USER_DEPRECATED: // 16384 // + return 'E_USER_DEPRECATED'; + } + return ""; + } + + /** + * Log. + * + * @param string $msg + * @return void + */ + public static function log($msg) + { + $msg = $msg . "\n"; + if (!self::$daemonize) { + self::safeEcho($msg); + } + file_put_contents((string)self::$logFile, date('Y-m-d H:i:s') . ' ' . 'pid:'. posix_getpid() . ' ' . $msg, FILE_APPEND | LOCK_EX); + } + + /** + * Safe Echo. + * + * @param $msg + */ + public static function safeEcho($msg) + { + if (!function_exists('posix_isatty') || posix_isatty(STDOUT)) { + echo $msg; + } + } + + /** + * Construct. + * + * @param string $socket_name + * @param array $context_option + */ + public function __construct($socket_name = '', $context_option = array()) + { + // Save all worker instances. + $this->workerId = spl_object_hash($this); + self::$_workers[$this->workerId] = $this; + self::$_pidMap[$this->workerId] = array(); + + // Get autoload root path. + $backtrace = debug_backtrace(); + $this->_autoloadRootPath = dirname($backtrace[0]['file']); + + // Context for socket. + if ($socket_name) { + $this->_socketName = $socket_name; + if (!isset($context_option['socket']['backlog'])) { + $context_option['socket']['backlog'] = self::DEFAULT_BACKLOG; + } + $this->_context = stream_context_create($context_option); + } + + // Set an empty onMessage callback. + $this->onMessage = function () { + }; + } + + /** + * Listen port. + * + * @throws Exception + */ + public function listen() + { + if (!$this->_socketName || $this->_mainSocket) { + return; + } + + // Autoload. + Autoloader::setRootPath($this->_autoloadRootPath); + + // Get the application layer communication protocol and listening address. + list($scheme, $address) = explode(':', $this->_socketName, 2); + // Check application layer protocol class. + if (!isset(self::$_builtinTransports[$scheme])) { + if(class_exists($scheme)){ + $this->protocol = $scheme; + } else { + $scheme = ucfirst($scheme); + $this->protocol = '\\Protocols\\' . $scheme; + if (!class_exists($this->protocol)) { + $this->protocol = "\\Workerman\\Protocols\\$scheme"; + if (!class_exists($this->protocol)) { + throw new Exception("class \\Protocols\\$scheme not exist"); + } + } + } + if (!isset(self::$_builtinTransports[$this->transport])) { + throw new \Exception('Bad worker->transport ' . var_export($this->transport, true)); + } + } else { + $this->transport = $scheme; + } + + $local_socket = self::$_builtinTransports[$this->transport] . ":" . $address; + + // Flag. + $flags = $this->transport === 'udp' ? STREAM_SERVER_BIND : STREAM_SERVER_BIND | STREAM_SERVER_LISTEN; + $errno = 0; + $errmsg = ''; + // SO_REUSEPORT. + if ($this->reusePort) { + stream_context_set_option($this->_context, 'socket', 'so_reuseport', 1); + } + + // Create an Internet or Unix domain server socket. + $this->_mainSocket = stream_socket_server($local_socket, $errno, $errmsg, $flags, $this->_context); + if (!$this->_mainSocket) { + throw new Exception($errmsg); + } + + if ($this->transport === 'ssl') { + stream_socket_enable_crypto($this->_mainSocket, false); + } + + // Try to open keepalive for tcp and disable Nagle algorithm. + if (function_exists('socket_import_stream') && self::$_builtinTransports[$this->transport] === 'tcp') { + $socket = socket_import_stream($this->_mainSocket); + @socket_set_option($socket, SOL_SOCKET, SO_KEEPALIVE, 1); + @socket_set_option($socket, SOL_TCP, TCP_NODELAY, 1); + } + + // Non blocking. + stream_set_blocking($this->_mainSocket, 0); + + // Register a listener to be notified when server socket is ready to read. + if (self::$globalEvent) { + if ($this->transport !== 'udp') { + self::$globalEvent->add($this->_mainSocket, EventInterface::EV_READ, array($this, 'acceptConnection')); + } else { + self::$globalEvent->add($this->_mainSocket, EventInterface::EV_READ, + array($this, 'acceptUdpConnection')); + } + } + } + + /** + * Get socket name. + * + * @return string + */ + public function getSocketName() + { + return $this->_socketName ? lcfirst($this->_socketName) : 'none'; + } + + /** + * Run worker instance. + * + * @return void + */ + public function run() + { + //Update process state. + self::$_status = self::STATUS_RUNNING; + + // Register shutdown function for checking errors. + register_shutdown_function(array("\\Workerman\\Worker", 'checkErrors')); + + // Set autoload root path. + Autoloader::setRootPath($this->_autoloadRootPath); + + // Create a global event loop. + if (!self::$globalEvent) { + $eventLoopClass = "\\Workerman\\Events\\" . ucfirst(self::getEventLoopName()); + self::$globalEvent = new $eventLoopClass; + // Register a listener to be notified when server socket is ready to read. + if ($this->_socketName) { + if ($this->transport !== 'udp') { + self::$globalEvent->add($this->_mainSocket, EventInterface::EV_READ, + array($this, 'acceptConnection')); + } else { + self::$globalEvent->add($this->_mainSocket, EventInterface::EV_READ, + array($this, 'acceptUdpConnection')); + } + } + } + + // Reinstall signal. + self::reinstallSignal(); + + // Init Timer. + Timer::init(self::$globalEvent); + + // Try to emit onWorkerStart callback. + if ($this->onWorkerStart) { + try { + call_user_func($this->onWorkerStart, $this); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + + // Main loop. + self::$globalEvent->loop(); + } + + /** + * Stop current worker instance. + * + * @return void + */ + public function stop() + { + // Try to emit onWorkerStop callback. + if ($this->onWorkerStop) { + try { + call_user_func($this->onWorkerStop, $this); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + // Remove listener for server socket. + self::$globalEvent->del($this->_mainSocket, EventInterface::EV_READ); + @fclose($this->_mainSocket); + } + + /** + * Accept a connection. + * + * @param resource $socket + * @return void + */ + public function acceptConnection($socket) + { + // Accept a connection on server socket. + $new_socket = @stream_socket_accept($socket, 0, $remote_address); + // Thundering herd. + if (!$new_socket) { + return; + } + + // TcpConnection. + $connection = new TcpConnection($new_socket, $remote_address); + $this->connections[$connection->id] = $connection; + $connection->worker = $this; + $connection->protocol = $this->protocol; + $connection->transport = $this->transport; + $connection->onMessage = $this->onMessage; + $connection->onClose = $this->onClose; + $connection->onError = $this->onError; + $connection->onBufferDrain = $this->onBufferDrain; + $connection->onBufferFull = $this->onBufferFull; + + // Try to emit onConnect callback. + if ($this->onConnect) { + try { + call_user_func($this->onConnect, $connection); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + } + + /** + * For udp package. + * + * @param resource $socket + * @return bool + */ + public function acceptUdpConnection($socket) + { + $recv_buffer = stream_socket_recvfrom($socket, self::MAX_UDP_PACKAGE_SIZE, 0, $remote_address); + if (false === $recv_buffer || empty($remote_address)) { + return false; + } + // UdpConnection. + $connection = new UdpConnection($socket, $remote_address); + $connection->protocol = $this->protocol; + if ($this->onMessage) { + if ($this->protocol) { + $parser = $this->protocol; + $recv_buffer = $parser::decode($recv_buffer, $connection); + // Discard bad packets. + if ($recv_buffer === false) + return true; + } + ConnectionInterface::$statistics['total_request']++; + try { + call_user_func($this->onMessage, $connection, $recv_buffer); + } catch (\Exception $e) { + self::log($e); + exit(250); + } catch (\Error $e) { + self::log($e); + exit(250); + } + } + return true; + } +} diff --git a/vendor/workerman/workerman/composer.json b/vendor/workerman/workerman/composer.json new file mode 100644 index 000000000..733ab8a6c --- /dev/null +++ b/vendor/workerman/workerman/composer.json @@ -0,0 +1,33 @@ +{ + "name" : "workerman/workerman", + "type" : "library", + "keywords": ["event-loop", "asynchronous"], + "homepage": "http://www.workerman.net", + "license" : "MIT", + "description": "An asynchronous event driven PHP framework for easily building fast, scalable network applications.", + "authors" : [ + { + "name" : "walkor", + "email" : "walkor@workerman.net", + "homepage" : "http://www.workerman.net", + "role": "Developer" + } + ], + "support" : { + "email" : "walkor@workerman.net", + "issues": "https://github.com/walkor/workerman/issues", + "forum" : "http://wenda.workerman.net/", + "wiki" : "http://doc3.workerman.net/index.html", + "source": "https://github.com/walkor/workerman" + }, + "require": { + "php": ">=5.3" + }, + "suggest": { + "ext-event": "For better performance." + }, + "autoload": { + "psr-4": {"Workerman\\": "./"} + }, + "minimum-stability":"dev" +} diff --git a/vendor/zoujingli/wechat-php-sdk/MIT-LICENSE.txt b/vendor/zoujingli/wechat-php-sdk/MIT-LICENSE.txt new file mode 100644 index 000000000..d07e2ebab --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/MIT-LICENSE.txt @@ -0,0 +1,21 @@ +The MIT License + +Copyright (c) 2009-2015 walkor and contributors (see https://github.com/walkor/workerman/contributors) + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/vendor/zoujingli/wechat-php-sdk/README.md b/vendor/zoujingli/wechat-php-sdk/README.md new file mode 100644 index 000000000..a579d7722 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/README.md @@ -0,0 +1,86 @@ +[![Latest Stable Version](https://poser.pugx.org/zoujingli/wechat-php-sdk/v/stable)](https://packagist.org/packages/zoujingli/wechat-php-sdk) +[![Total Downloads](https://poser.pugx.org/zoujingli/wechat-php-sdk/downloads)](https://packagist.org/packages/zoujingli/wechat-php-sdk) +[![Latest Unstable Version](https://poser.pugx.org/zoujingli/wechat-php-sdk/v/unstable)](https://packagist.org/packages/zoujingli/wechat-php-sdk) +[![License](https://poser.pugx.org/zoujingli/wechat-php-sdk/license)](https://packagist.org/packages/zoujingli/wechat-php-sdk) + +姝SDK`杩愯鏈搴曡姹俙PHP`鐗堟湰`5.3`, 寤鸿鍦╜PHP7`涓婅繍琛屼互鑾峰彇鏈浣虫ц兘銆 + +寰俊鐨勯儴鍒嗘帴鍙i渶瑕佺紦瀛樻暟鎹湪鏈湴锛屽洜姝ゅ鐩綍闇瑕佹湁鍐欐潈闄愩 + +鎴戜滑榧撳姳澶у浣跨敤`composer`鏉ョ鐞嗘偍鐨勭涓夋柟搴擄紝鏂逛究鍚庢湡鏇存柊鎿嶄綔锛堝挨鍏舵槸鎺ュ彛绫伙級銆 + +杩戞湡`access_token`缁忓父鏃犳晠澶辨晥锛宍SDK`宸插姞鍏ュけ璐ョ姸鎬佹娴嬶紝鑷姩閲嶆柊鑾峰彇`access_token`骞惰繑鍥炵粨鏋. + +姝SDK`宸插巻缁忔暟涓嚎涓婇」鐩獙璇佷笌鑰冮獙锛屽彲闈犳т笌绋冲畾鎬ф瀬楂橈紝娆㈣繋`fork`鎴朻star`姝ら」鐩 + + +**寰俊SDK寮鍙戝府鍔╁強浜ゆ祦** +-- +* **鍦ㄥ仛寰俊寮鍙戝墠锛屽繀闇鍏堥槄璇诲井淇″畼鏂规枃妗o紝姝DK涔熸槸鍩轰簬涔嬩笂杩涜鐨勫皝瑁呫** + +* **鏂囨。閾炬帴鍦板潃**锛歨ttp://www.kancloud.cn/zoujingli/wechat-php-sdk + +* **寮鍙戜氦娴丵Q缇わ細513350915锛堟柊锛** + +**鑻ュ鎮ㄦ湁甯姪锛屽彲浠ヨ禐鍔╁苟鏀寔涓嬩綔鑰呭摝锛岃阿璋紒 ^_^** +-- +![](https://git.kancloud.cn/repos/zoujingli/wechat-php-sdk/raw/master/image/%E8%B5%9E%E5%8A%A9.png?access-token=49255b63935edafaf42aec9376136528&v1) + + +**瀹樻柟鎺ュ彛鏂囨。閾炬帴** +-- +* 浣跨敤鍓嶉渶鍏堟墦寮寰俊甯愬彿鐨勫紑鍙戞ā寮忥紝璇︾粏姝ラ璇锋煡鐪嬪井淇″叕浼楀钩鍙版帴鍙d娇鐢ㄨ鏄庯細 +* 1. 寰俊鍏紬骞冲彴锛 http://mp.weixin.qq.com/wiki/ +* 2. 寰俊浼佷笟骞冲彴锛 http://qydev.weixin.qq.com/wiki/ +* 3. 寰俊寮鏀惧钩鍙帮細https://open.weixin.qq.com/ +* 4. 寰俊鏀粯鎺ュ叆鏂囨。锛歨ttps://mp.weixin.qq.com/cgi-bin/readtemplate?t=business/course2_tmpl&lang=zh_CN +* 5. 寰俊鍟嗘埛骞冲彴锛歨ttps://pay.weixin.qq.com + +**寰俊`SDK`椤圭洰婧愭枃浠舵墭绠** +-- +* SDK 涓哄紑婧愰」鐩紝浣犲彲浠ユ妸瀹冪敤浜庝换浣曞湴鍧锛屽苟涓嶅彈浠讳綍绾︽潫锛屾杩巂fork`椤圭洰銆 +* 閫氳繃 [Github](https://github.com/zoujingli/wechat-php-sdk) 涓嬭浇 SDK 婧愪唬鐮 +* 閫氳繃 [OSChina](http://git.oschina.net/zoujingli/wechat-php-sdk) 涓嬭浇 SDK 婧愪唬鐮 +* 閫氳繃 [Composer](https://getcomposer.org) 鍖呯鐞嗗伐鍏蜂笅杞 SDK 婧愪唬鐮 + +**寰俊`SDK`灏佽瀵规帴鍙婂姛鑳** +-- +* 鎺ュ叆楠岃瘉 锛堝垵绾ф潈闄愶級 +* 鑷姩鍥炲锛堟枃鏈佸浘鐗囥佽闊炽佽棰戙侀煶涔愩佸浘鏂囷級 锛堝垵绾ф潈闄愶級 +* 鑿滃崟鎿嶄綔锛堟煡璇€佸垱寤恒佸垹闄わ級 锛堣彍鍗曟潈闄愶級 +* 瀹㈡湇娑堟伅锛堟枃鏈佸浘鐗囥佽闊炽佽棰戙侀煶涔愩佸浘鏂囷級 锛堣璇佹潈闄愶級 +* 浜岀淮鐮侊紙鍒涘缓涓存椂銆佹案涔呬簩缁寸爜锛岃幏鍙栦簩缁寸爜URL锛 锛堟湇鍔″彿銆佽璇佹潈闄愶級 +* 闀块摼鎺ヨ浆鐭摼鎺ユ帴鍙 锛堟湇鍔″彿銆佽璇佹潈闄愶級 +* 鏍囩鎿嶄綔锛堟煡璇€佸垱寤恒佷慨鏀广佺Щ鍔ㄧ敤鎴峰埌鏍囩锛 锛堣璇佹潈闄愶級 +* 缃戦〉鎺堟潈锛堝熀鏈巿鏉冿紝鐢ㄦ埛淇℃伅鎺堟潈锛 锛堟湇鍔″彿銆佽璇佹潈闄愶級 +* 鐢ㄦ埛淇℃伅锛堟煡璇㈢敤鎴峰熀鏈俊鎭佽幏鍙栧叧娉ㄨ呭垪琛級 锛堣璇佹潈闄愶級 +* 澶氬鏈嶅姛鑳斤紙瀹㈡湇绠$悊銆佽幏鍙栧鏈嶈褰曘佸鏈嶄細璇濈鐞嗭級 锛堣璇佹潈闄愶級 +* 濯掍綋鏂囦欢锛堜笂浼犮佽幏鍙栵級 锛堣璇佹潈闄愶級 +* 楂樼骇缇ゅ彂 锛堣璇佹潈闄愶級 +* 妯℃澘娑堟伅锛堣缃墍灞炶涓氥佹坊鍔犳ā鏉裤佸彂閫佹ā鏉挎秷鎭級 锛堟湇鍔″彿銆佽璇佹潈闄愶級 +* 鍗″埜绠$悊锛堝垱寤恒佷慨鏀广佸垹闄ゃ佸彂鏀俱侀棬搴楃鐞嗙瓑锛 锛堣璇佹潈闄愶級 +* 璇箟鐞嗚В 锛堟湇鍔″彿銆佽璇佹潈闄愶級 +* 鑾峰彇寰俊鏈嶅姟鍣↖P鍒楄〃 锛堝垵绾ф潈闄愶級 +* 寰俊JSAPI鎺堟潈(鑾峰彇ticket銆佽幏鍙栫鍚) 锛堝垵绾ф潈闄愶級 +* 鏁版嵁缁熻(鐢ㄦ埛銆佸浘鏂囥佹秷鎭佹帴鍙e垎鏋愭暟鎹) 锛堣璇佹潈闄愶級 +* 寰俊鏀粯锛堢綉椤垫敮浠樸佹壂鐮佹敮浠樸佷氦鏄撻娆俱佺粰绮変笣鎵撴锛夛紙璁よ瘉鏈嶅姟鍙峰苟寮閫氭敮浠樺姛鑳斤級 + +**鎺ュ彛鏉冮檺澶囨敞锛** +-- +* 鍒濈骇鏉冮檺锛氬熀鏈潈闄愶紝浠讳綍姝e父鐨勫叕浼楀彿閮芥湁姝ゆ潈闄 +* 鑿滃崟鏉冮檺锛氭甯哥殑鏈嶅姟鍙枫佽璇佸悗鐨勮闃呭彿鎷ユ湁姝ゆ潈闄 +* 璁よ瘉鏉冮檺锛氬垎涓鸿闃呭彿銆佹湇鍔″彿璁よ瘉锛屽鍓嶇紑鏈嶅姟鍙峰垯浠呰璇佺殑鏈嶅姟鍙锋湁姝ゆ潈闄 +* 鏀粯鏉冮檺锛氫粎璁よ瘉鍚庣殑鏈嶅姟鍙峰彲浠ョ敵璇锋鏉冮檺 + +**寰俊寮鏀剧涓夋柟骞冲彴** --- 锛堟渚嬪強鏂囨。鏁寸悊涓級 +-- +* 鍏紬鍙锋巿鏉冩湇鍔 +* 鍏紬鍙锋帹閫佹秷鎭唬澶勭悊 +* 鍏紬鍙峰熀纭涓氬姟浠e鐞 +* 鍏紬鍙锋敮浠樹唬鍙戣捣 + +**寰俊`SDK`鐗堟潈澹版槑** +-- +* 姝DK鍩轰簬`MIT`鍗忚鍙戝竷锛屼换浣曚汉鍙互鐢ㄥ湪浠讳綍鍦版柟锛屼笉鍙楃害鏉 +* 姝DK閮ㄥ垎浠g爜鏉ヨ嚜浜掕仈缃戯紝鑻ユ湁寮傝锛屽彲浠ヨ仈绯讳綔鑰呰繘琛屽垹闄 + diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Cache.php b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Cache.php new file mode 100644 index 000000000..0335f6f4e --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Cache.php @@ -0,0 +1,93 @@ + + * @date 2016-08-20 17:50 + */ +class Cache { + + /** + * 缂撳瓨浣嶇疆 + * @var string + */ + static public $cachepath; + + /** + * 璁剧疆缂撳瓨 + * @param string $name + * @param string $value + * @param int $expired + * @return mixed + */ + static public function set($name, $value, $expired = 0) { + if (isset(Loader::$callback['CacheSet'])) { + return call_user_func_array(Loader::$callback['CacheSet'], func_get_args()); + } + $data = serialize(array('value' => $value, 'expired' => $expired > 0 ? time() + $expired : 0)); + return self::check() && file_put_contents(self::$cachepath . $name, $data); + } + + /** + * 璇诲彇缂撳瓨 + * @param string $name + * @return mixed + */ + static public function get($name) { + if (isset(Loader::$callback['CacheGet'])) { + return call_user_func_array(Loader::$callback['CacheGet'], func_get_args()); + } + if (self::check() && ($file = self::$cachepath . $name) && file_exists($file) && ($data = file_get_contents($file)) && !empty($data)) { + $data = unserialize($data); + if (isset($data['expired']) && ($data['expired'] > time() || $data['expired'] === 0)) { + return isset($data['value']) ? $data['value'] : null; + } + } + return null; + } + + /** + * 鍒犻櫎缂撳瓨 + * @param string $name + * @return mixed + */ + static public function del($name) { + if (isset(Loader::$callback['CacheDel'])) { + return call_user_func_array(Loader::$callback['CacheDel'], func_get_args()); + } + return self::check() && @unlink(self::$cachepath . $name); + } + + /** + * 杈撳嚭鍐呭鍒版棩蹇 + * @param string $line + * @param string $filename + * @return mixed + */ + static public function put($line, $filename = '') { + if (isset(Loader::$callback['CachePut'])) { + return call_user_func_array(Loader::$callback['CachePut'], func_get_args()); + } + empty($filename) && $filename = date('Ymd') . '.log'; + return self::check() && file_put_contents(self::$cachepath . $filename, '[' . date('Y/m/d H:i:s') . "] {$line}\n", FILE_APPEND); + } + + /** + * 妫鏌ョ紦瀛樼洰褰 + * @return bool + */ + static protected function check() { + empty(self::$cachepath) && self::$cachepath = dirname(__DIR__) . DIRECTORY_SEPARATOR . 'Cache' . DIRECTORY_SEPARATOR; + self::$cachepath = rtrim(self::$cachepath, '/\\') . DIRECTORY_SEPARATOR; + if (!is_dir(self::$cachepath) && !mkdir(self::$cachepath, 0755, TRUE)) { + return FALSE; + } + return TRUE; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Common.php b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Common.php new file mode 100644 index 000000000..7a50f5513 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Common.php @@ -0,0 +1,179 @@ + + * @date 2016/05/28 11:55 + */ +class Common { + + /** API鎺ュ彛URL闇瑕佷娇鐢ㄦ鍓嶇紑 */ + const API_BASE_URL_PREFIX = 'https://api.weixin.qq.com'; + const API_URL_PREFIX = 'https://api.weixin.qq.com/cgi-bin'; + const GET_TICKET_URL = '/ticket/getticket?'; + const AUTH_URL = '/token?grant_type=client_credential&'; + public $token; + public $encodingAesKey; + public $encrypt_type; + public $appid; + public $appsecret; + public $access_token; + public $postxml; + public $_msg; + public $errCode = 0; + public $errMsg = "no access"; + public $config = array(); + private $_retry = FALSE; + + /** + * 鏋勯犳柟娉 + * @param array $options + */ + public function __construct($options = array()) { + $config = Loader::config($options); + $this->token = isset($config['token']) ? $config['token'] : ''; + $this->appid = isset($config['appid']) ? $config['appid'] : ''; + $this->appsecret = isset($config['appsecret']) ? $config['appsecret'] : ''; + $this->encodingAesKey = isset($config['encodingaeskey']) ? $config['encodingaeskey'] : ''; + $this->config = $config; + } + + /** + * 鎺ュ彛楠岃瘉 + * @return bool + */ + public function valid() { + $encryptStr = ""; + if ($_SERVER['REQUEST_METHOD'] == "POST") { + $postStr = file_get_contents("php://input"); + $array = (array)simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA); + $this->encrypt_type = isset($_GET["encrypt_type"]) ? $_GET["encrypt_type"] : ''; + if ($this->encrypt_type == 'aes') { + $encryptStr = $array['Encrypt']; + !class_exists('Prpcrypt', FALSE) && require __DIR__ . '/Prpcrypt.php'; + $pc = new Prpcrypt($this->encodingAesKey); + $array = $pc->decrypt($encryptStr, $this->appid); + if (!isset($array[0]) || intval($array[0]) > 0) { + $this->errCode = $array[0]; + $this->errMsg = $array[1]; + Tools::log("Interface Authentication Failed. {$this->errMsg}[{$this->errCode}]", 'ERR'); + return false; + } + $this->postxml = $array[1]; + empty($this->appid) && $this->appid = $array[2]; + } else { + $this->postxml = $postStr; + } + } elseif (isset($_GET["echostr"])) { + if ($this->checkSignature()) { + exit($_GET["echostr"]); + } else { + return false; + } + } + if (!$this->checkSignature($encryptStr)) { + $this->errMsg = 'Interface authentication failed, please use the correct method to call.'; + return false; + } + return true; + } + + /** + * 楠岃瘉鏉ヨ嚜寰俊鏈嶅姟鍣 + * @param string $str + * @return bool + */ + private function checkSignature($str = '') { + // 濡傛灉瀛樺湪鍔犲瘑楠岃瘉鍒欑敤鍔犲瘑楠岃瘉娈 + $signature = isset($_GET["msg_signature"]) ? $_GET["msg_signature"] : (isset($_GET["signature"]) ? $_GET["signature"] : ''); + $timestamp = isset($_GET["timestamp"]) ? $_GET["timestamp"] : ''; + $nonce = isset($_GET["nonce"]) ? $_GET["nonce"] : ''; + $tmpArr = array($this->token, $timestamp, $nonce, $str); + sort($tmpArr, SORT_STRING); + if (sha1(implode($tmpArr)) == $signature) { + return true; + } else { + return false; + } + } + + /** + * 鑾峰彇鍏紬鍙疯闂 access_token + * @param string $appid 濡傚湪绫诲垵濮嬪寲鏃跺凡鎻愪緵锛屽垯鍙负绌 + * @param string $appsecret 濡傚湪绫诲垵濮嬪寲鏃跺凡鎻愪緵锛屽垯鍙负绌 + * @param string $token 鎵嬪姩鎸囧畾access_token锛岄潪蹇呰鎯呭喌涓嶅缓璁敤 + * @return bool|string + */ + public function getAccessToken($appid = '', $appsecret = '', $token = '') { + if (!$appid || !$appsecret) { + $appid = $this->appid; + $appsecret = $this->appsecret; + } + if ($token) { + return $this->access_token = $token; + } + $cache = 'wechat_access_token_' . $appid; + if (($access_token = Tools::getCache($cache)) && !empty($access_token)) { + return $this->access_token = $access_token; + } + # 妫娴嬩簨浠舵敞鍐 + if (isset(Loader::$callback[__FUNCTION__])) { + return $this->access_token = call_user_func_array(Loader::$callback[__FUNCTION__], array(&$this, &$cache)); + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::AUTH_URL . 'appid=' . $appid . '&secret=' . $appsecret); + if ($result) { + $json = json_decode($result, true); + if (!$json || isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + Tools::log("Get New AccessToken Error. {$this->errMsg}[{$this->errCode}]", 'ERR'); + return false; + } + $this->access_token = $json['access_token']; + Tools::log("Get New AccessToken Success."); + Tools::setCache($cache, $this->access_token, 5000); + return $this->access_token; + } + return false; + } + + /** + * 鎺ュ彛澶辫触閲嶈瘯 + * @param $method SDK鏂规硶鍚嶇О + * @param array $arguments SDK鏂规硶鍙傛暟 + * @return bool|mixed + */ + protected function checkRetry($method, $arguments = array()) { + if (!$this->_retry && in_array($this->errCode, array('40014', '40001', '41001', '42001'))) { + Tools::log("Run {$method} Faild. {$this->errMsg}[{$this->errCode}]", 'ERR'); + ($this->_retry = true) && $this->resetAuth(); + $this->errCode = 40001; + $this->errMsg = 'no access'; + Tools::log("Retry Run {$method} ..."); + return call_user_func_array(array($this, $method), $arguments); + } + return false; + } + + /** + * 鍒犻櫎楠岃瘉鏁版嵁 + * @param string $appid 濡傚湪绫诲垵濮嬪寲鏃跺凡鎻愪緵锛屽垯鍙负绌 + * @return bool + */ + public function resetAuth($appid = '') { + $authname = 'wechat_access_token_' . (empty($appid) ? $this->appid : $appid); + Tools::log("Reset Auth And Remove Old AccessToken."); + $this->access_token = ''; + Tools::removeCache($authname); + return true; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Prpcrypt.php b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Prpcrypt.php new file mode 100644 index 000000000..bc72e300d --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Prpcrypt.php @@ -0,0 +1,187 @@ + PKCS7Encoder::$block_size) { + $pad = 0; + } + return substr($text, 0, (strlen($text) - $pad)); + } + +} + +/** + * 鎺ユ敹鍜屾帹閫佺粰鍏紬骞冲彴娑堟伅鐨勫姞瑙e瘑 + * @category WechatSDK + * @subpackage library + * @date 2016/06/28 11:59 + */ +class Prpcrypt { + + public $key; + + function __construct($k) { + $this->key = base64_decode($k . "="); + } + + /** + * 瀵规槑鏂囪繘琛屽姞瀵 + * @param string $text 闇瑕佸姞瀵嗙殑鏄庢枃 + * @return string 鍔犲瘑鍚庣殑瀵嗘枃 + */ + public function encrypt($text, $appid) { + try { + $random = $this->getRandomStr(); + $text = $random . pack("N", strlen($text)) . $text . $appid; + $size = mcrypt_get_block_size(MCRYPT_RIJNDAEL_128, MCRYPT_MODE_CBC); + $module = mcrypt_module_open(MCRYPT_RIJNDAEL_128, '', MCRYPT_MODE_CBC, ''); + $iv = substr($this->key, 0, 16); + $pkc_encoder = new PKCS7Encoder(); + $text = $pkc_encoder->encode($text); + mcrypt_generic_init($module, $this->key, $iv); + $encrypted = mcrypt_generic($module, $text); + mcrypt_generic_deinit($module); + mcrypt_module_close($module); + return array(ErrorCode::$OK, base64_encode($encrypted)); + } catch (Exception $e) { + return array(ErrorCode::$EncryptAESError, ErrorCode::getErrText(ErrorCode::$EncryptAESError)); + } + } + + /** + * 瀵瑰瘑鏂囪繘琛岃В瀵 + * @param string $encrypted 闇瑕佽В瀵嗙殑瀵嗘枃 + * @return string 瑙e瘑寰楀埌鐨勬槑鏂 + */ + public function decrypt($encrypted, $appid) { + try { + $ciphertext_dec = base64_decode($encrypted); + $module = mcrypt_module_open(MCRYPT_RIJNDAEL_128, '', MCRYPT_MODE_CBC, ''); + $iv = substr($this->key, 0, 16); + mcrypt_generic_init($module, $this->key, $iv); + $decrypted = mdecrypt_generic($module, $ciphertext_dec); + mcrypt_generic_deinit($module); + mcrypt_module_close($module); + } catch (Exception $e) { + return array(ErrorCode::$DecryptAESError, ErrorCode::getErrText(ErrorCode::$DecryptAESError)); + } + try { + $pkc_encoder = new PKCS7Encoder(); + $result = $pkc_encoder->decode($decrypted); + if (strlen($result) < 16) { + return ""; + } + $content = substr($result, 16, strlen($result)); + $len_list = unpack("N", substr($content, 0, 4)); + $xml_len = $len_list[1]; + $xml_content = substr($content, 4, $xml_len); + $from_appid = substr($content, $xml_len + 4); + if (!$appid) { + $appid = $from_appid; + } + } catch (Exception $e) { + return array(ErrorCode::$IllegalBuffer, ErrorCode::getErrText(ErrorCode::$IllegalBuffer)); + } + return array(0, $xml_content, $from_appid); + } + + /** + * 闅忔満鐢熸垚16浣嶅瓧绗︿覆 + * @return string 鐢熸垚鐨勫瓧绗︿覆 + */ + function getRandomStr() { + $str = ""; + $str_pol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz"; + $max = strlen($str_pol) - 1; + for ($i = 0; $i < 16; $i++) { + $str .= $str_pol[mt_rand(0, $max)]; + } + return $str; + } + +} + +/** + * 浠呯敤浣滅被鍐呴儴浣跨敤锛屼笉鐢ㄤ簬瀹樻柟API鎺ュ彛鐨別rrCode鐮 + * @category WechatSDK + * @subpackage library + * @date 2016/06/28 11:59 + */ +class ErrorCode { + + public static $OK = 0; + public static $ValidateSignatureError = 40001; + public static $ParseXmlError = 40002; + public static $ComputeSignatureError = 40003; + public static $IllegalAesKey = 40004; + public static $ValidateAppidError = 40005; + public static $EncryptAESError = 40006; + public static $DecryptAESError = 40007; + public static $IllegalBuffer = 40008; + public static $EncodeBase64Error = 40009; + public static $DecodeBase64Error = 40010; + public static $GenReturnXmlError = 40011; + public static $errCode = array( + '0' => '澶勭悊鎴愬姛', + '40001' => '鏍¢獙绛惧悕澶辫触', + '40002' => '瑙f瀽xml澶辫触', + '40003' => '璁$畻绛惧悕澶辫触', + '40004' => '涓嶅悎娉曠殑AESKey', + '40005' => '鏍¢獙AppID澶辫触', + '40006' => 'AES鍔犲瘑澶辫触', + '40007' => 'AES瑙e瘑澶辫触', + '40008' => '鍏紬骞冲彴鍙戦佺殑xml涓嶅悎娉', + '40009' => 'Base64缂栫爜澶辫触', + '40010' => 'Base64瑙g爜澶辫触', + '40011' => '鍏紬甯愬彿鐢熸垚鍥炲寘xml澶辫触' + ); + + /** + * 鑾峰彇閿欒娑堟伅鍐呭 + * @param type $err + * @return bool + */ + public static function getErrText($err) { + if (isset(self::$errCode[$err])) { + return self::$errCode[$err]; + } else { + return false; + } + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Tools.php b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Tools.php new file mode 100644 index 000000000..c0012e7ad --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/Lib/Tools.php @@ -0,0 +1,268 @@ + + * @date 2016/05/28 11:55 + */ +class Tools { + + /** + * 浜х敓闅忔満瀛楃涓 + * @param int $length + * @param string $str + * @return string + */ + static public function createNoncestr($length = 32, $str = "") { + $chars = "abcdefghijklmnopqrstuvwxyz0123456789"; + for ($i = 0; $i < $length; $i++) { + $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1); + } + return $str; + } + + /** + * 鑾峰彇绛惧悕 + * @param array $arrdata 绛惧悕鏁扮粍 + * @param string $method 绛惧悕鏂规硶 + * @return bool|string 绛惧悕鍊 + */ + static public function getSignature($arrdata, $method = "sha1") { + if (!function_exists($method)) { + return false; + } + ksort($arrdata); + $params = array(); + foreach ($arrdata as $key => $value) { + $params[] = "{$key}={$value}"; + } + return $method(join('&', $params)); + } + + /** + * 鐢熸垚鏀粯绛惧悕 + * @param array $option + * @param string $partnerKey + * @return string + */ + static public function getPaySign($option, $partnerKey) { + ksort($option); + $buff = ''; + foreach ($option as $k => $v) { + $buff .= "{$k}={$v}&"; + } + return strtoupper(md5("{$buff}key={$partnerKey}")); + } + + /** + * XML缂栫爜 + * @param mixed $data 鏁版嵁 + * @param string $root 鏍硅妭鐐瑰悕 + * @param string $item 鏁板瓧绱㈠紩鐨勫瓙鑺傜偣鍚 + * @param string $id 鏁板瓧绱㈠紩瀛愯妭鐐筴ey杞崲鐨勫睘鎬у悕 + * @return string + */ + static public function arr2xml($data, $root = 'xml', $item = 'item', $id = 'id') { + return "<{$root}>" . self::_data_to_xml($data, $item, $id) . ""; + } + + static private function _data_to_xml($data, $item = 'item', $id = 'id', $content = '') { + foreach ($data as $key => $val) { + is_numeric($key) && $key = "{$item} {$id}=\"{$key}\""; + $content .= "<{$key}>"; + if (is_array($val) || is_object($val)) { + $content .= self::_data_to_xml($val); + } elseif (is_numeric($val)) { + $content .= $val; + } else { + $content .= ''; + } + list($_key,) = explode(' ', $key . ' '); + $content .= ""; + } + return $content; + } + + + /** + * 灏唜ml杞负array + * @param string $xml + * @return array + */ + static public function xml2arr($xml) { + return json_decode(Tools::json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true); + } + + /** + * 鐢熸垚瀹夊叏JSON鏁版嵁 + * @param array $array + * @return string + */ + static public function json_encode($array) { + return preg_replace_callback('/\\\\u([0-9a-f]{4})/i', create_function('$matches', 'return mb_convert_encoding(pack("H*", $matches[1]), "UTF-8", "UCS-2BE");'), json_encode($array)); + } + + /** + * 浠et鏂瑰紡鎻愪氦璇锋眰 + * @param $url + * @return bool|mixed + */ + static public function httpGet($url) { + $oCurl = curl_init(); + if (stripos($url, "https://") !== FALSE) { + curl_setopt($oCurl, CURLOPT_SSL_VERIFYPEER, FALSE); + curl_setopt($oCurl, CURLOPT_SSL_VERIFYHOST, FALSE); + curl_setopt($oCurl, CURLOPT_SSLVERSION, 1); + } + curl_setopt($oCurl, CURLOPT_URL, $url); + curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1); + $sContent = curl_exec($oCurl); + $aStatus = curl_getinfo($oCurl); + curl_close($oCurl); + if (intval($aStatus["http_code"]) == 200) { + return $sContent; + } else { + return false; + } + } + + /** + * 浠ost鏂瑰紡鎻愪氦璇锋眰 + * @param string $url + * @param array|string $postdata + * @return bool|mixed + */ + static public function httpPost($url, $postdata) { + $ch = curl_init(); + curl_setopt($ch, CURLOPT_URL, $url); + curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); + curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE); + curl_setopt($ch, CURLOPT_HEADER, FALSE); + curl_setopt($ch, CURLOPT_POST, TRUE); + if (is_array($postdata)) { + foreach ($postdata as &$value) { + if (is_string($value) && stripos($value, '@') === 0 && class_exists('CURLFile', FALSE)) { + $value = new CURLFile(realpath(trim($value, '@'))); + } + } + } + curl_setopt($ch, CURLOPT_POSTFIELDS, $postdata); + $data = curl_exec($ch); + curl_close($ch); + if ($data) { + return $data; + } + return false; + } + + /** + * 浣跨敤璇佷功锛屼互post鏂瑰紡鎻愪氦xml鍒板搴旂殑鎺ュ彛url + * @param string $url POST鎻愪氦鐨勫唴瀹 + * @param array $postdata 璇锋眰鐨勫湴鍧 + * @param string $ssl_cer 璇佷功Cer璺緞 | 璇佷功鍐呭 + * @param string $ssl_key 璇佷功Key璺緞 | 璇佷功鍐呭 + * @param int $second 璁剧疆璇锋眰瓒呮椂鏃堕棿 + * @return bool|mixed + */ + static public function httpsPost($url, $postdata, $ssl_cer = null, $ssl_key = null, $second = 30) { + $ch = curl_init(); + curl_setopt($ch, CURLOPT_TIMEOUT, $second); + curl_setopt($ch, CURLOPT_URL, $url); + curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); + curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); + curl_setopt($ch, CURLOPT_HEADER, FALSE); + /* 瑕佹眰缁撴灉涓哄瓧绗︿覆涓旇緭鍑哄埌灞忓箷涓 */ + curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE); + /* 璁剧疆璇佷功 */ + if (!is_null($ssl_cer) && file_exists($ssl_cer) && is_file($ssl_cer)) { + curl_setopt($ch, CURLOPT_SSLCERTTYPE, 'PEM'); + curl_setopt($ch, CURLOPT_SSLCERT, $ssl_cer); + } + if (!is_null($ssl_key) && file_exists($ssl_key) && is_file($ssl_key)) { + curl_setopt($ch, CURLOPT_SSLKEYTYPE, 'PEM'); + curl_setopt($ch, CURLOPT_SSLKEY, $ssl_key); + } + curl_setopt($ch, CURLOPT_POST, true); + if (is_array($postdata)) { + foreach ($postdata as &$data) { + if (is_string($data) && stripos($data, '@') === 0 && class_exists('CURLFile', FALSE)) { + $data = new CURLFile(realpath(trim($data, '@'))); + } + } + } + curl_setopt($ch, CURLOPT_POSTFIELDS, $postdata); + $result = curl_exec($ch); + curl_close($ch); + if ($result) { + return $result; + } else { + return false; + } + } + + /** + * 璇诲彇寰俊瀹㈡埛绔疘P + * @return null|string + */ + static public function getAddress() { + foreach (array('HTTP_X_FORWARDED_FOR', 'HTTP_CLIENT_IP', 'HTTP_X_CLIENT_IP', 'HTTP_X_CLUSTER_CLIENT_IP', 'REMOTE_ADDR') as $header) { + if (!isset($_SERVER[$header]) || ($spoof = $_SERVER[$header]) === NULL) { + continue; + } + sscanf($spoof, '%[^,]', $spoof); + if (!filter_var($spoof, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4)) { + $spoof = NULL; + } else { + return $spoof; + } + } + return '0.0.0.0'; + } + + /** + * 璁剧疆缂撳瓨锛屾寜闇閲嶈浇 + * @param string $cachename + * @param mixed $value + * @param int $expired + * @return bool + */ + static public function setCache($cachename, $value, $expired = 0) { + return Cache::set($cachename, $value, $expired); + } + + /** + * 鑾峰彇缂撳瓨锛屾寜闇閲嶈浇 + * @param string $cachename + * @return mixed + */ + static public function getCache($cachename) { + return Cache::get($cachename); + } + + /** + * 娓呴櫎缂撳瓨锛屾寜闇閲嶈浇 + * @param string $cachename + * @return bool + */ + static public function removeCache($cachename) { + return Cache::del($cachename); + } + + /** + * SDK鏃ュ織澶勭悊鏂规硶 + * @param string $msg 鏃ュ織琛屽唴瀹 + * @param string $type 鏃ュ織绾у埆 + */ + static public function log($msg, $type = 'MSG') { + Cache::put($type . ' - ' . $msg); + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/Loader.php b/vendor/zoujingli/wechat-php-sdk/Wechat/Loader.php new file mode 100644 index 000000000..428e48cc7 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/Loader.php @@ -0,0 +1,111 @@ + + * @date 2016/10/26 10:21 + */ +spl_autoload_register(function ($class) { + if (0 === stripos($class, 'Wechat\\')) { + $filename = dirname(__DIR__) . DIRECTORY_SEPARATOR . str_replace('\\', DIRECTORY_SEPARATOR, $class) . '.php'; + file_exists($filename) && require($filename); + } +}); + +/** + * 寰俊SDK鍔犺浇鍣 + * @author Anyon + * @date 2016-08-21 11:06 + */ +class Loader { + + /** + * 浜嬩欢娉ㄥ唽鍑芥暟 + * @var array + */ + static public $callback = array(); + + /** + * 閰嶇疆鍙傛暟 + * @var array + */ + static protected $config = array(); + + /** + * 瀵硅薄缂撳瓨 + * @var array + */ + static protected $cache = array(); + + /** + * 鍔ㄦ佹敞鍐孲DK浜嬩欢澶勭悊鍑芥暟 + * @param string $event 浜嬩欢鍚嶇О锛坓etAccessToken|getJsTicket锛 + * @param string $method 澶勭悊鏂规硶锛堝彲浠ユ槸鏅氭柟娉曟垨鑰呯被涓殑鏂规硶锛 + * @param string|null $class 澶勭悊瀵硅薄锛堝彲浠ョ洿鎺ヤ娇鐢ㄧ殑绫诲疄渚嬶級 + */ + static public function register($event, $method, $class = NULL) { + if (!empty($class) && class_exists($class, FALSE) && method_exists($class, $method)) { + self::$callback[$event] = array($class, $method); + } else { + self::$callback[$event] = $method; + } + } + + /** + * 鑾峰彇寰俊SDK鎺ュ彛瀵硅薄(鍒悕鍑芥暟) + * @param string $type 鎺ュ彛绫诲瀷(Card|Custom|Device|Extends|Media|Menu|Oauth|Pay|Receive|Script|User|Poi) + * @param array $config SDK閰嶇疆(token,appid,appsecret,encodingaeskey,mch_id,partnerkey,ssl_cer,ssl_key,qrc_img) + * @return WechatCard|WechatCustom|WechatDevice|WechatExtends|WechatMedia|WechatMenu|WechatOauth|WechatPay|WechatPoi|WechatReceive|WechatScript|WechatService|WechatUser + */ + static public function & get_instance($type, $config = array()) { + return self::get($type, $config); + } + + /** + * 鑾峰彇寰俊SDK鎺ュ彛瀵硅薄 + * @param string $type 鎺ュ彛绫诲瀷(Card|Custom|Device|Extends|Media|Menu|Oauth|Pay|Receive|Script|User|Poi) + * @param array $config SDK閰嶇疆(token,appid,appsecret,encodingaeskey,mch_id,partnerkey,ssl_cer,ssl_key,qrc_img) + * @return WechatCard|WechatCustom|WechatDevice|WechatExtends|WechatMedia|WechatMenu|WechatOauth|WechatPay|WechatPoi|WechatReceive|WechatScript|WechatService|WechatUser + */ + static public function & get($type, $config = array()) { + $index = md5(strtolower($type) . md5(json_encode(self::$config))); + if (!isset(self::$cache[$index])) { + $basicName = 'Wechat' . ucfirst(strtolower($type)); + $className = "\\Wechat\\{$basicName}"; + /* 娉ㄥ唽绫荤殑鏃犲懡鍚嶇┖闂村埆鍚嶏紝鍏煎鏈甫鍛藉悕绌洪棿鐨勮佺増鏈琒DK */ + !class_exists($basicName, FALSE) && class_alias($className, $basicName); + self::$cache[$index] = new $className(self::config($config)); + } + return self::$cache[$index]; + } + + /** + * 璁剧疆閰嶇疆鍙傛暟 + * @param array $config + * @return array + */ + static public function config($config = array()) { + !empty($config) && self::$config = array_merge(self::$config, $config); + if (!empty(self::$config['cachepath'])) { + Cache::$cachepath = self::$config['cachepath']; + } + if (empty(self::$config['component_verify_ticket'])) { + self::$config['component_verify_ticket'] = Cache::get('component_verify_ticket'); + } + if (empty(self::$config['token']) && !empty(self::$config['component_token'])) { + self::$config['token'] = self::$config['component_token']; + } + if (empty(self::$config['appsecret']) && !empty(self::$config['component_appsecret'])) { + self::$config['appsecret'] = self::$config['component_appsecret']; + } + if (empty(self::$config['encodingaeskey']) && !empty(self::$config['component_encodingaeskey'])) { + self::$config['encodingaeskey'] = self::$config['component_encodingaeskey']; + } + return self::$config; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatCard.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatCard.php new file mode 100644 index 000000000..cbbad872c --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatCard.php @@ -0,0 +1,774 @@ +access_token && !$this->getAccessToken()) { + return false; + } + $appid = empty($appid) ? $this->appid : $appid; + if ($jsapi_ticket) { + return $jsapi_ticket; + } + $authname = 'wechat_jsapi_ticket_wxcard_' . $appid; + if (($jsapi_ticket = Tools::getCache($authname))) { + return $jsapi_ticket; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::GET_TICKET_URL . "access_token={$this->access_token}" . '&type=wx_card'); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + $expire = $json['expires_in'] ? intval($json['expires_in']) - 100 : 3600; + Tools::setCache($authname, $json['ticket'], $expire); + return $json['ticket']; + } + return false; + } + + /** + * 鐢熸垚閫夋嫨鍗″嵎JS绛惧悕鍖 + * @param string $cardid 鍗″埜Id + * @param string $cardtype 鍗″埜绫诲瀷 + * @param string $shopid 闂ㄥ簵Id + * @return array + */ + public function createChooseCardJsPackage($cardid = NULL, $cardtype = NULL, $shopid = NULL) { + $data = array(); + $data['api_ticket'] = $this->getJsCardTicket(); + $data['app_id'] = $this->appid; + $data['timestamp'] = time(); + $data['nonceStr'] = Tools::createNoncestr(); + !empty($cardid) && $data['cardId'] = $cardid; + !empty($cardtype) && $data['cardType'] = $cardtype; + !empty($shopid) && $data['shopId'] = $shopid; + $data['cardSign'] = $this->getTicketSignature($data); + $data['signType'] = 'SHA1'; + unset($data['api_ticket'], $data['app_id']); + return $data; + } + + /** + * 鐢熸垚娣诲姞鍗″嵎JS绛惧悕鍖 + * @param string|null $cardid 鍗″嵎ID + * @param array $data 鍏跺畠闄愬畾鍙傛暟 + * @return array + */ + public function createAddCardJsPackage($cardid = NULL, $data = array()) { + + function _sign($cardid = NULL, $attr = array(), $self) { + unset($attr['outer_id']); + $attr['cardId'] = $cardid; + $attr['timestamp'] = time(); + $attr['api_ticket'] = $self->getJsCardTicket(); + $attr['nonce_str'] = Tools::createNoncestr(); + $attr['signature'] = $self->getTicketSignature($attr); + unset($attr['api_ticket']); + return $attr; + } + + $cardList = array(); + if (is_array($cardid)) { + foreach ($cardid as $id) { + $cardList[] = array('cardId' => $id, 'cardExt' => json_encode(_sign($id, $data, $this))); + } + } else { + $cardList[] = array('cardId' => $cardid, 'cardExt' => json_encode(_sign($cardid, $data, $this))); + } + return array('cardList' => $cardList); + } + + /** + * 鑾峰彇寰俊鍗″埜绛惧悕 + * @param array $arrdata 绛惧悕鏁扮粍 + * @param string $method 绛惧悕鏂规硶 + * @return bool|string 绛惧悕鍊 + */ + public function getTicketSignature($arrdata, $method = "sha1") { + if (!function_exists($method)) { + return false; + } + $newArray = array(); + foreach ($arrdata as $value) { + array_push($newArray, (string)$value); + } + sort($newArray, SORT_STRING); + return $method(implode($newArray)); + } + + /** + * 鍒涘缓鍗″埜 + * @param array $data 鍗″埜鏁版嵁 + * @return bool|array 杩斿洖鏁扮粍涓璫ard_id涓哄崱鍒窱D + */ + public function createCard($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CREATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏇存敼鍗″埜淇℃伅 + * 璋冪敤璇ユ帴鍙f洿鏂颁俊鎭悗浼氶噸鏂伴佸锛屽崱鍒哥姸鎬佸彉鏇翠负寰呭鏍搞傚凡琚敤鎴烽鍙栫殑鍗″埜浼氬疄鏃舵洿鏂扮エ闈俊鎭 + * @param string $data + * @return bool + */ + public function updateCard($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_UPDATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鍒犻櫎鍗″埜 + * 鍏佽鍟嗘埛鍒犻櫎浠绘剰涓绫诲崱鍒搞傚垹闄ゅ崱鍒稿悗锛岃鍗″埜瀵瑰簲宸茬敓鎴愮殑棰嗗彇鐢ㄤ簩缁寸爜銆佹坊鍔犲埌鍗″寘 JS API 鍧囦細澶辨晥銆 + * 娉ㄦ剰锛氬垹闄ゅ崱鍒镐笉鑳藉垹闄ゅ凡琚敤鎴烽鍙栵紝淇濆瓨鍦ㄥ井淇″鎴风涓殑鍗″埜锛屽凡棰嗗彇鐨勫崱鍒镐緷鏃ф湁鏁堛 + * @param string $card_id 鍗″埜ID + * @return bool + */ + public function delCard($card_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('card_id' => $card_id); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_DELETE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鑾峰彇绮変笣涓嬫墍鏈夊崱鍗峰垪琛 + * @param $openid 绮変笣openid + * @param string $card_id 鍗″嵎ID锛堝彲涓嶇粰锛 + * @return bool|array + */ + public function getCardList($openid, $card_id = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid' => $openid); + !empty($card_id) && $data['card_id'] = $card_id; + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_USER_GET_LIST . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode']) || empty($json['card_list'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鍥炬枃娑堟伅缇ゅ彂鍗″埜HTML + * @param string $card_id 鍗″嵎ID + * @return bool|array + */ + public function getCardMpHtml($card_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('card_id' => $card_id); + !empty($card_id) && $data['card_id'] = $card_id; + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_SEND_HTML . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode']) || empty($json['card_list'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍗″嵎code鏍告煡 + * @param string $card_id 鍗″嵎ID + * @param array $code_list 鍗″嵎code鍒楄〃锛堜竴缁存暟缁勶級 + * @return bool|array + */ + public function checkCardCodeList($card_id, $code_list) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('card_id' => $card_id, 'code' => $code_list); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CHECKCODE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode']) || empty($json['card_list'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏌ヨ鍗″埜璇︽儏 + * @param string $card_id 鍗″嵎ID + * @return bool|array + */ + public function getCardInfo($card_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('card_id' => $card_id); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_GET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇棰滆壊鍒楄〃 + * 鑾峰緱鍗″埜鐨勬渶鏂伴鑹插垪琛紝鐢ㄤ簬鍒涘缓鍗″埜 + * @return bool|array + */ + public function getCardColors() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::CARD_GETCOLORS . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鐢熸垚鍗″埜浜岀淮鐮 + * 鎴愬姛鍒欑洿鎺ヨ繑鍥瀟icket鍊硷紝鍙互鐢 getQRUrl($ticket) 鎹㈠彇浜岀淮鐮乽rl + * @param string $card_id 鍗″埜ID 蹇呴』 + * @param string $code 鎸囧畾鍗″埜 code 鐮侊紝鍙兘琚涓娆°倁se_custom_code 瀛楁涓 true 鐨勫崱鍒稿繀椤诲~鍐欙紝闈炶嚜瀹氫箟 code 涓嶅繀濉啓銆 + * @param string $openid 鎸囧畾棰嗗彇鑰呯殑 openid锛屽彧鏈夎鐢ㄦ埛鑳介鍙栥俠ind_openid 瀛楁涓 true 鐨勫崱鍒稿繀椤诲~鍐欙紝闈炶嚜瀹氫箟 openid 涓嶅繀濉啓銆 + * @param int $expire_seconds 鎸囧畾浜岀淮鐮佺殑鏈夋晥鏃堕棿锛岃寖鍥存槸 60 ~ 1800 绉掋備笉濉粯璁や负姘镐箙鏈夋晥銆 + * @param bool $is_unique_code 鎸囧畾涓嬪彂浜岀淮鐮侊紝鐢熸垚鐨勪簩缁寸爜闅忔満鍒嗛厤涓涓 code锛岄鍙栧悗涓嶅彲鍐嶆鎵弿銆傚~鍐 true 鎴 false銆傞粯璁 false銆 + * @param string $balance 绾㈠寘浣欓锛屼互鍒嗕负鍗曚綅銆傜孩鍖呯被鍨嬪繀濉紙LUCKY_MONEY锛夛紝鍏朵粬鍗″埜绫诲瀷涓嶅~銆 + * @return bool|string + */ + public function createCardQrcode($card_id, $code = '', $openid = '', $expire_seconds = 0, $is_unique_code = false, $balance = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $card = array('card_id' => $card_id); + !empty($code) && $card['code'] = $code; + !empty($openid) && $card['openid'] = $openid; + !empty($is_unique_code) && $card['is_unique_code'] = $is_unique_code; + !empty($balance) && $card['balance'] = $balance; + $data = array('action_name' => "QR_CARD"); + !empty($expire_seconds) && $data['expire_seconds'] = $expire_seconds; + $data['action_info'] = array('card' => $card); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_QRCODE_CREATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 娑堣 code + * 鑷畾涔 code锛坲se_custom_code 涓 true锛夌殑浼樻儬鍒革紝鍦 code 琚牳閿鏃讹紝蹇呴』璋冪敤姝ゆ帴鍙c + * @param string $code 瑕佹秷鑰楃殑搴忓垪鍙 + * @param string $card_id 瑕佹秷鑰楀簭鍒楀彿鎵杩扮殑 card_id锛屽垱寤哄崱鍒告椂use_custom_code 濉啓 true 鏃跺繀濉 + * @return bool|array + * { + * "errcode":0, + * "errmsg":"ok", + * "card":{"card_id":"pFS7Fjg8kV1IdDz01r4SQwMkuCKc"}, + * "openid":"oFS7Fjl0WsZ9AMZqrI80nbIq8xrA" + * } + */ + public function consumeCardCode($code, $card_id = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('code' => $code); + !empty($card_id) && $data['card_id'] = $card_id; + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CODE_CONSUME . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * code 瑙g爜 + * @param string $encrypt_code 閫氳繃 choose_card_info 鑾峰彇鐨勫姞瀵嗗瓧绗︿覆 + * @return bool|array + * { + * "errcode":0, + * "errmsg":"ok", + * "code":"751234212312" + * } + */ + public function decryptCardCode($encrypt_code) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('encrypt_code' => $encrypt_code,); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CODE_DECRYPT . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏌ヨ code 鐨勬湁鏁堟э紙闈炶嚜瀹氫箟 code锛 + * @param string $code + * @return bool|array + * { + * "errcode":0, + * "errmsg":"ok", + * "openid":"oFS7Fjl0WsZ9AMZqrI80nbIq8xrA", //鐢ㄦ埛 openid + * "card":{ + * "card_id":"pFS7Fjg8kV1IdDz01r4SQwMkuCKc", + * "begin_time": 1404205036, //璧峰浣跨敤鏃堕棿 + * "end_time": 1404205036, //缁撴潫鏃堕棿 + * } + * } + */ + public function checkCardCode($code) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('code' => $code); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CODE_GET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鎵归噺鏌ヨ鍗″垪琛 + * @param int $offset 寮濮嬫媺鍙栫殑鍋忕Щ锛岄粯璁や负0浠庡ご寮濮 + * @param int $count 闇瑕佹煡璇㈢殑鍗$墖鐨勬暟閲忥紙鏁伴噺鏈澶50,榛樿50锛 + * @return bool|array + * { + * "errcode":0, + * "errmsg":"ok", + * "card_id_list":["ph_gmt7cUVrlRk8swPwx7aDyF-pg"], //鍗 id 鍒楄〃 + * "total_num":1 //璇ュ晢鎴峰悕涓 card_id 鎬绘暟 + * } + */ + public function getCardIdList($offset = 0, $count = 50) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $count > 50 && $count = 50; + $data = array('offset' => $offset, 'count' => $count); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_BATCHGET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏇存敼 code + * 涓虹‘淇濊浆璧犲悗鐨勫畨鍏ㄦэ紝寰俊鍏佽鑷畾涔塩ode鐨勫晢鎴峰宸蹭笅鍙戠殑code杩涜鏇存敼銆 + * 娉細涓洪伩鍏嶇敤鎴风枒鎯戯紝寤鸿浠呭湪鍙戠敓杞禒琛屼负鍚庯紙鍙戠敓杞禒鍚庯紝寰俊浼氶氳繃浜嬩欢鎺ㄩ佺殑鏂瑰紡鍛婄煡鍟嗘埛琚浆璧犵殑鍗″埜code锛夊鐢ㄦ埛鐨刢ode杩涜鏇存敼銆 + * @param string $code 鍗″埜鐨 code 缂栫爜 + * @param string $card_id 鍗″埜 ID + * @param string $new_code 鏂扮殑鍗″埜 code 缂栫爜 + * @return bool + */ + public function updateCardCode($code, $card_id, $new_code) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('code' => $code, 'card_id' => $card_id, 'new_code' => $new_code); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CODE_UPDATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 璁剧疆鍗″埜澶辨晥 + * 璁剧疆鍗″埜澶辨晥鐨勬搷浣滀笉鍙 + * @param string $code 闇瑕佽缃负澶辨晥鐨 code + * @param string $card_id 鑷畾涔 code 鐨勫崱鍒稿繀濉傞潪鑷畾涔 code 鐨勫崱鍒镐笉濉 + * @return bool + */ + public function unavailableCardCode($code, $card_id = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('code' => $code); + !empty($card_id) && $data['card_id'] = $card_id; + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_CODE_UNAVAILABLE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 搴撳瓨淇敼 + * @param string $data + * @return bool + */ + public function modifyCardStock($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_MODIFY_STOCK . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鏇存柊闂ㄧエ + * @param string $data + * @return bool + */ + public function updateMeetingCard($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_MEETINGCARD_UPDATEUSER . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 婵娲/缁戝畾浼氬憳鍗 + * @param string $data 鍏蜂綋缁撴瀯璇峰弬鐪嬪崱鍒稿紑鍙戞枃妗(6.1.1 婵娲/缁戝畾浼氬憳鍗)绔犺妭 + * @return bool + */ + public function activateMemberCard($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_MEMBERCARD_ACTIVATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 浼氬憳鍗′氦鏄 + * 浼氬憳鍗′氦鏄撳悗姣忔绉垎鍙婁綑棰濆彉鏇撮渶閫氳繃鎺ュ彛閫氱煡寰俊锛屼究浜庡悗缁秷鎭氱煡鍙婂叾浠栨墿灞曞姛鑳姐 + * @param string $data 鍏蜂綋缁撴瀯璇峰弬鐪嬪崱鍒稿紑鍙戞枃妗(6.1.2 浼氬憳鍗′氦鏄)绔犺妭 + * @return bool|array + */ + public function updateMemberCard($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_MEMBERCARD_UPDATEUSER . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 璁剧疆鍗″埜娴嬭瘯鐧藉悕鍗 + * @param array $openid 娴嬭瘯鐨 openid 鍒楄〃 + * @param array $user 娴嬭瘯鐨勫井淇″彿鍒楄〃 + * @return bool + */ + public function setCardTestWhiteList($openid = array(), $user = array()) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array(); + count($openid) > 0 && $data['openid'] = $openid; + count($user) > 0 && $data['username'] = $user; + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_TESTWHILELIST_SET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鏇存柊绾㈠寘閲戦 + * @param string $code 绾㈠寘鐨勫簭鍒楀彿 + * @param int $balance 绾㈠寘浣欓 + * @param string $card_id 鑷畾涔 code 鐨勫崱鍒稿繀濉傞潪鑷畾涔 code 鍙笉濉 + * @return bool|array + */ + public function updateLuckyMoney($code, $balance, $card_id = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('code' => $code, 'balance' => $balance); + !empty($card_id) && $data['card_id'] = $card_id; + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_LUCKYMONEY_UPDATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 璁剧疆鑷姪鏍搁攢鎺ュ彛 + * @param string $card_id 鍗″埜ID + * @param bool $is_openid 鏄惁寮鍚嚜鍔╂牳閿鍔熻兘锛屽~true/false锛岄粯璁や负false + * @param bool $need_verify_cod 鐢ㄦ埛鏍搁攢鏃舵槸鍚﹂渶瑕佽緭鍏ラ獙璇佺爜锛屽~true/false锛岄粯璁や负false + * @param bool $need_remark_amount 鐢ㄦ埛鏍搁攢鏃舵槸鍚﹂渶瑕佸娉ㄦ牳閿閲戦锛屽~true/false锛岄粯璁や负false + * @return bool|array + */ + public function setSelfconsumecell($card_id, $is_openid = false, $need_verify_cod = false, $need_remark_amount = false) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array( + 'card_id' => $card_id, + 'is_open' => $is_openid, + 'need_verify_cod' => $need_verify_cod, + 'need_remark_amount' => $need_remark_amount, + ); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_SET_SELFCONSUMECELL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + /** + * 璁剧疆涔板崟鎺ュ彛 + * @DateTime 2016-12-02T19:19:45+0800 + * @param [type] $card_id [description] + * @param boolean $is_openid [description] + */ + public function setPaycell($card_id,$is_openid = true){ + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array( + 'card_id' => $card_id, + 'is_open' => $is_openid, + ); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_PAYCELL_SET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 璁剧疆寮鍗″瓧娈典俊鎭帴鍙 + * @DateTime 2016-12-02T20:31:43+0800 + * @param [type] $data [description] + */ + public function setMembercardActivateuserform($data){ + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CARD_MEMBERCARD_ACTIVATEUSERFORM_SET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatCustom.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatCustom.php new file mode 100644 index 000000000..004827184 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatCustom.php @@ -0,0 +1,338 @@ +access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::CUSTOM_SERVICE_GET_RECORD . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return false; + } + return $json; + } + return false; + } + + /** + * 鑾峰彇澶氬鏈嶅鏈嶅熀鏈俊鎭 + * + * @return bool|array + */ + public function getCustomServiceKFlist() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::CUSTOM_SERVICE_GET_KFLIST . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇澶氬鏈嶅湪绾垮鏈嶆帴寰呬俊鎭 + * + * @return bool|array + */ + public function getCustomServiceOnlineKFlist() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::CUSTOM_SERVICE_GET_ONLINEKFLIST . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍒涘缓鎸囧畾澶氬鏈嶄細璇 + * @tutorial 褰撶敤鎴峰凡琚叾浠栧鏈嶆帴寰呮垨鎸囧畾瀹㈡湇涓嶅湪绾垮垯浼氬け璐 + * @param string $openid //鐢ㄦ埛openid + * @param string $kf_account //瀹㈡湇璐﹀彿 + * @param string $text //闄勫姞淇℃伅锛屾枃鏈細灞曠ず鍦ㄥ鏈嶄汉鍛樼殑澶氬鏈嶅鎴风锛屽彲涓虹┖ + * @return bool|array + */ + public function createKFSession($openid, $kf_account, $text = '') { + $data = array("openid" => $openid, "kf_account" => $kf_account); + if ($text) { + $data["text"] = $text; + } + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CUSTOM_SESSION_CREATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍏抽棴鎸囧畾澶氬鏈嶄細璇 + * @tutorial 褰撶敤鎴疯鍏朵粬瀹㈡湇鎺ュ緟鏃跺垯浼氬け璐 + * @param string $openid //鐢ㄦ埛openid + * @param string $kf_account //瀹㈡湇璐﹀彿 + * @param string $text //闄勫姞淇℃伅锛屾枃鏈細灞曠ず鍦ㄥ鏈嶄汉鍛樼殑澶氬鏈嶅鎴风锛屽彲涓虹┖ + * @return bool | array //鎴愬姛杩斿洖json鏁扮粍 + * { + * "errcode": 0, + * "errmsg": "ok", + * } + */ + public function closeKFSession($openid, $kf_account, $text = '') { + $data = array("openid" => $openid, "kf_account" => $kf_account); + if ($text) { + $data["text"] = $text; + } + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CUSTOM_SESSION_CLOSE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鐢ㄦ埛浼氳瘽鐘舵 + * @param string $openid //鐢ㄦ埛openid + * @return bool | array //鎴愬姛杩斿洖json鏁扮粍 + * { + * "errcode" : 0, + * "errmsg" : "ok", + * "kf_account" : "test1@test", //姝e湪鎺ュ緟鐨勫鏈 + * "createtime": 123456789, //浼氳瘽鎺ュ叆鏃堕棿 + * } + */ + public function getKFSession($openid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::CUSTOM_SESSION_GET . "access_token={$this->access_token}" . '&openid=' . $openid); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鎸囧畾瀹㈡湇鐨勪細璇濆垪琛 + * @param string $kf_account //鐢ㄦ埛openid + * @return bool | array //鎴愬姛杩斿洖json鏁扮粍 + * array( + * 'sessionlist' => array ( + * array ( + * 'openid'=>'OPENID', //瀹㈡埛 openid + * 'createtime'=>123456789, //浼氳瘽鍒涘缓鏃堕棿锛孶NIX 鏃堕棿鎴 + * ), + * array ( + * 'openid'=>'OPENID', //瀹㈡埛 openid + * 'createtime'=>123456789, //浼氳瘽鍒涘缓鏃堕棿锛孶NIX 鏃堕棿鎴 + * ), + * ) + * ) + */ + public function getKFSessionlist($kf_account) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::CUSTOM_SESSION_GET_LIST . "access_token={$this->access_token}" . '&kf_account=' . $kf_account); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鏈帴鍏ヤ細璇濆垪琛 + * @return bool|array + */ + public function getKFSessionWait() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::CUSTOM_SESSION_GET_WAIT . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 娣诲姞瀹㈡湇璐﹀彿 + * + * @param string $account 瀹屾暣瀹㈡湇璐﹀彿(璐﹀彿鍓嶇紑@鍏紬鍙峰井淇″彿锛岃处鍙峰墠缂鏈澶10涓瓧绗) + * @param string $nickname 瀹㈡湇鏄电О锛屾渶闀6涓眽瀛楁垨12涓嫳鏂囧瓧绗 + * @param string $password 瀹㈡湇璐﹀彿鏄庢枃鐧诲綍瀵嗙爜锛屼細鑷姩鍔犲瘑 + * @return bool|array + */ + public function addKFAccount($account, $nickname, $password) { + $data = array("kf_account" => $account, "nickname" => $nickname, "password" => md5($password)); + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CS_KF_ACCOUNT_ADD_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 淇敼瀹㈡湇璐﹀彿淇℃伅 + * + * @param string $account //瀹屾暣瀹㈡湇璐﹀彿锛屾牸寮忎负锛氳处鍙峰墠缂@鍏紬鍙峰井淇″彿锛岃处鍙峰墠缂鏈澶10涓瓧绗︼紝蹇呴』鏄嫳鏂囨垨鑰呮暟瀛楀瓧绗 + * @param string $nickname //瀹㈡湇鏄电О锛屾渶闀6涓眽瀛楁垨12涓嫳鏂囧瓧绗 + * @param string $password //瀹㈡湇璐﹀彿鏄庢枃鐧诲綍瀵嗙爜锛屼細鑷姩鍔犲瘑 + * @return bool|array + * 鎴愬姛杩斿洖缁撴灉 + * { + * "errcode": 0, + * "errmsg": "ok", + * } + */ + public function updateKFAccount($account, $nickname, $password) { + $data = array("kf_account" => $account, "nickname" => $nickname, "password" => md5($password)); + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CS_KF_ACCOUNT_UPDATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍒犻櫎瀹㈡湇璐﹀彿 + * @param string $account 瀹屾暣瀹㈡湇璐﹀彿(璐﹀彿鍓嶇紑@鍏紬鍙峰井淇″彿锛岃处鍙峰墠缂鏈澶10涓瓧绗) + * @return bool|array + */ + public function deleteKFAccount($account) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::CS_KF_ACCOUNT_DEL_URL . "access_token={$this->access_token}" . '&kf_account=' . $account); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 涓婁紶瀹㈡湇澶村儚 + * @param string $account 瀹屾暣瀹㈡湇璐﹀彿(璐﹀彿鍓嶇紑@鍏紬鍙峰井淇″彿锛岃处鍙峰墠缂鏈澶10涓瓧绗) + * @param string $imgfile 澶村儚鏂囦欢瀹屾暣璺緞,濡傦細'D:\user.jpg'銆傚ご鍍忔枃浠跺繀椤籎PG鏍煎紡锛屽儚绱犲缓璁640*640 + * @return bool|array + */ + public function setKFHeadImg($account, $imgfile) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::CS_KF_ACCOUNT_UPLOAD_HEADIMG_URL . "access_token={$this->access_token}" . '&kf_account=' . $account, array('media' => '@' . $imgfile), true); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatDevice.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatDevice.php new file mode 100644 index 000000000..91d8b6afd --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatDevice.php @@ -0,0 +1,392 @@ + + * @date 2016-08-22 10:35 + */ +class WechatDevice extends Common { + + const SHAKEAROUND_DEVICE_APPLYID = '/shakearound/device/applyid?'; //鐢宠璁惧ID + const SHAKEAROUND_DEVICE_UPDATE = '/shakearound/device/update?'; //缂栬緫璁惧淇℃伅 + const SHAKEAROUND_DEVICE_SEARCH = '/shakearound/device/search?'; //鏌ヨ璁惧鍒楄〃 + const SHAKEAROUND_DEVICE_BINDLOCATION = '/shakearound/device/bindlocation?'; //閰嶇疆璁惧涓庨棬搴桰D鐨勫叧绯 + const SHAKEAROUND_DEVICE_BINDPAGE = '/shakearound/device/bindpage?'; //閰嶇疆璁惧涓庨〉闈㈢殑缁戝畾鍏崇郴 + const SHAKEAROUND_MATERIAL_ADD = '/shakearound/material/add?'; //涓婁紶鎽囦竴鎽囧浘鐗囩礌鏉 + const SHAKEAROUND_PAGE_ADD = '/shakearound/page/add?'; //澧炲姞椤甸潰 + const SHAKEAROUND_PAGE_UPDATE = '/shakearound/page/update?'; //缂栬緫椤甸潰 + const SHAKEAROUND_PAGE_SEARCH = '/shakearound/page/search?'; //鏌ヨ椤甸潰鍒楄〃 + const SHAKEAROUND_PAGE_DELETE = '/shakearound/page/delete?'; //鍒犻櫎椤甸潰 + const SHAKEAROUND_USER_GETSHAKEINFO = '/shakearound/user/getshakeinfo?'; //鑾峰彇鎽囧懆杈圭殑璁惧鍙婄敤鎴蜂俊鎭 + const SHAKEAROUND_STATISTICS_DEVICE = '/shakearound/statistics/device?'; //浠ヨ澶囦负缁村害鐨勬暟鎹粺璁℃帴鍙 + const SHAKEAROUND_STATISTICS_PAGE = '/shakearound/statistics/page?'; //浠ラ〉闈负缁村害鐨勬暟鎹粺璁℃帴鍙 + + + /** + * 鐢宠璁惧ID + * @param array $data + * @return bool|array + */ + public function applyShakeAroundDevice($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_DEVICE_APPLYID . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 缂栬緫璁惧淇℃伅 + * @param array $data + * @return bool + */ + public function updateShakeAroundDevice($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_DEVICE_UPDATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + + /** + * 鏌ヨ璁惧鍒楄〃 + * @param $data + * @return bool|array + */ + public function searchShakeAroundDevice($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_DEVICE_SEARCH . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 閰嶇疆璁惧涓庨棬搴楃殑鍏宠仈鍏崇郴 + * @param string $device_id 璁惧缂栧彿锛岃嫢濉簡UUID銆乵ajor銆乵inor锛屽垯鍙笉濉澶囩紪鍙凤紝鑻ヤ簩鑰呴兘濉紝鍒欎互璁惧缂栧彿涓轰紭鍏 + * @param int $poi_id 寰呭叧鑱旂殑闂ㄥ簵ID + * @param string $uuid UUID銆乵ajor銆乵inor锛屼笁涓俊鎭渶濉啓瀹屾暣锛岃嫢濉簡璁惧缂栧彿锛屽垯鍙笉濉淇℃伅 + * @param int $major + * @param int $minor + * @return bool|array + */ + public function bindLocationShakeAroundDevice($device_id, $poi_id, $uuid = '', $major = 0, $minor = 0) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if (!$device_id) { + if (!$uuid || !$major || !$minor) { + return false; + } + $device_identifier = array('uuid' => $uuid, 'major' => $major, 'minor' => $minor); + } else { + $device_identifier = array( + 'device_id' => $device_id + ); + } + $data = array('device_identifier' => $device_identifier, 'poi_id' => $poi_id); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_DEVICE_BINDLOCATION . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; //杩欎釜鍙互鏇存敼涓鸿繑鍥瀟rue + } + return false; + } + + /** + * 閰嶇疆璁惧涓庨〉闈㈢殑鍏宠仈鍏崇郴 + * @param string $device_id 璁惧缂栧彿锛岃嫢濉簡UUID銆乵ajor銆乵inor锛屽垯鍙笉濉澶囩紪鍙凤紝鑻ヤ簩鑰呴兘濉紝鍒欎互璁惧缂栧彿涓轰紭鍏 + * @param array $page_ids 寰呭叧鑱旂殑椤甸潰鍒楄〃 + * @param int $bind 鍏宠仈鎿嶄綔鏍囧織浣嶏紝 0 涓鸿В闄ゅ叧鑱斿叧绯伙紝1 涓哄缓绔嬪叧鑱斿叧绯 + * @param int $append 鏂板鎿嶄綔鏍囧織浣嶏紝 0 涓鸿鐩栵紝1 涓烘柊澧 + * @param string $uuid UUID銆乵ajor銆乵inor锛屼笁涓俊鎭渶濉啓瀹屾暣锛岃嫢濉簡璁惧缂栧彿锛屽垯鍙笉濉淇℃伅 + * @param int $major + * @param int $minor + * @return bool|array + */ + public function bindPageShakeAroundDevice($device_id, $page_ids = array(), $bind = 1, $append = 1, $uuid = '', $major = 0, $minor = 0) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if (!$device_id) { + if (!$uuid || !$major || !$minor) { + return false; + } + $device_identifier = array('uuid' => $uuid, 'major' => $major, 'minor' => $minor); + } else { + $device_identifier = array('device_id' => $device_id); + } + $data = array('device_identifier' => $device_identifier, 'page_ids' => $page_ids, 'bind' => $bind, 'append' => $append); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_DEVICE_BINDPAGE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 涓婁紶鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑鍥剧墖绱犳潗 + * @param array $data {"media":'@Path\filename.jpg'} + * @return bool|array + */ + public function uploadShakeAroundMedia($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_MATERIAL_ADD . "access_token={$this->access_token}", $data, true); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 澧炲姞鎽囦竴鎽囧嚭鏉ョ殑椤甸潰淇℃伅 + * @param string $title 鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑涓绘爣棰橈紝涓嶈秴杩6 涓瓧 + * @param string $description 鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑鍓爣棰橈紝涓嶈秴杩7 涓瓧 + * @param string $icon_url 鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑鍥剧墖锛 鏍煎紡闄愬畾涓猴細jpg,jpeg,png,gif; 寤鸿120*120 锛 闄愬埗涓嶈秴杩200*200 + * @param string $page_url 璺宠浆閾炬帴 + * @param string $comment 椤甸潰鐨勫娉ㄤ俊鎭紝涓嶈秴杩15 涓瓧,鍙笉濉 + * @return bool|array + */ + public function addShakeAroundPage($title, $description, $icon_url, $page_url, $comment = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array("title" => $title, "description" => $description, "icon_url" => $icon_url, "page_url" => $page_url, "comment" => $comment); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_PAGE_ADD . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 缂栬緫鎽囦竴鎽囧嚭鏉ョ殑椤甸潰淇℃伅 + * @param int $page_id + * @param string $title 鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑涓绘爣棰橈紝涓嶈秴杩6 涓瓧 + * @param string $description 鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑鍓爣棰橈紝涓嶈秴杩7 涓瓧 + * @param string $icon_url 鍦ㄦ憞涓鎽囬〉闈㈠睍绀虹殑鍥剧墖锛 鏍煎紡闄愬畾涓猴細jpg,jpeg,png,gif; 寤鸿120*120 锛 闄愬埗涓嶈秴杩200*200 + * @param string $page_url 璺宠浆閾炬帴 + * @param string $comment 椤甸潰鐨勫娉ㄤ俊鎭紝涓嶈秴杩15 涓瓧,鍙笉濉 + * @return bool|array + */ + public function updateShakeAroundPage($page_id, $title, $description, $icon_url, $page_url, $comment = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array("page_id" => $page_id, "title" => $title, "description" => $description, "icon_url" => $icon_url, "page_url" => $page_url, "comment" => $comment); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_PAGE_UPDATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 鏌ヨ宸叉湁鐨勯〉闈 + * @param array $page_ids + * @param int $begin + * @param int $count + * @return bool|mixed + */ + public function searchShakeAroundPage($page_ids = array(), $begin = 0, $count = 1) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if (!empty($page_ids)) { + $data = array('page_ids' => $page_ids); + } else { + $data = array('begin' => $begin, 'count' => $count); + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_PAGE_SEARCH . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 鍒犻櫎宸叉湁鐨勯〉闈 + * @param array $page_ids + * @return bool|array + */ + public function deleteShakeAroundPage($page_ids = array()) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('page_ids' => $page_ids); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_PAGE_DELETE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 鑾峰彇璁惧淇℃伅 + * @param string $ticket 鎽囧懆杈逛笟鍔$殑ticket(鍙湪鎽囧埌鐨刄RL涓緱鍒帮紝ticket鐢熸晥鏃堕棿涓30 鍒嗛挓) + * @return bool|array + */ + public function getShakeInfoShakeAroundUser($ticket) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('ticket' => $ticket); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_USER_GETSHAKEINFO . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 浠ヨ澶囦负缁村害鐨勬暟鎹粺璁℃帴鍙 + * @param int $device_id 璁惧缂栧彿锛岃嫢濉簡UUID銆乵ajor銆乵inor锛屽嵆鍙笉濉澶囩紪鍙凤紝浜岃呴夊叾涓 + * @param int $begin_date 璧峰鏃ユ湡鏃堕棿鎴筹紝鏈闀挎椂闂磋法搴︿负30 澶 + * @param int $end_date 缁撴潫鏃ユ湡鏃堕棿鎴筹紝鏈闀挎椂闂磋法搴︿负30 澶 + * @param string $uuid UUID銆乵ajor銆乵inor锛屼笁涓俊鎭渶濉啓瀹屾垚锛岃嫢濉簡璁惧缂栬緫锛屽嵆鍙笉濉淇℃伅锛屼簩鑰呴夊叾涓 + * @param int $major + * @param int $minor + * @return bool|array + */ + public function deviceShakeAroundStatistics($device_id, $begin_date, $end_date, $uuid = '', $major = 0, $minor = 0) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if (!$device_id) { + if (!$uuid || !$major || !$minor) { + return false; + } + $device_identifier = array('uuid' => $uuid, 'major' => $major, 'minor' => $minor); + } else { + $device_identifier = array('device_id' => $device_id); + } + $data = array('device_identifier' => $device_identifier, 'begin_date' => $begin_date, 'end_date' => $end_date); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_STATISTICS_DEVICE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + + /** + * 浠ラ〉闈负缁村害鐨勬暟鎹粺璁℃帴鍙 + * @param int $page_id 鎸囧畾椤甸潰鐨処D + * @param int $begin_date 璧峰鏃ユ湡鏃堕棿鎴筹紝鏈闀挎椂闂磋法搴︿负30 澶 + * @param int $end_date 缁撴潫鏃ユ湡鏃堕棿鎴筹紝鏈闀挎椂闂磋法搴︿负30 澶 + * @return bool|array + */ + public function pageShakeAroundStatistics($page_id, $begin_date, $end_date) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('page_id' => $page_id, 'begin_date' => $begin_date, 'end_date' => $end_date); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SHAKEAROUND_STATISTICS_DEVICE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatExtends.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatExtends.php new file mode 100644 index 000000000..6a8e6ca8f --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatExtends.php @@ -0,0 +1,197 @@ + + * @date 2016-08-22 10:32 + */ +class WechatExtends extends Common { + + const QR_LIMIT_SCENE = 1; + + /** 璇箟鐞嗚В */ + const SEMANTIC_API_URL = '/semantic/semproxy/search?'; + const QRCODE_IMG_URL = 'https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket='; + const QRCODE_CREATE_URL = '/qrcode/create?'; + const SHORT_URL = '/shorturl?'; + const QR_SCENE = 0; + + /** 鏁版嵁鍒嗘瀽鎺ュ彛 */ + static $DATACUBE_URL_ARR = array(//鐢ㄦ埛鍒嗘瀽 + 'user' => array( + 'summary' => '/datacube/getusersummary?', //鑾峰彇鐢ㄦ埛澧炲噺鏁版嵁锛坓etusersummary锛 + 'cumulate' => '/datacube/getusercumulate?', //鑾峰彇绱鐢ㄦ埛鏁版嵁锛坓etusercumulate锛 + ), + 'article' => array(//鍥炬枃鍒嗘瀽 + 'summary' => '/datacube/getarticlesummary?', //鑾峰彇鍥炬枃缇ゅ彂姣忔棩鏁版嵁锛坓etarticlesummary锛 + 'total' => '/datacube/getarticletotal?', //鑾峰彇鍥炬枃缇ゅ彂鎬绘暟鎹紙getarticletotal锛 + 'read' => '/datacube/getuserread?', //鑾峰彇鍥炬枃缁熻鏁版嵁锛坓etuserread锛 + 'readhour' => '/datacube/getuserreadhour?', //鑾峰彇鍥炬枃缁熻鍒嗘椂鏁版嵁锛坓etuserreadhour锛 + 'share' => '/datacube/getusershare?', //鑾峰彇鍥炬枃鍒嗕韩杞彂鏁版嵁锛坓etusershare锛 + 'sharehour' => '/datacube/getusersharehour?', //鑾峰彇鍥炬枃鍒嗕韩杞彂鍒嗘椂鏁版嵁锛坓etusersharehour锛 + ), + 'upstreammsg' => array(//娑堟伅鍒嗘瀽 + 'summary' => '/datacube/getupstreammsg?', //鑾峰彇娑堟伅鍙戦佹鍐垫暟鎹紙getupstreammsg锛 + 'hour' => '/datacube/getupstreammsghour?', //鑾峰彇娑堟伅鍒嗛佸垎鏃舵暟鎹紙getupstreammsghour锛 + 'week' => '/datacube/getupstreammsgweek?', //鑾峰彇娑堟伅鍙戦佸懆鏁版嵁锛坓etupstreammsgweek锛 + 'month' => '/datacube/getupstreammsgmonth?', //鑾峰彇娑堟伅鍙戦佹湀鏁版嵁锛坓etupstreammsgmonth锛 + 'dist' => '/datacube/getupstreammsgdist?', //鑾峰彇娑堟伅鍙戦佸垎甯冩暟鎹紙getupstreammsgdist锛 + 'distweek' => '/datacube/getupstreammsgdistweek?', //鑾峰彇娑堟伅鍙戦佸垎甯冨懆鏁版嵁锛坓etupstreammsgdistweek锛 + 'distmonth' => '/datacube/getupstreammsgdistmonth?', //鑾峰彇娑堟伅鍙戦佸垎甯冩湀鏁版嵁锛坓etupstreammsgdistmonth锛 + ), + 'interface' => array(//鎺ュ彛鍒嗘瀽 + 'summary' => '/datacube/getinterfacesummary?', //鑾峰彇鎺ュ彛鍒嗘瀽鏁版嵁锛坓etinterfacesummary锛 + 'summaryhour' => '/datacube/getinterfacesummaryhour?', //鑾峰彇鎺ュ彛鍒嗘瀽鍒嗘椂鏁版嵁锛坓etinterfacesummaryhour锛 + ) + ); + + /** + * 鑾峰彇浜岀淮鐮佸浘鐗 + * @param string $ticket 浼犲叆鐢眊etQRCode鏂规硶鐢熸垚鐨則icket鍙傛暟 + * @return string url 杩斿洖http鍦板潃 + */ + public function getQRUrl($ticket) { + return self::QRCODE_IMG_URL . urlencode($ticket); + } + + /** + * 闀块摼鎺ヨ浆鐭摼鎺ユ帴鍙 + * @param string $long_url 浼犲叆瑕佽浆鎹㈢殑闀縰rl + * @return bool|string url 鎴愬姛鍒欒繑鍥炶浆鎹㈠悗鐨勭煭url + */ + public function getShortUrl($long_url) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array( + 'action' => 'long2short', + 'long_url' => $long_url + ); + $result = Tools::httpPost(self::API_URL_PREFIX . self::SHORT_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json['short_url']; + } + return false; + } + + /** + * 鍒涘缓浜岀淮鐮乼icket + * @param int|string $scene_id 鑷畾涔夎拷韪猧d,涓存椂浜岀淮鐮佸彧鑳界敤鏁板煎瀷 + * @param int $type 0:涓存椂浜岀淮鐮侊紱1:姘镐箙浜岀淮鐮(姝ゆ椂expire鍙傛暟鏃犳晥)锛2:姘镐箙浜岀淮鐮(姝ゆ椂expire鍙傛暟鏃犳晥) + * @param int $expire 涓存椂浜岀淮鐮佹湁鏁堟湡锛屾渶澶т负2592000绉(30澶) + * @return bool|array ('ticket'=>'qrcode瀛椾覆','expire_seconds'=>2592000,'url'=>'浜岀淮鐮佸浘鐗囪В鏋愬悗鐨勫湴鍧') + */ + public function getQRCode($scene_id, $type = 0, $expire = 2592000) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $type = ($type && is_string($scene_id)) ? 2 : $type; + $data = array( + 'action_name' => $type ? ($type == 2 ? "QR_LIMIT_STR_SCENE" : "QR_LIMIT_SCENE") : "QR_SCENE", + 'expire_seconds' => $expire, + 'action_info' => array('scene' => ($type == 2 ? array('scene_str' => $scene_id) : array('scene_id' => $scene_id))) + ); + if ($type == 1) { + unset($data['expire_seconds']); + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::QRCODE_CREATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 璇箟鐞嗚В鎺ュ彛 + * @param string $uid 鐢ㄦ埛鍞竴id锛堥潪寮鍙戣卛d锛夛紝鐢ㄦ埛鍖哄垎鍏紬鍙蜂笅鐨勪笉鍚岀敤鎴凤紙寤鸿濉叆鐢ㄦ埛openid锛 + * @param string $query 杈撳叆鏂囨湰涓 + * @param string $category 闇瑕佷娇鐢ㄧ殑鏈嶅姟绫诲瀷锛屽涓敤鈥滐紝鈥濋殧寮锛屼笉鑳戒负绌 + * @param float $latitude 绾害鍧愭爣锛屼笌缁忓害鍚屾椂浼犲叆锛涗笌鍩庡競浜岄変竴浼犲叆 + * @param float $longitude 缁忓害鍧愭爣锛屼笌绾害鍚屾椂浼犲叆锛涗笌鍩庡競浜岄変竴浼犲叆 + * @param string $city 鍩庡競鍚嶇О锛屼笌缁忕含搴︿簩閫変竴浼犲叆 + * @param string $region 鍖哄煙鍚嶇О锛屽湪鍩庡競瀛樺湪鐨勬儏鍐典笅鍙渷鐣ワ紱涓庣粡绾害浜岄変竴浼犲叆 + * @return bool|array + */ + public function querySemantic($uid, $query, $category, $latitude = 0.00, $longitude = 0.00, $city = "", $region = "") { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array( + 'query' => $query, + 'category' => $category, + 'appid' => $this->appid, + 'uid' => '' + ); + //鍦扮悊鍧愭爣鎴栧煄甯傚悕绉颁簩閫変竴 + if ($latitude) { + $data['latitude'] = $latitude; + $data['longitude'] = $longitude; + } elseif ($city) { + $data['city'] = $city; + } elseif ($region) { + $data['region'] = $region; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::SEMANTIC_API_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇缁熻鏁版嵁 + * @param string $type 鏁版嵁鍒嗙被(user|article|upstreammsg|interface)鍒嗗埆涓(鐢ㄦ埛鍒嗘瀽|鍥炬枃鍒嗘瀽|娑堟伅鍒嗘瀽|鎺ュ彛鍒嗘瀽) + * @param string $subtype 鏁版嵁瀛愬垎绫伙紝鍙傝 DATACUBE_URL_ARR 甯搁噺瀹氫箟閮ㄥ垎 鎴栬匯EADME.md璇存槑鏂囨。 + * @param string $begin_date 寮濮嬫椂闂 + * @param string $end_date 缁撴潫鏃堕棿 + * @return bool|array 鎴愬姛杩斿洖鏌ヨ缁撴灉鏁扮粍锛屽叾瀹氫箟璇风湅瀹樻柟鏂囨。 + */ + public function getDatacube($type, $subtype, $begin_date, $end_date = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if (!isset(self::$DATACUBE_URL_ARR[$type]) || !isset(self::$DATACUBE_URL_ARR[$type][$subtype])) { + return false; + } + $data = array( + 'begin_date' => $begin_date, + 'end_date' => $end_date ? $end_date : $begin_date + ); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::$DATACUBE_URL_ARR[$type][$subtype] . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return isset($json['list']) ? $json['list'] : $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatMedia.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatMedia.php new file mode 100644 index 000000000..5a8613276 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatMedia.php @@ -0,0 +1,370 @@ + + * @date 2016/10/26 14:47 + */ +class WechatMedia extends Common { + + const UPLOAD_MEDIA_URL = 'http://file.api.weixin.qq.com/cgi-bin'; + const MEDIA_UPLOAD_URL = '/media/upload?'; + const MEDIA_UPLOADIMG_URL = '/media/uploadimg?'; //鍥剧墖涓婁紶鎺ュ彛 + const MEDIA_GET_URL = '/media/get?'; + const MEDIA_VIDEO_UPLOAD = '/media/uploadvideo?'; + const MEDIA_FOREVER_UPLOAD_URL = '/material/add_material?'; + const MEDIA_FOREVER_NEWS_UPLOAD_URL = '/material/add_news?'; + const MEDIA_FOREVER_NEWS_UPDATE_URL = '/material/update_news?'; + const MEDIA_FOREVER_GET_URL = '/material/get_material?'; + const MEDIA_FOREVER_DEL_URL = '/material/del_material?'; + const MEDIA_FOREVER_COUNT_URL = '/material/get_materialcount?'; + const MEDIA_FOREVER_BATCHGET_URL = '/material/batchget_material?'; + const MEDIA_UPLOADNEWS_URL = '/media/uploadnews?'; + + /** + * 涓婁紶涓存椂绱犳潗锛屾湁鏁堟湡涓3澶(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 娉ㄦ剰锛氫笂浼犲ぇ鏂囦欢鏃跺彲鑳介渶瑕佸厛璋冪敤 set_time_limit(0) 閬垮厤瓒呮椂 + * 娉ㄦ剰锛氭暟缁勭殑閿间换鎰忥紝浣嗘枃浠跺悕鍓嶅繀椤诲姞@锛屼娇鐢ㄥ崟寮曞彿浠ラ伩鍏嶆湰鍦拌矾寰勬枩鏉犺杞箟 + * 娉ㄦ剰锛氫复鏃剁礌鏉愮殑media_id鏄彲澶嶇敤鐨勶紒 + * @param array $data {"media":'@Path\filename.jpg'} + * @param string $type 绫诲瀷锛氬浘鐗:image 璇煶:voice 瑙嗛:video 缂╃暐鍥:thumb + * @return bool|array + */ + public function uploadMedia($data, $type) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + //鍘熷厛鐨勪笂浼犲濯掍綋鏂囦欢鎺ュ彛浣跨敤 self::UPLOAD_MEDIA_URL 鍓嶇紑 + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_UPLOAD_URL . "access_token={$this->access_token}" . '&type=' . $type, $data, true); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇涓存椂绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param string $media_id 濯掍綋鏂囦欢id + * @param bool $is_video 鏄惁涓鸿棰戞枃浠讹紝榛樿涓哄惁 + * @return bool|array + */ + public function getMedia($media_id, $is_video = false) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + //鍘熷厛鐨勪笂浼犲濯掍綋鏂囦欢鎺ュ彛浣跨敤 self::UPLOAD_MEDIA_URL 鍓嶇紑 + //濡傛灉瑕佽幏鍙栫殑绱犳潗鏄棰戞枃浠舵椂锛屼笉鑳戒娇鐢╤ttps鍗忚锛屽繀椤绘洿鎹㈡垚http鍗忚 + $url_prefix = $is_video ? str_replace('https', 'http', self::API_URL_PREFIX) : self::API_URL_PREFIX; + $result = Tools::httpGet($url_prefix . self::MEDIA_GET_URL . "access_token={$this->access_token}" . '&media_id=' . $media_id); + if ($result) { + if (is_string($result)) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + } + return $result; + } + return false; + } + + /** + * 涓婁紶鍥剧墖锛屾湰鎺ュ彛鎵涓婁紶鐨勫浘鐗囦笉鍗犵敤鍏紬鍙风殑绱犳潗搴撲腑鍥剧墖鏁伴噺鐨5000涓殑闄愬埗銆傚浘鐗囦粎鏀寔jpg/png鏍煎紡锛屽ぇ灏忓繀椤诲湪1MB浠ヤ笅銆 (璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 娉ㄦ剰锛氫笂浼犲ぇ鏂囦欢鏃跺彲鑳介渶瑕佸厛璋冪敤 set_time_limit(0) 閬垮厤瓒呮椂 + * 娉ㄦ剰锛氭暟缁勭殑閿间换鎰忥紝浣嗘枃浠跺悕鍓嶅繀椤诲姞@锛屼娇鐢ㄥ崟寮曞彿浠ラ伩鍏嶆湰鍦拌矾寰勬枩鏉犺杞箟 + * @param array $data {"media":'@Path\filename.jpg'} + * @return bool|array + */ + public function uploadImg($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + /* 鍘熷厛鐨勪笂浼犲濯掍綋鏂囦欢鎺ュ彛浣跨敤 self::UPLOAD_MEDIA_URL 鍓嶇紑 */ + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_UPLOADIMG_URL . "access_token={$this->access_token}", $data, true); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 涓婁紶姘镐箙绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 鏂板鐨勬案涔呯礌鏉愪篃鍙互鍦ㄥ叕浼楀钩鍙板畼缃戠礌鏉愮鐞嗘ā鍧椾腑鐪嬪埌 + * 娉ㄦ剰锛氫笂浼犲ぇ鏂囦欢鏃跺彲鑳介渶瑕佸厛璋冪敤 set_time_limit(0) 閬垮厤瓒呮椂 + * 娉ㄦ剰锛氭暟缁勭殑閿间换鎰忥紝浣嗘枃浠跺悕鍓嶅繀椤诲姞@锛屼娇鐢ㄥ崟寮曞彿浠ラ伩鍏嶆湰鍦拌矾寰勬枩鏉犺杞箟 + * @param array $data {"media":'@Path\filename.jpg'} + * @param string $type 绫诲瀷锛氬浘鐗:image 璇煶:voice 瑙嗛:video 缂╃暐鍥:thumb + * @param bool $is_video 鏄惁涓鸿棰戞枃浠讹紝榛樿涓哄惁 + * @param array $video_info 瑙嗛淇℃伅鏁扮粍锛岄潪瑙嗛绱犳潗涓嶉渶瑕佹彁渚 array('title'=>'瑙嗛鏍囬','introduction'=>'鎻忚堪') + * @return bool|array + */ + public function uploadForeverMedia($data, $type, $is_video = false, $video_info = array()) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if ($is_video) { + $data['description'] = Tools::json_encode($video_info); + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_FOREVER_UPLOAD_URL . "access_token={$this->access_token}" . '&type=' . $type, $data, true); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 涓婁紶姘镐箙鍥炬枃绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 鏂板鐨勬案涔呯礌鏉愪篃鍙互鍦ㄥ叕浼楀钩鍙板畼缃戠礌鏉愮鐞嗘ā鍧椾腑鐪嬪埌 + * @param array $data 娑堟伅缁撴瀯{"articles":[{...}]} + * @return bool|array + */ + public function uploadForeverArticles($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_FOREVER_NEWS_UPLOAD_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 淇敼姘镐箙鍥炬枃绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 姘镐箙绱犳潗涔熷彲浠ュ湪鍏紬骞冲彴瀹樼綉绱犳潗绠$悊妯″潡涓湅鍒 + * @param string $media_id 鍥炬枃绱犳潗id + * @param array $data 娑堟伅缁撴瀯{"articles":[{...}]} + * @param int $index 鏇存柊鐨勬枃绔犲湪鍥炬枃绱犳潗鐨勪綅缃紝绗竴绡囦负0锛屼粎澶氬浘鏂囦娇鐢 + * @return bool|array + */ + public function updateForeverArticles($media_id, $data, $index = 0) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + if (!isset($data['media_id'])) { + $data['media_id'] = $media_id; + } + if (!isset($data['index'])) { + $data['index'] = $index; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_FOREVER_NEWS_UPDATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇姘镐箙绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 杩斿洖鍥炬枃娑堟伅鏁扮粍鎴栦簩杩涘埗鏁版嵁锛屽け璐ヨ繑鍥瀎alse + * @param string $media_id 濯掍綋鏂囦欢id + * @param bool $is_video 鏄惁涓鸿棰戞枃浠讹紝榛樿涓哄惁 + * @return bool|array|raw data + */ + public function getForeverMedia($media_id, $is_video = false) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('media_id' => $media_id); + //#TODO 鏆備笉纭畾姝ゆ帴鍙f槸鍚﹂渶瑕佽瑙嗛鏂囦欢璧癶ttp鍗忚 + //濡傛灉瑕佽幏鍙栫殑绱犳潗鏄棰戞枃浠舵椂锛屼笉鑳戒娇鐢╤ttps鍗忚锛屽繀椤绘洿鎹㈡垚http鍗忚 + //$url_prefix = $is_video?str_replace('https','http',self::API_URL_PREFIX):self::API_URL_PREFIX; + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_FOREVER_GET_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + if (is_string($result)) { + $json = json_decode($result, true); + if ($json) { + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } else { + return $result; + } + } + return $result; + } + return false; + } + + /** + * 鍒犻櫎姘镐箙绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param string $media_id 濯掍綋鏂囦欢id + * @return bool + */ + public function delForeverMedia($media_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('media_id' => $media_id); + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_FOREVER_DEL_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鑾峰彇姘镐箙绱犳潗鍒楄〃(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param string $type 绱犳潗鐨勭被鍨,鍥剧墖锛坕mage锛夈佽棰戯紙video锛夈佽闊 锛坴oice锛夈佸浘鏂囷紙news锛 + * @param int $offset 鍏ㄩ儴绱犳潗鐨勫亸绉讳綅缃紝0琛ㄧず浠庣涓涓礌鏉 + * @param int $count 杩斿洖绱犳潗鐨勬暟閲忥紝鍙栧煎湪1鍒20涔嬮棿 + * @return bool|array + * 杩斿洖鏁扮粍鏍煎紡: + * array( + * 'total_count'=>0, //璇ョ被鍨嬬殑绱犳潗鐨勬绘暟 + * 'item_count'=>0, //鏈璋冪敤鑾峰彇鐨勭礌鏉愮殑鏁伴噺 + * 'item'=>array() //绱犳潗鍒楄〃鏁扮粍锛屽唴瀹瑰畾涔夎鍙傝冨畼鏂规枃妗 + * ) + */ + public function getForeverList($type, $offset, $count) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array( + 'type' => $type, + 'offset' => $offset, + 'count' => $count, + ); + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_FOREVER_BATCHGET_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇姘镐箙绱犳潗鎬绘暟(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @return bool|array + * 杩斿洖鏁扮粍鏍煎紡: + * array( + * 'voice_count'=>0, //璇煶鎬绘暟閲 + * 'video_count'=>0, //瑙嗛鎬绘暟閲 + * 'image_count'=>0, //鍥剧墖鎬绘暟閲 + * 'news_count'=>0 //鍥炬枃鎬绘暟閲 + * ) + */ + public function getForeverCount() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::MEDIA_FOREVER_COUNT_URL . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 涓婁紶鍥炬枃娑堟伅绱犳潗锛岀敤浜庣兢鍙(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param array $data 娑堟伅缁撴瀯{"articles":[{...}]} + * @return bool|array + */ + public function uploadArticles($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MEDIA_UPLOADNEWS_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 涓婁紶瑙嗛绱犳潗(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param array $data 娑堟伅缁撴瀯 + * { + * "media_id"=>"", //閫氳繃涓婁紶濯掍綋鎺ュ彛寰楀埌鐨凪ediaId + * "title"=>"TITLE", //瑙嗛鏍囬 + * "description"=>"Description" //瑙嗛鎻忚堪 + * } + * @return bool|array + * { + * "type":"video", + * "media_id":"mediaid", + * "created_at":1398848981 + * } + */ + public function uploadMpVideo($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::UPLOAD_MEDIA_URL . self::MEDIA_VIDEO_UPLOAD . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatMenu.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatMenu.php new file mode 100644 index 000000000..e87045622 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatMenu.php @@ -0,0 +1,164 @@ + + * @date 2016/06/28 11:52 + */ +class WechatMenu extends Common { + + /** 鍒涘缓鑷畾涔夎彍鍗 */ + const MENU_ADD_URL = '/menu/create?'; + /* 鑾峰彇鑷畾涔夎彍鍗 */ + const MENU_GET_URL = '/menu/get?'; + /* 鍒犻櫎鑷畾涔夎彍鍗 */ + const MENU_DEL_URL = '/menu/delete?'; + + /** 娣诲姞涓ц彍鍗 */ + const COND_MENU_ADD_URL = '/menu/addconditional?'; + /* 鍒犻櫎涓ц彍鍗 */ + const COND_MENU_DEL_URL = '/menu/delconditional?'; + /* 娴嬭瘯涓ц彍鍗 */ + const COND_MENU_TRY_URL = '/menu/trymatch?'; + + /** + * 鍒涘缓鑷畾涔夎彍鍗 + * @param array $data 鑿滃崟鏁扮粍鏁版嵁 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421141013&token=&lang=zh_CN 鏂囨。 + * @return bool + */ + public function createMenu($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MENU_ADD_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鑾峰彇鎵鏈夎彍鍗 + * @return bool|array + */ + public function getMenu() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::MENU_GET_URL . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍒犻櫎鎵鏈夎彍鍗 + * @return bool + */ + public function deleteMenu() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::MENU_DEL_URL . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鍒涘缓涓ц彍鍗 + * @param array $data 鑿滃崟鏁扮粍鏁版嵁 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1455782296&token=&lang=zh_CN 鏂囨。 + * @return bool|string + */ + public function createCondMenu($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::COND_MENU_ADD_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode']) || empty($json['menuid'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json['menuid']; + } + return false; + } + + /** + * 鍒犻櫎涓ц彍鍗 + * @param string $menuid 鑿滃崟ID + * @return bool + */ + public function deleteCondMenu($menuid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('menuid' => $menuid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::COND_MENU_DEL_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 娴嬭瘯骞惰繑鍥炰釜鎬у寲鑿滃崟 + * @param string $openid 绮変笣openid + * @return bool + */ + public function tryCondMenu($openid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('user_id' => $openid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::COND_MENU_TRY_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatOauth.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatOauth.php new file mode 100644 index 000000000..8f5f14949 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatOauth.php @@ -0,0 +1,120 @@ +appid}&redirect_uri={$redirect_uri}&response_type=code&scope={$scope}&state={$state}#wechat_redirect"; + } + + /** + * 閫氳繃 code 鑾峰彇 AccessToken 鍜 openid + * @return bool|array + */ + public function getOauthAccessToken() { + $code = isset($_GET['code']) ? $_GET['code'] : ''; + if (empty($code)) { + Tools::log("getOauthAccessToken Fail, Because there is no access to the code value in get."); + return false; + } + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::OAUTH_TOKEN_URL . "appid={$this->appid}&secret={$this->appsecret}&code={$code}&grant_type=authorization_code"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + Tools::log("WechatOauth::getOauthAccessToken Fail.{$this->errMsg} [{$this->errCode}]", 'ERR'); + return false; + } + return $json; + } + return false; + } + + /** + * 鍒锋柊access token骞剁画鏈 + * @param string $refresh_token + * @return bool|array + */ + public function getOauthRefreshToken($refresh_token) { + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::OAUTH_REFRESH_URL . "appid={$this->appid}&grant_type=refresh_token&refresh_token={$refresh_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + Tools::log("WechatOauth::getOauthRefreshToken Fail.{$this->errMsg} [{$this->errCode}]", 'ERR'); + return false; + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鎺堟潈鍚庣殑鐢ㄦ埛璧勬枡 + * @param string $access_token + * @param string $openid + * @return bool|array {openid,nickname,sex,province,city,country,headimgurl,privilege,[unionid]} + * 娉ㄦ剰锛歶nionid瀛楁 鍙湁鍦ㄧ敤鎴峰皢鍏紬鍙风粦瀹氬埌寰俊寮鏀惧钩鍙拌处鍙峰悗锛屾墠浼氬嚭鐜般傚缓璁皟鐢ㄥ墠鐢╥sset()妫娴嬩竴涓 + */ + public function getOauthUserInfo($access_token, $openid) { + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::OAUTH_USERINFO_URL . "access_token={$access_token}&openid={$openid}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + Tools::log("WechatOauth::getOauthUserInfo Fail.{$this->errMsg} [{$this->errCode}]", 'ERR'); + return false; + } + return $json; + } + return false; + } + + /** + * 妫楠屾巿鏉冨嚟璇佹槸鍚︽湁鏁 + * @param string $access_token + * @param string $openid + * @return bool 鏄惁鏈夋晥 + */ + public function getOauthAuth($access_token, $openid) { + $result = Tools::httpGet(self::API_BASE_URL_PREFIX . self::OAUTH_AUTH_URL . "access_token={$access_token}&openid={$openid}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + Tools::log("WechatOauth::getOauthAuth Fail.{$this->errMsg} [{$this->errCode}]", 'ERR'); + return false; + } else if ($json['errcode'] == 0) { + return true; + } + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatPay.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatPay.php new file mode 100644 index 000000000..a8c125a00 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatPay.php @@ -0,0 +1,490 @@ + + * @date 2015/05/13 12:12:00 + */ +class WechatPay { + + /** 鏀粯鎺ュ彛鍩虹鍦板潃 */ + const MCH_BASE_URL = 'https://api.mch.weixin.qq.com'; + + /** 鍏紬鍙穉ppid */ + public $appid; + + /** 鍟嗘埛韬唤ID */ + public $mch_id; + + /** 鍟嗘埛鏀粯瀵嗛挜Key */ + public $partnerKey; + + /** 璇佷功璺緞 */ + public $ssl_cer; + public $ssl_key; + + /** 鎵ц閿欒娑堟伅鍙婁唬鐮 */ + public $errMsg; + public $errCode; + + /** + * WechatPay constructor. + * @param array $options + */ + public function __construct($options = array()) { + $config = Loader::config($options); + $this->appid = isset($config['appid']) ? $config['appid'] : ''; + $this->mch_id = isset($config['mch_id']) ? $config['mch_id'] : ''; + $this->partnerKey = isset($config['partnerkey']) ? $config['partnerkey'] : ''; + $this->ssl_cer = isset($config['ssl_cer']) ? $config['ssl_cer'] : ''; + $this->ssl_key = isset($config['ssl_key']) ? $config['ssl_key'] : ''; + } + + /** + * 璁剧疆鏍囬厤鐨勮姹傚弬鏁帮紝鐢熸垚绛惧悕锛岀敓鎴愭帴鍙e弬鏁皒ml + * @param array $data + * @return string + */ + protected function createXml($data) { + if (!isset($data['wxappid']) && !isset($data['mch_appid']) && !isset($data['appid'])) { + $data['appid'] = $this->appid; + } + if (!isset($data['mchid']) && !isset($data['mch_id'])) { + $data['mch_id'] = $this->mch_id; + } + isset($data['nonce_str']) || $data['nonce_str'] = Tools::createNoncestr(); + $data["sign"] = Tools::getPaySign($data, $this->partnerKey); + return Tools::arr2xml($data); + } + + /** + * POST鎻愪氦XML + * @param array $data + * @param string $url + * @return mixed + */ + public function postXml($data, $url) { + return Tools::httpPost($url, $this->createXml($data)); + } + + /** + * 浣跨敤璇佷功post璇锋眰XML + * @param array $data + * @param string $url + * @return mixed + */ + function postXmlSSL($data, $url) { + return Tools::httpsPost($url, $this->createXml($data), $this->ssl_cer, $this->ssl_key); + } + + /** + * POST鎻愪氦鑾峰彇Array缁撴灉 + * @param array $data 闇瑕佹彁浜ょ殑鏁版嵁 + * @param string $url + * @param string $method + * @return array + */ + public function getArrayResult($data, $url, $method = 'postXml') { + return Tools::xml2arr($this->$method($data, $url)); + } + + /** + * 瑙f瀽杩斿洖鐨勭粨鏋 + * @param array $result + * @return bool|array + */ + protected function _parseResult($result) { + if (empty($result)) { + $this->errCode = 'result error'; + $this->errMsg = '瑙f瀽杩斿洖缁撴灉澶辫触'; + return false; + } + if ($result['return_code'] !== 'SUCCESS') { + $this->errCode = $result['return_code']; + $this->errMsg = $result['return_msg']; + return false; + } + if (isset($result['err_code']) && $result['err_code'] !== 'SUCCESS') { + $this->errMsg = $result['err_code_des']; + $this->errCode = $result['err_code']; + return false; + } + return $result; + } + + /** + * 鏀粯閫氱煡楠岃瘉澶勭悊 + * @return bool|array + */ + public function getNotify() { + $notifyInfo = (array)simplexml_load_string(file_get_contents("php://input"), 'SimpleXMLElement', LIBXML_NOCDATA); + if (empty($notifyInfo)) { + Tools::log('Payment notification forbidden access.', 'ERR'); + $this->errCode = '404'; + $this->errMsg = 'Payment notification forbidden access.'; + return false; + } + if (empty($notifyInfo['sign'])) { + Tools::log('Payment notification signature is missing.' . var_export($notifyInfo, true), 'ERR'); + $this->errCode = '403'; + $this->errMsg = 'Payment notification signature is missing.'; + return false; + } + $data = $notifyInfo; + unset($data['sign']); + if ($notifyInfo['sign'] !== Tools::getPaySign($data, $this->partnerKey)) { + Tools::log('Payment notification signature verification failed.' . var_export($notifyInfo, true), 'ERR'); + $this->errCode = '403'; + $this->errMsg = 'Payment signature verification failed.'; + return false; + } + Tools::log('Payment notification signature verification success.' . var_export($notifyInfo, true), 'MSG'); + $this->errCode = '0'; + $this->errMsg = ''; + return $notifyInfo; + } + + + /** + * 鏀粯XML缁熶竴鍥炲 + * @param array $data 闇瑕佸洖澶嶇殑XML鍐呭鏁扮粍 + * @param bool $isReturn 鏄惁杩斿洖XML鍐呭锛岄粯璁や笉杩斿洖 + * @return string + */ + public function replyXml(array $data, $isReturn = false) { + $xml = Tools::arr2xml($data); + if ($isReturn) { + return $xml; + } + ob_clean(); + exit($xml); + } + + /** + * 鑾峰彇棰勬敮浠業D + * @param string $openid 鐢ㄦ埛openid锛孞SAPI蹇呭~ + * @param string $body 鍟嗗搧鏍囬 + * @param string $out_trade_no 绗笁鏂硅鍗曞彿 + * @param int $total_fee 璁㈠崟鎬讳环 + * @param string $notify_url 鏀粯鎴愬姛鍥炶皟鍦板潃 + * @param string $trade_type 鏀粯绫诲瀷JSAPI|NATIVE|APP + * @param string $goods_tag 鍟嗗搧鏍囪锛屼唬閲戝埜鎴栫珛鍑忎紭鎯犲姛鑳界殑鍙傛暟 + * @return bool|string + */ + public function getPrepayId($openid, $body, $out_trade_no, $total_fee, $notify_url, $trade_type = "JSAPI", $goods_tag = null) { + $postdata = array( + "body" => $body, + "out_trade_no" => $out_trade_no, + "total_fee" => $total_fee, + "notify_url" => $notify_url, + "trade_type" => $trade_type, + "spbill_create_ip" => Tools::getAddress() + ); + empty($goods_tag) || $postdata['goods_tag'] = $goods_tag; + empty($openid) || $postdata['openid'] = $openid; + $result = $this->getArrayResult($postdata, self::MCH_BASE_URL . '/pay/unifiedorder'); + if (false === $this->_parseResult($result)) { + return false; + } + return ($trade_type === 'JSAPI') ? $result['prepay_id'] : $result['code_url']; + } + + /** + * 鑾峰彇浜岀淮鐮侀鏀粯ID + * @param string $openid 鐢ㄦ埛openid锛孞SAPI蹇呭~ + * @param string $body 鍟嗗搧鏍囬 + * @param string $out_trade_no 绗笁鏂硅鍗曞彿 + * @param int $total_fee 璁㈠崟鎬讳环 + * @param string $notify_url 鏀粯鎴愬姛鍥炶皟鍦板潃 + * @param string $goods_tag 鍟嗗搧鏍囪锛屼唬閲戝埜鎴栫珛鍑忎紭鎯犲姛鑳界殑鍙傛暟 + * @return bool|string + */ + public function getQrcPrepayId($openid, $body, $out_trade_no, $total_fee, $notify_url, $goods_tag = null) { + $postdata = array( + "body" => $body, + "out_trade_no" => $out_trade_no, + "total_fee" => $total_fee, + "notify_url" => $notify_url, + "trade_type" => 'NATIVE', + "spbill_create_ip" => Tools::getAddress() + ); + empty($goods_tag) || $postdata['goods_tag'] = $goods_tag; + empty($openid) || $postdata['openid'] = $openid; + $result = $this->getArrayResult($postdata, self::MCH_BASE_URL . '/pay/unifiedorder'); + if (false === $this->_parseResult($result) || empty($result['prepay_id'])) { + return false; + } + return $result['prepay_id']; + } + + /** + * 鑾峰彇鏀粯瑙勪簩缁寸爜 + * @param string $product_id 鍟嗘埛瀹氫箟鐨勫晢鍝乮d 鎴栬呰鍗曞彿 + * @return string + */ + public function getQrcPayUrl($product_id) { + $data = array( + 'appid' => $this->appid, + 'mch_id' => $this->mch_id, + 'time_stamp' => (string)time(), + 'nonce_str' => Tools::createNoncestr(), + 'product_id' => (string)$product_id, + ); + $data['sign'] = Tools::getPaySign($data, $this->partnerKey); + return "weixin://wxpay/bizpayurl?" . http_build_query($data); + } + + + /** + * 鍒涘缓JSAPI鏀粯鍙傛暟鍖 + * @param string $prepay_id + * @return array + */ + public function createMchPay($prepay_id) { + $option = array(); + $option["appId"] = $this->appid; + $option["timeStamp"] = (string)time(); + $option["nonceStr"] = Tools::createNoncestr(); + $option["package"] = "prepay_id={$prepay_id}"; + $option["signType"] = "MD5"; + $option["paySign"] = Tools::getPaySign($option, $this->partnerKey); + $option['timestamp'] = $option['timeStamp']; + return $option; + } + + /** + * 鍏抽棴璁㈠崟 + * @param string $out_trade_no + * @return bool + */ + public function closeOrder($out_trade_no) { + $data = array('out_trade_no' => $out_trade_no); + $result = $this->getArrayResult($data, self::MCH_BASE_URL . '/pay/closeorder'); + if (false === $this->_parseResult($result)) { + return false; + } + return ($result['return_code'] === 'SUCCESS'); + } + + /** + * 鏌ヨ璁㈠崟璇︽儏 + * @param $out_trade_no + * @return bool|array + */ + public function queryOrder($out_trade_no) { + $data = array('out_trade_no' => $out_trade_no); + $result = $this->getArrayResult($data, self::MCH_BASE_URL . '/pay/orderquery'); + if (false === $this->_parseResult($result)) { + return false; + } + return $result; + } + + /** + * 璁㈠崟閫娆炬帴鍙 + * @param string $out_trade_no 鍟嗘埛璁㈠崟鍙 + * @param string $transaction_id 寰俊璁㈠崟鍙 + * @param string $out_refund_no 鍟嗘埛閫娆捐鍗曞彿 + * @param int $total_fee 鍟嗘埛璁㈠崟鎬婚噾棰 + * @param int $refund_fee 閫娆鹃噾棰 + * @param int|null $op_user_id 鎿嶄綔鍛業D锛岄粯璁ゅ晢鎴稩D + * @param string $refund_account 閫娆捐祫閲戞潵婧 + * 浠呴拡瀵硅佽祫閲戞祦鍟嗘埛浣跨敤 + * REFUND_SOURCE_UNSETTLED_FUNDS --- 鏈粨绠楄祫閲戦娆撅紙榛樿浣跨敤鏈粨绠楄祫閲戦娆撅級 + * REFUND_SOURCE_RECHARGE_FUNDS --- 鍙敤浣欓閫娆 + * @return bool + */ + public function refund($out_trade_no, $transaction_id, $out_refund_no, $total_fee, $refund_fee, $op_user_id = null, $refund_account = '') { + $data = array(); + $data['out_trade_no'] = $out_trade_no; + $data['transaction_id'] = $transaction_id; + $data['out_refund_no'] = $out_refund_no; + $data['total_fee'] = $total_fee; + $data['refund_fee'] = $refund_fee; + $data['op_user_id'] = empty($op_user_id) ? $this->mch_id : $op_user_id; + !empty($refund_account) && $data['refund_account'] = $refund_account; + $result = $this->getArrayResult($data, self::MCH_BASE_URL . '/secapi/pay/refund', 'postXmlSSL'); + if (false === $this->_parseResult($result)) { + return false; + } + return ($result['return_code'] === 'SUCCESS'); + } + + /** + * 閫娆炬煡璇㈡帴鍙 + * @param string $out_trade_no + * @return bool|array + */ + public function refundQuery($out_trade_no) { + $data = array(); + $data['out_trade_no'] = $out_trade_no; + $result = $this->getArrayResult($data, self::MCH_BASE_URL . '/pay/refundquery'); + if (false === $this->_parseResult($result)) { + return false; + } + return $result; + } + + /** + * 鑾峰彇瀵硅处鍗 + * @param string $bill_date 璐﹀崟鏃ユ湡锛屽 20141110 + * @param string $bill_type ALL|SUCCESS|REFUND|REVOKED + * @return bool|array + */ + public function getBill($bill_date, $bill_type = 'ALL') { + $data = array(); + $data['bill_date'] = $bill_date; + $data['bill_type'] = $bill_type; + $result = $this->postXml($data, self::MCH_BASE_URL . '/pay/downloadbill'); + $json = Tools::xml2arr($result); + if (!empty($json) && false === $this->_parseResult($json)) { + return false; + } + return $json; + } + + /** + * 鍙戦佺幇閲戠孩鍖 + * @param string $openid 绾㈠寘鎺ユ敹鑰匫PENID + * @param int $amount 绾㈠寘鎬婚噾棰 + * @param string $billno 鍟嗘埛璁㈠崟鍙 + * @param string $sendname 鍟嗘埛鍚嶇О + * @param string $wishing 绾㈠寘绁濈璇 + * @param string $actname 娲诲姩鍚嶇О + * @param string $remark 澶囨敞淇℃伅 + * @return bool|array + * @link https://pay.weixin.qq.com/wiki/doc/api/tools/cash_coupon.php?chapter=13_5 + */ + public function sendRedPack($openid, $amount, $billno, $sendname, $wishing, $actname, $remark) { + $data = array(); + $data['mch_billno'] = $billno; // 鍟嗘埛璁㈠崟鍙 mch_id+yyyymmdd+10浣嶄竴澶╁唴涓嶈兘閲嶅鐨勬暟瀛 + $data['wxappid'] = $this->appid; + $data['send_name'] = $sendname; //鍟嗘埛鍚嶇О + $data['re_openid'] = $openid; //绾㈠寘鎺ユ敹鑰 + $data['total_amount'] = $amount; //绾㈠寘閲戦 + $data['total_num'] = '1'; //鍙戞斁浜烘暟鎹 + $data['wishing'] = $wishing; //绾㈠寘绁濈璇 + $data['client_ip'] = Tools::getAddress(); //璋冪敤鎺ュ彛鐨勬満鍣↖p鍦板潃 + $data['act_name'] = $actname; //娲诲姩鍚嶇О + $data['remark'] = $remark; //澶囨敞淇℃伅 + $result = $this->postXmlSSL($data, self::MCH_BASE_URL . '/mmpaymkttransfers/sendredpack'); + $json = Tools::xml2arr($result); + if (!empty($json) && false === $this->_parseResult($json)) { + return false; + } + return $json; + } + + /** + * 鐜伴噾绾㈠寘鐘舵佹煡璇 + * @param string $billno + * @return bool|array + * @link https://pay.weixin.qq.com/wiki/doc/api/tools/cash_coupon.php?chapter=13_7&index=6 + */ + public function queryRedPack($billno) { + $data['mch_billno'] = $billno; + $data['bill_type'] = 'MCHT'; + $result = $this->postXmlSSL($data, self::MCH_BASE_URL . '/mmpaymkttransfers/gethbinfo'); + $json = Tools::xml2arr($result); + if (!empty($json) && false === $this->_parseResult($json)) { + return false; + } + return $json; + } + + /** + * 浼佷笟浠樻 + * @param string $openid 绾㈠寘鎺ユ敹鑰匫PENID + * @param int $amount 绾㈠寘鎬婚噾棰 + * @param string $billno 鍟嗘埛璁㈠崟鍙 + * @param string $desc 澶囨敞淇℃伅 + * @return bool|array + * @link https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=14_2 + */ + public function transfers($openid, $amount, $billno, $desc) { + $data = array(); + $data['mchid'] = $this->mch_id; + $data['mch_appid'] = $this->appid; + $data['partner_trade_no'] = $billno; + $data['openid'] = $openid; + $data['amount'] = $amount; + $data['check_name'] = 'NO_CHECK'; #涓嶉獙璇佸鍚 + $data['spbill_create_ip'] = Tools::getAddress(); //璋冪敤鎺ュ彛鐨勬満鍣↖p鍦板潃 + $data['desc'] = $desc; //澶囨敞淇℃伅 + $result = $this->postXmlSSL($data, self::MCH_BASE_URL . '/mmpaymkttransfers/promotion/transfers'); + $json = Tools::xml2arr($result); + if (!empty($json) && false === $this->_parseResult($json)) { + return false; + } + return $json; + } + + /** + * 浼佷笟浠樻鏌ヨ + * @param string $billno + * @return bool|array + * @link https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=14_3 + */ + public function queryTransfers($billno) { + $data['appid'] = $this->appid; + $data['mch_id'] = $this->mch_id; + $data['partner_trade_no'] = $billno; + $result = $this->postXmlSSL($data, self::MCH_BASE_URL . '/mmpaymkttransfers/gettransferinfo'); + $json = Tools::xml2arr($result); + if (!empty($json) && false === $this->_parseResult($json)) { + return false; + } + return $json; + } + + /** + * 浜岀淮鐮侀摼鎺ヨ浆鎴愮煭閾炬帴 + * @param string $url 闇瑕佸鐞嗙殑闀块摼鎺 + * @return bool|string + */ + public function shortUrl($url) { + $data = array(); + $data['long_url'] = $url; + $result = $this->getArrayResult($data, self::MCH_BASE_URL . '/tools/shorturl'); + if (!$result || $result['return_code'] !== 'SUCCESS') { + $this->errCode = $result['return_code']; + $this->errMsg = $result['return_msg']; + return false; + } + if (isset($result['err_code']) && $result['err_code'] !== 'SUCCESS') { + $this->errMsg = $result['err_code_des']; + $this->errCode = $result['err_code']; + return false; + } + return $result['short_url']; + } + + /** + * 鍙戞斁浠i噾鍒 + * @param int $coupon_stock_id 浠i噾鍒告壒娆d + * @param string $partner_trade_no 鍟嗘埛姝ゆ鍙戞斁鍑嵁鍙凤紙鏍煎紡锛氬晢鎴穒d+鏃ユ湡+娴佹按鍙凤級锛屽晢鎴蜂晶闇淇濇寔鍞竴鎬 + * @param string $openid Openid淇℃伅 + * @param string $op_user_id 鎿嶄綔鍛樺笎鍙, 榛樿涓哄晢鎴峰彿 鍙湪鍟嗘埛骞冲彴閰嶇疆鎿嶄綔鍛樺搴旂殑api鏉冮檺 + * @return bool|array + * @link https://pay.weixin.qq.com/wiki/doc/api/tools/sp_coupon.php?chapter=12_3 + */ + public function sendCoupon($coupon_stock_id, $partner_trade_no, $openid, $op_user_id = null) { + $data = array(); + $data['appid'] = $this->appid; + $data['coupon_stock_id'] = $coupon_stock_id; + $data['openid_count'] = 1; + $data['partner_trade_no'] = $partner_trade_no; + $data['openid'] = $openid; + $data['op_user_id'] = empty($op_user_id) ? $this->mch_id : $op_user_id; + $result = $this->postXmlSSL($data, self::MCH_BASE_URL . '/mmpaymkttransfers/send_coupon'); + $json = Tools::xml2arr($result); + if (!empty($json) && false === $this->_parseResult($json)) { + return false; + } + return $json; + } +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatPoi.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatPoi.php new file mode 100644 index 000000000..1bce669fa --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatPoi.php @@ -0,0 +1,174 @@ + + * @date 2016/10/26 15:43 + */ +class WechatPoi extends Common { + + /** 鍒涘缓闂ㄥ簵 */ + const POI_ADD = '/cgi-bin/poi/addpoi?'; + + /** 鏌ヨ闂ㄥ簵淇℃伅 */ + const POI_GET = '/cgi-bin/poi/getpoi?'; + + /** 鑾峰彇闂ㄥ簵鍒楄〃 */ + const POI_GET_LIST = '/cgi-bin/poi/getpoilist?'; + + /** 淇敼闂ㄥ簵淇℃伅 */ + const POI_UPDATE = '/cgi-bin/poi/updatepoi?'; + + /** 鍒犻櫎闂ㄥ簵 */ + const POI_DELETE = '/cgi-bin/poi/delpoi?'; + + /** 鑾峰彇闂ㄥ簵绫荤洰琛 */ + const POI_CATEGORY = '/cgi-bin/poi/getwxcategory?'; + + /** + * 鍒涘缓闂ㄥ簵 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1444378120&token=&lang=zh_CN + * @param array $data + * @return bool + */ + public function addPoi($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::POI_ADD . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍒犻櫎闂ㄥ簵 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1444378120&token=&lang=zh_CN + * @param string $poi_id JSON鏁版嵁鏍煎紡 + * @return bool|array + */ + public function delPoi($poi_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('poi_id' => $poi_id); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::POI_DELETE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 淇敼闂ㄥ簵鏈嶅姟淇℃伅 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1444378120&token=&lang=zh_CN + * @param array $data + * @return bool + */ + public function updatePoi($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::POI_UPDATE . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 鏌ヨ闂ㄥ簵淇℃伅 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1444378120&token=&lang=zh_CN + * @param string $poi_id + * @return bool + */ + public function getPoi($poi_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('poi_id' => $poi_id); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::POI_GET . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏌ヨ闂ㄥ簵鍒楄〃 + * @link https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1444378120&token=&lang=zh_CN + * @param int $begin 寮濮嬩綅缃紝0 鍗充负浠庣涓鏉″紑濮嬫煡璇 + * @param int $limit 杩斿洖鏁版嵁鏉℃暟锛屾渶澶у厑璁50锛岄粯璁や负20 + * @return bool|array + */ + public function getPoiList($begin = 0, $limit = 50) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $limit > 50 && $limit = 50; + $data = array('begin' => $begin, 'limit' => $limit); + $result = Tools::httpPost(self::API_BASE_URL_PREFIX . self::POI_GET_LIST . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鍟嗗闂ㄥ簵绫荤洰琛 + * @return bool|string + */ + public function getCategory() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::POI_CATEGORY . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatReceive.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatReceive.php new file mode 100644 index 000000000..2a351c9fa --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatReceive.php @@ -0,0 +1,1005 @@ + + * @date 2016/06/28 11:29 + */ +class WechatReceive extends Common { + + /** 娑堟伅鎺ㄩ佸湴鍧 */ + const CUSTOM_SEND_URL = '/message/custom/send?'; + const MASS_SEND_URL = '/message/mass/send?'; + const TEMPLATE_SET_INDUSTRY_URL = '/message/template/api_set_industry?'; + const TEMPLATE_ADD_TPL_URL = '/message/template/api_add_template?'; + const TEMPLATE_SEND_URL = '/message/template/send?'; + const MASS_SEND_GROUP_URL = '/message/mass/sendall?'; + const MASS_DELETE_URL = '/message/mass/delete?'; + const MASS_PREVIEW_URL = '/message/mass/preview?'; + const MASS_QUERY_URL = '/message/mass/get?'; + + /** 娑堟伅鍥炲绫诲瀷 */ + const MSGTYPE_TEXT = 'text'; + const MSGTYPE_IMAGE = 'image'; + const MSGTYPE_LOCATION = 'location'; + const MSGTYPE_LINK = 'link'; + const MSGTYPE_EVENT = 'event'; + const MSGTYPE_MUSIC = 'music'; + const MSGTYPE_NEWS = 'news'; + const MSGTYPE_VOICE = 'voice'; + const MSGTYPE_VIDEO = 'video'; + + /** 鏂囦欢杩囨护 */ + protected $_text_filter = true; + + /** 娑堟伅瀵硅薄 */ + private $_receive; + + /** + * 鑾峰彇寰俊鏈嶅姟鍣ㄥ彂鏉ョ殑鍐呭 + * @return $this + */ + public function getRev() { + if ($this->_receive) { + return $this; + } + $postStr = !empty($this->postxml) ? $this->postxml : file_get_contents("php://input"); + !empty($postStr) && $this->_receive = (array) simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA); + return $this; + } + + /** + * 鑾峰彇寰俊鏈嶅姟鍣ㄥ彂鏉ョ殑淇℃伅鏁版嵁 + * @return array + */ + public function getRevData() { + return $this->_receive; + } + + /** + * 鑾峰彇娑堟伅鍙戦佽 + * @return bool|string + */ + public function getRevFrom() { + if (isset($this->_receive['FromUserName'])) { + return $this->_receive['FromUserName']; + } else { + return false; + } + } + + /** + * 鑾峰彇娑堟伅鎺ュ彈鑰 + * @return bool|string + */ + public function getRevTo() { + if (isset($this->_receive['ToUserName'])) { + return $this->_receive['ToUserName']; + } else { + return false; + } + } + + /** + * 鑾峰彇鎺ユ敹娑堟伅鐨勭被鍨 + * @return bool|string + */ + public function getRevType() { + if (isset($this->_receive['MsgType'])) { + return $this->_receive['MsgType']; + } else { + return false; + } + } + + /** + * 鑾峰彇娑堟伅ID + * @return bool|string + */ + public function getRevID() { + if (isset($this->_receive['MsgId'])) { + return $this->_receive['MsgId']; + } else { + return false; + } + } + + /** + * 鑾峰彇娑堟伅鍙戦佹椂闂 + * @return bool|string + */ + public function getRevCtime() { + if (isset($this->_receive['CreateTime'])) { + return $this->_receive['CreateTime']; + } else { + return false; + } + } + + /** + * 鑾峰彇鍗″埜浜嬩欢鎺ㄩ - 鍗″嵎瀹℃牳鏄惁閫氳繃 + * 褰揈vent涓 card_pass_check(瀹℃牳閫氳繃) 鎴 card_not_pass_check(鏈氳繃) + * @return bool|string 杩斿洖鍗″埜ID + */ + public function getRevCardPass() { + if (isset($this->_receive['CardId'])) { + return $this->_receive['CardId']; + } else { + return false; + } + } + + /** + * 鑾峰彇鍗″埜浜嬩欢鎺ㄩ - 棰嗗彇鍗″埜 + * 褰揈vent涓 user_get_card(鐢ㄦ埛棰嗗彇鍗″埜) + * @return bool|array + */ + public function getRevCardGet() { + $array = array(); + if (isset($this->_receive['CardId'])) { + $array['CardId'] = $this->_receive['CardId']; + } + if (isset($this->_receive['IsGiveByFriend'])) { + $array['IsGiveByFriend'] = $this->_receive['IsGiveByFriend']; + } + $array['OldUserCardCode'] = $this->_receive['OldUserCardCode']; + if (isset($this->_receive['UserCardCode']) && !empty($this->_receive['UserCardCode'])) { + $array['UserCardCode'] = $this->_receive['UserCardCode']; + } + if (isset($array) && count($array) > 0) { + return $array; + } else { + return false; + } + } + + /** + * 鑾峰彇鍗″埜浜嬩欢鎺ㄩ - 鍒犻櫎鍗″埜 + * 褰揈vent涓 user_del_card(鐢ㄦ埛鍒犻櫎鍗″埜) + * @return bool|array + */ + public function getRevCardDel() { + if (isset($this->_receive['CardId'])) { //鍗″埜 ID + $array['CardId'] = $this->_receive['CardId']; + } + if (isset($this->_receive['UserCardCode']) && !empty($this->_receive['UserCardCode'])) { + $array['UserCardCode'] = $this->_receive['UserCardCode']; + } + if (isset($array) && count($array) > 0) { + return $array; + } else { + return false; + } + } + + /** + * 鑾峰彇鎺ユ敹娑堟伅鍐呭姝f枃 + * @return bool + */ + public function getRevContent() { + if (isset($this->_receive['Content'])) { + return $this->_receive['Content']; + } else if (isset($this->_receive['Recognition'])) { //鑾峰彇璇煶璇嗗埆鏂囧瓧鍐呭锛岄渶鐢宠寮閫 + return $this->_receive['Recognition']; + } else { + return false; + } + } + + /** + * 鑾峰彇鎺ユ敹娑堟伅鍥剧墖 + * @return array|bool + */ + public function getRevPic() { + if (isset($this->_receive['PicUrl'])) { + return array( + 'mediaid' => $this->_receive['MediaId'], + 'picurl' => (string) $this->_receive['PicUrl'], //闃叉picurl涓虹┖瀵艰嚧瑙f瀽鍑洪敊 + ); + } else { + return false; + } + } + + /** + * 鑾峰彇鎺ユ敹娑堟伅閾炬帴 + * @return bool|array + */ + public function getRevLink() { + if (isset($this->_receive['Url'])) { + return array( + 'url' => $this->_receive['Url'], + 'title' => $this->_receive['Title'], + 'description' => $this->_receive['Description'] + ); + } else { + return false; + } + } + + /** + * 鑾峰彇鎺ユ敹鍦扮悊浣嶇疆 + * @return bool|array + */ + public function getRevGeo() { + if (isset($this->_receive['Location_X'])) { + return array( + 'x' => $this->_receive['Location_X'], + 'y' => $this->_receive['Location_Y'], + 'scale' => $this->_receive['Scale'], + 'label' => $this->_receive['Label'] + ); + } else { + return false; + } + } + + /** + * 鑾峰彇涓婃姤鍦扮悊浣嶇疆浜嬩欢 + * @return bool|array + */ + public function getRevEventGeo() { + if (isset($this->_receive['Latitude'])) { + return array( + 'x' => $this->_receive['Latitude'], + 'y' => $this->_receive['Longitude'], + 'precision' => $this->_receive['Precision'], + ); + } + return false; + } + + /** + * 鑾峰彇鎺ユ敹浜嬩欢鎺ㄩ + * @return bool|array + */ + public function getRevEvent() { + if (isset($this->_receive['Event'])) { + $array['event'] = $this->_receive['Event']; + } + if (isset($this->_receive['EventKey'])) { + $array['key'] = $this->_receive['EventKey']; + } + if (isset($array) && count($array) > 0) { + return $array; + } + return false; + } + + /** + * 鑾峰彇鑷畾涔夎彍鍗曠殑鎵爜鎺ㄤ簨浠朵俊鎭 + * + * 浜嬩欢绫诲瀷涓轰互涓嬩袱绉嶆椂鍒欒皟鐢ㄦ鏂规硶鏈夋晥 + * Event 浜嬩欢绫诲瀷锛宻cancode_push + * Event 浜嬩欢绫诲瀷锛宻cancode_waitmsg + * @return bool|array + */ + public function getRevScanInfo() { + if (isset($this->_receive['ScanCodeInfo'])) { + if (!is_array($this->_receive['ScanCodeInfo'])) { + $array = (array) $this->_receive['ScanCodeInfo']; + $this->_receive['ScanCodeInfo'] = $array; + } else { + $array = $this->_receive['ScanCodeInfo']; + } + } + if (isset($array) && count($array) > 0) { + return $array; + } + return false; + } + + /** + * 鑾峰彇鑷畾涔夎彍鍗曠殑鍥剧墖鍙戦佷簨浠朵俊鎭 + * + * 浜嬩欢绫诲瀷涓轰互涓嬩笁绉嶆椂鍒欒皟鐢ㄦ鏂规硶鏈夋晥 + * Event 浜嬩欢绫诲瀷锛宲ic_sysphoto 寮瑰嚭绯荤粺鎷嶇収鍙戝浘鐨勪簨浠舵帹閫 + * Event 浜嬩欢绫诲瀷锛宲ic_photo_or_album 寮瑰嚭鎷嶇収鎴栬呯浉鍐屽彂鍥剧殑浜嬩欢鎺ㄩ + * Event 浜嬩欢绫诲瀷锛宲ic_weixin 寮瑰嚭寰俊鐩稿唽鍙戝浘鍣ㄧ殑浜嬩欢鎺ㄩ + * + * @return bool|array + * array ( + * 'Count' => '2', + * 'PicList' =>array ( + * 'item' =>array ( + * 0 =>array ('PicMd5Sum' => 'aaae42617cf2a14342d96005af53624c'), + * 1 =>array ('PicMd5Sum' => '149bd39e296860a2adc2f1bb81616ff8'), + * ), + * ), + * ) + * + */ + public function getRevSendPicsInfo() { + if (isset($this->_receive['SendPicsInfo'])) { + if (!is_array($this->_receive['SendPicsInfo'])) { + $array = (array) $this->_receive['SendPicsInfo']; + if (isset($array['PicList'])) { + $array['PicList'] = (array) $array['PicList']; + $item = $array['PicList']['item']; + $array['PicList']['item'] = array(); + foreach ($item as $key => $value) { + $array['PicList']['item'][$key] = (array) $value; + } + } + $this->_receive['SendPicsInfo'] = $array; + } else { + $array = $this->_receive['SendPicsInfo']; + } + } + if (isset($array) && count($array) > 0) { + return $array; + } + return false; + } + + /** + * 鑾峰彇鑷畾涔夎彍鍗曠殑鍦扮悊浣嶇疆閫夋嫨鍣ㄤ簨浠舵帹閫 + * + * 浜嬩欢绫诲瀷涓轰互涓嬫椂鍒欏彲浠ヨ皟鐢ㄦ鏂规硶鏈夋晥 + * Event 浜嬩欢绫诲瀷锛宭ocation_select 寮瑰嚭鍦扮悊浣嶇疆閫夋嫨鍣ㄧ殑浜嬩欢鎺ㄩ + * + * @return bool|array + * array ( + * 'Location_X' => '33.731655000061', + * 'Location_Y' => '113.29955200008047', + * 'Scale' => '16', + * 'Label' => '鏌愭煇甯傛煇鏌愬尯鏌愭煇璺', + * 'Poiname' => '', + * ) + * + */ + public function getRevSendGeoInfo() { + if (isset($this->_receive['SendLocationInfo'])) { + if (!is_array($this->_receive['SendLocationInfo'])) { + $array = (array) $this->_receive['SendLocationInfo']; + if (empty($array['Poiname'])) { + $array['Poiname'] = ""; + } + if (empty($array['Label'])) { + $array['Label'] = ""; + } + $this->_receive['SendLocationInfo'] = $array; + } else { + $array = $this->_receive['SendLocationInfo']; + } + } + if (isset($array) && count($array) > 0) { + return $array; + } + return false; + } + + /** + * 鑾峰彇鎺ユ敹璇煶鎺ㄩ + * @return bool|array + */ + public function getRevVoice() { + if (isset($this->_receive['MediaId'])) { + return array( + 'mediaid' => $this->_receive['MediaId'], + 'format' => $this->_receive['Format'], + ); + } + return false; + } + + /** + * 鑾峰彇鎺ユ敹瑙嗛鎺ㄩ + * @return array|bool + */ + public function getRevVideo() { + if (isset($this->_receive['MediaId'])) { + return array( + 'mediaid' => $this->_receive['MediaId'], + 'thumbmediaid' => $this->_receive['ThumbMediaId'] + ); + } + return false; + } + + /** + * 鑾峰彇鎺ユ敹TICKET + * @return bool|string + */ + public function getRevTicket() { + if (isset($this->_receive['Ticket'])) { + return $this->_receive['Ticket']; + } + return false; + } + + /** + * 鑾峰彇浜岀淮鐮佺殑鍦烘櫙鍊 + * @return bool|string + */ + public function getRevSceneId() { + if (isset($this->_receive['EventKey'])) { + return str_replace('qrscene_', '', $this->_receive['EventKey']); + } + return false; + } + + /** + * 鑾峰彇涓诲姩鎺ㄩ佺殑娑堟伅ID + * 缁忚繃楠岃瘉锛岃繖涓拰鏅氱殑娑堟伅MsgId涓嶄竴鏍 + * 褰揈vent涓 MASSSENDJOBFINISH 鎴 TEMPLATESENDJOBFINISH + * @return bool|string + */ + public function getRevTplMsgID() { + if (isset($this->_receive['MsgID'])) { + return $this->_receive['MsgID']; + } + return false; + } + + /** + * 鑾峰彇妯℃澘娑堟伅鍙戦佺姸鎬 + * @return bool|string + */ + public function getRevStatus() { + if (isset($this->_receive['Status'])) { + return $this->_receive['Status']; + } + return false; + } + + /** + * 鑾峰彇缇ゅ彂鎴栨ā鏉挎秷鎭彂閫佺粨鏋 + * 褰揈vent涓 MASSSENDJOBFINISH 鎴 TEMPLATESENDJOBFINISH锛屽嵆楂樼骇缇ゅ彂/妯℃澘娑堟伅 + * @return bool|array + */ + public function getRevResult() { + if (isset($this->_receive['Status'])) { //鍙戦佹槸鍚︽垚鍔燂紝鍏蜂綋鐨勮繑鍥炲艰鍙傝 楂樼骇缇ゅ彂/妯℃澘娑堟伅 鐨勪簨浠舵帹閫佽鏄 + $array['Status'] = $this->_receive['Status']; + } + if (isset($this->_receive['MsgID'])) { //鍙戦佺殑娑堟伅id + $array['MsgID'] = $this->_receive['MsgID']; + } + //浠ヤ笅浠呭綋缇ゅ彂娑堟伅鏃舵墠浼氭湁鐨勪簨浠跺唴瀹 + if (isset($this->_receive['TotalCount'])) { //鍒嗙粍鎴杘penid鍒楄〃鍐呯矇涓濇暟閲 + $array['TotalCount'] = $this->_receive['TotalCount']; + } + if (isset($this->_receive['FilterCount'])) { //杩囨护锛堣繃婊ゆ槸鎸囩壒瀹氬湴鍖恒佹у埆鐨勮繃婊ゃ佺敤鎴疯缃嫆鏀剁殑杩囨护锛岀敤鎴锋帴鏀跺凡瓒4鏉$殑杩囨护锛夊悗锛屽噯澶囧彂閫佺殑绮変笣鏁 + $array['FilterCount'] = $this->_receive['FilterCount']; + } + if (isset($this->_receive['SentCount'])) { //鍙戦佹垚鍔熺殑绮変笣鏁 + $array['SentCount'] = $this->_receive['SentCount']; + } + if (isset($this->_receive['ErrorCount'])) { //鍙戦佸け璐ョ殑绮変笣鏁 + $array['ErrorCount'] = $this->_receive['ErrorCount']; + } + if (isset($array) && count($array) > 0) { + return $array; + } + return false; + } + + /** + * 鑾峰彇澶氬鏈嶄細璇濈姸鎬佹帹閫佷簨浠 - 鎺ュ叆浼氳瘽 + * 褰揈vent涓 kfcreatesession 鍗虫帴鍏ヤ細璇 + * @return bool|string + */ + public function getRevKFCreate() { + if (isset($this->_receive['KfAccount'])) { + return $this->_receive['KfAccount']; + } + return false; + } + + /** + * 鑾峰彇澶氬鏈嶄細璇濈姸鎬佹帹閫佷簨浠 - 鍏抽棴浼氳瘽 + * 褰揈vent涓 kfclosesession 鍗冲叧闂細璇 + * @return bool|string + */ + public function getRevKFClose() { + if (isset($this->_receive['KfAccount'])) { + return $this->_receive['KfAccount']; + } + return false; + } + + /** + * 鑾峰彇澶氬鏈嶄細璇濈姸鎬佹帹閫佷簨浠 - 杞帴浼氳瘽 + * 褰揈vent涓 kfswitchsession 鍗宠浆鎺ヤ細璇 + * @return bool|array + */ + public function getRevKFSwitch() { + if (isset($this->_receive['FromKfAccount'])) { //鍘熸帴鍏ュ鏈 + $array['FromKfAccount'] = $this->_receive['FromKfAccount']; + } + if (isset($this->_receive['ToKfAccount'])) { //杞帴鍒板鏈 + $array['ToKfAccount'] = $this->_receive['ToKfAccount']; + } + if (isset($array) && count($array) > 0) { + return $array; + } + return false; + } + + /** + * 鍙戦佸鏈嶆秷鎭 + * @param array $data 娑堟伅缁撴瀯{"touser":"OPENID","msgtype":"news","news":{...}} + * @return bool|array + */ + public function sendCustomMessage($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::CUSTOM_SEND_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 妯℃澘娑堟伅 璁剧疆鎵灞炶涓 + * @param string $id1 鍏紬鍙锋ā鏉挎秷鎭墍灞炶涓氱紪鍙凤紝鍙傜湅瀹樻柟寮鍙戞枃妗 琛屼笟浠g爜 + * @param string $id2 鍚$id1銆備絾濡傛灉鍙湁涓涓涓氾紝姝ゅ弬鏁板彲鐪佺暐 + * @return bool|mixed + */ + public function setTMIndustry($id1, $id2 = '') { + if ($id1) { + $data['industry_id1'] = $id1; + } + if ($id2) { + $data['industry_id2'] = $id2; + } + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::TEMPLATE_SET_INDUSTRY_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 妯℃澘娑堟伅 娣诲姞娑堟伅妯℃澘 + * 鎴愬姛杩斿洖娑堟伅妯℃澘鐨勮皟鐢╥d + * @param string $tpl_id 妯℃澘搴撲腑妯℃澘鐨勭紪鍙凤紝鏈夆淭M**鈥濆拰鈥淥PENTMTM**鈥濈瓑褰㈠紡 + * @return bool|string + */ + public function addTemplateMessage($tpl_id) { + $data = array('template_id_short' => $tpl_id); + if (!$this->access_token && !$this->getAccessToken()) + return false; + $result = Tools::httpPost(self::API_URL_PREFIX . self::TEMPLATE_ADD_TPL_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json['template_id']; + } + return false; + } + + /** + * 鍙戦佹ā鏉挎秷鎭 + * @param array $data 娑堟伅缁撴瀯 + * { + * "touser":"OPENID", + * "template_id":"ngqIpbwh8bUfcSsECmogfXcV14J0tQlEpBO27izEYtY", + * "url":"http://weixin.qq.com/download", + * "topcolor":"#FF0000", + * "data":{ + * "鍙傛暟鍚1": { + * "value":"鍙傛暟", + * "color":"#173177" //鍙傛暟棰滆壊 + * }, + * "Date":{ + * "value":"06鏈07鏃 19鏃24鍒", + * "color":"#173177" + * }, + * "CardNumber":{ + * "value":"0426", + * "color":"#173177" + * }, + * "Type":{ + * "value":"娑堣垂", + * "color":"#173177" + * } + * } + * } + * @return bool|array + */ + public function sendTemplateMessage($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::TEMPLATE_SEND_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 杞彂澶氬鏈嶆秷鎭 + * @param string $customer_account + * @return $this + */ + public function transfer_customer_service($customer_account = '') { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'CreateTime' => time(), + 'MsgType' => 'transfer_customer_service', + ); + if ($customer_account) { + $msg['TransInfo'] = array('KfAccount' => $customer_account); + } + $this->Message($msg); + return $this; + } + + /** + * 楂樼骇缇ゅ彂娑堟伅, 鏍规嵁OpenID鍒楄〃缇ゅ彂鍥炬枃娑堟伅(璁㈤槄鍙蜂笉鍙敤) + * 娉ㄦ剰锛氳棰戦渶瑕佸湪璋冪敤uploadMedia()鏂规硶鍚庯紝鍐嶄娇鐢 uploadMpVideo() 鏂规硶鐢熸垚锛 + * 鐒跺悗鑾峰緱鐨 mediaid 鎵嶈兘鐢ㄤ簬缇ゅ彂锛屼笖娑堟伅绫诲瀷涓 mpvideo 绫诲瀷銆 + * @param array $data 娑堟伅缁撴瀯 + * { + * "touser"=>array( + * "OPENID1", + * "OPENID2" + * ), + * "msgtype"=>"mpvideo", + * // 鍦ㄤ笅闈5绉嶇被鍨嬩腑閫夋嫨瀵瑰簲鐨勫弬鏁板唴瀹 + * // mpnews | voice | image | mpvideo => array( "media_id"=>"MediaId") + * // text => array ( "content" => "hello") + * } + * @return bool|array + */ + public function sendMassMessage($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MASS_SEND_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 楂樼骇缇ゅ彂娑堟伅, 鏍规嵁缇ょ粍id缇ゅ彂鍥炬枃娑堟伅(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 娉ㄦ剰锛氳棰戦渶瑕佸湪璋冪敤uploadMedia()鏂规硶鍚庯紝鍐嶄娇鐢 uploadMpVideo() 鏂规硶鐢熸垚锛 + * 鐒跺悗鑾峰緱鐨 mediaid 鎵嶈兘鐢ㄤ簬缇ゅ彂锛屼笖娑堟伅绫诲瀷涓 mpvideo 绫诲瀷銆 + * @param array $data 娑堟伅缁撴瀯 + * { + * "filter"=>array( + * "is_to_all"=>False, //鏄惁缇ゅ彂缁欐墍鏈夌敤鎴.True涓嶇敤鍒嗙粍id锛孎alse闇濉啓鍒嗙粍id + * "group_id"=>"2" //缇ゅ彂鐨勫垎缁刬d + * ), + * "msgtype"=>"mpvideo", + * // 鍦ㄤ笅闈5绉嶇被鍨嬩腑閫夋嫨瀵瑰簲鐨勫弬鏁板唴瀹 + * // mpnews | voice | image | mpvideo => array( "media_id"=>"MediaId") + * // text => array ( "content" => "hello") + * } + * @return bool|array + */ + public function sendGroupMassMessage($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MASS_SEND_GROUP_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 楂樼骇缇ゅ彂娑堟伅, 鍒犻櫎缇ゅ彂鍥炬枃娑堟伅(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param string $msg_id 娑堟伅ID + * @return bool + */ + public function deleteMassMessage($msg_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MASS_DELETE_URL . "access_token={$this->access_token}", Tools::json_encode(array('msg_id' => $msg_id))); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return true; + } + return false; + } + + /** + * 楂樼骇缇ゅ彂娑堟伅, 棰勮缇ゅ彂娑堟伅(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * 娉ㄦ剰锛氳棰戦渶瑕佸湪璋冪敤uploadMedia()鏂规硶鍚庯紝鍐嶄娇鐢 uploadMpVideo() 鏂规硶鐢熸垚锛 + * 鐒跺悗鑾峰緱鐨 mediaid 鎵嶈兘鐢ㄤ簬缇ゅ彂锛屼笖娑堟伅绫诲瀷涓 mpvideo 绫诲瀷銆 + * @param type $data + * @娑堟伅缁撴瀯 + * { + * "touser"=>"OPENID", + * "msgtype"=>"mpvideo", + * // 鍦ㄤ笅闈5绉嶇被鍨嬩腑閫夋嫨瀵瑰簲鐨勫弬鏁板唴瀹 + * // mpnews | voice | image | mpvideo => array( "media_id"=>"MediaId") + * // text => array ( "content" => "hello") + * } + * @return bool|array + */ + public function previewMassMessage($data) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MASS_PREVIEW_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 楂樼骇缇ゅ彂娑堟伅, 鏌ヨ缇ゅ彂娑堟伅鍙戦佺姸鎬(璁よ瘉鍚庣殑璁㈤槄鍙峰彲鐢) + * @param string $msg_id 娑堟伅ID + * @return bool|array + * { + * "msg_id":201053012, //缇ゅ彂娑堟伅鍚庤繑鍥炵殑娑堟伅id + * "msg_status":"SEND_SUCCESS" //娑堟伅鍙戦佸悗鐨勭姸鎬侊紝SENDING琛ㄧず姝e湪鍙戦 SEND_SUCCESS琛ㄧず鍙戦佹垚鍔 + * } + */ + public function queryMassMessage($msg_id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpPost(self::API_URL_PREFIX . self::MASS_QUERY_URL . "access_token={$this->access_token}", Tools::json_encode(array('msg_id' => $msg_id))); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 璁剧疆鍙戦佹秷鎭 + * @param string|array $msg 娑堟伅鏁扮粍 + * @param bool $append 鏄惁鍦ㄥ師娑堟伅鏁扮粍杩藉姞 + * @return array + */ + public function Message($msg = '', $append = false) { + if (is_null($msg)) { + $this->_msg = array(); + } elseif (is_array($msg)) { + if ($append) { + $this->_msg = array_merge($this->_msg, $msg); + } else { + $this->_msg = $msg; + } + return $this->_msg; + } + return $this->_msg; + } + + /** + * 璁剧疆鏂囨湰娑堟伅 + * @param string $text 鏂囨湰鍐呭 + * @return $this + */ + public function text($text = '') { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'MsgType' => self::MSGTYPE_TEXT, + 'Content' => $this->_auto_text_filter($text), + 'CreateTime' => time(), + ); + $this->Message($msg); + return $this; + } + + /** + * 璁剧疆鍥剧墖娑堟伅 + * @param string $mediaid 鍥剧墖濯掍綋ID + * @return $this + */ + public function image($mediaid = '') { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'MsgType' => self::MSGTYPE_IMAGE, + 'Image' => array('MediaId' => $mediaid), + 'CreateTime' => time(), + ); + $this->Message($msg); + return $this; + } + + /** + * 璁剧疆璇煶鍥炲娑堟伅 + * @param string $mediaid 璇煶濯掍綋ID + * @return $this + */ + public function voice($mediaid = '') { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'MsgType' => self::MSGTYPE_VOICE, + 'Voice' => array('MediaId' => $mediaid), + 'CreateTime' => time(), + ); + $this->Message($msg); + return $this; + } + + /** + * 璁剧疆瑙嗛鍥炲娑堟伅 + * @param string $mediaid 瑙嗛濯掍綋ID + * @param string $title 瑙嗛鏍囬 + * @param string $description 瑙嗛鎻忚堪 + * @return $this + */ + public function video($mediaid = '', $title = '', $description = '') { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'MsgType' => self::MSGTYPE_VIDEO, + 'Video' => array( + 'MediaId' => $mediaid, + 'Title' => $title, + 'Description' => $description + ), + 'CreateTime' => time(), + ); + $this->Message($msg); + return $this; + } + + /** + * 璁剧疆闊充箰鍥炲娑堟伅 + * @param string $title 闊充箰鏍囬 + * @param string $desc 闊充箰鎻忚堪 + * @param string $musicurl 闊充箰鍦板潃 + * @param string $hgmusicurl 楂樻竻闊充箰鍦板潃 + * @param string $thumbmediaid 闊充箰鍥剧墖缂╃暐鍥剧殑濯掍綋id锛堝彲閫夛級 + * @return $this + */ + public function music($title, $desc, $musicurl, $hgmusicurl = '', $thumbmediaid = '') { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'CreateTime' => time(), + 'MsgType' => self::MSGTYPE_MUSIC, + 'Music' => array( + 'Title' => $title, + 'Description' => $desc, + 'MusicUrl' => $musicurl, + 'HQMusicUrl' => $hgmusicurl + ), + ); + if ($thumbmediaid) { + $msg['Music']['ThumbMediaId'] = $thumbmediaid; + } + $this->Message($msg); + return $this; + } + + /** + * 璁剧疆鍥炲鍥炬枃 + * @param array $newsData + * @return $this + */ + public function news($newsData = array()) { + $msg = array( + 'ToUserName' => $this->getRevFrom(), + 'FromUserName' => $this->getRevTo(), + 'CreateTime' => time(), + 'MsgType' => self::MSGTYPE_NEWS, + 'ArticleCount' => count($newsData), + 'Articles' => $newsData, + ); + $this->Message($msg); + return $this; + } + + /** + * 鍥炲寰俊鏈嶅姟鍣 + * @param array $msg 瑕佸彂閫佺殑淇℃伅锛堥粯璁ゅ彇$this->_msg锛 + * @param bool $return 鏄惁杩斿洖淇℃伅鑰屼笉鎶涘嚭鍒版祻瑙堝櫒锛堥粯璁:鍚︼級 + * @return bool|string + */ + public function reply($msg = array(), $return = false) { + if (empty($msg)) { + if (empty($this->_msg)) { //闃叉涓嶅厛璁剧疆鍥炲鍐呭锛岀洿鎺ヨ皟鐢╮eply鏂规硶瀵艰嚧寮傚父 + return false; + } + $msg = $this->_msg; + } + $xmldata = Tools::arr2xml($msg); + if ($this->encrypt_type == 'aes') { //濡傛灉鏉ユ簮娑堟伅涓哄姞瀵嗘柟寮 + !class_exists('Prpcrypt', FALSE) && require __DIR__ . '/Lib/Prpcrypt.php'; + $pc = new Prpcrypt($this->encodingAesKey); + // 濡傛灉鏄涓夋柟骞冲彴锛屽姞瀵嗗緱浣跨敤 component_appid + $array = $pc->encrypt($xmldata, empty($this->config['component_appid']) ? $this->appid : $this->config['component_appid']); + $ret = $array[0]; + if ($ret != 0) { + Tools::log('encrypt err!'); + return false; + } + $timestamp = time(); + $nonce = rand(77, 999) * rand(605, 888) * rand(11, 99); + $encrypt = $array[1]; + $tmpArr = array($this->token, $timestamp, $nonce, $encrypt); //姣旀櫘閫氬叕浼楀钩鍙板浜嗕竴涓姞瀵嗙殑瀵嗘枃 + sort($tmpArr, SORT_STRING); + $signature = sha1(implode($tmpArr)); + $format = "%s"; + $xmldata = sprintf($format, $encrypt, $signature, $timestamp, $nonce); + } + if ($return) { + return $xmldata; + } + echo $xmldata; + } + + /** + * 杩囨护鏂囧瓧鍥炲\r\n鎹㈣绗 + * @param string $text + * @return string + */ + private function _auto_text_filter($text) { + if (!$this->_text_filter) { + return $text; + } + return str_replace("\r\n", "\n", $text); + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatScript.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatScript.php new file mode 100644 index 000000000..fa2522ab8 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatScript.php @@ -0,0 +1,120 @@ + + * @date 2016/06/28 11:24 + */ +class WechatScript extends Common { + + /** + * JSAPI鎺堟潈TICKET + * @var string + */ + public $jsapi_ticket; + + /** + * 鍒犻櫎JSAPI鎺堟潈TICKET + * @param string $appid + * @return bool + */ + public function resetJsTicket($appid = '') { + $this->jsapi_ticket = ''; + $authname = 'wechat_jsapi_ticket_' . empty($appid) ? $this->appid : $appid; + Tools::removeCache($authname); + return true; + } + + /** + * 鑾峰彇JSAPI鎺堟潈TICKET + * @param string $appid 鐢ㄤ簬澶氫釜appid鏃朵娇鐢,鍙┖ + * @param string $jsapi_ticket 鎵嬪姩鎸囧畾jsapi_ticket锛岄潪蹇呰鎯呭喌涓嶅缓璁敤 + * @param string $access_token 鑾峰彇 jsapi_ticket 鎸囧畾 access_token + * @return bool|string + */ + public function getJsTicket($appid = '', $jsapi_ticket = '', $access_token = '') { + if (empty($access_token)) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $access_token = $this->access_token; + } + if (empty($appid)) { + $appid = $this->appid; + } + # 鎵嬪姩鎸囧畾token锛屼紭鍏堜娇鐢 + if ($jsapi_ticket) { + $this->jsapi_ticket = $jsapi_ticket; + return $this->jsapi_ticket; + } + # 灏濊瘯浠庣紦瀛樹腑璇诲彇 + $cache = 'wechat_jsapi_ticket_' . $appid; + $jt = Tools::getCache($cache); + if ($jt) { + return $this->jsapi_ticket = $jt; + } + # 妫娴嬩簨浠舵敞鍐 + if (isset(Loader::$callback[__FUNCTION__])) { + return $this->jsapi_ticket = call_user_func_array(Loader::$callback[__FUNCTION__], array(&$this, &$cache)); + } + # 璋冩帴鍙h幏鍙 + $result = Tools::httpGet(self::API_URL_PREFIX . self::GET_TICKET_URL . "access_token={$access_token}" . '&type=jsapi'); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + $this->jsapi_ticket = $json['ticket']; + Tools::setCache($cache, $this->jsapi_ticket, $json['expires_in'] ? intval($json['expires_in']) - 100 : 3600); + return $this->jsapi_ticket; + } + return false; + } + + /** + * 鑾峰彇JsApi浣跨敤绛惧悕 + * @param string $url 缃戦〉鐨刄RL锛岃嚜鍔ㄥ鐞#鍙婂叾鍚庨潰閮ㄥ垎 + * @param int $timestamp 褰撳墠鏃堕棿鎴 (涓虹┖鍒欒嚜鍔ㄧ敓鎴) + * @param string $noncestr 闅忔満涓 (涓虹┖鍒欒嚜鍔ㄧ敓鎴) + * @param string $appid 鐢ㄤ簬澶氫釜appid鏃朵娇鐢,鍙┖ + * @param string $access_token 鑾峰彇 jsapi_ticket 鎸囧畾 access_token + * @return array|bool 杩斿洖绛惧悕瀛椾覆 + */ + public function getJsSign($url, $timestamp = 0, $noncestr = '', $appid = '', $access_token = '') { + if (!$this->jsapi_ticket && !$this->getJsTicket($appid, '', $access_token) || empty($url)) { + return false; + } + $data = array( + "jsapi_ticket" => $this->jsapi_ticket, + "timestamp" => empty($timestamp) ? time() : $timestamp, + "noncestr" => '' . empty($noncestr) ? Tools::createNoncestr(16) : $noncestr, + "url" => trim($url), + ); + return array( + "url" => $url, + 'debug' => false, + "appId" => empty($appid) ? $this->appid : $appid, + "nonceStr" => $data['noncestr'], + "timestamp" => $data['timestamp'], + "signature" => Tools::getSignature($data, 'sha1'), + 'jsApiList' => array( + 'onMenuShareTimeline', 'onMenuShareAppMessage', 'onMenuShareQQ', 'onMenuShareWeibo', 'onMenuShareQZone', + 'hideOptionMenu', 'showOptionMenu', 'hideMenuItems', 'showMenuItems', 'hideAllNonBaseMenuItem', 'showAllNonBaseMenuItem', + 'chooseImage', 'previewImage', 'uploadImage', 'downloadImage', 'closeWindow', 'scanQRCode', 'chooseWXPay', + 'translateVoice', 'getNetworkType', 'openLocation', 'getLocation', + 'openProductSpecificView', 'addCard', 'chooseCard', 'openCard', + 'startRecord', 'stopRecord', 'onVoiceRecordEnd', 'playVoice', 'pauseVoice', 'stopVoice', 'onVoicePlayEnd', 'uploadVoice', 'downloadVoice', + ) + ); + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatService.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatService.php new file mode 100644 index 000000000..eea5556ef --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatService.php @@ -0,0 +1,382 @@ + + * @date 2016/10/18 00:35:55 + */ +class WechatService { + + const URL_PREFIX = 'https://api.weixin.qq.com/cgi-bin/component'; + // 鑾峰彇鏈嶅姟access_token + const COMPONENT_TOKEN_URL = '/api_component_token'; + // 鑾峰彇锛堝埛鏂帮級鎺堟潈鍏紬鍙风殑浠ょ墝 + const REFRESH_ACCESS_TOKEN = '/api_authorizer_token'; + // 鑾峰彇棰勬巿鏉冪爜 + const PREAUTH_CODE_URL = '/api_create_preauthcode'; + // 鑾峰彇鍏紬鍙风殑鎺堟潈淇℃伅 + const QUERY_AUTH_URL = '/api_query_auth'; + // 鑾峰彇鎺堟潈鏂圭殑璐︽埛淇℃伅 + const GET_AUTHORIZER_INFO_URL = '/api_get_authorizer_info'; + // 鑾峰彇鎺堟潈鏂圭殑閫夐」璁剧疆淇℃伅 + const GET_AUTHORIZER_OPTION_URL = '/api_get_authorizer_option'; + // 璁剧疆鎺堟潈鏂圭殑閫夐」淇℃伅 + const SET_AUTHORIZER_OPTION_URL = '/api_set_authorizer_option'; + + // 寰俊鍚庡彴鎺ㄩ佺殑ticket 姣忓崄鍒嗛挓鏇存柊涓娆 + public $errCode; + // 鏈嶅姟appid + public $errMsg; + // 鏈嶅姟appsecret + protected $component_verify_ticket; + // 鍏紬鍙锋秷鎭牎楠孴oken + protected $component_appid; + // 鍏紬鍙锋秷鎭姞瑙e瘑Key + protected $component_appsecret; + // 鏈嶅姟浠ょ墝 + protected $component_token; + // 鎺堟潈鏂筧ppid + protected $component_encodingaeskey; + // 鎺堟潈鏂逛护鐗 + protected $component_access_token; + // 鍒锋柊浠ょ墝 + protected $authorizer_appid; + // JSON鏁版嵁 + protected $pre_auth_code; + // 閿欒娑堟伅 + protected $data; + + /** + * WechatService constructor. + * @param array $options + */ + public function __construct($options = array()) { + $options = Loader::config($options); + $this->component_encodingaeskey = !empty($options['component_encodingaeskey']) ? $options['component_encodingaeskey'] : ''; + $this->component_verify_ticket = !empty($options['component_verify_ticket']) ? $options['component_verify_ticket'] : ''; + $this->component_appsecret = !empty($options['component_appsecret']) ? $options['component_appsecret'] : ''; + $this->component_token = !empty($options['component_token']) ? $options['component_token'] : ''; + $this->component_appid = !empty($options['component_appid']) ? $options['component_appid'] : ''; + } + + /** + * 鎺ユ敹鍏紬骞冲彴鎺ㄩ佺殑 Ticket + * @return bool|array + */ + public function getComonentTicket() { + $receive = new WechatReceive(array( + 'appid' => $this->component_appid, + 'appsecret' => $this->component_appsecret, + 'encodingaeskey' => $this->component_encodingaeskey, + 'token' => $this->component_token, + 'cachepath' => Cache::$cachepath + )); + # 浼氳瘽鍐呭瑙e瘑鐘舵佸垽鏂 + if (false === $receive->valid()) { + $this->errCode = $receive->errCode; + $this->errMsg = $receive->errMsg; + Tools::log("Get Wechat Push ComponentVerifyTicket Faild. {$this->errMsg} [$this->errCode]", 'Err'); + return false; + } + $data = $receive->getRev()->getRevData(); + if ($data['InfoType'] === 'component_verify_ticket' && !empty($data['ComponentVerifyTicket'])) { + # 璁板綍鎺ㄩ佹棩蹇楀埌寰俊SDK + Tools::log("Get Wechat Push ComponentVerifyTicket Success. "); + Tools::setCache('component_verify_ticket', $data['ComponentVerifyTicket']); + } + return $data; + } + + /** + * 鑾峰彇锛堝埛鏂帮級鎺堟潈鍏紬鍙风殑浠ょ墝 + * @娉ㄦ剰1. 鎺堟潈鍏紬鍙疯闂產ccess token2灏忔椂鏈夋晥 + * @娉ㄦ剰2. 涓瀹氫繚瀛樺ソ鏂扮殑鍒锋柊浠ょ墝 + * @param string $authorizer_appid 鎺堟潈鏂笰PPID + * @param string $authorizer_refresh_token 鎺堟潈鏂瑰埛鏂颁护鐗 + * @return bool|string + */ + public function refreshAccessToken($authorizer_appid, $authorizer_refresh_token) { + empty($this->component_access_token) && $this->getComponentAccessToken(); + if (empty($this->component_access_token)) { + return false; + } + $data = array(); + $data['component_appid'] = $this->component_appid; + $data['authorizer_appid'] = $authorizer_appid; + $data['authorizer_refresh_token'] = $authorizer_refresh_token; + $url = self::URL_PREFIX . self::REFRESH_ACCESS_TOKEN . "?component_access_token={$this->component_access_token}"; + $result = Tools::httpPost($url, Tools::json_encode($data)); + if (($result = $this->_decode($result)) === false) { + Tools::log("Get getAuthorizerOption Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + } + return $result; + } + + /** + * 鑾峰彇鎴栧埛鏂版湇鍔 AccessToken + * @return bool|string + */ + public function getComponentAccessToken() { + $cacheKey = 'wechat_component_access_token'; + $this->component_access_token = Tools::getCache($cacheKey); + if (empty($this->component_access_token)) { + $data = array(); + $data['component_appid'] = $this->component_appid; + $data['component_appsecret'] = $this->component_appsecret; + $data['component_verify_ticket'] = $this->component_verify_ticket; + $url = self::URL_PREFIX . self::COMPONENT_TOKEN_URL; + $result = Tools::httpPost($url, Tools::json_encode($data)); + if (($this->component_access_token = $this->_decode($result, 'component_access_token')) === false) { + Tools::log("Get getComponentAccessToken Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + return false; + } + Tools::setCache($cacheKey, $this->component_access_token, 7200); + } + return $this->component_access_token; + } + + /** + * 瑙f瀽JSON鏁版嵁 + * @param string $result + * @param string|null $field + * @return bool|array + */ + private function _decode($result, $field = null) { + $this->data = json_decode($result, true); + if (!empty($this->data['errcode'])) { + $this->errCode = $this->data['errcode']; + $this->errMsg = $this->data['errmsg']; + return false; + } + if ($this->data && !is_null($field)) { + if (isset($this->data[$field])) { + return $this->data[$field]; + } else { + return false; + } + } + return $this->data; + } + + /** + * 鑾峰彇鍏紬鍙风殑鎺堟潈淇℃伅 + * + * @param string $authorization_code + * @return bool|array + */ + public function getAuthorizationInfo($authorization_code) { + empty($this->component_access_token) && $this->getComponentAccessToken(); + if (empty($this->component_access_token)) { + return false; + } + $data = array(); + $data['component_appid'] = $this->component_appid; + $data['authorization_code'] = $authorization_code; + $url = self::URL_PREFIX . self::QUERY_AUTH_URL . "?component_access_token={$this->component_access_token}"; + $result = Tools::httpPost($url, Tools::json_encode($data)); + $authorization_info = $this->_decode($result, 'authorization_info'); + if (empty($authorization_info)) { + Tools::log("Get getAuthorizationInfo Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + return false; + } + $authorization_info['func_info'] = $this->_parseFuncInfo($authorization_info['func_info']); + return $authorization_info; + } + + /** + * 瑙f瀽鎺堟潈淇℃伅锛岃繑鍥炰互閫楀彿鍒嗗壊鐨勬暟鎹 + * @param array $func_info + * @return string + */ + private function _parseFuncInfo($func_info) { + $authorization_list = array(); + foreach ($func_info as $func) { + foreach ($func as $f) { + $authorization_list[] = $f['id']; + } + } + return join($authorization_list, ','); + } + + /** + * 鑾峰彇鎺堟潈鏂圭殑璐︽埛淇℃伅 + * @param string $authorizer_appid + * @return bool + */ + public function getWechatInfo($authorizer_appid) { + empty($this->component_access_token) && $this->getComponentAccessToken(); + $data = array(); + $data['component_access_token'] = $this->component_access_token; + $data['component_appid'] = $this->component_appid; + $data['authorizer_appid'] = $authorizer_appid; + $url = self::URL_PREFIX . self::GET_AUTHORIZER_INFO_URL . "?component_access_token={$this->component_access_token}"; + $result = Tools::httpPost($url, Tools::json_encode($data)); + $authorizer_info = $this->_decode($result, 'authorizer_info'); + if (empty($authorizer_info)) { + Tools::log("Get WechatInfo Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + return false; + } + $author_data = array_merge($authorizer_info, $this->data['authorization_info']); + $author_data['service_type_info'] = $author_data['service_type_info']['id']; + $author_data['verify_type_info'] = $author_data['verify_type_info']['id']; + $author_data['func_info'] = $this->_parseFuncInfo($author_data['func_info']); + $author_data['business_info'] = json_encode($author_data['business_info']); + return $author_data; + } + + /** + * 鑾峰彇鎺堟潈鏂圭殑閫夐」璁剧疆淇℃伅 + * @param string $authorizer_appid + * @param string $option_name + * @return bool + */ + public function getAuthorizerOption($authorizer_appid, $option_name) { + empty($this->component_access_token) && $this->getComponentAccessToken(); + if (empty($this->authorizer_appid)) { + return false; + } + $data = array(); + $data['component_appid'] = $this->component_appid; + $data['authorizer_appid'] = $authorizer_appid; + $data['option_name'] = $option_name; + $url = self::URL_PREFIX . self::GET_AUTHORIZER_OPTION_URL . "?component_access_token={$this->component_access_token}"; + $result = Tools::httpPost($url, Tools::json_encode($data)); + if (($result = $this->_decode($result)) === false) { + Tools::log("Get getAuthorizerOption Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + } + return $result; + } + + /** + * 璁剧疆鎺堟潈鏂圭殑閫夐」淇℃伅 + * @param string $authorizer_appid + * @param string $option_name + * @param string $option_value + * @return bool + */ + public function setAuthorizerOption($authorizer_appid, $option_name, $option_value) { + empty($this->component_access_token) && $this->getComponentAccessToken(); + if (empty($this->authorizer_appid)) { + return false; + } + $data = array(); + $data['component_appid'] = $this->component_appid; + $data['authorizer_appid'] = $authorizer_appid; + $data['option_name'] = $option_name; + $data['option_value'] = $option_value; + $url = self::URL_PREFIX . self::SET_AUTHORIZER_OPTION_URL . "?component_access_token={$this->component_access_token}"; + $result = Tools::httpPost($url, Tools::json_encode($data)); + if (($result = $this->_decode($result)) === false) { + Tools::log("Get setAuthorizerOption Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + } + return $result; + } + + /** + * 鑾峰彇鎺堟潈鍥炶烦鍦板潃 + * @param string $redirect_uri + * @return bool + */ + public function getAuthRedirect($redirect_uri) { + empty($this->pre_auth_code) && $this->getPreauthCode(); + if (empty($this->pre_auth_code)) { + return false; + } + return "https://mp.weixin.qq.com/cgi-bin/componentloginpage?component_appid={$this->component_appid}&pre_auth_code={$this->pre_auth_code}&redirect_uri={$redirect_uri}"; + } + + /** + * 鑾峰彇棰勬巿鏉冪爜 + * + * @return bool|string + */ + public function getPreauthCode() { + empty($this->component_access_token) && $this->getComponentAccessToken(); + if (empty($this->component_access_token)) { + return false; + } + $data = array(); + $data['component_appid'] = $this->component_appid; + $url = self::URL_PREFIX . self::PREAUTH_CODE_URL . "?component_access_token={$this->component_access_token}"; + $result = Tools::httpPost($url, Tools::json_encode($data)); + $this->pre_auth_code = $this->_decode($result, 'pre_auth_code'); + if (empty($this->pre_auth_code)) { + Tools::log("Get getPreauthCode Faild. {$this->errMsg} [$this->errCode]", 'ERR'); + } + return $this->pre_auth_code; + } + + /** + * oauth 鎺堟潈璺宠浆鎺ュ彛 + * @param string $appid + * @param string $redirect_uri + * @param string $scope snsapi_userinfo|snsapi_base + * @return string + */ + public function getOauthRedirect($appid, $redirect_uri, $scope = 'snsapi_userinfo') { + return "https://open.weixin.qq.com/connect/oauth2/authorize?appid={$appid}&redirect_uri=" . urlencode($redirect_uri) + . "&response_type=code&scope={$scope}&state={$appid}&component_appid={$this->component_appid}#wechat_redirect"; + } + + /** + * 閫氳繃code鑾峰彇Access Token + * @param string $appid + * @return bool|array + */ + public function getOauthAccessToken($appid) { + $code = isset($_GET['code']) ? $_GET['code'] : ''; + if (empty($code)) { + return false; + } + empty($this->component_access_token) && $this->getComponentAccessToken(); + if (empty($this->component_access_token)) { + return false; + } + $url = "https://api.weixin.qq.com/sns/oauth2/component/access_token?" + . "appid={$appid}&code={$code}&" + . "grant_type=authorization_code&" + . "component_appid={$this->component_appid}&" + . "component_access_token={$this->component_access_token}"; + $json = $this->parseJson(Tools::httpGet($url)); + if ($json !== false) { + return $json; + } + return false; + } + + /** + * 瑙f瀽JSON鏁版嵁 + * @param string $result + * @return bool + */ + private function parseJson($result) { + $json = json_decode($result, true); + if (!empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return false; + } + return $json; + } + + /** + * 鑾峰彇鍏虫敞鑰呰缁嗕俊鎭 + * @param string $openid + * @param string $oauthAccessToken + * @return bool|array {subscribe,openid,nickname,sex,city,province,country,language,headimgurl,subscribe_time,[unionid]} + * 娉ㄦ剰锛歶nionid瀛楁 鍙湁鍦ㄧ敤鎴峰皢鍏紬鍙风粦瀹氬埌鍏紬鍙风涓夋柟骞冲彴璐﹀彿鍚庯紝鎵嶄細鍑虹幇銆傚缓璁皟鐢ㄥ墠鐢╥sset()妫娴嬩竴涓 + */ + public function getOauthUserInfo($openid, $oauthAccessToken) { + $url = "https://api.weixin.qq.com/sns/userinfo?access_token={$oauthAccessToken}&openid={$openid}&lang=zh_CN"; + return $this->parseJson(Tools::httpGet($url)); + } + + +} diff --git a/vendor/zoujingli/wechat-php-sdk/Wechat/WechatUser.php b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatUser.php new file mode 100644 index 000000000..2b9c2bc06 --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/Wechat/WechatUser.php @@ -0,0 +1,549 @@ + + * @date 2016/06/28 11:20 + */ +class WechatUser extends Common { + + /** 鑾峰彇绮変笣鍒楄〃 */ + const USER_GET_URL = '/user/get?'; + /* 鑾峰彇绮変笣淇℃伅 */ + const USER_INFO_URL = '/user/info?'; + /* 鏇存柊绮変笣鏍囨敞 */ + const USER_UPDATEREMARK_URL = '/user/info/updateremark?'; + + /** 鍒涘缓鏍囩 */ + const TAGS_CREATE_URL = '/tags/create?'; + /* 鑾峰彇鏍囩鍒楄〃 */ + const TAGS_GET_URL = '/tags/get?'; + /* 鏇存柊鏍囩 */ + const TAGS_UPDATE_URL = '/tags/update?'; + /* 鍒犻櫎鏍囩 */ + const TAGS_DELETE_URL = '/tags/delete?'; + /* 鑾峰彇鏍囩涓嬬殑绮変笣鍒楄〃 */ + const TAGS_GET_USER_URL = '/user/tag/get?'; + /* 鎵归噺涓虹矇涓濇墦鏍囩 */ + const TAGS_MEMBER_BATCHTAGGING = '/tags/members/batchtagging?'; + /* 鎵归噺涓虹矇涓濆彇娑堟爣绛 */ + const TAGS_MEMBER_BATCHUNTAGGING = '/tags/members/batchuntagging?'; + /* 鑾峰彇绮変笣鐨勬爣绛惧垪琛 */ + const TAGS_LIST = '/tags/getidlist?'; + + /** 鑾峰彇鍒嗙粍鍒楄〃 */ + const GROUP_GET_URL = '/groups/get?'; + /* 鑾峰彇绮変笣鎵鍦ㄧ殑鍒嗙粍 */ + const USER_GROUP_URL = '/groups/getid?'; + /* 鍒涘缓鍒嗙粍 */ + const GROUP_CREATE_URL = '/groups/create?'; + /* 鏇存柊鍒嗙粍 */ + const GROUP_UPDATE_URL = '/groups/update?'; + /* 鍒犻櫎鍒嗙粍 */ + const GROUP_DELETE_URL = '/groups/delete?'; + /* 淇敼绮変笣鎵鍦ㄥ垎缁 */ + const GROUP_MEMBER_UPDATE_URL = '/groups/members/update?'; + /* 鎵归噺淇敼绮変笣鎵鍦ㄥ垎缁 */ + const GROUP_MEMBER_BATCHUPDATE_URL = '/groups/members/batchupdate?'; + + /** 鑾峰彇榛戝悕鍗曞垪琛 */ + const BACKLIST_GET_URL = '/tags/members/getblacklist?'; + /* 鎵归噺鎷夐粦绮変笣 */ + const BACKLIST_ADD_URL = '/tags/members/batchblacklist?'; + /* 鎵归噺鍙栨秷鎷夐粦绮変笣 */ + const BACKLIST_DEL_URL = '/tags/members/batchunblacklist?'; + + /** + * 鎵归噺鑾峰彇鍏虫敞绮変笣鍒楄〃 + * @param string $next_openid + * @return bool|array + */ + public function getUserList($next_openid = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::USER_GET_URL . "access_token={$this->access_token}" . '&next_openid=' . $next_openid); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鍏虫敞鑰呰缁嗕俊鎭 + * @param string $openid + * @return bool|array {subscribe,openid,nickname,sex,city,province,country,language,headimgurl,subscribe_time,[unionid]} + * @娉ㄦ剰锛歶nionid瀛楁 鍙湁鍦ㄧ矇涓濆皢鍏紬鍙风粦瀹氬埌寰俊寮鏀惧钩鍙拌处鍙峰悗锛屾墠浼氬嚭鐜般傚缓璁皟鐢ㄥ墠鐢╥sset()妫娴嬩竴涓 + */ + public function getUserInfo($openid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::USER_INFO_URL . "access_token={$this->access_token}&openid={$openid}"); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 璁剧疆绮変笣澶囨敞鍚 + * @param string $openid + * @param string $remark 澶囨敞鍚 + * @return bool|array + */ + public function updateUserRemark($openid, $remark) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid' => $openid, 'remark' => $remark); + $result = Tools::httpPost(self::API_URL_PREFIX . self::USER_UPDATEREMARK_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇绮変笣鍒嗙粍鍒楄〃 + * @return bool|array + */ + public function getGroup() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::GROUP_GET_URL . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍒犻櫎绮変笣鍒嗙粍 + * @param type $id + * @return bool + */ + public function delGroup($id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('group' => array('id' => $id)); + $result = Tools::httpPost(self::API_URL_PREFIX . self::GROUP_DELETE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇绮変笣鎵鍦ㄥ垎缁 + * @param string $openid + * @return bool|int 鎴愬姛鍒欒繑鍥炵矇涓濆垎缁刬d + */ + public function getUserGroup($openid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid' => $openid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::USER_GROUP_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } else if (isset($json['groupid'])) { + return $json['groupid']; + } + } + return false; + } + + /** + * 鏂板鑷畾鍒嗙粍 + * @param string $name 鍒嗙粍鍚嶇О + * @return bool|array + */ + public function createGroup($name) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('group' => array('name' => $name)); + $result = Tools::httpPost(self::API_URL_PREFIX . self::GROUP_CREATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏇存敼鍒嗙粍鍚嶇О + * @param int $groupid 鍒嗙粍id + * @param string $name 鍒嗙粍鍚嶇О + * @return bool|array + */ + public function updateGroup($groupid, $name) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('group' => array('id' => $groupid, 'name' => $name)); + $result = Tools::httpPost(self::API_URL_PREFIX . self::GROUP_UPDATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 绉诲姩绮変笣鍒嗙粍 + * @param int $groupid 鍒嗙粍id + * @param string $openid 绮変笣openid + * @return bool|array + */ + public function updateGroupMembers($groupid, $openid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid' => $openid, 'to_groupid' => $groupid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::GROUP_MEMBER_UPDATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鎵归噺绉诲姩绮変笣鍒嗙粍 + * @param string $groupid 鍒嗙粍ID + * @param string $openid_list 绮変笣openid鏁扮粍(涓娆′笉鑳借秴杩50涓) + * @return bool|array + */ + public function batchUpdateGroupMembers($groupid, $openid_list) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid_list' => $openid_list, 'to_groupid' => $groupid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::GROUP_MEMBER_BATCHUPDATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏂板鑷畾鏍囩 + * @param string $name 鏍囩鍚嶇О + * @return bool|array + */ + public function createTags($name) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('tag' => array('name' => $name)); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_CREATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鏇存柊鏍囩 + * @param string $id 鏍囩id + * @param string $name 鏍囩鍚嶇О + * @return bool|array + */ + public function updateTag($id, $name) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('tag' => array('id' => $id, 'name' => $name)); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_UPDATE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇绮変笣鏍囩鍒楄〃 + * @return bool|array + */ + public function getTags() { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $result = Tools::httpGet(self::API_URL_PREFIX . self::TAGS_GET_URL . "access_token={$this->access_token}"); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鍒犻櫎绮変笣鏍囩 + * @param string $id + * @return bool + */ + public function delTag($id) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('tag' => array('id' => $id)); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_DELETE_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇鏍囩涓嬬殑绮変笣鍒楄〃 + * @param string $tagid + * @param string $next_openid + * @return bool + */ + public function getTagUsers($tagid, $next_openid = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('tagid' => $tagid, 'next_openid' => $next_openid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_GET_USER_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鎵归噺涓虹矇涓濇墦鏍囩 + * @param string $tagid 鏍囩ID + * @param array $openid_list 绮変笣openid鏁扮粍锛屼竴娆′笉鑳借秴杩50涓 + * @return bool|array + */ + public function batchAddUserTag($tagid, $openid_list) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid_list' => $openid_list, 'tagid' => $tagid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_MEMBER_BATCHTAGGING . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鎵归噺涓虹矇涓濆彇娑堟爣绛 + * @param string $tagid 鏍囩ID + * @param array $openid_list 绮変笣openid鏁扮粍锛屼竴娆′笉鑳借秴杩50涓 + * @return bool|array + */ + public function batchDeleteUserTag($tagid, $openid_list) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid_list' => $openid_list, 'tagid' => $tagid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_MEMBER_BATCHUNTAGGING . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鑾峰彇绮変笣鐨勬爣绛惧垪琛 + * @param string $openid 绮変笣openid + * @return bool|array + */ + public function getUserTags($openid) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid' => $openid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::TAGS_LIST . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !isset($json['tagid_list']) || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json['tagid_list']; + } + return false; + } + + /** + * 鎵归噺鑾峰彇榛戝悕鍗曠矇涓 + * @param string $begin_openid + * @return bool|array + */ + public function getBacklist($begin_openid = '') { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = empty($begin_openid) ? array() : array('begin_openid' => $begin_openid); + $result = Tools::httpPost(self::API_URL_PREFIX . self::BACKLIST_GET_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (isset($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鎵归噺鎷夐粦绮変笣 + * @param string $openids + * @return bool|array + */ + public function addBacklist($openids) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid_list' => $openids); + $result = Tools::httpPost(self::API_URL_PREFIX . self::BACKLIST_ADD_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + + /** + * 鎵归噺鍙栨秷鎷夐粦绮変笣 + * @param string $openids + * @return bool|array + */ + public function delBacklist($openids) { + if (!$this->access_token && !$this->getAccessToken()) { + return false; + } + $data = array('openid_list' => $openids); + $result = Tools::httpPost(self::API_URL_PREFIX . self::BACKLIST_DEL_URL . "access_token={$this->access_token}", Tools::json_encode($data)); + if ($result) { + $json = json_decode($result, true); + if (!$json || !empty($json['errcode'])) { + $this->errCode = $json['errcode']; + $this->errMsg = $json['errmsg']; + return $this->checkRetry(__FUNCTION__, func_get_args()); + } + return $json; + } + return false; + } + +} diff --git a/vendor/zoujingli/wechat-php-sdk/composer.json b/vendor/zoujingli/wechat-php-sdk/composer.json new file mode 100644 index 000000000..b3f9dfd1b --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/composer.json @@ -0,0 +1,18 @@ +{ + "type": "project", + "name": "zoujingli/wechat-php-sdk", + "homepage": "http://www.kancloud.cn/zoujingli/wechat-php-sdk", + "description": "WeChat development of SDK", + "license": "MIT", + "keywords": [ + "wechat-php-sdk" + ], + "require": { + "php": ">=5.3.3" + }, + "autoload": { + "psr-4": { + "Wechat\\": "./Wechat" + } + } +} diff --git a/vendor/zoujingli/wechat-php-sdk/include.php b/vendor/zoujingli/wechat-php-sdk/include.php new file mode 100644 index 000000000..b870b555e --- /dev/null +++ b/vendor/zoujingli/wechat-php-sdk/include.php @@ -0,0 +1,3 @@ + '', + 'appid' => '', + 'appsecret' => '', + 'encodingaeskey' => '', +); + +# 鍔犺浇瀵瑰簲鎿嶄綔鎺ュ彛 +$wechat = &\Wechat\Loader::get('User', $config); +$userlist = $wechat->getUserList(); + +var_dump($userlist); +var_dump($wechat->errMsg); +var_dump($wechat->errCode); + +exit; + +// 绗笁鏂瑰钩鍙 JSSDK 绛惧悕鍖 + +$wechat = Db::table('wechat_config')->where('authorizer_appid', 'wx60a43dd8161666d4')->find(); +// 绗笁鏂规巿鏉冭幏鍙栧埌鐨 Access_token +$access_token = $wechat['authorizer_access_token']; +// 鍙備笌鎺堟潈鐨勫叕浼楀彿 APPID +$authorizer_appid = $wechat['authorizer_appid']; +// 褰撳墠寰俊椤甸潰URL鍦板潃锛堝畬鏁达級 +$current_url = url('', '', true, true); +// 瀹炰緥SDK鑴氭湰 +$script = load_wechat('Script', $authorizer_appid); +// 鑾峰彇JS绛惧悕鍖 +$result = $script->getJsSign($current_url, 0, '', $authorizer_appid, $access_token); +dump($result); +dump([$script->errMsg, $script->errCode]); + +$json = json_encode($result, JSON_PRETTY_PRINT); +echo ''; +echo " + +"; \ No newline at end of file