2011-03-14 15:43:56 -07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
2012-01-16 12:37:14 -08:00
|
|
|
* Copyright 2012 Facebook, Inc.
|
2011-03-14 15:43:56 -07:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
final class PhabricatorDaemonControl {
|
|
|
|
|
|
|
|
|
|
|
|
public function executeListCommand() {
|
|
|
|
$symbols = $this->loadAvailableDaemonClasses();
|
|
|
|
|
|
|
|
$symbols = igroup($symbols, 'library');
|
|
|
|
|
|
|
|
echo "\n";
|
|
|
|
foreach ($symbols as $library => $symbol_list) {
|
|
|
|
echo phutil_console_format("Daemons in library __%s__:\n", $library);
|
|
|
|
foreach ($symbol_list as $symbol) {
|
|
|
|
echo " ".$symbol['name']."\n";
|
|
|
|
}
|
|
|
|
echo "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function executeStatusCommand() {
|
|
|
|
$daemons = $this->loadRunningDaemons();
|
|
|
|
|
|
|
|
if (!$daemons) {
|
|
|
|
echo "There are no running Phabricator daemons.\n";
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(
|
|
|
|
"%-5s\t%-24s\t%s\n",
|
|
|
|
"PID",
|
|
|
|
"Started",
|
|
|
|
"Daemon");
|
|
|
|
foreach ($daemons as $daemon) {
|
2011-03-27 00:23:39 -07:00
|
|
|
$name = $daemon->getName();
|
|
|
|
if (!$daemon->isRunning()) {
|
|
|
|
$name = '<DEAD> '.$name;
|
|
|
|
if ($daemon->getPIDFile()) {
|
|
|
|
Filesystem::remove($daemon->getPIDFile());
|
|
|
|
}
|
|
|
|
}
|
2011-03-14 15:43:56 -07:00
|
|
|
printf(
|
|
|
|
"%5s\t%-24s\t%s\n",
|
|
|
|
$daemon->getPID(),
|
|
|
|
$daemon->getEpochStarted()
|
|
|
|
? date('M j Y, g:i:s A', $daemon->getEpochStarted())
|
|
|
|
: null,
|
2011-03-27 00:23:39 -07:00
|
|
|
$name);
|
2011-03-14 15:43:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-10-01 02:35:53 -04:00
|
|
|
public function executeStopCommand($pids = null) {
|
2011-03-14 15:43:56 -07:00
|
|
|
$daemons = $this->loadRunningDaemons();
|
|
|
|
if (!$daemons) {
|
|
|
|
echo "There are no running Phabricator daemons.\n";
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-10-01 02:35:53 -04:00
|
|
|
$daemons = mpull($daemons, null, 'getPID');
|
2011-03-14 15:43:56 -07:00
|
|
|
|
2011-10-01 02:35:53 -04:00
|
|
|
$running = array();
|
|
|
|
if ($pids == null) {
|
|
|
|
$running = $daemons;
|
|
|
|
} else {
|
|
|
|
// We were given a PID or set of PIDs to kill.
|
|
|
|
foreach ($pids as $key => $pid) {
|
2012-01-19 11:06:05 -08:00
|
|
|
if (!preg_match('/^\d+$/', $pid)) {
|
|
|
|
echo "'{$pid}' is not a valid PID.\n";
|
|
|
|
continue;
|
|
|
|
} else if (empty($daemons[$pid])) {
|
|
|
|
echo "'{$pid}' is not Phabricator-controlled PID. Not killing.\n";
|
2011-10-01 02:35:53 -04:00
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
$running[] = $daemons[$pid];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($running)) {
|
|
|
|
echo "No daemons to kill.\n";
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-16 12:37:14 -08:00
|
|
|
$all_daemons = $running;
|
|
|
|
|
2011-03-14 15:43:56 -07:00
|
|
|
foreach ($running as $key => $daemon) {
|
|
|
|
$pid = $daemon->getPID();
|
|
|
|
$name = $daemon->getName();
|
|
|
|
|
|
|
|
echo "Stopping daemon '{$name}' ({$pid})...\n";
|
|
|
|
if (!$daemon->isRunning()) {
|
|
|
|
echo "Daemon is not running.\n";
|
|
|
|
unset($running[$key]);
|
|
|
|
} else {
|
|
|
|
posix_kill($pid, SIGINT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$start = time();
|
|
|
|
do {
|
|
|
|
foreach ($running as $key => $daemon) {
|
|
|
|
$pid = $daemon->getPID();
|
|
|
|
if (!$daemon->isRunning()) {
|
|
|
|
echo "Daemon {$pid} exited normally.\n";
|
|
|
|
unset($running[$key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (empty($running)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
usleep(100000);
|
|
|
|
} while (time() < $start + 15);
|
|
|
|
|
|
|
|
foreach ($running as $key => $daemon) {
|
|
|
|
$pid = $daemon->getPID();
|
|
|
|
echo "KILLing daemon {$pid}.\n";
|
|
|
|
posix_kill($pid, SIGKILL);
|
|
|
|
}
|
|
|
|
|
2012-01-16 12:37:14 -08:00
|
|
|
foreach ($all_daemons as $daemon) {
|
2011-03-14 15:43:56 -07:00
|
|
|
if ($daemon->getPIDFile()) {
|
|
|
|
Filesystem::remove($daemon->getPIDFile());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public function executeHelpCommand() {
|
|
|
|
echo phutil_console_format(<<<EOHELP
|
|
|
|
**NAME**
|
|
|
|
**phd** - phabricator daemon launcher
|
|
|
|
|
|
|
|
**COMMAND REFERENCE**
|
|
|
|
|
2012-05-09 10:29:37 -07:00
|
|
|
**start**
|
|
|
|
Start the normal collection of daemons that Phabricator uses. This
|
|
|
|
is appropriate for most installs. If you want to customize what
|
|
|
|
is launched, you can use **launch** for fine-grained control.
|
|
|
|
|
|
|
|
**restart**
|
|
|
|
Stop all running daemons, then start a standard loadout.
|
|
|
|
|
|
|
|
**stop** [PID ...]
|
|
|
|
Stop all running daemons if no PIDs are given, or a particular
|
|
|
|
PID or set of PIDs, if they are supplied.
|
|
|
|
|
2011-05-15 09:29:48 -07:00
|
|
|
**launch** [__n__] __daemon__ [argv ...]
|
2011-06-13 10:01:06 -07:00
|
|
|
**debug** __daemon__ [argv ...]
|
2011-03-14 15:43:56 -07:00
|
|
|
Start a daemon (or n copies of a daemon).
|
2011-06-13 10:01:06 -07:00
|
|
|
With **debug**, do not daemonize. Use this if you're having trouble
|
|
|
|
getting daemons working.
|
2011-03-14 15:43:56 -07:00
|
|
|
|
|
|
|
**list**
|
|
|
|
List available daemons.
|
|
|
|
|
|
|
|
**status**
|
|
|
|
List running daemons.
|
|
|
|
|
|
|
|
**help**
|
|
|
|
Show this help.
|
|
|
|
|
2011-03-15 13:38:14 -07:00
|
|
|
**repository-launch-master**
|
2012-05-09 10:29:37 -07:00
|
|
|
DEPRECATED. Use 'phd start'.
|
2011-03-15 13:38:14 -07:00
|
|
|
|
|
|
|
**repository-launch-readonly**
|
2012-05-09 10:29:37 -07:00
|
|
|
DEPRECATED. Use 'phd launch pulllocal --no-discovery'.
|
2011-03-14 15:43:56 -07:00
|
|
|
|
|
|
|
EOHELP
|
|
|
|
);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-06-13 18:39:23 -07:00
|
|
|
public function pingConduit() {
|
|
|
|
// It's fairly common to have issues here, e.g. because Phabricator isn't
|
|
|
|
// running, isn't accessible, you put the domain in your hostsfile but it
|
|
|
|
// isn't available on the production host, etc. If any of this doesn't work,
|
|
|
|
// conduit will throw.
|
|
|
|
|
|
|
|
$conduit = new ConduitClient(PhabricatorEnv::getURI('/api/'));
|
|
|
|
$conduit->callMethodSynchronous('conduit.ping', array());
|
|
|
|
}
|
|
|
|
|
2011-06-13 18:07:58 -07:00
|
|
|
public function launchDaemon($daemon, array $argv, $debug = false) {
|
2011-03-15 13:38:14 -07:00
|
|
|
$symbols = $this->loadAvailableDaemonClasses();
|
|
|
|
$symbols = ipull($symbols, 'name', 'name');
|
|
|
|
if (empty($symbols[$daemon])) {
|
2012-01-25 11:50:59 -08:00
|
|
|
throw new Exception(
|
|
|
|
"Daemon '{$daemon}' is not loaded, misspelled or abstract.");
|
2011-03-15 13:38:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
$libphutil_root = dirname(phutil_get_library_root('phutil'));
|
|
|
|
$launch_daemon = $libphutil_root.'/scripts/daemon/';
|
|
|
|
|
|
|
|
foreach ($argv as $key => $arg) {
|
|
|
|
$argv[$key] = escapeshellarg($arg);
|
|
|
|
}
|
|
|
|
|
2012-06-19 12:56:41 -07:00
|
|
|
$flags = array();
|
|
|
|
if ($debug || PhabricatorEnv::getEnvConfig('phd.trace')) {
|
|
|
|
$flags[] = '--trace';
|
|
|
|
}
|
2011-04-15 16:22:37 -07:00
|
|
|
|
2012-06-19 12:56:41 -07:00
|
|
|
if ($debug || PhabricatorEnv::getEnvConfig('phd.verbose')) {
|
|
|
|
$flags[] = '--verbose';
|
|
|
|
}
|
2011-04-15 16:22:37 -07:00
|
|
|
|
2012-06-19 12:56:41 -07:00
|
|
|
if (!$debug) {
|
|
|
|
$flags[] = '--daemonize';
|
|
|
|
}
|
|
|
|
|
|
|
|
$bootloader = PhutilBootloader::getInstance();
|
|
|
|
foreach ($bootloader->getAllLibraries() as $library) {
|
|
|
|
if ($library == 'phutil') {
|
|
|
|
// No need to load libphutil, it's necessarily loaded implicitly by the
|
|
|
|
// daemon itself.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$flags[] = csprintf(
|
2011-04-15 16:22:37 -07:00
|
|
|
'--load-phutil-library=%s',
|
|
|
|
phutil_get_library_root($library));
|
|
|
|
}
|
|
|
|
|
2012-06-19 12:56:41 -07:00
|
|
|
$flags[] = csprintf('--conduit-uri=%s', PhabricatorEnv::getURI('/api/'));
|
2011-06-26 20:14:11 -07:00
|
|
|
|
|
|
|
if (!$debug) {
|
2012-06-19 12:56:41 -07:00
|
|
|
$log_dir = $this->getControlDirectory('log').'/daemons.log';
|
|
|
|
$flags[] = csprintf('--log=%s', $log_dir);
|
2011-06-26 20:14:11 -07:00
|
|
|
}
|
|
|
|
|
2012-06-19 12:56:41 -07:00
|
|
|
$pid_dir = $this->getControlDirectory('pid');
|
|
|
|
|
|
|
|
// TODO: This should be a much better user experience.
|
|
|
|
Filesystem::assertExists($pid_dir);
|
|
|
|
Filesystem::assertIsDirectory($pid_dir);
|
|
|
|
Filesystem::assertWritable($pid_dir);
|
|
|
|
|
|
|
|
$flags[] = csprintf('--phd=%s', $pid_dir);
|
|
|
|
|
|
|
|
$command = csprintf(
|
|
|
|
'./launch_daemon.php %s %C %C',
|
|
|
|
$daemon,
|
|
|
|
implode(' ', $flags),
|
|
|
|
implode(' ', $argv));
|
2011-03-15 20:55:05 -07:00
|
|
|
|
2011-06-13 10:01:06 -07:00
|
|
|
if ($debug) {
|
|
|
|
// Don't terminate when the user sends ^C; it will be sent to the
|
|
|
|
// subprocess which will terminate normally.
|
|
|
|
pcntl_signal(
|
|
|
|
SIGINT,
|
|
|
|
array('PhabricatorDaemonControl', 'ignoreSignal'));
|
|
|
|
|
|
|
|
echo "\n libphutil/scripts/daemon/ \$ {$command}\n\n";
|
|
|
|
|
|
|
|
phutil_passthru('(cd %s && exec %C)', $launch_daemon, $command);
|
|
|
|
} else {
|
|
|
|
$future = new ExecFuture('exec %C', $command);
|
|
|
|
// Play games to keep 'ps' looking reasonable.
|
|
|
|
$future->setCWD($launch_daemon);
|
|
|
|
$future->resolvex();
|
|
|
|
}
|
|
|
|
}
|
2011-03-15 20:55:05 -07:00
|
|
|
|
2011-06-13 10:01:06 -07:00
|
|
|
public static function ignoreSignal($signo) {
|
|
|
|
return;
|
2011-03-15 13:38:14 -07:00
|
|
|
}
|
2011-03-14 15:43:56 -07:00
|
|
|
|
2011-06-13 18:39:23 -07:00
|
|
|
public function getControlDirectory($dir) {
|
2011-06-12 21:11:32 -07:00
|
|
|
$path = PhabricatorEnv::getEnvConfig('phd.pid-directory').'/'.$dir;
|
|
|
|
if (!Filesystem::pathExists($path)) {
|
|
|
|
list($err) = exec_manual('mkdir -p %s', $path);
|
|
|
|
if ($err) {
|
|
|
|
throw new Exception(
|
|
|
|
"phd requires the directory '{$path}' to exist, but it does not ".
|
|
|
|
"exist and could not be created. Create this directory or update ".
|
|
|
|
"'phd.pid-directory' in your configuration to point to an existing ".
|
|
|
|
"directory.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $path;
|
2011-03-14 15:43:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function loadAvailableDaemonClasses() {
|
|
|
|
$loader = new PhutilSymbolLoader();
|
|
|
|
return $loader
|
|
|
|
->setAncestorClass('PhutilDaemon')
|
2012-01-25 11:50:59 -08:00
|
|
|
->setConcreteOnly(true)
|
2011-03-14 15:43:56 -07:00
|
|
|
->selectSymbolsWithoutLoading();
|
|
|
|
}
|
|
|
|
|
2012-05-09 10:29:37 -07:00
|
|
|
public function loadRunningDaemons() {
|
2011-03-14 15:43:56 -07:00
|
|
|
$results = array();
|
|
|
|
|
|
|
|
$pid_dir = $this->getControlDirectory('pid');
|
|
|
|
$pid_files = Filesystem::listDirectory($pid_dir);
|
|
|
|
if (!$pid_files) {
|
|
|
|
return $results;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($pid_files as $pid_file) {
|
|
|
|
$pid_data = Filesystem::readFile($pid_dir.'/'.$pid_file);
|
|
|
|
$dict = json_decode($pid_data, true);
|
|
|
|
if (!is_array($dict)) {
|
|
|
|
// Just return a hanging reference, since control code needs to be
|
|
|
|
// robust against unusual system states.
|
|
|
|
$dict = array();
|
|
|
|
}
|
|
|
|
$ref = PhabricatorDaemonReference::newFromDictionary($dict);
|
|
|
|
$ref->setPIDFile($pid_dir.'/'.$pid_file);
|
|
|
|
$results[] = $ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $results;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function killDaemon(PhabricatorDaemonReference $ref) {
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|