| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * StatusNet, the distributed open-source microblogging tool | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * I/O manager to wrap around socket-reading and polling queue & connection managers. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * PHP version 5 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * LICENCE: This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program 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 Affero General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @category  QueueManager | 
					
						
							|  |  |  |  * @package   StatusNet | 
					
						
							|  |  |  |  * @author    Brion Vibber <brion@status.net> | 
					
						
							|  |  |  |  * @copyright 2009 StatusNet, Inc. | 
					
						
							|  |  |  |  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 | 
					
						
							|  |  |  |  * @link      http://status.net/ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  | abstract class IoMaster | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  | { | 
					
						
							|  |  |  |     public $id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected $multiSite = false; | 
					
						
							|  |  |  |     protected $managers = array(); | 
					
						
							|  |  |  |     protected $singletons = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected $pollTimeouts = array(); | 
					
						
							|  |  |  |     protected $lastPoll = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |     public $shutdown = false; // Did we do a graceful shutdown?
 | 
					
						
							|  |  |  |     public $respawn = true; // Should we respawn after shutdown?
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param string $id process ID to use in logging/monitoring | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct($id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->id = $id; | 
					
						
							|  |  |  |         $this->monitor = new QueueMonitor(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public function init($multiSite=null) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($multiSite !== null) { | 
					
						
							|  |  |  |             $this->multiSite = $multiSite; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 16:49:00 -08:00
										 |  |  |         $this->initManagers(); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2010-02-17 16:49:00 -08:00
										 |  |  |      * Initialize IoManagers which are appropriate to this instance; | 
					
						
							|  |  |  |      * pass class names or instances into $this->instantiate(). | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * If setup and configuration may vary between sites in multi-site | 
					
						
							|  |  |  |      * mode, it's the subclass's responsibility to set them up here. | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2010-02-17 16:49:00 -08:00
										 |  |  |      * Switching site configurations is an acceptable side effect. | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     abstract function initManagers(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Instantiate an i/o manager class for the current site. | 
					
						
							|  |  |  |      * If a multi-site capable handler is already present, | 
					
						
							|  |  |  |      * we don't need to build a new one. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |      * @param mixed $manager class name (to run $class::get()) or object | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |     protected function instantiate($manager) | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |         if (is_string($manager)) { | 
					
						
							|  |  |  |             $manager = call_user_func(array($class, 'get')); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |         $caps = $manager->multiSite(); | 
					
						
							|  |  |  |         if ($caps == IoManager::SINGLE_ONLY) { | 
					
						
							|  |  |  |             if ($this->multiSite) { | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |                 throw new Exception("$class can't run with --all; aborting."); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |         } else if ($caps == IoManager::INSTANCE_PER_PROCESS) { | 
					
						
							|  |  |  |             $manager->addSite(); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |         if (!in_array($manager, $this->managers, true)) { | 
					
						
							|  |  |  |             // Only need to save singletons once
 | 
					
						
							|  |  |  |             $this->managers[] = $manager; | 
					
						
							| 
									
										
										
										
											2010-01-23 01:03:41 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Basic run loop... | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Initialize all io managers, then sit around waiting for input. | 
					
						
							|  |  |  |      * Between events or timeouts, pass control back to idle() method | 
					
						
							|  |  |  |      * to allow for any additional background processing. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function service() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->logState('init'); | 
					
						
							|  |  |  |         $this->start(); | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |         $this->checkMemory(false); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |         while (!$this->shutdown) { | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |             $timeouts = array_values($this->pollTimeouts); | 
					
						
							|  |  |  |             $timeouts[] = 60; // default max timeout
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Wait for something on one of our sockets
 | 
					
						
							|  |  |  |             $sockets = array(); | 
					
						
							|  |  |  |             $managers = array(); | 
					
						
							|  |  |  |             foreach ($this->managers as $manager) { | 
					
						
							|  |  |  |                 foreach ($manager->getSockets() as $socket) { | 
					
						
							|  |  |  |                     $sockets[] = $socket; | 
					
						
							|  |  |  |                     $managers[] = $manager; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 $timeouts[] = intval($manager->timeout()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $timeout = min($timeouts); | 
					
						
							|  |  |  |             if ($sockets) { | 
					
						
							|  |  |  |                 $read = $sockets; | 
					
						
							|  |  |  |                 $write = array(); | 
					
						
							|  |  |  |                 $except = array(); | 
					
						
							|  |  |  |                 $this->logState('listening'); | 
					
						
							| 
									
										
										
										
											2013-10-05 13:12:16 +02:00
										 |  |  |                 //common_debug("Waiting up to $timeout seconds for socket data...");
 | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |                 $ready = stream_select($read, $write, $except, $timeout, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if ($ready === false) { | 
					
						
							|  |  |  |                     common_log(LOG_ERR, "Error selecting on sockets"); | 
					
						
							|  |  |  |                 } else if ($ready > 0) { | 
					
						
							|  |  |  |                     foreach ($read as $socket) { | 
					
						
							|  |  |  |                         $index = array_search($socket, $sockets, true); | 
					
						
							|  |  |  |                         if ($index !== false) { | 
					
						
							|  |  |  |                             $this->logState('queue'); | 
					
						
							|  |  |  |                             $managers[$index]->handleInput($socket); | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             common_log(LOG_ERR, "Saw input on a socket we didn't listen to"); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($timeout > 0 && empty($sockets)) { | 
					
						
							|  |  |  |                 // If we had no listeners, sleep until the pollers' next requested wakeup.
 | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |                 common_log(LOG_DEBUG, "Sleeping $timeout seconds until next poll cycle..."); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |                 $this->logState('sleep'); | 
					
						
							|  |  |  |                 sleep($timeout); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $this->logState('poll'); | 
					
						
							|  |  |  |             $this->poll(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $this->logState('idle'); | 
					
						
							|  |  |  |             $this->idle(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |             $this->checkMemory(); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->logState('shutdown'); | 
					
						
							|  |  |  |         $this->finish(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Check runtime memory usage, possibly triggering a graceful shutdown | 
					
						
							|  |  |  |      * and thread respawn if we've crossed the soft limit. | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @param boolean $respawn if false we'll shut down instead of respawning | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |     protected function checkMemory($respawn=true) | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |     { | 
					
						
							|  |  |  |         $memoryLimit = $this->softMemoryLimit(); | 
					
						
							|  |  |  |         if ($memoryLimit > 0) { | 
					
						
							|  |  |  |             $usage = memory_get_usage(); | 
					
						
							|  |  |  |             if ($usage > $memoryLimit) { | 
					
						
							|  |  |  |                 common_log(LOG_INFO, "Queue thread hit soft memory limit ($usage > $memoryLimit); gracefully restarting."); | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |                 if ($respawn) { | 
					
						
							|  |  |  |                     $this->requestRestart(); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $this->requestShutdown(); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |             } else if (common_config('queue', 'debug_memory')) { | 
					
						
							| 
									
										
										
										
											2010-02-16 09:01:59 -08:00
										 |  |  |                 $fmt = number_format($usage); | 
					
						
							|  |  |  |                 common_log(LOG_DEBUG, "Memory usage $fmt"); | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Return fully-parsed soft memory limit in bytes. | 
					
						
							|  |  |  |      * @return intval 0 or -1 if not set | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function softMemoryLimit() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $softLimit = trim(common_config('queue', 'softlimit')); | 
					
						
							|  |  |  |         if (substr($softLimit, -1) == '%') { | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |             $limit = $this->parseMemoryLimit(ini_get('memory_limit')); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |             if ($limit > 0) { | 
					
						
							|  |  |  |                 return intval(substr($softLimit, 0, -1) * $limit / 100); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-13 21:35:47 -08:00
										 |  |  |             return $this->parseMemoryLimit($softLimit); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |         return $softLimit; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Interpret PHP shorthand for memory_limit and friends. | 
					
						
							|  |  |  |      * Why don't they just expose the actual numeric value? :P | 
					
						
							|  |  |  |      * @param string $mem | 
					
						
							|  |  |  |      * @return int | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |     public function parseMemoryLimit($mem) | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |     { | 
					
						
							|  |  |  |         // http://www.php.net/manual/en/faq.using.php#faq.using.shorthandbytes
 | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |         $mem = strtolower(trim($mem)); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |         $size = array('k' => 1024, | 
					
						
							|  |  |  |                       'm' => 1024*1024, | 
					
						
							|  |  |  |                       'g' => 1024*1024*1024); | 
					
						
							|  |  |  |         if (empty($mem)) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } else if (is_numeric($mem)) { | 
					
						
							|  |  |  |             return intval($mem); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-21 16:42:50 -08:00
										 |  |  |             $mult = substr($mem, -1); | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |             if (isset($size[$mult])) { | 
					
						
							|  |  |  |                 return substr($mem, 0, -1) * $size[$mult]; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 return intval($mem); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function start() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         foreach ($this->managers as $index => $manager) { | 
					
						
							|  |  |  |             $manager->start($this); | 
					
						
							|  |  |  |             // @fixme error check
 | 
					
						
							|  |  |  |             if ($manager->pollInterval()) { | 
					
						
							|  |  |  |                 // We'll want to check for input on the first pass
 | 
					
						
							|  |  |  |                 $this->pollTimeouts[$index] = 0; | 
					
						
							|  |  |  |                 $this->lastPoll[$index] = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function finish() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         foreach ($this->managers as $manager) { | 
					
						
							|  |  |  |             $manager->finish(); | 
					
						
							|  |  |  |             // @fixme error check
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Called during the idle portion of the runloop to see which handlers | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function poll() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         foreach ($this->managers as $index => $manager) { | 
					
						
							|  |  |  |             $interval = $manager->pollInterval(); | 
					
						
							|  |  |  |             if ($interval <= 0) { | 
					
						
							|  |  |  |                 // Not a polling manager.
 | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (isset($this->pollTimeouts[$index])) { | 
					
						
							|  |  |  |                 $timeout = $this->pollTimeouts[$index]; | 
					
						
							|  |  |  |                 if (time() - $this->lastPoll[$index] < $timeout) { | 
					
						
							|  |  |  |                     // Not time to poll yet.
 | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 $timeout = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             $hit = $manager->poll(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $this->lastPoll[$index] = time(); | 
					
						
							|  |  |  |             if ($hit) { | 
					
						
							|  |  |  |                 // Do the next poll quickly, there may be more input!
 | 
					
						
							|  |  |  |                 $this->pollTimeouts[$index] = 0; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 // Empty queue. Exponential backoff up to the maximum poll interval.
 | 
					
						
							|  |  |  |                 if ($timeout > 0) { | 
					
						
							|  |  |  |                     $timeout = min($timeout * 2, $interval); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $timeout = 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 $this->pollTimeouts[$index] = $timeout; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Called after each handled item or empty polling cycle. | 
					
						
							|  |  |  |      * This is a good time to e.g. service your XMPP connection. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function idle() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         foreach ($this->managers as $manager) { | 
					
						
							|  |  |  |             $manager->idle(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Send thread state update to the monitoring server, if configured. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string $state ('init', 'queue', 'shutdown' etc) | 
					
						
							|  |  |  |      * @param string $substate (optional, eg queue name 'omb' 'sms' etc) | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function logState($state, $substate='') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->monitor->logState($this->id, $state, $substate); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Send thread stats. | 
					
						
							|  |  |  |      * Thread ID will be implicit; other owners can be listed as well | 
					
						
							|  |  |  |      * for per-queue and per-site records. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string $key counter name | 
					
						
							| 
									
										
										
										
											2010-01-23 01:03:41 -05:00
										 |  |  |      * @param array $owners list of owner keys like 'queue:xmpp' or 'site:stat01' | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function stats($key, $owners=array()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $owners[] = "thread:" . $this->id; | 
					
						
							|  |  |  |         $this->monitor->stats($key, $owners); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-26 11:49:49 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * For IoManagers to request a graceful shutdown at end of event loop. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function requestShutdown() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->shutdown = true; | 
					
						
							|  |  |  |         $this->respawn = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * For IoManagers to request a graceful restart at end of event loop. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function requestRestart() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->shutdown = true; | 
					
						
							|  |  |  |         $this->respawn = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 |