2017-07-21 21:15:43 +00:00
|
|
|
<?php
|
2017-11-17 01:23:18 +00:00
|
|
|
/** @license MIT
|
|
|
|
* Copyright 2017 J. King, Dustin Wilson et al.
|
|
|
|
* See LICENSE and AUTHORS files for details */
|
|
|
|
|
2017-07-21 21:15:43 +00:00
|
|
|
declare(strict_types=1);
|
2021-04-14 15:17:01 +00:00
|
|
|
|
2017-07-21 21:15:43 +00:00
|
|
|
namespace JKingWeb\Arsse;
|
|
|
|
|
2019-03-25 14:45:05 +00:00
|
|
|
use JKingWeb\Arsse\REST\Fever\User as Fever;
|
2019-04-01 21:24:19 +00:00
|
|
|
use JKingWeb\Arsse\ImportExport\OPML;
|
2021-02-11 02:40:51 +00:00
|
|
|
use JKingWeb\Arsse\REST\Miniflux\Token as Miniflux;
|
2021-06-15 20:58:54 +00:00
|
|
|
use JKingWeb\Arsse\Service\Daemon;
|
2023-11-01 22:40:16 +00:00
|
|
|
use GetOpt\GetOpt;
|
|
|
|
use GetOpt\Command;
|
|
|
|
use GetOpt\Operand;
|
|
|
|
use GetOpt\Option;
|
2023-12-26 14:14:59 +00:00
|
|
|
use GetOpt\ArgumentException;
|
2018-11-05 14:08:50 +00:00
|
|
|
|
2017-07-21 21:15:43 +00:00
|
|
|
class CLI {
|
2023-12-26 14:14:59 +00:00
|
|
|
protected $cli;
|
2019-04-01 21:24:19 +00:00
|
|
|
|
2023-12-26 14:14:59 +00:00
|
|
|
public function __construct() {
|
2023-12-26 01:59:07 +00:00
|
|
|
$cli = new GetOpt([], []);
|
2023-11-01 22:40:16 +00:00
|
|
|
$cli->addOptions([
|
|
|
|
Option::create("h", "help"),
|
|
|
|
Option::create(null, "version"),
|
|
|
|
]);
|
|
|
|
$cli->addCommands([
|
|
|
|
Command::create("user list", [$this, "userList"]),
|
|
|
|
Command::create("user add", [$this, "userAdd"])
|
|
|
|
->addOperand(Operand::create("username", operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("password", Operand::OPTIONAL))
|
|
|
|
->addOption(Option::create(null, "admin")),
|
|
|
|
Command::create("user remove", [$this, "userRemove"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED)),
|
|
|
|
Command::create("user show", [$this, "userShow"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED)),
|
|
|
|
Command::create("user set", [$this, "userSet"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("property", Operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("value", Operand::REQUIRED)),
|
|
|
|
Command::create("user unset", [$this, "userUnset"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("property", Operand::REQUIRED)),
|
|
|
|
Command::create("user set-pass", [$this, "userSetPass"])
|
|
|
|
->addOperand(Operand::create("username", operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("password", Operand::OPTIONAL))
|
|
|
|
->addOption(Option::create(null, "fever")),
|
|
|
|
Command::create("user unset-pass", [$this, "userUnsetPass"])
|
|
|
|
->addOperand(Operand::create("username", operand::REQUIRED))
|
|
|
|
->addOption(Option::create(null, "fever")),
|
|
|
|
Command::create("user auth", [$this, "userAuth"])
|
|
|
|
->addOperand(Operand::create("username", operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("password", Operand::REQUIRED))
|
|
|
|
->addOption(Option::create(null, "fever")),
|
|
|
|
Command::create("token list", [$this, "tokenList"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED)),
|
|
|
|
Command::create("token create", [$this, "tokenCreate"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("label", Operand::OPTIONAL)),
|
|
|
|
Command::create("token revoke", [$this, "tokenRevoke"])
|
|
|
|
->addOperand(Operand::create("username", Operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("token", Operand::OPTIONAL)),
|
|
|
|
Command::create("import", [$this, "import"])
|
|
|
|
->addOperand(Operand::create("username", operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("file", Operand::OPTIONAL))
|
|
|
|
->addOption(Option::create("f", "flat"))
|
|
|
|
->addOption(Option::create("r", "replace")),
|
|
|
|
Command::create("export", [$this, "export"])
|
|
|
|
->addOperand(Operand::create("username", operand::REQUIRED))
|
|
|
|
->addOperand(Operand::create("file", Operand::OPTIONAL))
|
|
|
|
->addOption(Option::create("f", "flat")),
|
|
|
|
Command::create("daemon", [$this, "daemon"])
|
|
|
|
->addOption(Option::create(null, "fork", GetOpt::REQUIRED_ARGUMENT)->setArgumentName("pidfile")),
|
|
|
|
Command::create("feed refresh-all", [$this, "feedRefreshAll"]),
|
|
|
|
Command::create("feed refresh", [$this, "feedRefresh"])
|
|
|
|
->addOperand(Operand::create("n", Operand::REQUIRED)),
|
|
|
|
Command::create("conf save-defaults", [$this, "confSaveDefaults"])
|
|
|
|
->addOperand(Operand::create("file", Operand::OPTIONAL)),
|
2018-11-06 17:32:28 +00:00
|
|
|
]);
|
2023-12-26 14:14:59 +00:00
|
|
|
$this->cli = $cli;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function usage(string $prog, bool $help = false): string {
|
|
|
|
$cli = $this->cli;
|
|
|
|
$out = "Usage:\n";
|
|
|
|
foreach ($cli->getCommands() as $cmd) {
|
|
|
|
$out .= " $prog ".$cmd->getName();
|
|
|
|
foreach ($cmd->getOperands() as $op) {
|
|
|
|
$rep = "<".$op->getName().">";
|
|
|
|
if (!$op->isRequired()) {
|
|
|
|
$rep = "[$rep]";
|
|
|
|
}
|
|
|
|
$out .= " $rep";
|
|
|
|
}
|
|
|
|
foreach ($cmd->getOptions() as $op) {
|
|
|
|
$arg = $op->getArgument();
|
|
|
|
$rep = "--".$op->getName()."";
|
|
|
|
if ($op->getMode() === GetOpt::REQUIRED_ARGUMENT) {
|
|
|
|
$rep = "$rep=".$arg->getName();
|
|
|
|
}
|
|
|
|
if ($short = $op->getShort()) {
|
|
|
|
$srep = "-$short";
|
|
|
|
$rep = "$srep|$rep";
|
|
|
|
}
|
|
|
|
$out .= " [$rep]";
|
|
|
|
}
|
|
|
|
$out .= "\n";
|
|
|
|
}
|
|
|
|
foreach ($cli->getOptionObjects() as $op) {
|
|
|
|
$rep = "--".$op->getName()."";
|
|
|
|
if ($short = $op->getShort()) {
|
|
|
|
$srep = "-$short";
|
|
|
|
$rep = "$srep|$rep";
|
|
|
|
}
|
|
|
|
$out .= " $prog $rep\n";
|
|
|
|
}
|
|
|
|
if ($help) {
|
|
|
|
$out .= <<<HELP_TEXT
|
|
|
|
\nThe Arsse command-line interface can be used to perform various administrative
|
|
|
|
tasks such as starting the newsfeed refresh service, managing users, and
|
|
|
|
importing or exporting data.
|
|
|
|
|
|
|
|
See the manual page for more details:
|
|
|
|
|
|
|
|
man arsse\n
|
|
|
|
HELP_TEXT;
|
|
|
|
}
|
|
|
|
return $out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @codeCoverageIgnore */
|
|
|
|
protected function loadConf(): bool {
|
|
|
|
Arsse::bootstrap();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function resolveFile($file, string $mode): string {
|
|
|
|
// TODO: checking read/write permissions on the provided path may be useful
|
|
|
|
$stdinOrStdout = in_array($mode, ["r", "r+"]) ? "php://input" : "php://output";
|
|
|
|
return ($file === "-" ? null : $file) ?? $stdinOrStdout;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function dispatch(array $argv = null): int {
|
|
|
|
$cli = $this->cli;
|
|
|
|
$argv = $argv ?? $_SERVER['argv'];
|
|
|
|
$prog = array_shift($argv);
|
2018-11-05 14:08:50 +00:00
|
|
|
try {
|
2023-12-26 14:14:59 +00:00
|
|
|
$cli->process($argv);
|
2021-07-03 16:38:48 +00:00
|
|
|
// ensure the require extensions are loaded
|
|
|
|
Arsse::checkExtensions(...Arsse::REQUIRED_EXTENSIONS);
|
2023-12-26 01:59:07 +00:00
|
|
|
$cmd = $cli->getCommand();
|
|
|
|
$cmd = $cmd ? $cmd->getName() : "";
|
2021-06-06 19:20:53 +00:00
|
|
|
if ($cmd && !in_array($cmd, ["", "conf save-defaults", "daemon"])) {
|
2021-06-06 20:38:11 +00:00
|
|
|
// only certain commands don't require configuration to be loaded; daemon loads configuration after forking (if applicable)
|
2019-04-01 21:24:19 +00:00
|
|
|
$this->loadConf();
|
|
|
|
}
|
2021-07-03 16:38:48 +00:00
|
|
|
// run the requested command
|
2019-04-01 21:24:19 +00:00
|
|
|
switch ($cmd) {
|
2021-02-10 17:46:28 +00:00
|
|
|
case "":
|
2023-12-26 01:59:07 +00:00
|
|
|
if ($cli->getOption("version")) {
|
2021-02-10 17:46:28 +00:00
|
|
|
echo Arsse::VERSION.\PHP_EOL;
|
2023-12-26 01:59:07 +00:00
|
|
|
} else {
|
2023-12-26 14:14:59 +00:00
|
|
|
echo $this->usage($prog, (bool) $cli->getOption("help"));
|
2021-02-10 17:46:28 +00:00
|
|
|
}
|
2018-11-06 17:32:28 +00:00
|
|
|
return 0;
|
2018-11-05 14:08:50 +00:00
|
|
|
case "daemon":
|
2023-12-26 01:59:07 +00:00
|
|
|
if (($fork = $cli->getOption("fork")) !== null) {
|
|
|
|
return $this->serviceFork($fork);
|
2021-06-15 20:58:54 +00:00
|
|
|
} else {
|
|
|
|
$this->loadConf();
|
|
|
|
Arsse::$obj->get(Service::class)->watch(true);
|
2021-06-06 22:54:24 +00:00
|
|
|
}
|
2018-11-06 17:32:28 +00:00
|
|
|
return 0;
|
2018-11-05 14:08:50 +00:00
|
|
|
case "feed refresh":
|
2023-12-26 01:59:07 +00:00
|
|
|
return (int) !Arsse::$db->feedUpdate((int) $cli->getOperand("n"), true);
|
2019-03-02 19:59:44 +00:00
|
|
|
case "feed refresh-all":
|
2021-02-07 04:51:23 +00:00
|
|
|
Arsse::$obj->get(Service::class)->watch(false);
|
2019-03-02 19:59:44 +00:00
|
|
|
return 0;
|
2018-11-05 14:08:50 +00:00
|
|
|
case "conf save-defaults":
|
2023-12-26 01:59:07 +00:00
|
|
|
$file = $this->resolveFile($cli->getOperand("file"), "w");
|
2021-02-07 04:51:23 +00:00
|
|
|
return (int) !Arsse::$obj->get(Conf::class)->exportFile($file, true);
|
2019-04-01 21:24:19 +00:00
|
|
|
case "export":
|
2023-12-26 01:59:07 +00:00
|
|
|
$u = $cli->getOperand("username");
|
|
|
|
$file = $this->resolveFile($cli->getOperand("file"), "w");
|
|
|
|
return (int) !Arsse::$obj->get(OPML::class)->exportFile($file, $u, (bool) $cli->getOption("flat"));
|
2019-05-01 14:46:44 +00:00
|
|
|
case "import":
|
2023-12-26 01:59:07 +00:00
|
|
|
$u = $cli->getOperand("username");
|
|
|
|
$file = $this->resolveFile($cli->getOperand("file"), "r");
|
|
|
|
return (int) !Arsse::$obj->get(OPML::class)->importFile($file, $u, (bool) $cli->getOption("flat"), (bool) $cli->getOption("replace"));
|
2021-02-11 02:40:51 +00:00
|
|
|
case "token list":
|
2023-12-26 14:14:59 +00:00
|
|
|
return $this->tokenList($cli->getOperand("username"));
|
2021-02-11 02:40:51 +00:00
|
|
|
case "token create":
|
2023-12-26 01:59:07 +00:00
|
|
|
echo Arsse::$obj->get(Miniflux::class)->tokenGenerate($cli->getOperand("username"), $cli->getOperand("label")).\PHP_EOL;
|
2021-02-11 02:40:51 +00:00
|
|
|
return 0;
|
|
|
|
case "token revoke":
|
2023-12-26 01:59:07 +00:00
|
|
|
Arsse::$db->tokenRevoke($cli->getOperand("username"), "miniflux.login", $cli->getOperand("token"));
|
2021-02-11 02:40:51 +00:00
|
|
|
return 0;
|
2021-02-10 17:46:28 +00:00
|
|
|
case "user add":
|
2023-12-26 01:59:07 +00:00
|
|
|
$out = $this->userAddOrSetPassword("add", $cli->getOperand("username"), $cli->getOperand("password"));
|
|
|
|
if ($cli->getOption("admin")) {
|
|
|
|
Arsse::$user->propertiesSet($cli->getOperand("username"), ['admin' => true]);
|
2021-02-10 17:46:28 +00:00
|
|
|
}
|
|
|
|
return $out;
|
|
|
|
case "user set-pass":
|
2023-12-26 01:59:07 +00:00
|
|
|
if ($cli->getOption("fever")) {
|
|
|
|
$passwd = Arsse::$obj->get(Fever::class)->register($cli->getOperand("username"), $cli->getOperand("password"));
|
|
|
|
if ($cli->getOperand("password") === null) {
|
2021-02-10 17:46:28 +00:00
|
|
|
echo $passwd.\PHP_EOL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
} else {
|
2023-12-26 01:59:07 +00:00
|
|
|
return $this->userAddOrSetPassword("passwordSet", $cli->getOperand("username"), $cli->getOperand("password"));
|
2021-02-10 17:46:28 +00:00
|
|
|
}
|
|
|
|
// no break
|
|
|
|
case "user unset-pass":
|
2023-12-26 01:59:07 +00:00
|
|
|
if ($cli->getOption("fever")) {
|
|
|
|
Arsse::$obj->get(Fever::class)->unregister($cli->getOperand("username"));
|
2021-02-10 17:46:28 +00:00
|
|
|
} else {
|
2023-12-26 01:59:07 +00:00
|
|
|
Arsse::$user->passwordUnset($cli->getOperand("username"));
|
2021-02-10 17:46:28 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
case "user remove":
|
2023-12-26 01:59:07 +00:00
|
|
|
return (int) !Arsse::$user->remove($cli->getOperand("username"));
|
2021-02-10 17:46:28 +00:00
|
|
|
case "user show":
|
2023-12-26 01:59:07 +00:00
|
|
|
return $this->userShowProperties($cli->getOperand("username"));
|
2021-02-10 17:46:28 +00:00
|
|
|
case "user set":
|
2023-12-26 01:59:07 +00:00
|
|
|
return (int) !Arsse::$user->propertiesSet($cli->getOperand("username"), [$cli->getOperand("property") => $cli->getOperand("value")]);
|
2021-02-10 17:46:28 +00:00
|
|
|
case "user unset":
|
2023-12-26 01:59:07 +00:00
|
|
|
return (int) !Arsse::$user->propertiesSet($cli->getOperand("username"), [$cli->getOperand("property") => null]);
|
2021-02-10 17:46:28 +00:00
|
|
|
case "user auth":
|
2023-12-26 01:59:07 +00:00
|
|
|
return $this->userAuthenticate($cli->getOperand("username"), $cli->getOperand("password"), (bool) $cli->getOption("fever"));
|
2021-02-10 17:46:28 +00:00
|
|
|
case "user list":
|
|
|
|
return $this->userList();
|
2021-02-11 02:40:51 +00:00
|
|
|
default:
|
|
|
|
throw new Exception("constantUnknown", $cmd); // @codeCoverageIgnore
|
2017-08-28 23:38:58 +00:00
|
|
|
}
|
2023-12-26 14:14:59 +00:00
|
|
|
} catch (ArgumentException $e) {
|
|
|
|
// invalid command was entered
|
|
|
|
$this->logError(trim($this->usage($prog)));
|
|
|
|
return 1;
|
2018-11-05 14:08:50 +00:00
|
|
|
} catch (AbstractException $e) {
|
2023-12-26 14:14:59 +00:00
|
|
|
// some other error
|
2018-12-08 01:03:04 +00:00
|
|
|
$this->logError($e->getMessage());
|
2018-11-05 14:08:50 +00:00
|
|
|
return $e->getCode();
|
2017-07-21 21:15:43 +00:00
|
|
|
}
|
2019-10-17 17:00:56 +00:00
|
|
|
} // @codeCoverageIgnore
|
2017-07-21 21:15:43 +00:00
|
|
|
|
2018-12-08 01:03:04 +00:00
|
|
|
/** @codeCoverageIgnore */
|
2020-01-20 18:34:03 +00:00
|
|
|
protected function logError(string $msg): void {
|
2019-01-23 21:34:54 +00:00
|
|
|
fwrite(STDERR, $msg.\PHP_EOL);
|
2018-12-08 01:03:04 +00:00
|
|
|
}
|
|
|
|
|
2021-06-15 20:58:54 +00:00
|
|
|
protected function serviceFork(string $pidfile): int {
|
|
|
|
// initialize the object factory
|
|
|
|
Arsse::$obj = Arsse::$obj ?? new Factory;
|
|
|
|
// create a Daemon object which contains various helper functions
|
|
|
|
$daemon = Arsse::$obj->get(Daemon::class);
|
|
|
|
// resolve the PID file to its absolute path; this also checks its readability and writability
|
2021-06-16 16:48:09 +00:00
|
|
|
$pidfile = $daemon->checkPIDFilePath($pidfile);
|
2021-06-15 20:58:54 +00:00
|
|
|
// daemonize
|
|
|
|
$daemon->fork($pidfile);
|
|
|
|
// start the fetching service as normal
|
|
|
|
$this->loadConf();
|
|
|
|
Arsse::$obj->get(Service::class)->watch(true);
|
|
|
|
// after the service has been shut down, delete the PID file and exit cleanly
|
|
|
|
unlink($pidfile);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-11-05 14:08:50 +00:00
|
|
|
protected function userAddOrSetPassword(string $method, string $user, string $password = null, string $oldpass = null): int {
|
2018-11-06 17:32:28 +00:00
|
|
|
$passwd = Arsse::$user->$method(...array_slice(func_get_args(), 1));
|
2017-08-29 14:50:31 +00:00
|
|
|
if (is_null($password)) {
|
2017-09-28 23:25:31 +00:00
|
|
|
echo $passwd.\PHP_EOL;
|
2017-08-28 23:38:58 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2018-11-05 14:08:50 +00:00
|
|
|
|
|
|
|
protected function userList(): int {
|
|
|
|
$list = Arsse::$user->list();
|
|
|
|
if ($list) {
|
|
|
|
echo implode(\PHP_EOL, $list).\PHP_EOL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-03-25 14:45:05 +00:00
|
|
|
protected function userAuthenticate(string $user, string $password, bool $fever = false): int {
|
2021-02-07 04:51:23 +00:00
|
|
|
$result = $fever ? Arsse::$obj->get(Fever::class)->authenticate($user, $password) : Arsse::$user->auth($user, $password);
|
2019-03-25 14:45:05 +00:00
|
|
|
if ($result) {
|
2018-11-05 14:08:50 +00:00
|
|
|
echo Arsse::$lang->msg("CLI.Auth.Success").\PHP_EOL;
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
echo Arsse::$lang->msg("CLI.Auth.Failure").\PHP_EOL;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
2021-02-10 16:24:01 +00:00
|
|
|
|
|
|
|
protected function userShowProperties(string $user): int {
|
|
|
|
$data = Arsse::$user->propertiesGet($user);
|
|
|
|
$len = array_reduce(array_keys($data), function($carry, $item) {
|
|
|
|
return max($carry, strlen($item));
|
|
|
|
}, 0) + 2;
|
|
|
|
foreach ($data as $k => $v) {
|
|
|
|
echo str_pad($k, $len, " ");
|
|
|
|
echo var_export($v, true).\PHP_EOL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2021-02-11 02:40:51 +00:00
|
|
|
|
|
|
|
protected function tokenList(string $user): int {
|
|
|
|
$list = Arsse::$obj->get(Miniflux::class)->tokenList($user);
|
|
|
|
usort($list, function($v1, $v2) {
|
|
|
|
return $v1['label'] <=> $v2['label'];
|
|
|
|
});
|
|
|
|
foreach ($list as $t) {
|
|
|
|
echo $t['id']." ".$t['label'].\PHP_EOL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2017-08-29 14:50:31 +00:00
|
|
|
}
|