mirror of
				https://github.com/pmmp/PocketMine-MP.git
				synced 2025-10-20 15:41:33 +00:00 
			
		
		
		
	Removed pocketmine subdirectory, map PSR-4 style
This commit is contained in:
		
							
								
								
									
										529
									
								
								src/MemoryManager.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										529
									
								
								src/MemoryManager.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,529 @@ | ||||
| <?php | ||||
|  | ||||
| /* | ||||
|  * | ||||
|  *  ____            _        _   __  __ _                  __  __ ____ | ||||
|  * |  _ \ ___   ___| | _____| |_|  \/  (_)_ __   ___      |  \/  |  _ \ | ||||
|  * | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) | | ||||
|  * |  __/ (_) | (__|   <  __/ |_| |  | | | | | |  __/_____| |  | |  __/ | ||||
|  * |_|   \___/ \___|_|\_\___|\__|_|  |_|_|_| |_|\___|     |_|  |_|_| | ||||
|  * | ||||
|  * This program 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 | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * @author PocketMine Team | ||||
|  * @link http://www.pocketmine.net/ | ||||
|  * | ||||
|  * | ||||
| */ | ||||
|  | ||||
| declare(strict_types=1); | ||||
|  | ||||
| namespace pocketmine; | ||||
|  | ||||
| use pocketmine\event\server\LowMemoryEvent; | ||||
| use pocketmine\scheduler\DumpWorkerMemoryTask; | ||||
| use pocketmine\scheduler\GarbageCollectionTask; | ||||
| use pocketmine\timings\Timings; | ||||
| use pocketmine\utils\Process; | ||||
| use pocketmine\utils\Utils; | ||||
| use function arsort; | ||||
| use function count; | ||||
| use function fclose; | ||||
| use function file_exists; | ||||
| use function file_put_contents; | ||||
| use function fopen; | ||||
| use function fwrite; | ||||
| use function gc_collect_cycles; | ||||
| use function gc_disable; | ||||
| use function gc_enable; | ||||
| use function get_class; | ||||
| use function get_declared_classes; | ||||
| use function implode; | ||||
| use function ini_get; | ||||
| use function ini_set; | ||||
| use function is_array; | ||||
| use function is_object; | ||||
| use function is_resource; | ||||
| use function is_string; | ||||
| use function json_encode; | ||||
| use function min; | ||||
| use function mkdir; | ||||
| use function preg_match; | ||||
| use function print_r; | ||||
| use function round; | ||||
| use function spl_object_hash; | ||||
| use function sprintf; | ||||
| use function strlen; | ||||
| use function strtoupper; | ||||
| use function substr; | ||||
| use const JSON_PRETTY_PRINT; | ||||
| use const JSON_UNESCAPED_SLASHES; | ||||
| use const SORT_NUMERIC; | ||||
|  | ||||
| class MemoryManager{ | ||||
|  | ||||
| 	/** @var Server */ | ||||
| 	private $server; | ||||
|  | ||||
| 	/** @var int */ | ||||
| 	private $memoryLimit; | ||||
| 	/** @var int */ | ||||
| 	private $globalMemoryLimit; | ||||
| 	/** @var int */ | ||||
| 	private $checkRate; | ||||
| 	/** @var int */ | ||||
| 	private $checkTicker = 0; | ||||
| 	/** @var bool */ | ||||
| 	private $lowMemory = false; | ||||
|  | ||||
| 	/** @var bool */ | ||||
| 	private $continuousTrigger = true; | ||||
| 	/** @var int */ | ||||
| 	private $continuousTriggerRate; | ||||
| 	/** @var int */ | ||||
| 	private $continuousTriggerCount = 0; | ||||
| 	/** @var int */ | ||||
| 	private $continuousTriggerTicker = 0; | ||||
|  | ||||
| 	/** @var int */ | ||||
| 	private $garbageCollectionPeriod; | ||||
| 	/** @var int */ | ||||
| 	private $garbageCollectionTicker = 0; | ||||
| 	/** @var bool */ | ||||
| 	private $garbageCollectionTrigger; | ||||
| 	/** @var bool */ | ||||
| 	private $garbageCollectionAsync; | ||||
|  | ||||
| 	/** @var int */ | ||||
| 	private $lowMemChunkRadiusOverride; | ||||
| 	/** @var bool */ | ||||
| 	private $lowMemChunkGC; | ||||
|  | ||||
| 	/** @var bool */ | ||||
| 	private $lowMemDisableChunkCache; | ||||
| 	/** @var bool */ | ||||
| 	private $lowMemClearWorldCache; | ||||
|  | ||||
| 	/** @var bool */ | ||||
| 	private $dumpWorkers = true; | ||||
|  | ||||
| 	/** @var \Logger */ | ||||
| 	private $logger; | ||||
|  | ||||
| 	public function __construct(Server $server){ | ||||
| 		$this->server = $server; | ||||
| 		$this->logger = new \PrefixedLogger($server->getLogger(), "Memory Manager"); | ||||
|  | ||||
| 		$this->init(); | ||||
| 	} | ||||
|  | ||||
| 	private function init() : void{ | ||||
| 		$this->memoryLimit = ((int) $this->server->getProperty("memory.main-limit", 0)) * 1024 * 1024; | ||||
|  | ||||
| 		$defaultMemory = 1024; | ||||
|  | ||||
| 		if(preg_match("/([0-9]+)([KMGkmg])/", $this->server->getConfigString("memory-limit", ""), $matches) > 0){ | ||||
| 			$m = (int) $matches[1]; | ||||
| 			if($m <= 0){ | ||||
| 				$defaultMemory = 0; | ||||
| 			}else{ | ||||
| 				switch(strtoupper($matches[2])){ | ||||
| 					case "K": | ||||
| 						$defaultMemory = $m / 1024; | ||||
| 						break; | ||||
| 					case "M": | ||||
| 						$defaultMemory = $m; | ||||
| 						break; | ||||
| 					case "G": | ||||
| 						$defaultMemory = $m * 1024; | ||||
| 						break; | ||||
| 					default: | ||||
| 						$defaultMemory = $m; | ||||
| 						break; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		$hardLimit = ((int) $this->server->getProperty("memory.main-hard-limit", $defaultMemory)); | ||||
|  | ||||
| 		if($hardLimit <= 0){ | ||||
| 			ini_set("memory_limit", '-1'); | ||||
| 		}else{ | ||||
| 			ini_set("memory_limit", $hardLimit . "M"); | ||||
| 		} | ||||
|  | ||||
| 		$this->globalMemoryLimit = ((int) $this->server->getProperty("memory.global-limit", 0)) * 1024 * 1024; | ||||
| 		$this->checkRate = (int) $this->server->getProperty("memory.check-rate", 20); | ||||
| 		$this->continuousTrigger = (bool) $this->server->getProperty("memory.continuous-trigger", true); | ||||
| 		$this->continuousTriggerRate = (int) $this->server->getProperty("memory.continuous-trigger-rate", 30); | ||||
|  | ||||
| 		$this->garbageCollectionPeriod = (int) $this->server->getProperty("memory.garbage-collection.period", 36000); | ||||
| 		$this->garbageCollectionTrigger = (bool) $this->server->getProperty("memory.garbage-collection.low-memory-trigger", true); | ||||
| 		$this->garbageCollectionAsync = (bool) $this->server->getProperty("memory.garbage-collection.collect-async-worker", true); | ||||
|  | ||||
| 		$this->lowMemChunkRadiusOverride = (int) $this->server->getProperty("memory.max-chunks.chunk-radius", 4); | ||||
| 		$this->lowMemChunkGC = (bool) $this->server->getProperty("memory.max-chunks.trigger-chunk-collect", true); | ||||
|  | ||||
| 		$this->lowMemDisableChunkCache = (bool) $this->server->getProperty("memory.world-caches.disable-chunk-cache", true); | ||||
| 		$this->lowMemClearWorldCache = (bool) $this->server->getProperty("memory.world-caches.low-memory-trigger", true); | ||||
|  | ||||
| 		$this->dumpWorkers = (bool) $this->server->getProperty("memory.memory-dump.dump-async-worker", true); | ||||
| 		gc_enable(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @return bool | ||||
| 	 */ | ||||
| 	public function isLowMemory() : bool{ | ||||
| 		return $this->lowMemory; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @return int | ||||
| 	 */ | ||||
| 	public function getGlobalMemoryLimit() : int{ | ||||
| 		return $this->globalMemoryLimit; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @return bool | ||||
| 	 */ | ||||
| 	public function canUseChunkCache() : bool{ | ||||
| 		return !$this->lowMemory or !$this->lowMemDisableChunkCache; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Returns the allowed chunk radius based on the current memory usage. | ||||
| 	 * | ||||
| 	 * @param int $distance | ||||
| 	 * | ||||
| 	 * @return int | ||||
| 	 */ | ||||
| 	public function getViewDistance(int $distance) : int{ | ||||
| 		return ($this->lowMemory and $this->lowMemChunkRadiusOverride > 0) ? (int) min($this->lowMemChunkRadiusOverride, $distance) : $distance; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Triggers garbage collection and cache cleanup to try and free memory. | ||||
| 	 * | ||||
| 	 * @param int  $memory | ||||
| 	 * @param int  $limit | ||||
| 	 * @param bool $global | ||||
| 	 * @param int  $triggerCount | ||||
| 	 */ | ||||
| 	public function trigger(int $memory, int $limit, bool $global = false, int $triggerCount = 0) : void{ | ||||
| 		$this->logger->debug(sprintf("%sLow memory triggered, limit %gMB, using %gMB", | ||||
| 			$global ? "Global " : "", round(($limit / 1024) / 1024, 2), round(($memory / 1024) / 1024, 2))); | ||||
| 		if($this->lowMemClearWorldCache){ | ||||
| 			foreach($this->server->getWorldManager()->getWorlds() as $world){ | ||||
| 				$world->clearCache(true); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if($this->lowMemChunkGC){ | ||||
| 			foreach($this->server->getWorldManager()->getWorlds() as $world){ | ||||
| 				$world->doChunkGarbageCollection(); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		$ev = new LowMemoryEvent($memory, $limit, $global, $triggerCount); | ||||
| 		$ev->call(); | ||||
|  | ||||
| 		$cycles = 0; | ||||
| 		if($this->garbageCollectionTrigger){ | ||||
| 			$cycles = $this->triggerGarbageCollector(); | ||||
| 		} | ||||
|  | ||||
| 		$this->logger->debug(sprintf("Freed %gMB, $cycles cycles", round(($ev->getMemoryFreed() / 1024) / 1024, 2))); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Called every tick to update the memory manager state. | ||||
| 	 */ | ||||
| 	public function check() : void{ | ||||
| 		Timings::$memoryManagerTimer->startTiming(); | ||||
|  | ||||
| 		if(($this->memoryLimit > 0 or $this->globalMemoryLimit > 0) and ++$this->checkTicker >= $this->checkRate){ | ||||
| 			$this->checkTicker = 0; | ||||
| 			$memory = Process::getMemoryUsage(true); | ||||
| 			$trigger = false; | ||||
| 			if($this->memoryLimit > 0 and $memory[0] > $this->memoryLimit){ | ||||
| 				$trigger = 0; | ||||
| 			}elseif($this->globalMemoryLimit > 0 and $memory[1] > $this->globalMemoryLimit){ | ||||
| 				$trigger = 1; | ||||
| 			} | ||||
|  | ||||
| 			if($trigger !== false){ | ||||
| 				if($this->lowMemory and $this->continuousTrigger){ | ||||
| 					if(++$this->continuousTriggerTicker >= $this->continuousTriggerRate){ | ||||
| 						$this->continuousTriggerTicker = 0; | ||||
| 						$this->trigger($memory[$trigger], $this->memoryLimit, $trigger > 0, ++$this->continuousTriggerCount); | ||||
| 					} | ||||
| 				}else{ | ||||
| 					$this->lowMemory = true; | ||||
| 					$this->continuousTriggerCount = 0; | ||||
| 					$this->trigger($memory[$trigger], $this->memoryLimit, $trigger > 0); | ||||
| 				} | ||||
| 			}else{ | ||||
| 				$this->lowMemory = false; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if($this->garbageCollectionPeriod > 0 and ++$this->garbageCollectionTicker >= $this->garbageCollectionPeriod){ | ||||
| 			$this->garbageCollectionTicker = 0; | ||||
| 			$this->triggerGarbageCollector(); | ||||
| 		} | ||||
|  | ||||
| 		Timings::$memoryManagerTimer->stopTiming(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @return int | ||||
| 	 */ | ||||
| 	public function triggerGarbageCollector() : int{ | ||||
| 		Timings::$garbageCollectorTimer->startTiming(); | ||||
|  | ||||
| 		if($this->garbageCollectionAsync){ | ||||
| 			$pool = $this->server->getAsyncPool(); | ||||
| 			if(($w = $pool->shutdownUnusedWorkers()) > 0){ | ||||
| 				$this->logger->debug("Shut down $w idle async pool workers"); | ||||
| 			} | ||||
| 			foreach($pool->getRunningWorkers() as $i){ | ||||
| 				$pool->submitTaskToWorker(new GarbageCollectionTask(), $i); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		$cycles = gc_collect_cycles(); | ||||
|  | ||||
| 		Timings::$garbageCollectorTimer->stopTiming(); | ||||
|  | ||||
| 		return $cycles; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Dumps the server memory into the specified output folder. | ||||
| 	 * | ||||
| 	 * @param string $outputFolder | ||||
| 	 * @param int    $maxNesting | ||||
| 	 * @param int    $maxStringSize | ||||
| 	 */ | ||||
| 	public function dumpServerMemory(string $outputFolder, int $maxNesting, int $maxStringSize) : void{ | ||||
| 		$logger = new \PrefixedLogger($this->server->getLogger(), "Memory Dump"); | ||||
| 		$logger->notice("After the memory dump is done, the server might crash"); | ||||
| 		self::dumpMemory($this->server, $outputFolder, $maxNesting, $maxStringSize, $logger); | ||||
|  | ||||
| 		if($this->dumpWorkers){ | ||||
| 			$pool = $this->server->getAsyncPool(); | ||||
| 			foreach($pool->getRunningWorkers() as $i){ | ||||
| 				$pool->submitTaskToWorker(new DumpWorkerMemoryTask($outputFolder, $maxNesting, $maxStringSize), $i); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Static memory dumper accessible from any thread. | ||||
| 	 * | ||||
| 	 * @param mixed   $startingObject | ||||
| 	 * @param string  $outputFolder | ||||
| 	 * @param int     $maxNesting | ||||
| 	 * @param int     $maxStringSize | ||||
| 	 * @param \Logger $logger | ||||
| 	 * | ||||
| 	 * @throws \ReflectionException | ||||
| 	 */ | ||||
| 	public static function dumpMemory($startingObject, string $outputFolder, int $maxNesting, int $maxStringSize, \Logger $logger) : void{ | ||||
| 		$hardLimit = ini_get('memory_limit'); | ||||
| 		ini_set('memory_limit', '-1'); | ||||
| 		gc_disable(); | ||||
|  | ||||
| 		if(!file_exists($outputFolder)){ | ||||
| 			mkdir($outputFolder, 0777, true); | ||||
| 		} | ||||
|  | ||||
| 		$obData = fopen($outputFolder . "/objects.js", "wb+"); | ||||
|  | ||||
| 		$data = []; | ||||
|  | ||||
| 		$objects = []; | ||||
|  | ||||
| 		$refCounts = []; | ||||
|  | ||||
| 		$instanceCounts = []; | ||||
|  | ||||
| 		$staticProperties = []; | ||||
| 		$staticCount = 0; | ||||
|  | ||||
| 		foreach(get_declared_classes() as $className){ | ||||
| 			$reflection = new \ReflectionClass($className); | ||||
| 			$staticProperties[$className] = []; | ||||
| 			foreach($reflection->getProperties() as $property){ | ||||
| 				if(!$property->isStatic() or $property->getDeclaringClass()->getName() !== $className){ | ||||
| 					continue; | ||||
| 				} | ||||
|  | ||||
| 				if(!$property->isPublic()){ | ||||
| 					$property->setAccessible(true); | ||||
| 				} | ||||
|  | ||||
| 				$staticCount++; | ||||
| 				self::continueDump($property->getValue(), $staticProperties[$className][$property->getName()], $objects, $refCounts, 0, $maxNesting, $maxStringSize); | ||||
| 			} | ||||
|  | ||||
| 			if(count($staticProperties[$className]) === 0){ | ||||
| 				unset($staticProperties[$className]); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		file_put_contents($outputFolder . "/staticProperties.js", json_encode($staticProperties, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); | ||||
| 		$logger->info("Wrote $staticCount static properties"); | ||||
|  | ||||
| 		if(isset($GLOBALS)){ //This might be null if we're on a different thread | ||||
| 			$globalVariables = []; | ||||
| 			$globalCount = 0; | ||||
|  | ||||
| 			$ignoredGlobals = [ | ||||
| 				'GLOBALS' => true, | ||||
| 				'_SERVER' => true, | ||||
| 				'_REQUEST' => true, | ||||
| 				'_POST' => true, | ||||
| 				'_GET' => true, | ||||
| 				'_FILES' => true, | ||||
| 				'_ENV' => true, | ||||
| 				'_COOKIE' => true, | ||||
| 				'_SESSION' => true | ||||
| 			]; | ||||
|  | ||||
| 			foreach($GLOBALS as $varName => $value){ | ||||
| 				if(isset($ignoredGlobals[$varName])){ | ||||
| 					continue; | ||||
| 				} | ||||
|  | ||||
| 				$globalCount++; | ||||
| 				self::continueDump($value, $globalVariables[$varName], $objects, $refCounts, 0, $maxNesting, $maxStringSize); | ||||
| 			} | ||||
|  | ||||
| 			file_put_contents($outputFolder . "/globalVariables.js", json_encode($globalVariables, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); | ||||
| 			$logger->info("Wrote $globalCount global variables"); | ||||
| 		} | ||||
|  | ||||
| 		self::continueDump($startingObject, $data, $objects, $refCounts, 0, $maxNesting, $maxStringSize); | ||||
|  | ||||
| 		do{ | ||||
| 			$continue = false; | ||||
| 			foreach($objects as $hash => $object){ | ||||
| 				if(!is_object($object)){ | ||||
| 					continue; | ||||
| 				} | ||||
| 				$continue = true; | ||||
|  | ||||
| 				$className = get_class($object); | ||||
| 				if(!isset($instanceCounts[$className])){ | ||||
| 					$instanceCounts[$className] = 1; | ||||
| 				}else{ | ||||
| 					$instanceCounts[$className]++; | ||||
| 				} | ||||
|  | ||||
| 				$objects[$hash] = true; | ||||
|  | ||||
| 				$reflection = new \ReflectionObject($object); | ||||
|  | ||||
| 				$info = [ | ||||
| 					"information" => "$hash@$className", | ||||
| 					"properties" => [] | ||||
| 				]; | ||||
|  | ||||
| 				if($reflection->getParentClass()){ | ||||
| 					$info["parent"] = $reflection->getParentClass()->getName(); | ||||
| 				} | ||||
|  | ||||
| 				if(count($reflection->getInterfaceNames()) > 0){ | ||||
| 					$info["implements"] = implode(", ", $reflection->getInterfaceNames()); | ||||
| 				} | ||||
|  | ||||
| 				for($original = $reflection; $reflection !== false; $reflection = $reflection->getParentClass()){ | ||||
| 					foreach($reflection->getProperties() as $property){ | ||||
| 						if($property->isStatic()){ | ||||
| 							continue; | ||||
| 						} | ||||
|  | ||||
| 						$name = $property->getName(); | ||||
| 						if($reflection !== $original and !$property->isPublic()){ | ||||
| 							$name = $reflection->getName() . ":" . $name; | ||||
| 						} | ||||
| 						if(!$property->isPublic()){ | ||||
| 							$property->setAccessible(true); | ||||
| 						} | ||||
|  | ||||
| 						self::continueDump($property->getValue($object), $info["properties"][$name], $objects, $refCounts, 0, $maxNesting, $maxStringSize); | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				fwrite($obData, "$hash@$className: " . json_encode($info, JSON_UNESCAPED_SLASHES) . "\n"); | ||||
| 			} | ||||
|  | ||||
|  | ||||
| 		}while($continue); | ||||
|  | ||||
| 		$logger->info("Wrote " . count($objects) . " objects"); | ||||
|  | ||||
| 		fclose($obData); | ||||
|  | ||||
| 		file_put_contents($outputFolder . "/serverEntry.js", json_encode($data, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); | ||||
| 		file_put_contents($outputFolder . "/referenceCounts.js", json_encode($refCounts, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); | ||||
|  | ||||
| 		arsort($instanceCounts, SORT_NUMERIC); | ||||
| 		file_put_contents($outputFolder . "/instanceCounts.js", json_encode($instanceCounts, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT)); | ||||
|  | ||||
| 		$logger->info("Finished!"); | ||||
|  | ||||
| 		ini_set('memory_limit', $hardLimit); | ||||
| 		gc_enable(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @param mixed    $from | ||||
| 	 * @param mixed    &$data | ||||
| 	 * @param object[] &$objects | ||||
| 	 * @param int[]    &$refCounts | ||||
| 	 * @param int      $recursion | ||||
| 	 * @param int      $maxNesting | ||||
| 	 * @param int      $maxStringSize | ||||
| 	 */ | ||||
| 	private static function continueDump($from, &$data, array &$objects, array &$refCounts, int $recursion, int $maxNesting, int $maxStringSize) : void{ | ||||
| 		if($maxNesting <= 0){ | ||||
| 			$data = "(error) NESTING LIMIT REACHED"; | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		--$maxNesting; | ||||
|  | ||||
| 		if(is_object($from)){ | ||||
| 			if(!isset($objects[$hash = spl_object_hash($from)])){ | ||||
| 				$objects[$hash] = $from; | ||||
| 				$refCounts[$hash] = 0; | ||||
| 			} | ||||
|  | ||||
| 			++$refCounts[$hash]; | ||||
|  | ||||
| 			$data = "(object) $hash@" . get_class($from); | ||||
| 		}elseif(is_array($from)){ | ||||
| 			if($recursion >= 5){ | ||||
| 				$data = "(error) ARRAY RECURSION LIMIT REACHED"; | ||||
| 				return; | ||||
| 			} | ||||
| 			$data = []; | ||||
| 			foreach($from as $key => $value){ | ||||
| 				self::continueDump($value, $data[$key], $objects, $refCounts, $recursion + 1, $maxNesting, $maxStringSize); | ||||
| 			} | ||||
| 		}elseif(is_string($from)){ | ||||
| 			$data = "(string) len(" . strlen($from) . ") " . substr(Utils::printable($from), 0, $maxStringSize); | ||||
| 		}elseif(is_resource($from)){ | ||||
| 			$data = "(resource) " . print_r($from, true); | ||||
| 		}else{ | ||||
| 			$data = $from; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
		Reference in New Issue
	
	Block a user