2018-11-06 12:32:28 -05:00
|
|
|
<?php
|
|
|
|
/** @license MIT
|
|
|
|
* Copyright 2017 J. King, Dustin Wilson et al.
|
|
|
|
* See LICENSE and AUTHORS files for details */
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace JKingWeb\Arsse\TestCase\CLI;
|
|
|
|
|
|
|
|
use JKingWeb\Arsse\Arsse;
|
|
|
|
use JKingWeb\Arsse\Conf;
|
|
|
|
use JKingWeb\Arsse\User;
|
|
|
|
use JKingWeb\Arsse\Database;
|
|
|
|
use JKingWeb\Arsse\Service;
|
|
|
|
use JKingWeb\Arsse\CLI;
|
2019-03-25 10:45:05 -04:00
|
|
|
use JKingWeb\Arsse\REST\Fever\User as FeverUser;
|
2019-04-01 17:24:19 -04:00
|
|
|
use JKingWeb\Arsse\ImportExport\OPML;
|
2018-11-06 12:32:28 -05:00
|
|
|
use Phake;
|
|
|
|
|
|
|
|
/** @covers \JKingWeb\Arsse\CLI */
|
|
|
|
class TestCLI extends \JKingWeb\Arsse\Test\AbstractTest {
|
|
|
|
public function setUp() {
|
2018-11-23 10:01:17 -05:00
|
|
|
self::clearData(false);
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->cli = Phake::partialMock(CLI::class);
|
|
|
|
Phake::when($this->cli)->logError->thenReturn(null);
|
2018-11-06 12:32:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public function assertConsole(CLI $cli, string $command, int $exitStatus, string $output = "", bool $pattern = false) {
|
|
|
|
$argv = \Clue\Arguments\split($command);
|
|
|
|
$output = strlen($output) ? $output.\PHP_EOL : "";
|
|
|
|
if ($pattern) {
|
2018-12-05 17:28:11 -05:00
|
|
|
$this->expectOutputRegex($output);
|
2018-11-06 12:32:28 -05:00
|
|
|
} else {
|
|
|
|
$this->expectOutputString($output);
|
|
|
|
}
|
|
|
|
$this->assertSame($exitStatus, $cli->dispatch($argv));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function assertLoaded(bool $loaded) {
|
|
|
|
$r = new \ReflectionClass(Arsse::class);
|
|
|
|
$props = array_keys($r->getStaticProperties());
|
|
|
|
foreach ($props as $prop) {
|
|
|
|
if ($loaded) {
|
|
|
|
$this->assertNotNull(Arsse::$$prop, "Global $prop object should be loaded");
|
|
|
|
} else {
|
|
|
|
$this->assertNull(Arsse::$$prop, "Global $prop object should not be loaded");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testPrintVersion() {
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, "arsse.php --version", 0, Arsse::VERSION);
|
2018-11-06 12:32:28 -05:00
|
|
|
$this->assertLoaded(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @dataProvider provideHelpText */
|
|
|
|
public function testPrintHelp(string $cmd, string $name) {
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, 0, str_replace("arsse.php", $name, CLI::USAGE));
|
2018-11-06 12:32:28 -05:00
|
|
|
$this->assertLoaded(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideHelpText() {
|
|
|
|
return [
|
|
|
|
["arsse.php --help", "arsse.php"],
|
|
|
|
["arsse --help", "arsse"],
|
|
|
|
["thearsse --help", "thearsse"],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testStartTheDaemon() {
|
|
|
|
$srv = Phake::mock(Service::class);
|
|
|
|
Phake::when($srv)->watch->thenReturn(new \DateTimeImmutable);
|
2019-04-01 17:24:19 -04:00
|
|
|
Phake::when($this->cli)->getInstance(Service::class)->thenReturn($srv);
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, "arsse.php daemon", 0);
|
2018-11-06 12:32:28 -05:00
|
|
|
$this->assertLoaded(true);
|
|
|
|
Phake::verify($srv)->watch(true);
|
2019-04-01 17:24:19 -04:00
|
|
|
Phake::verify($this->cli)->getInstance(Service::class);
|
2018-11-06 12:32:28 -05:00
|
|
|
}
|
|
|
|
|
2019-03-02 14:59:44 -05:00
|
|
|
public function testRefreshAllFeeds() {
|
|
|
|
$srv = Phake::mock(Service::class);
|
|
|
|
Phake::when($srv)->watch->thenReturn(new \DateTimeImmutable);
|
2019-04-01 17:24:19 -04:00
|
|
|
Phake::when($this->cli)->getInstance(Service::class)->thenReturn($srv);
|
2019-03-02 14:59:44 -05:00
|
|
|
$this->assertConsole($this->cli, "arsse.php feed refresh-all", 0);
|
|
|
|
$this->assertLoaded(true);
|
|
|
|
Phake::verify($srv)->watch(false);
|
2019-04-01 17:24:19 -04:00
|
|
|
Phake::verify($this->cli)->getInstance(Service::class);
|
2019-03-02 14:59:44 -05:00
|
|
|
}
|
|
|
|
|
2018-11-06 12:32:28 -05:00
|
|
|
/** @dataProvider provideFeedUpdates */
|
|
|
|
public function testRefreshAFeed(string $cmd, int $exitStatus, string $output) {
|
|
|
|
Arsse::$db = Phake::mock(Database::class);
|
|
|
|
Phake::when(Arsse::$db)->feedUpdate(1, true)->thenReturn(true);
|
|
|
|
Phake::when(Arsse::$db)->feedUpdate(2, true)->thenThrow(new \JKingWeb\Arsse\Feed\Exception("http://example.com/", new \PicoFeed\Client\InvalidUrlException));
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
2018-11-06 12:32:28 -05:00
|
|
|
$this->assertLoaded(true);
|
|
|
|
Phake::verify(Arsse::$db)->feedUpdate;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideFeedUpdates() {
|
|
|
|
return [
|
|
|
|
["arsse.php feed refresh 1", 0, ""],
|
|
|
|
["arsse.php feed refresh 2", 10502, ""],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @dataProvider provideDefaultConfigurationSaves */
|
|
|
|
public function testSaveTheDefaultConfiguration(string $cmd, int $exitStatus, string $file) {
|
|
|
|
$conf = Phake::mock(Conf::class);
|
|
|
|
Phake::when($conf)->exportFile("php://output", true)->thenReturn(true);
|
|
|
|
Phake::when($conf)->exportFile("good.conf", true)->thenReturn(true);
|
|
|
|
Phake::when($conf)->exportFile("bad.conf", true)->thenThrow(new \JKingWeb\Arsse\Conf\Exception("fileUnwritable"));
|
2019-04-01 17:24:19 -04:00
|
|
|
Phake::when($this->cli)->getInstance(Conf::class)->thenReturn($conf);
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus);
|
2018-11-06 12:32:28 -05:00
|
|
|
$this->assertLoaded(false);
|
|
|
|
Phake::verify($conf)->exportFile($file, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideDefaultConfigurationSaves() {
|
|
|
|
return [
|
|
|
|
["arsse.php conf save-defaults", 0, "php://output"],
|
|
|
|
["arsse.php conf save-defaults -", 0, "php://output"],
|
|
|
|
["arsse.php conf save-defaults good.conf", 0, "good.conf"],
|
|
|
|
["arsse.php conf save-defaults bad.conf", 10304, "bad.conf"],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @dataProvider provideUserList */
|
|
|
|
public function testListUsers(string $cmd, array $list, int $exitStatus, string $output) {
|
2018-12-07 19:21:44 -05:00
|
|
|
// FIXME: Phake is somehow unable to mock the User class correctly, so we use PHPUnit's mocks instead
|
2018-11-06 16:35:33 -05:00
|
|
|
Arsse::$user = $this->createMock(User::class);
|
|
|
|
Arsse::$user->method("list")->willReturn($list);
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
2018-11-06 12:32:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public function provideUserList() {
|
|
|
|
$list = ["john.doe@example.com", "jane.doe@example.com"];
|
|
|
|
$str = implode(PHP_EOL, $list);
|
|
|
|
return [
|
|
|
|
["arsse.php user list", $list, 0, $str],
|
|
|
|
["arsse.php user", $list, 0, $str],
|
|
|
|
["arsse.php user list", [], 0, ""],
|
|
|
|
["arsse.php user", [], 0, ""],
|
|
|
|
];
|
|
|
|
}
|
2018-11-06 16:35:33 -05:00
|
|
|
|
|
|
|
/** @dataProvider provideUserAdditions */
|
|
|
|
public function testAddAUser(string $cmd, int $exitStatus, string $output) {
|
2018-12-07 19:21:44 -05:00
|
|
|
// FIXME: Phake is somehow unable to mock the User class correctly, so we use PHPUnit's mocks instead
|
2018-11-06 16:35:33 -05:00
|
|
|
Arsse::$user = $this->createMock(User::class);
|
|
|
|
Arsse::$user->method("add")->will($this->returnCallback(function($user, $pass = null) {
|
|
|
|
switch ($user) {
|
|
|
|
case "john.doe@example.com":
|
|
|
|
throw new \JKingWeb\Arsse\User\Exception("alreadyExists");
|
|
|
|
case "jane.doe@example.com":
|
|
|
|
return is_null($pass) ? "random password" : $pass;
|
|
|
|
}
|
|
|
|
}));
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
2018-11-06 16:35:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public function provideUserAdditions() {
|
|
|
|
return [
|
|
|
|
["arsse.php user add john.doe@example.com", 10403, ""],
|
|
|
|
["arsse.php user add jane.doe@example.com", 0, "random password"],
|
|
|
|
["arsse.php user add jane.doe@example.com superman", 0, ""],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @dataProvider provideUserAuthentication */
|
|
|
|
public function testAuthenticateAUser(string $cmd, int $exitStatus, string $output) {
|
2018-12-07 19:21:44 -05:00
|
|
|
// FIXME: Phake is somehow unable to mock the User class correctly, so we use PHPUnit's mocks instead
|
2018-11-06 16:35:33 -05:00
|
|
|
Arsse::$user = $this->createMock(User::class);
|
|
|
|
Arsse::$user->method("auth")->will($this->returnCallback(function($user, $pass) {
|
|
|
|
return (
|
2019-01-11 10:38:06 -05:00
|
|
|
($user === "john.doe@example.com" && $pass === "secret") ||
|
|
|
|
($user === "jane.doe@example.com" && $pass === "superman")
|
2018-11-06 16:35:33 -05:00
|
|
|
);
|
|
|
|
}));
|
2019-03-25 10:45:05 -04:00
|
|
|
$fever = \Phake::mock(FeverUser::class);
|
|
|
|
\Phake::when($fever)->authenticate->thenReturn(false);
|
|
|
|
\Phake::when($fever)->authenticate("john.doe@example.com", "ashalla")->thenReturn(true);
|
2019-03-25 14:24:58 -04:00
|
|
|
\Phake::when($fever)->authenticate("jane.doe@example.com", "thx1138")->thenReturn(true);
|
2019-04-01 17:24:19 -04:00
|
|
|
\Phake::when($this->cli)->getInstance(FeverUser::class)->thenReturn($fever);
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
2018-11-06 16:35:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public function provideUserAuthentication() {
|
|
|
|
$l = new \JKingWeb\Arsse\Lang;
|
2019-03-25 10:45:05 -04:00
|
|
|
$success = $l("CLI.Auth.Success");
|
|
|
|
$failure = $l("CLI.Auth.Failure");
|
2018-11-06 16:35:33 -05:00
|
|
|
return [
|
2019-03-25 10:45:05 -04:00
|
|
|
["arsse.php user auth john.doe@example.com secret", 0, $success],
|
|
|
|
["arsse.php user auth john.doe@example.com superman", 1, $failure],
|
|
|
|
["arsse.php user auth jane.doe@example.com secret", 1, $failure],
|
|
|
|
["arsse.php user auth jane.doe@example.com superman", 0, $success],
|
|
|
|
["arsse.php user auth john.doe@example.com ashalla --fever", 0, $success],
|
|
|
|
["arsse.php user auth john.doe@example.com thx1138 --fever", 1, $failure],
|
|
|
|
["arsse.php user auth --fever jane.doe@example.com ashalla", 1, $failure],
|
|
|
|
["arsse.php user auth --fever jane.doe@example.com thx1138", 0, $success],
|
2018-11-06 16:35:33 -05:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @dataProvider provideUserRemovals */
|
|
|
|
public function testRemoveAUser(string $cmd, int $exitStatus, string $output) {
|
2018-12-07 19:21:44 -05:00
|
|
|
// FIXME: Phake is somehow unable to mock the User class correctly, so we use PHPUnit's mocks instead
|
2018-11-06 16:35:33 -05:00
|
|
|
Arsse::$user = $this->createMock(User::class);
|
|
|
|
Arsse::$user->method("remove")->will($this->returnCallback(function($user) {
|
2019-01-11 10:38:06 -05:00
|
|
|
if ($user === "john.doe@example.com") {
|
2018-11-06 16:35:33 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
throw new \JKingWeb\Arsse\User\Exception("doesNotExist");
|
|
|
|
}));
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
2018-11-06 16:35:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public function provideUserRemovals() {
|
|
|
|
return [
|
|
|
|
["arsse.php user remove john.doe@example.com", 0, ""],
|
|
|
|
["arsse.php user remove jane.doe@example.com", 10402, ""],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @dataProvider provideUserPasswordChanges */
|
|
|
|
public function testChangeAUserPassword(string $cmd, int $exitStatus, string $output) {
|
2019-03-25 15:03:41 -04:00
|
|
|
$passwordChange = function($user, $pass = null) {
|
2018-11-06 16:35:33 -05:00
|
|
|
switch ($user) {
|
|
|
|
case "jane.doe@example.com":
|
|
|
|
throw new \JKingWeb\Arsse\User\Exception("doesNotExist");
|
|
|
|
case "john.doe@example.com":
|
|
|
|
return is_null($pass) ? "random password" : $pass;
|
|
|
|
}
|
2019-03-25 15:03:41 -04:00
|
|
|
};
|
|
|
|
// FIXME: Phake is somehow unable to mock the User class correctly, so we use PHPUnit's mocks instead
|
|
|
|
Arsse::$user = $this->createMock(User::class);
|
|
|
|
Arsse::$user->method("passwordSet")->will($this->returnCallback($passwordChange));
|
|
|
|
$fever = \Phake::mock(FeverUser::class);
|
|
|
|
\Phake::when($fever)->register->thenReturnCallback($passwordChange);
|
2019-04-01 17:24:19 -04:00
|
|
|
\Phake::when($this->cli)->getInstance(FeverUser::class)->thenReturn($fever);
|
2018-12-07 20:03:04 -05:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
2018-11-06 16:35:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
public function provideUserPasswordChanges() {
|
|
|
|
return [
|
2019-03-25 15:03:41 -04:00
|
|
|
["arsse.php user set-pass john.doe@example.com", 0, "random password"],
|
|
|
|
["arsse.php user set-pass john.doe@example.com superman", 0, ""],
|
|
|
|
["arsse.php user set-pass jane.doe@example.com", 10402, ""],
|
|
|
|
["arsse.php user set-pass john.doe@example.com --fever", 0, "random password"],
|
|
|
|
["arsse.php user set-pass --fever john.doe@example.com superman", 0, ""],
|
|
|
|
["arsse.php user set-pass jane.doe@example.com --fever", 10402, ""],
|
2018-11-06 16:35:33 -05:00
|
|
|
];
|
|
|
|
}
|
2019-03-25 10:45:05 -04:00
|
|
|
|
2019-03-25 15:03:41 -04:00
|
|
|
/** @dataProvider provideUserPasswordClearings */
|
|
|
|
public function testClearAUserPassword(string $cmd, int $exitStatus, string $output) {
|
|
|
|
$passwordClear = function($user) {
|
|
|
|
switch ($user) {
|
|
|
|
case "jane.doe@example.com":
|
|
|
|
throw new \JKingWeb\Arsse\User\Exception("doesNotExist");
|
|
|
|
case "john.doe@example.com":
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
// FIXME: Phake is somehow unable to mock the User class correctly, so we use PHPUnit's mocks instead
|
|
|
|
Arsse::$user = $this->createMock(User::class);
|
2019-03-25 17:07:28 -04:00
|
|
|
Arsse::$user->method("passwordUnset")->will($this->returnCallback($passwordClear));
|
2019-03-25 15:03:41 -04:00
|
|
|
$fever = \Phake::mock(FeverUser::class);
|
2019-03-25 17:07:28 -04:00
|
|
|
\Phake::when($fever)->unregister->thenReturnCallback($passwordClear);
|
2019-04-01 17:24:19 -04:00
|
|
|
\Phake::when($this->cli)->getInstance(FeverUser::class)->thenReturn($fever);
|
2019-03-25 15:03:41 -04:00
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus, $output);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideUserPasswordClearings() {
|
|
|
|
return [
|
|
|
|
["arsse.php user unset-pass john.doe@example.com", 0, ""],
|
|
|
|
["arsse.php user unset-pass jane.doe@example.com", 10402, ""],
|
|
|
|
["arsse.php user unset-pass john.doe@example.com --fever", 0, ""],
|
|
|
|
["arsse.php user unset-pass jane.doe@example.com --fever", 10402, ""],
|
|
|
|
];
|
2019-03-25 10:45:05 -04:00
|
|
|
}
|
2019-04-01 17:24:19 -04:00
|
|
|
|
|
|
|
/** @dataProvider provideOpmlExports */
|
|
|
|
public function testExportToOpml(string $cmd, int $exitStatus, string $file, string $user, bool $flat) {
|
|
|
|
$opml = Phake::mock(OPML::class);
|
|
|
|
Phake::when($opml)->exportFile("php://output", $user, $flat)->thenReturn(true);
|
|
|
|
Phake::when($opml)->exportFile("good.opml", $user, $flat)->thenReturn(true);
|
|
|
|
Phake::when($opml)->exportFile("bad.opml", $user, $flat)->thenThrow(new \JKingWeb\Arsse\ImportExport\Exception("fileUnwritable"));
|
|
|
|
Phake::when($this->cli)->getInstance(OPML::class)->thenReturn($opml);
|
|
|
|
$this->assertConsole($this->cli, $cmd, $exitStatus);
|
|
|
|
$this->assertLoaded(true);
|
|
|
|
Phake::verify($opml)->exportFile($file, $user, $flat);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideOpmlExports() {
|
|
|
|
return [
|
|
|
|
["arsse.php export john.doe@example.com", 0, "php://output", "john.doe@example.com", false],
|
|
|
|
["arsse.php export john.doe@example.com -", 0, "php://output", "john.doe@example.com", false],
|
|
|
|
["arsse.php export john.doe@example.com good.opml", 0, "good.opml", "john.doe@example.com", false],
|
|
|
|
["arsse.php export john.doe@example.com bad.opml", 10604, "bad.opml", "john.doe@example.com", false],
|
|
|
|
["arsse.php export john.doe@example.com --flat", 0, "php://output", "john.doe@example.com", true],
|
|
|
|
["arsse.php export john.doe@example.com - --flat", 0, "php://output", "john.doe@example.com", true],
|
|
|
|
["arsse.php export --flat john.doe@example.com good.opml", 0, "good.opml", "john.doe@example.com", true],
|
|
|
|
["arsse.php export john.doe@example.com bad.opml --flat", 10604, "bad.opml", "john.doe@example.com", true],
|
|
|
|
["arsse.php export jane.doe@example.com", 0, "php://output", "jane.doe@example.com", false],
|
|
|
|
["arsse.php export jane.doe@example.com -", 0, "php://output", "jane.doe@example.com", false],
|
|
|
|
["arsse.php export jane.doe@example.com good.opml", 0, "good.opml", "jane.doe@example.com", false],
|
|
|
|
["arsse.php export jane.doe@example.com bad.opml", 10604, "bad.opml", "jane.doe@example.com", false],
|
|
|
|
["arsse.php export jane.doe@example.com --flat", 0, "php://output", "jane.doe@example.com", true],
|
|
|
|
["arsse.php export jane.doe@example.com - --flat", 0, "php://output", "jane.doe@example.com", true],
|
|
|
|
["arsse.php export --flat jane.doe@example.com good.opml", 0, "good.opml", "jane.doe@example.com", true],
|
|
|
|
["arsse.php export jane.doe@example.com bad.opml --flat", 10604, "bad.opml", "jane.doe@example.com", true],
|
|
|
|
];
|
|
|
|
}
|
2018-11-06 12:32:28 -05:00
|
|
|
}
|