| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305 |
- <?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
- */
- namespace app\process;
- use FilesystemIterator;
- use RecursiveDirectoryIterator;
- use RecursiveIteratorIterator;
- use SplFileInfo;
- use Workerman\Timer;
- use Workerman\Worker;
- /**
- * Class FileMonitor
- * @package process
- */
- class Monitor
- {
- /**
- * @var array
- */
- protected array $paths = [];
- /**
- * @var array
- */
- protected array $extensions = [];
- /**
- * @var array
- */
- protected array $loadedFiles = [];
- /**
- * @var int
- */
- protected int $ppid = 0;
- /**
- * Pause monitor
- * @return void
- */
- public static function pause(): void
- {
- file_put_contents(static::lockFile(), time());
- }
- /**
- * Resume monitor
- * @return void
- */
- public static function resume(): void
- {
- clearstatcache();
- if (is_file(static::lockFile())) {
- unlink(static::lockFile());
- }
- }
- /**
- * Whether monitor is paused
- * @return bool
- */
- public static function isPaused(): bool
- {
- clearstatcache();
- return file_exists(static::lockFile());
- }
- /**
- * Lock file
- * @return string
- */
- protected static function lockFile(): string
- {
- return runtime_path('monitor.lock');
- }
- /**
- * FileMonitor constructor.
- * @param $monitorDir
- * @param $monitorExtensions
- * @param array $options
- */
- public function __construct($monitorDir, $monitorExtensions, array $options = [])
- {
- $this->ppid = function_exists('posix_getppid') ? posix_getppid() : 0;
- static::resume();
- $this->paths = (array)$monitorDir;
- $this->extensions = $monitorExtensions;
- foreach (get_included_files() as $index => $file) {
- $this->loadedFiles[$file] = $index;
- if (strpos($file, 'webman-framework/src/support/App.php')) {
- break;
- }
- }
- if (!Worker::getAllWorkers()) {
- return;
- }
- $disableFunctions = explode(',', ini_get('disable_functions'));
- if (in_array('exec', $disableFunctions, true)) {
- echo "\nMonitor file change turned off because exec() has been disabled by disable_functions setting in " . PHP_CONFIG_FILE_PATH . "/php.ini\n";
- } else {
- if ($options['enable_file_monitor'] ?? true) {
- Timer::add(1, function () {
- $this->checkAllFilesChange();
- });
- }
- }
- $memoryLimit = $this->getMemoryLimit($options['memory_limit'] ?? null);
- if ($memoryLimit && ($options['enable_memory_monitor'] ?? true)) {
- Timer::add(60, [$this, 'checkMemory'], [$memoryLimit]);
- }
- }
- /**
- * @param $monitorDir
- * @return bool
- */
- public function checkFilesChange($monitorDir): bool
- {
- static $lastMtime, $tooManyFilesCheck;
- if (!$lastMtime) {
- $lastMtime = time();
- }
- clearstatcache();
- if (!is_dir($monitorDir)) {
- if (!is_file($monitorDir)) {
- return false;
- }
- $iterator = [new SplFileInfo($monitorDir)];
- } else {
- // recursive traversal directory
- $dirIterator = new RecursiveDirectoryIterator($monitorDir, FilesystemIterator::SKIP_DOTS | FilesystemIterator::FOLLOW_SYMLINKS);
- $iterator = new RecursiveIteratorIterator($dirIterator);
- }
- $count = 0;
- foreach ($iterator as $file) {
- $count ++;
- /** var SplFileInfo $file */
- if (is_dir($file->getRealPath())) {
- continue;
- }
- // check mtime
- if (in_array($file->getExtension(), $this->extensions, true) && $lastMtime < $file->getMTime()) {
- $lastMtime = $file->getMTime();
- if (DIRECTORY_SEPARATOR === '/' && isset($this->loadedFiles[$file->getRealPath()])) {
- echo "$file updated but cannot be reloaded because only auto-loaded files support reload.\n";
- continue;
- }
- $var = 0;
- exec('"'.PHP_BINARY . '" -l ' . $file, $out, $var);
- if ($var) {
- continue;
- }
- // send SIGUSR1 signal to master process for reload
- if (DIRECTORY_SEPARATOR === '/') {
- if ($masterPid = $this->getMasterPid()) {
- echo $file . " updated and reload\n";
- posix_kill($masterPid, SIGUSR1);
- } else {
- echo "Master process has gone away and can not reload\n";
- }
- return true;
- }
- echo $file . " updated and reload\n";
- return true;
- }
- }
- if (!$tooManyFilesCheck && $count > 1000) {
- echo "Monitor: There are too many files ($count files) in $monitorDir which makes file monitoring very slow\n";
- $tooManyFilesCheck = 1;
- }
- return false;
- }
- /**
- * @return int
- */
- public function getMasterPid(): int
- {
- if ($this->ppid === 0) {
- return 0;
- }
- if (function_exists('posix_kill') && !posix_kill($this->ppid, 0)) {
- echo "Master process has gone away\n";
- return $this->ppid = 0;
- }
- if (PHP_OS_FAMILY !== 'Linux') {
- return $this->ppid;
- }
- $cmdline = "/proc/$this->ppid/cmdline";
- if (!is_readable($cmdline) || !($content = file_get_contents($cmdline)) || (!str_contains($content, 'WorkerMan') && !str_contains($content, 'php'))) {
- // Process not exist
- $this->ppid = 0;
- }
- return $this->ppid;
- }
- /**
- * @return bool
- */
- public function checkAllFilesChange(): bool
- {
- if (static::isPaused()) {
- return false;
- }
- foreach ($this->paths as $path) {
- if ($this->checkFilesChange($path)) {
- return true;
- }
- }
- return false;
- }
- /**
- * @param $memoryLimit
- * @return void
- */
- public function checkMemory($memoryLimit): void
- {
- if (static::isPaused() || $memoryLimit <= 0) {
- return;
- }
- $masterPid = $this->getMasterPid();
- if ($masterPid <= 0) {
- echo "Master process has gone away\n";
- return;
- }
- $childrenFile = "/proc/$masterPid/task/$masterPid/children";
- if (!is_file($childrenFile) || !($children = file_get_contents($childrenFile))) {
- return;
- }
- foreach (explode(' ', $children) as $pid) {
- $pid = (int)$pid;
- $statusFile = "/proc/$pid/status";
- if (!is_file($statusFile) || !($status = file_get_contents($statusFile))) {
- continue;
- }
- $mem = 0;
- if (preg_match('/VmRSS\s*?:\s*?(\d+?)\s*?kB/', $status, $match)) {
- $mem = $match[1];
- }
- $mem = (int)($mem / 1024);
- if ($mem >= $memoryLimit) {
- posix_kill($pid, SIGINT);
- }
- }
- }
- /**
- * Get memory limit
- * @param $memoryLimit
- * @return int
- */
- protected function getMemoryLimit($memoryLimit): int
- {
- if ($memoryLimit === 0) {
- return 0;
- }
- $usePhpIni = false;
- if (!$memoryLimit) {
- $memoryLimit = ini_get('memory_limit');
- $usePhpIni = true;
- }
- if ($memoryLimit == -1) {
- return 0;
- }
- $unit = strtolower($memoryLimit[strlen($memoryLimit) - 1]);
- $memoryLimit = (int)$memoryLimit;
- if ($unit === 'g') {
- $memoryLimit = 1024 * $memoryLimit;
- } else if ($unit === 'k') {
- $memoryLimit = ($memoryLimit / 1024);
- } else if ($unit === 'm') {
- $memoryLimit = (int)($memoryLimit);
- } else if ($unit === 't') {
- $memoryLimit = (1024 * 1024 * $memoryLimit);
- } else {
- $memoryLimit = ($memoryLimit / (1024 * 1024));
- }
- if ($memoryLimit < 50) {
- $memoryLimit = 50;
- }
- if ($usePhpIni) {
- $memoryLimit = (0.8 * $memoryLimit);
- }
- return (int)$memoryLimit;
- }
- }
|