mirror of
https://we.phorge.it/source/arcanist.git
synced 2025-01-09 14:21:01 +01:00
8e0e07664a
Summary: Ref T13098. Historically, Phabricator was split into three parts: - Phabricator, the server. - Arcanist, the client. - libphutil, libraries shared between the client and server. One imagined use case for this was that `libphutil` might become a general-purpose library that other projects would use. However, this didn't really happen, and it seems unlikely to at this point: Phabricator has become a relatively more sophisticated application platform; we didn't end up seeing or encouraging much custom development; what custom development there is basically embraces all of Phabricator since there are huge advantages to doing so; and a general "open source is awful" sort of factor here in the sense that open source users often don't have goals well aligned to our goals. Turning "arc" into a client platform and building package management solidify us in this direction of being a standalone platform, not a standalone utility library. Phabricator also depends on `arcanist/`. If it didn't, there would be a small advantage to saying "shared code + client for client, shared code + server for server", but there's no such distinction and it seems unlikely that one will ever exist. Even if it did, I think this has little value. Nowadays, I think this separation has no advantages for us and one significant cost: it makes installing `arcanist` more difficult for end-users. This will need some more finesssing (Phabricator will need some changes for compatibility, and a lot of stuff that still says "libphutil" or "phutil" may eventually want to say "arcanist"), and some stuff (like xhpast) is probably straight-up broken right now and needs some tweaking, but I don't anticipate any major issues here. There was never anything particularly magical about libphutil as a separate standalone library. Test Plan: Ran `arc`, it gets about as far as it did before. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T13098 Differential Revision: https://secure.phabricator.com/D19688
131 lines
3 KiB
PHP
Executable file
131 lines
3 KiB
PHP
Executable file
#!/usr/bin/env php
|
|
<?php
|
|
|
|
if (function_exists('pcntl_async_signals')) {
|
|
pcntl_async_signals(true);
|
|
} else {
|
|
declare(ticks = 1);
|
|
}
|
|
|
|
require_once dirname(__FILE__).'/../../__init_script__.php';
|
|
|
|
if (!posix_isatty(STDOUT)) {
|
|
$sid = posix_setsid();
|
|
if ($sid <= 0) {
|
|
throw new Exception(pht('Failed to create new process session!'));
|
|
}
|
|
}
|
|
|
|
$args = new PhutilArgumentParser($argv);
|
|
$args->setTagline(pht('daemon executor'));
|
|
$args->setSynopsis(<<<EOHELP
|
|
**exec_daemon.php** [__options__] __daemon__ ...
|
|
Run an instance of __daemon__.
|
|
EOHELP
|
|
);
|
|
$args->parse(
|
|
array(
|
|
array(
|
|
'name' => 'trace',
|
|
'help' => pht('Enable debug tracing.'),
|
|
),
|
|
array(
|
|
'name' => 'trace-memory',
|
|
'help' => pht('Enable debug memory tracing.'),
|
|
),
|
|
array(
|
|
'name' => 'verbose',
|
|
'help' => pht('Enable verbose activity logging.'),
|
|
),
|
|
array(
|
|
'name' => 'label',
|
|
'short' => 'l',
|
|
'param' => 'label',
|
|
'help' => pht(
|
|
'Optional process label. Makes "%s" nicer, no behavioral effects.',
|
|
'ps'),
|
|
),
|
|
array(
|
|
'name' => 'daemon',
|
|
'wildcard' => true,
|
|
),
|
|
));
|
|
|
|
$trace_memory = $args->getArg('trace-memory');
|
|
$trace_mode = $args->getArg('trace') || $trace_memory;
|
|
$verbose = $args->getArg('verbose');
|
|
|
|
if (function_exists('posix_isatty') && posix_isatty(STDIN)) {
|
|
fprintf(STDERR, pht('Reading daemon configuration from stdin...')."\n");
|
|
}
|
|
$config = @file_get_contents('php://stdin');
|
|
$config = id(new PhutilJSONParser())->parse($config);
|
|
|
|
PhutilTypeSpec::checkMap(
|
|
$config,
|
|
array(
|
|
'log' => 'optional string|null',
|
|
'argv' => 'optional list<wild>',
|
|
'load' => 'optional list<string>',
|
|
'down' => 'optional int',
|
|
));
|
|
|
|
$log = idx($config, 'log');
|
|
|
|
if ($log) {
|
|
ini_set('error_log', $log);
|
|
PhutilErrorHandler::setErrorListener(array('PhutilDaemon', 'errorListener'));
|
|
}
|
|
|
|
$load = idx($config, 'load', array());
|
|
foreach ($load as $library) {
|
|
$library = Filesystem::resolvePath($library);
|
|
phutil_load_library($library);
|
|
}
|
|
|
|
PhutilErrorHandler::initialize();
|
|
|
|
$daemon = $args->getArg('daemon');
|
|
if (!$daemon) {
|
|
throw new PhutilArgumentUsageException(
|
|
pht('Specify which class of daemon to start.'));
|
|
} else if (count($daemon) > 1) {
|
|
throw new PhutilArgumentUsageException(
|
|
pht('Specify exactly one daemon to start.'));
|
|
} else {
|
|
$daemon = head($daemon);
|
|
if (!class_exists($daemon)) {
|
|
throw new PhutilArgumentUsageException(
|
|
pht(
|
|
'No class "%s" exists in any known library.',
|
|
$daemon));
|
|
} else if (!is_subclass_of($daemon, 'PhutilDaemon')) {
|
|
throw new PhutilArgumentUsageException(
|
|
pht(
|
|
'Class "%s" is not a subclass of "%s".',
|
|
$daemon,
|
|
'PhutilDaemon'));
|
|
}
|
|
}
|
|
|
|
$argv = idx($config, 'argv', array());
|
|
$daemon = newv($daemon, array($argv));
|
|
|
|
if ($trace_mode) {
|
|
$daemon->setTraceMode();
|
|
}
|
|
|
|
if ($trace_memory) {
|
|
$daemon->setTraceMemory();
|
|
}
|
|
|
|
if ($verbose) {
|
|
$daemon->setVerbose(true);
|
|
}
|
|
|
|
$down_duration = idx($config, 'down');
|
|
if ($down_duration) {
|
|
$daemon->setScaledownDuration($down_duration);
|
|
}
|
|
|
|
$daemon->execute();
|