123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528 |
- <?php
- /**
- * This file is part of webman.
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the MIT-LICENSE.txt
- * Redistributions of files must retain the above copyright notice.
- *
- * @author walkor<walkor@workerman.net>
- * @copyright walkor<walkor@workerman.net>
- * @link http://www.workerman.net/
- * @license http://www.opensource.org/licenses/mit-license.php MIT License
- */
- use support\Container;
- use support\Request;
- use support\Response;
- use support\Translation;
- use support\view\Blade;
- use support\view\Raw;
- use support\view\ThinkPHP;
- use support\view\Twig;
- use Twig\Error\LoaderError;
- use Twig\Error\RuntimeError;
- use Twig\Error\SyntaxError;
- use Webman\App;
- use Webman\Config;
- use Webman\Route;
- use Workerman\Protocols\Http\Session;
- use Workerman\Worker;
- // Project base path
- define('BASE_PATH', dirname(__DIR__));
- /**
- * return the program execute directory
- * @param string $path
- * @return string
- */
- function run_path(string $path = ''): string
- {
- static $runPath = '';
- if (!$runPath) {
- $runPath = is_phar() ? dirname(Phar::running(false)) : BASE_PATH;
- }
- return path_combine($runPath, $path);
- }
- /**
- * if the param $path equal false,will return this program current execute directory
- * @param string|false $path
- * @return string
- */
- function base_path($path = ''): string
- {
- if (false === $path) {
- return run_path();
- }
- return path_combine(BASE_PATH, $path);
- }
- /**
- * App path
- * @param string $path
- * @return string
- */
- function app_path(string $path = ''): string
- {
- return path_combine(BASE_PATH . DIRECTORY_SEPARATOR . 'app', $path);
- }
- /**
- * Public path
- * @param string $path
- * @return string
- */
- function public_path(string $path = ''): string
- {
- static $publicPath = '';
- if (!$publicPath) {
- $publicPath = \config('app.public_path') ?: run_path('public');
- }
- return path_combine($publicPath, $path);
- }
- /**
- * Config path
- * @param string $path
- * @return string
- */
- function config_path(string $path = ''): string
- {
- return path_combine(BASE_PATH . DIRECTORY_SEPARATOR . 'config', $path);
- }
- /**
- * Runtime path
- * @param string $path
- * @return string
- */
- function runtime_path(string $path = ''): string
- {
- static $runtimePath = '';
- if (!$runtimePath) {
- $runtimePath = \config('app.runtime_path') ?: run_path('runtime');
- }
- return path_combine($runtimePath, $path);
- }
- /**
- * Generate paths based on given information
- * @param string $front
- * @param string $back
- * @return string
- */
- function path_combine(string $front, string $back): string
- {
- return $front . ($back ? (DIRECTORY_SEPARATOR . ltrim($back, DIRECTORY_SEPARATOR)) : $back);
- }
- /**
- * Response
- * @param int $status
- * @param array $headers
- * @param string $body
- * @return Response
- */
- function response(string $body = '', int $status = 200, array $headers = []): Response
- {
- return new Response($status, $headers, $body);
- }
- /**
- * Json response
- * @param $data
- * @param int $options
- * @return Response
- */
- function json($data, int $options = JSON_UNESCAPED_UNICODE): Response
- {
- return new Response(200, ['Content-Type' => 'application/json'], json_encode($data, $options));
- }
- /**
- * Xml response
- * @param $xml
- * @return Response
- */
- function xml($xml): Response
- {
- if ($xml instanceof SimpleXMLElement) {
- $xml = $xml->asXML();
- }
- return new Response(200, ['Content-Type' => 'text/xml'], $xml);
- }
- /**
- * Jsonp response
- * @param $data
- * @param string $callbackName
- * @return Response
- */
- function jsonp($data, string $callbackName = 'callback'): Response
- {
- if (!is_scalar($data) && null !== $data) {
- $data = json_encode($data);
- }
- return new Response(200, [], "$callbackName($data)");
- }
- /**
- * Redirect response
- * @param string $location
- * @param int $status
- * @param array $headers
- * @return Response
- */
- function redirect(string $location, int $status = 302, array $headers = []): Response
- {
- $response = new Response($status, ['Location' => $location]);
- if (!empty($headers)) {
- $response->withHeaders($headers);
- }
- return $response;
- }
- /**
- * View response
- * @param string $template
- * @param array $vars
- * @param string|null $app
- * @param string|null $plugin
- * @return Response
- */
- function view(string $template, array $vars = [], string $app = null, string $plugin = null): Response
- {
- $request = \request();
- $plugin = $plugin === null ? ($request->plugin ?? '') : $plugin;
- $handler = \config($plugin ? "plugin.$plugin.view.handler" : 'view.handler');
- return new Response(200, [], $handler::render($template, $vars, $app, $plugin));
- }
- /**
- * Raw view response
- * @param string $template
- * @param array $vars
- * @param string|null $app
- * @return Response
- * @throws Throwable
- */
- function raw_view(string $template, array $vars = [], string $app = null): Response
- {
- return new Response(200, [], Raw::render($template, $vars, $app));
- }
- /**
- * Blade view response
- * @param string $template
- * @param array $vars
- * @param string|null $app
- * @return Response
- */
- function blade_view(string $template, array $vars = [], string $app = null): Response
- {
- return new Response(200, [], Blade::render($template, $vars, $app));
- }
- /**
- * Think view response
- * @param string $template
- * @param array $vars
- * @param string|null $app
- * @return Response
- */
- function think_view(string $template, array $vars = [], string $app = null): Response
- {
- return new Response(200, [], ThinkPHP::render($template, $vars, $app));
- }
- /**
- * Twig view response
- * @param string $template
- * @param array $vars
- * @param string|null $app
- * @return Response
- * @throws LoaderError
- * @throws RuntimeError
- * @throws SyntaxError
- */
- function twig_view(string $template, array $vars = [], string $app = null): Response
- {
- return new Response(200, [], Twig::render($template, $vars, $app));
- }
- /**
- * Get request
- * @return \Webman\Http\Request|Request|null
- */
- function request()
- {
- return App::request();
- }
- /**
- * Get config
- * @param string|null $key
- * @param $default
- * @return array|mixed|null
- */
- function config(string $key = null, $default = null)
- {
- return Config::get($key, $default);
- }
- /**
- * Create url
- * @param string $name
- * @param ...$parameters
- * @return string
- */
- function route(string $name, ...$parameters): string
- {
- $route = Route::getByName($name);
- if (!$route) {
- return '';
- }
- if (!$parameters) {
- return $route->url();
- }
- if (is_array(current($parameters))) {
- $parameters = current($parameters);
- }
- return $route->url($parameters);
- }
- /**
- * Session
- * @param mixed $key
- * @param mixed $default
- * @return mixed|bool|Session
- */
- function session($key = null, $default = null)
- {
- $session = \request()->session();
- if (null === $key) {
- return $session;
- }
- if (is_array($key)) {
- $session->put($key);
- return null;
- }
- if (strpos($key, '.')) {
- $keyArray = explode('.', $key);
- $value = $session->all();
- foreach ($keyArray as $index) {
- if (!isset($value[$index])) {
- return $default;
- }
- $value = $value[$index];
- }
- return $value;
- }
- return $session->get($key, $default);
- }
- /**
- * Translation
- * @param string $id
- * @param array $parameters
- * @param string|null $domain
- * @param string|null $locale
- * @return string
- */
- function trans(string $id, array $parameters = [], string $domain = null, string $locale = null): string
- {
- $res = Translation::trans($id, $parameters, $domain, $locale);
- return $res === '' ? $id : $res;
- }
- /**
- * Locale
- * @param string|null $locale
- * @return string
- */
- function locale(string $locale = null): string
- {
- if (!$locale) {
- return Translation::getLocale();
- }
- Translation::setLocale($locale);
- return $locale;
- }
- /**
- * 404 not found
- * @return Response
- */
- function not_found(): Response
- {
- return new Response(404, [], file_get_contents(public_path() . '/404.html'));
- }
- /**
- * Copy dir
- * @param string $source
- * @param string $dest
- * @param bool $overwrite
- * @return void
- */
- function copy_dir(string $source, string $dest, bool $overwrite = false)
- {
- if (is_dir($source)) {
- if (!is_dir($dest)) {
- mkdir($dest);
- }
- $files = scandir($source);
- foreach ($files as $file) {
- if ($file !== "." && $file !== "..") {
- copy_dir("$source/$file", "$dest/$file", $overwrite);
- }
- }
- } else if (file_exists($source) && ($overwrite || !file_exists($dest))) {
- copy($source, $dest);
- }
- }
- /**
- * Remove dir
- * @param string $dir
- * @return bool
- */
- function remove_dir(string $dir): bool
- {
- if (is_link($dir) || is_file($dir)) {
- return unlink($dir);
- }
- $files = array_diff(scandir($dir), array('.', '..'));
- foreach ($files as $file) {
- (is_dir("$dir/$file") && !is_link($dir)) ? remove_dir("$dir/$file") : unlink("$dir/$file");
- }
- return rmdir($dir);
- }
- /**
- * Bind worker
- * @param $worker
- * @param $class
- */
- function worker_bind($worker, $class)
- {
- $callbackMap = [
- 'onConnect',
- 'onMessage',
- 'onClose',
- 'onError',
- 'onBufferFull',
- 'onBufferDrain',
- 'onWorkerStop',
- 'onWebSocketConnect',
- 'onWorkerReload'
- ];
- foreach ($callbackMap as $name) {
- if (method_exists($class, $name)) {
- $worker->$name = [$class, $name];
- }
- }
- if (method_exists($class, 'onWorkerStart')) {
- call_user_func([$class, 'onWorkerStart'], $worker);
- }
- }
- /**
- * Start worker
- * @param $processName
- * @param $config
- * @return void
- */
- function worker_start($processName, $config)
- {
- $worker = new Worker($config['listen'] ?? null, $config['context'] ?? []);
- $propertyMap = [
- 'count',
- 'user',
- 'group',
- 'reloadable',
- 'reusePort',
- 'transport',
- 'protocol',
- ];
- $worker->name = $processName;
- foreach ($propertyMap as $property) {
- if (isset($config[$property])) {
- $worker->$property = $config[$property];
- }
- }
- $worker->onWorkerStart = function ($worker) use ($config) {
- require_once base_path('/support/bootstrap.php');
- if (isset($config['handler'])) {
- if (!class_exists($config['handler'])) {
- echo "process error: class {$config['handler']} not exists\r\n";
- return;
- }
- $instance = Container::make($config['handler'], $config['constructor'] ?? []);
- worker_bind($worker, $instance);
- }
- };
- }
- /**
- * Get realpath
- * @param string $filePath
- * @return string
- */
- function get_realpath(string $filePath): string
- {
- if (strpos($filePath, 'phar://') === 0) {
- return $filePath;
- } else {
- return realpath($filePath);
- }
- }
- /**
- * Is phar
- * @return bool
- */
- function is_phar(): bool
- {
- return class_exists(Phar::class, false) && Phar::running();
- }
- /**
- * Get cpu count
- * @return int
- */
- function cpu_count(): int
- {
- // Windows does not support the number of processes setting.
- if (DIRECTORY_SEPARATOR === '\\') {
- return 1;
- }
- $count = 4;
- if (is_callable('shell_exec')) {
- if (strtolower(PHP_OS) === 'darwin') {
- $count = (int)shell_exec('sysctl -n machdep.cpu.core_count');
- } else {
- $count = (int)shell_exec('nproc');
- }
- }
- return $count > 0 ? $count : 4;
- }
- /**
- * Get request parameters, if no parameter name is passed, an array of all values is returned, default values is supported
- * @param string|null $param param's name
- * @param mixed|null $default default value
- * @return mixed|null
- */
- function input(string $param = null, $default = null)
- {
- return is_null($param) ? request()->all() : request()->input($param, $default);
- }
|