| 
									
										
										
										
											2008-09-04 14:40:31 -04:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  | // This file is part of GNU social - https://www.gnu.org/software/social
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // GNU social 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.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // GNU social 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 GNU social.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | defined('GNUSOCIAL') || die(); | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Daemon | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public $daemonize = true; | 
					
						
							|  |  |  |     public $_id = 'generic'; | 
					
						
							| 
									
										
										
										
											2009-06-28 16:12:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function __construct($daemonize = true) | 
					
						
							| 
									
										
										
										
											2009-06-28 16:12:23 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |         $this->daemonize = $daemonize; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function name() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function get_id() | 
					
						
							| 
									
										
										
										
											2009-07-17 18:07:53 -07:00
										 |  |  |     { | 
					
						
							|  |  |  |         return $this->_id; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function set_id($id) | 
					
						
							| 
									
										
										
										
											2009-07-17 18:07:53 -07:00
										 |  |  |     { | 
					
						
							|  |  |  |         $this->_id = $id; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:08:42 +03:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Reconnect to the database for each child process, | 
					
						
							|  |  |  |      * or they'll get very confused trying to use the | 
					
						
							|  |  |  |      * same socket. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function resetDb() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         global $_DB_DATAOBJECT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Can't be called statically
 | 
					
						
							|  |  |  |         $user = new User(); | 
					
						
							|  |  |  |         $conn = $user->getDatabaseConnection(); | 
					
						
							|  |  |  |         $conn->disconnect(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Remove the disconnected connection from the list
 | 
					
						
							|  |  |  |         foreach ($_DB_DATAOBJECT['CONNECTIONS'] as $k => $v) { | 
					
						
							|  |  |  |             if ($v === $conn) { | 
					
						
							|  |  |  |                 unset($_DB_DATAOBJECT['CONNECTIONS'][$k]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Reconnect main memcached, or threads will stomp on
 | 
					
						
							|  |  |  |         // each other and corrupt their requests.
 | 
					
						
							|  |  |  |         $cache = Cache::instance(); | 
					
						
							|  |  |  |         if ($cache) { | 
					
						
							|  |  |  |             $cache->reconnect(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Also reconnect memcached for status_network table.
 | 
					
						
							|  |  |  |         if (!empty(Status_network::$cache)) { | 
					
						
							|  |  |  |             Status_network::$cache->close(); | 
					
						
							|  |  |  |             Status_network::$cache = null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function background() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |         // Database connection will likely get lost after forking
 | 
					
						
							|  |  |  |         $this->resetDb(); | 
					
						
							| 
									
										
										
										
											2013-06-20 11:07:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 21:14:27 +03:00
										 |  |  |         // Double-forking.
 | 
					
						
							|  |  |  |         foreach (['single', 'double'] as $v) { | 
					
						
							|  |  |  |             switch ($pid = pcntl_fork()) { | 
					
						
							|  |  |  |                 case -1: // error
 | 
					
						
							|  |  |  |                     common_log(LOG_ERR, 'Could not fork.'); | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 case 0:  // child
 | 
					
						
							|  |  |  |                     if ($v === 'single') { | 
					
						
							|  |  |  |                         posix_setsid(); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 default: // parent
 | 
					
						
							|  |  |  |                     if ($v === 'double') { | 
					
						
							|  |  |  |                         common_log(LOG_INFO, 'Successfully forked.'); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     die(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-10 21:14:27 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function alreadyRunning() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $pidfilename = $this->pidFilename(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!$pidfilename) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!file_exists($pidfilename)) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $contents = file_get_contents($pidfilename); | 
					
						
							|  |  |  |         if (posix_kill(trim($contents), 0)) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function writePidFile() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $pidfilename = $this->pidFilename(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!$pidfilename) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return file_put_contents($pidfilename, posix_getpid() . "\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function clearPidFile() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $pidfilename = $this->pidFilename(); | 
					
						
							|  |  |  |         if (!$pidfilename) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return unlink($pidfilename); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function pidFilename() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $piddir = common_config('daemon', 'piddir'); | 
					
						
							|  |  |  |         if (!$piddir) { | 
					
						
							|  |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $name = $this->name(); | 
					
						
							|  |  |  |         if (!$name) { | 
					
						
							|  |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $piddir . '/' . $name . '.pid'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function changeUser() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $groupname = common_config('daemon', 'group'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($groupname) { | 
					
						
							|  |  |  |             $group_info = posix_getgrnam($groupname); | 
					
						
							|  |  |  |             if (!$group_info) { | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |                 common_log( | 
					
						
							|  |  |  |                     LOG_WARNING, | 
					
						
							|  |  |  |                     'Ignoring unknown group for daemon: ' . $groupname | 
					
						
							|  |  |  |                 ); | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 common_log(LOG_INFO, "Setting group to " . $groupname); | 
					
						
							|  |  |  |                 posix_setgid($group_info['gid']); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-08-08 23:17:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $username = common_config('daemon', 'user'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($username) { | 
					
						
							|  |  |  |             $user_info = posix_getpwnam($username); | 
					
						
							|  |  |  |             if (!$user_info) { | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |                 common_log( | 
					
						
							|  |  |  |                     LOG_WARNING, | 
					
						
							|  |  |  |                     'Ignoring unknown user for daemon: ' . $username | 
					
						
							|  |  |  |                 ); | 
					
						
							| 
									
										
										
										
											2012-08-08 23:17:51 -04:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 common_log(LOG_INFO, "Setting user to " . $username); | 
					
						
							|  |  |  |                 posix_setuid($user_info['uid']); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function runOnce() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         if ($this->alreadyRunning()) { | 
					
						
							|  |  |  |             common_log(LOG_INFO, $this->name() . ' already running. Exiting.'); | 
					
						
							|  |  |  |             exit(0); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-06-28 16:12:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($this->daemonize) { | 
					
						
							| 
									
										
										
										
											2009-06-28 17:41:16 -04:00
										 |  |  |             common_log(LOG_INFO, 'Backgrounding daemon "'.$this->name().'"'); | 
					
						
							| 
									
										
										
										
											2009-06-28 16:12:23 -04:00
										 |  |  |             $this->background(); | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-06-28 16:12:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $this->writePidFile(); | 
					
						
							|  |  |  |         $this->changeUser(); | 
					
						
							|  |  |  |         $this->run(); | 
					
						
							|  |  |  |         $this->clearPidFile(); | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 20:25:28 +03:00
										 |  |  |     public function run() | 
					
						
							| 
									
										
										
										
											2008-12-20 19:46:07 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-09-04 14:40:31 -04:00
										 |  |  | } |