2017-09-25 10:08:37 -04:00
< ? php
2017-11-16 20:51:03 -05:00
/** @ license MIT
* Copyright 2017 J . King , Dustin Wilson et al .
* See LICENSE and AUTHORS files for details */
2017-09-25 10:08:37 -04:00
declare ( strict_types = 1 );
2017-12-21 22:47:19 -05:00
namespace JKingWeb\Arsse\TestCase\REST\TinyTinyRSS ;
2017-09-25 10:08:37 -04:00
2017-12-21 22:47:19 -05:00
use JKingWeb\Arsse\Arsse ;
use JKingWeb\Arsse\User ;
use JKingWeb\Arsse\Database ;
2017-09-25 10:08:37 -04:00
use JKingWeb\Arsse\Test\Result ;
use JKingWeb\Arsse\Misc\Date ;
2019-02-25 22:41:12 -05:00
use JKingWeb\Arsse\Context\Context ;
2017-09-25 10:08:37 -04:00
use JKingWeb\Arsse\Db\ExceptionInput ;
use JKingWeb\Arsse\Db\Transaction ;
2017-10-11 12:55:50 -04:00
use JKingWeb\Arsse\REST\TinyTinyRSS\API ;
2021-02-27 15:24:45 -05:00
use JKingWeb\Arsse\Feed\Exception as FeedException ;
2018-01-03 23:13:08 -05:00
use Psr\Http\Message\ResponseInterface ;
2020-01-20 10:40:05 -05:00
use Laminas\Diactoros\Response\JsonResponse as Response ;
use Laminas\Diactoros\Response\EmptyResponse ;
2021-02-08 19:07:49 -05:00
2021-02-08 19:14:11 -05:00
/** @ covers \JKingWeb\Arsse\REST\TinyTinyRSS\API < extended >
2017-10-03 10:43:09 -04:00
* @ covers \JKingWeb\Arsse\REST\TinyTinyRSS\Exception */
2017-12-21 22:47:19 -05:00
class TestAPI extends \JKingWeb\Arsse\Test\AbstractTest {
2020-12-21 21:49:57 -05:00
protected const NOW = " 2020-12-21T23:09:17.189065Z " ;
2017-09-25 10:08:37 -04:00
protected $h ;
2021-02-27 15:24:45 -05:00
protected $userId = " john.doe@example.com " ;
2017-10-11 12:55:50 -04:00
protected $folders = [
[ 'id' => 5 , 'parent' => 3 , 'children' => 0 , 'feeds' => 1 , 'name' => " Local " ],
[ 'id' => 6 , 'parent' => 3 , 'children' => 0 , 'feeds' => 2 , 'name' => " National " ],
[ 'id' => 4 , 'parent' => null , 'children' => 0 , 'feeds' => 0 , 'name' => " Photography " ],
[ 'id' => 3 , 'parent' => null , 'children' => 2 , 'feeds' => 0 , 'name' => " Politics " ],
[ 'id' => 2 , 'parent' => 1 , 'children' => 0 , 'feeds' => 1 , 'name' => " Rocketry " ],
[ 'id' => 1 , 'parent' => null , 'children' => 1 , 'feeds' => 1 , 'name' => " Science " ],
];
protected $topFolders = [
[ 'id' => 4 , 'parent' => null , 'children' => 0 , 'feeds' => 0 , 'name' => " Photography " ],
[ 'id' => 3 , 'parent' => null , 'children' => 2 , 'feeds' => 0 , 'name' => " Politics " ],
[ 'id' => 1 , 'parent' => null , 'children' => 1 , 'feeds' => 1 , 'name' => " Science " ],
];
protected $subscriptions = [
2021-01-15 23:15:22 -05:00
[ 'id' => 3 , 'folder' => 1 , 'top_folder' => 1 , 'unread' => 2 , 'updated' => " 2016-05-23 06:40:02 " , 'err_msg' => 'argh' , 'title' => 'Ars Technica' , 'url' => " http://example.com/3 " , 'icon_url' => 'http://example.com/3.png' ],
[ 'id' => 4 , 'folder' => 6 , 'top_folder' => 3 , 'unread' => 6 , 'updated' => " 2017-10-09 15:58:34 " , 'err_msg' => '' , 'title' => 'CBC News' , 'url' => " http://example.com/4 " , 'icon_url' => 'http://example.com/4.png' ],
[ 'id' => 6 , 'folder' => null , 'top_folder' => null , 'unread' => 0 , 'updated' => " 2010-02-12 20:08:47 " , 'err_msg' => '' , 'title' => 'Eurogamer' , 'url' => " http://example.com/6 " , 'icon_url' => 'http://example.com/6.png' ],
[ 'id' => 1 , 'folder' => 2 , 'top_folder' => 1 , 'unread' => 5 , 'updated' => " 2017-09-15 22:54:16 " , 'err_msg' => '' , 'title' => 'NASA JPL' , 'url' => " http://example.com/1 " , 'icon_url' => null ],
[ 'id' => 5 , 'folder' => 6 , 'top_folder' => 3 , 'unread' => 12 , 'updated' => " 2017-07-07 17:07:17 " , 'err_msg' => '' , 'title' => 'Ottawa Citizen' , 'url' => " http://example.com/5 " , 'icon_url' => '' ],
[ 'id' => 2 , 'folder' => 5 , 'top_folder' => 3 , 'unread' => 10 , 'updated' => " 2011-11-11 11:11:11 " , 'err_msg' => 'oops' , 'title' => 'Toronto Star' , 'url' => " http://example.com/2 " , 'icon_url' => 'http://example.com/2.png' ],
2017-10-11 12:55:50 -04:00
];
protected $labels = [
2017-10-30 13:11:27 -04:00
[ 'id' => 3 , 'articles' => 100 , 'read' => 94 , 'unread' => 6 , 'name' => " Fascinating " ],
[ 'id' => 5 , 'articles' => 0 , 'read' => 0 , 'unread' => 0 , 'name' => " Interesting " ],
2017-11-02 17:17:46 -04:00
[ 'id' => 1 , 'articles' => 2 , 'read' => 2 , 'unread' => 0 , 'name' => " Logical " ],
2017-10-11 12:55:50 -04:00
];
protected $usedLabels = [
2017-11-02 17:17:46 -04:00
[ 'id' => 3 , 'articles' => 100 , 'read' => 94 , 'unread' => 6 , 'name' => " Fascinating " ],
[ 'id' => 1 , 'articles' => 2 , 'read' => 2 , 'unread' => 0 , 'name' => " Logical " ],
2017-10-11 12:55:50 -04:00
];
2017-11-02 17:17:46 -04:00
protected $starred = [ 'total' => 10 , 'unread' => 4 , 'read' => 6 ];
2017-11-15 15:38:49 -05:00
protected $articles = [
[
2020-03-01 15:16:50 -05:00
'id' => 101 ,
'url' => 'http://example.com/1' ,
'title' => 'Article title 1' ,
2017-11-15 15:38:49 -05:00
'subscription_title' => " Feed 11 " ,
2020-03-01 15:16:50 -05:00
'author' => '' ,
'content' => '<p>Article content 1</p>' ,
'guid' => '' ,
'published_date' => '2000-01-01 00:00:00' ,
'edited_date' => '2000-01-01 00:00:01' ,
'modified_date' => '2000-01-01 01:00:00' ,
'unread' => 1 ,
'starred' => 0 ,
'edition' => 101 ,
'subscription' => 8 ,
'fingerprint' => 'f5cb8bfc1c7396dc9816af212a3e2ac5221585c2a00bf7ccb6aabd95dcfcd6a6:fb0bc8f8cb08913dc5a497db700e327f1d34e4987402687d494a5891f24714d4:18fdd4fa93d693128c43b004399e5c9cea6c261ddfa002518d3669f55d8c2207' ,
'media_url' => null ,
'media_type' => null ,
'note' => " " ,
2017-11-15 15:38:49 -05:00
],
[
2020-03-01 15:16:50 -05:00
'id' => 102 ,
'url' => 'http://example.com/2' ,
'title' => 'Article title 2' ,
2017-11-15 15:38:49 -05:00
'subscription_title' => " Feed 11 " ,
2020-03-01 15:16:50 -05:00
'author' => 'J. King' ,
'content' => '<p>Article content 2</p>' ,
'guid' => '5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7' ,
'published_date' => '2000-01-02 00:00:00' ,
'edited_date' => '2000-01-02 00:00:02' ,
'modified_date' => '2000-01-02 02:00:00' ,
'unread' => 0 ,
'starred' => 0 ,
'edition' => 202 ,
'subscription' => 8 ,
'fingerprint' => '0e86d2de822a174fe3c44a466953e63ca1f1a58a19cbf475fce0855d4e3d5153:13075894189c47ffcfafd1dfe7fbb539f7c74a69d35a399b3abf8518952714f9:2abd0a8cba83b8214a66c8f0293ba63e467d720540e29ff8ddcdab069d4f1c9e' ,
'media_url' => " http://example.com/text " ,
'media_type' => " text/plain " ,
'note' => " Note 2 " ,
2017-11-15 15:38:49 -05:00
],
];
2017-11-22 20:18:16 -05:00
// text from https://corrigeur.fr/lorem-ipsum-traduction-origine.php
protected $richContent = <<< LONG_STRING
< section >
< p >
2018-10-26 14:58:04 -04:00
< b > Pour </ b > vous faire mieux
connaitre d’ ou\u { 300 } vient
l’ erreur de ceux qui
bla\u { 302 } ment la
volupte\u { 301 }, et qui louent
en quelque sorte la douleur ,
je vais entrer dans une
explication plus
e\u { 301 } tendue , et vous faire
voir tout ce qui a
e\u { 301 } te\u { 301 } dit
la\u { 300 } - dessus par
l’ inventeur de la
ve\u { 301 } rite\u { 301 }, et , pour
ainsi dire , par l’ architecte
2017-11-22 20:18:16 -05:00
de la vie heureuse .
</ p >
</ section >
LONG_STRING ;
2017-09-25 10:08:37 -04:00
2017-12-31 13:58:37 -05:00
protected function v ( $value ) {
return $value ;
}
2021-02-27 15:24:45 -05:00
public function setUp () : void {
parent :: setUp ();
self :: setConf ();
// create mock timestamps
$this -> objMock -> get -> with ( \DateTimeImmutable :: class ) -> returns ( new \DateTimeImmutable ( self :: NOW ));
// create a mock user manager
$this -> userId = " john.doe@example.com " ;
$this -> userMock = $this -> mock ( User :: class );
$this -> userMock -> auth -> returns ( true );
// create a mock database interface
$this -> dbMock = $this -> mock ( Database :: class );
$this -> dbMock -> begin -> returns ( $this -> mock ( Transaction :: class ));
$this -> dbMock -> sessionResume -> throws ( new \JKingWeb\Arsse\User\ExceptionSession ( " invalid " ));
$this -> dbMock -> sessionResume -> with ( " PriestsOfSyrinx " ) -> returns ([
'id' => " PriestsOfSyrinx " ,
'created' => " 2000-01-01 00:00:00 " ,
'expires' => " 2112-12-21 21:12:00 " ,
'user' => $this -> userId ,
]);
$this -> h = new API ();
}
2018-10-26 14:40:20 -04:00
protected function req ( $data , string $method = " POST " , string $target = " " , string $strData = null , string $user = null ) : ResponseInterface {
2021-02-27 15:24:45 -05:00
Arsse :: $obj = $this -> objMock -> get ();
Arsse :: $db = $this -> dbMock -> get ();
Arsse :: $user = $this -> userMock -> get ();
Arsse :: $user -> id = $this -> userId ;
2019-09-25 18:30:53 -04:00
$prefix = " /tt-rss/api " ;
$url = $prefix . $target ;
$body = $strData ? ? json_encode ( $data );
$req = $this -> serverRequest ( $method , $url , $prefix , [], [ 'HTTP_CONTENT_TYPE' => " application/x-www-form-urlencoded " ], $body , " application/json " , [], $user );
2018-01-04 23:08:53 -05:00
return $this -> h -> dispatch ( $req );
2017-11-23 14:05:26 -05:00
}
2018-10-26 14:40:20 -04:00
2020-01-20 13:34:03 -05:00
protected function reqAuth ( $data , $user ) : ResponseInterface {
2018-10-26 14:40:20 -04:00
return $this -> req ( $data , " POST " , " " , null , $user );
}
2018-10-26 14:58:04 -04:00
2017-09-26 22:45:54 -04:00
protected function respGood ( $content = null , $seq = 0 ) : Response {
2018-01-03 23:13:08 -05:00
return new Response ([
2020-03-01 15:16:50 -05:00
'seq' => $seq ,
'status' => 0 ,
2017-09-25 10:08:37 -04:00
'content' => $content ,
]);
}
2017-09-26 22:45:54 -04:00
protected function respErr ( string $msg , $content = [], $seq = 0 ) : Response {
2017-09-25 10:08:37 -04:00
$err = [ 'error' => $msg ];
2018-01-03 23:13:08 -05:00
return new Response ([
2020-03-01 15:16:50 -05:00
'seq' => $seq ,
'status' => 1 ,
2017-09-25 10:08:37 -04:00
'content' => array_merge ( $err , $content , $err ),
]);
}
2020-01-20 13:52:48 -05:00
public function testHandleInvalidPaths () : void {
2017-11-30 17:54:56 -05:00
$exp = $this -> respErr ( " MALFORMED_INPUT " , [], null );
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( null , " POST " , " " , " " ));
$this -> assertMessage ( $exp , $this -> req ( null , " POST " , " / " , " " ));
$this -> assertMessage ( $exp , $this -> req ( null , " POST " , " /index.php " , " " ));
2018-01-03 23:13:08 -05:00
$exp = new EmptyResponse ( 404 );
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( null , " POST " , " /bad/path " , " " ));
2017-11-30 17:54:56 -05:00
}
2020-01-20 13:52:48 -05:00
public function testHandleOptionsRequest () : void {
2018-01-03 23:13:08 -05:00
$exp = new EmptyResponse ( 204 , [
'Allow' => " POST " ,
'Accept' => " application/json, text/json " ,
2017-11-23 18:07:56 -05:00
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( null , " OPTIONS " , " " , " " ));
2017-11-23 18:07:56 -05:00
}
2020-01-20 13:52:48 -05:00
public function testHandleInvalidData () : void {
2017-11-29 09:22:59 -05:00
$exp = $this -> respErr ( " MALFORMED_INPUT " , [], null );
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( null , " POST " , " " , " This is not valid JSON data " ));
$this -> assertMessage ( $exp , $this -> req ( null , " POST " , " " , " " )); // lack of data is also an error
2017-11-23 18:07:56 -05:00
}
2018-10-26 14:40:20 -04:00
/** @dataProvider provideLoginRequests */
2020-01-20 13:52:48 -05:00
public function testLogIn ( array $conf , $httpUser , array $data , $sessions ) : void {
2021-02-27 15:24:45 -05:00
$this -> userId = null ;
2018-11-22 19:55:54 -05:00
self :: setConf ( $conf );
2021-02-27 15:24:45 -05:00
$this -> userMock -> auth -> returns ( false );
$this -> userMock -> auth -> with ( " john.doe@example.com " , " secret " ) -> returns ( true );
$this -> userMock -> auth -> with ( " jane.doe@example.com " , " superman " ) -> returns ( true );
$this -> dbMock -> sessionCreate -> with ( " john.doe@example.com " ) -> returns ( " PriestsOfSyrinx " , " SolarFederation " );
$this -> dbMock -> sessionCreate -> with ( " jane.doe@example.com " ) -> returns ( " ClockworkAngels " , " SevenCitiesOfGold " );
2018-10-26 14:40:20 -04:00
if ( $sessions instanceof EmptyResponse ) {
$exp1 = $sessions ;
$exp2 = $sessions ;
} elseif ( $sessions ) {
$exp1 = $this -> respGood ([ 'session_id' => $sessions [ 0 ], 'api_level' => \JKingWeb\Arsse\REST\TinyTinyRSS\API :: LEVEL ]);
$exp2 = $this -> respGood ([ 'session_id' => $sessions [ 1 ], 'api_level' => \JKingWeb\Arsse\REST\TinyTinyRSS\API :: LEVEL ]);
} else {
$exp1 = $this -> respErr ( " LOGIN_ERROR " );
$exp2 = $this -> respErr ( " LOGIN_ERROR " );
}
$data [ 'op' ] = " login " ;
$this -> assertMessage ( $exp1 , $this -> reqAuth ( $data , $httpUser ));
2017-11-30 19:31:40 -05:00
// base64 passwords are also accepted
2018-10-26 14:58:36 -04:00
if ( isset ( $data [ 'password' ])) {
2018-10-26 14:40:20 -04:00
$data [ 'password' ] = base64_encode ( $data [ 'password' ]);
}
$this -> assertMessage ( $exp2 , $this -> reqAuth ( $data , $httpUser ));
2017-09-25 10:08:37 -04:00
// logging in should never try to resume a session
2021-02-27 15:24:45 -05:00
$this -> dbMock -> sessionResume -> never () -> called ();
2017-09-25 10:08:37 -04:00
}
2019-10-16 14:42:43 -04:00
public function provideLoginRequests () : iterable {
2018-10-26 14:40:20 -04:00
return $this -> generateLoginRequests ( " login " );
}
/** @dataProvider provideResumeRequests */
2020-01-20 13:52:48 -05:00
public function testValidateASession ( array $conf , $httpUser , string $data , $result ) : void {
2021-02-27 15:24:45 -05:00
$this -> userId = null ;
2018-11-22 19:55:54 -05:00
self :: setConf ( $conf );
2021-02-27 15:24:45 -05:00
$this -> dbMock -> sessionResume -> with ( " PriestsOfSyrinx " ) -> returns ([
2020-03-01 15:16:50 -05:00
'id' => " PriestsOfSyrinx " ,
2018-10-26 14:40:20 -04:00
'created' => " 2000-01-01 00:00:00 " ,
'expires' => " 2112-12-21 21:12:00 " ,
'user' => " john.doe@example.com " ,
]);
2021-02-27 15:24:45 -05:00
$this -> dbMock -> sessionResume -> with ( " ClockworkAngels " ) -> returns ([
2020-03-01 15:16:50 -05:00
'id' => " ClockworkAngels " ,
2018-10-26 14:40:20 -04:00
'created' => " 2000-01-01 00:00:00 " ,
'expires' => " 2112-12-21 21:12:00 " ,
'user' => " jane.doe@example.com " ,
]);
$data = [
'op' => " isLoggedIn " ,
'sid' => $data ,
];
if ( $result instanceof EmptyResponse ) {
$exp1 = $result ;
$exp2 = null ;
} elseif ( $result ) {
$exp1 = $this -> respGood ([ 'status' => true ]);
$exp2 = $result ;
} else {
$exp1 = $this -> respErr ( " NOT_LOGGED_IN " );
$exp2 = ( $httpUser ) ? $httpUser : null ;
}
$this -> assertMessage ( $exp1 , $this -> reqAuth ( $data , $httpUser ));
$this -> assertSame ( $exp2 , Arsse :: $user -> id );
}
2019-10-16 14:42:43 -04:00
public function provideResumeRequests () : iterable {
2018-10-26 14:40:20 -04:00
return $this -> generateLoginRequests ( " isLoggedIn " );
}
2020-01-20 13:34:03 -05:00
public function generateLoginRequests ( string $type ) : array {
2018-10-26 14:40:20 -04:00
$john = " john.doe@example.com " ;
$johnGood = [
2020-03-01 15:16:50 -05:00
'user' => $john ,
2018-10-26 14:40:20 -04:00
'password' => " secret " ,
];
$johnBad = [
2020-03-01 15:16:50 -05:00
'user' => $john ,
2018-10-26 14:40:20 -04:00
'password' => " superman " ,
];
$johnSess = [ " PriestsOfSyrinx " , " SolarFederation " ];
$jane = " jane.doe@example.com " ;
$janeGood = [
2020-03-01 15:16:50 -05:00
'user' => $jane ,
2018-10-26 14:40:20 -04:00
'password' => " superman " ,
];
$janeBad = [
2020-03-01 15:16:50 -05:00
'user' => $jane ,
2018-10-26 14:40:20 -04:00
'password' => " secret " ,
];
$janeSess = [ " ClockworkAngels " , " SevenCitiesOfGold " ];
$missingU = [
'password' => " secret " ,
];
$missingP = [
'user' => $john ,
];
$sidJohn = " PriestsOfSyrinx " ;
$sidJane = " ClockworkAngels " ;
$sidBad = " TheWatchmaker " ;
$defaults = [
2020-03-01 15:16:50 -05:00
'userPreAuth' => false ,
2018-10-26 14:40:20 -04:00
'userHTTPAuthRequired' => false ,
2020-03-01 15:16:50 -05:00
'userSessionEnforced' => true ,
2018-10-26 14:40:20 -04:00
];
$preAuth = [
2020-03-01 15:16:50 -05:00
'userPreAuth' => true ,
2018-10-26 14:40:20 -04:00
'userHTTPAuthRequired' => false , // implied true by pre-auth
2020-03-01 15:16:50 -05:00
'userSessionEnforced' => true ,
2018-10-26 14:40:20 -04:00
];
$httpReq = [
2020-03-01 15:16:50 -05:00
'userPreAuth' => false ,
2018-10-26 14:40:20 -04:00
'userHTTPAuthRequired' => true ,
2020-03-01 15:16:50 -05:00
'userSessionEnforced' => true ,
2018-10-26 14:40:20 -04:00
];
$noSess = [
2020-03-01 15:16:50 -05:00
'userPreAuth' => false ,
2018-10-26 14:40:20 -04:00
'userHTTPAuthRequired' => false ,
2020-03-01 15:16:50 -05:00
'userSessionEnforced' => false ,
2018-10-26 14:40:20 -04:00
];
$fullHttp = [
2020-03-01 15:16:50 -05:00
'userPreAuth' => false ,
2018-10-26 14:40:20 -04:00
'userHTTPAuthRequired' => true ,
2020-03-01 15:16:50 -05:00
'userSessionEnforced' => false ,
2018-10-26 14:40:20 -04:00
];
$http401 = new EmptyResponse ( 401 );
2019-01-11 10:38:06 -05:00
if ( $type === " login " ) {
2018-10-26 14:40:20 -04:00
return [
// conf, user, data, result
[ $defaults , null , $johnGood , $johnSess ],
[ $defaults , null , $johnBad , false ],
[ $defaults , null , $janeGood , $janeSess ],
[ $defaults , null , $janeBad , false ],
[ $defaults , null , $missingU , false ],
[ $defaults , null , $missingP , false ],
[ $defaults , $john , $johnGood , $johnSess ],
[ $defaults , $john , $johnBad , false ],
[ $defaults , $john , $janeGood , $janeSess ],
[ $defaults , $john , $janeBad , false ],
[ $defaults , $john , $missingU , false ],
[ $defaults , $john , $missingP , false ],
[ $defaults , $jane , $johnGood , $johnSess ],
[ $defaults , $jane , $johnBad , false ],
[ $defaults , $jane , $janeGood , $janeSess ],
[ $defaults , $jane , $janeBad , false ],
[ $defaults , $jane , $missingU , false ],
[ $defaults , $jane , $missingP , false ],
[ $defaults , " " , $johnGood , $http401 ],
[ $defaults , " " , $johnBad , $http401 ],
[ $defaults , " " , $janeGood , $http401 ],
[ $defaults , " " , $janeBad , $http401 ],
[ $defaults , " " , $missingU , $http401 ],
[ $defaults , " " , $missingP , $http401 ],
[ $preAuth , null , $johnGood , $http401 ],
[ $preAuth , null , $johnBad , $http401 ],
[ $preAuth , null , $janeGood , $http401 ],
[ $preAuth , null , $janeBad , $http401 ],
[ $preAuth , null , $missingU , $http401 ],
[ $preAuth , null , $missingP , $http401 ],
[ $preAuth , $john , $johnGood , $johnSess ],
[ $preAuth , $john , $johnBad , $johnSess ],
[ $preAuth , $john , $janeGood , false ],
[ $preAuth , $john , $janeBad , false ],
[ $preAuth , $john , $missingU , false ],
[ $preAuth , $john , $missingP , $johnSess ],
[ $preAuth , $jane , $johnGood , false ],
[ $preAuth , $jane , $johnBad , false ],
[ $preAuth , $jane , $janeGood , $janeSess ],
[ $preAuth , $jane , $janeBad , $janeSess ],
[ $preAuth , $jane , $missingU , false ],
[ $preAuth , $jane , $missingP , false ],
[ $preAuth , " " , $johnGood , $http401 ],
[ $preAuth , " " , $johnBad , $http401 ],
[ $preAuth , " " , $janeGood , $http401 ],
[ $preAuth , " " , $janeBad , $http401 ],
[ $preAuth , " " , $missingU , $http401 ],
[ $preAuth , " " , $missingP , $http401 ],
[ $httpReq , null , $johnGood , $http401 ],
[ $httpReq , null , $johnBad , $http401 ],
[ $httpReq , null , $janeGood , $http401 ],
[ $httpReq , null , $janeBad , $http401 ],
[ $httpReq , null , $missingU , $http401 ],
[ $httpReq , null , $missingP , $http401 ],
[ $httpReq , $john , $johnGood , $johnSess ],
[ $httpReq , $john , $johnBad , false ],
[ $httpReq , $john , $janeGood , $janeSess ],
[ $httpReq , $john , $janeBad , false ],
[ $httpReq , $john , $missingU , false ],
[ $httpReq , $john , $missingP , false ],
[ $httpReq , $jane , $johnGood , $johnSess ],
[ $httpReq , $jane , $johnBad , false ],
[ $httpReq , $jane , $janeGood , $janeSess ],
[ $httpReq , $jane , $janeBad , false ],
[ $httpReq , $jane , $missingU , false ],
[ $httpReq , $jane , $missingP , false ],
[ $httpReq , " " , $johnGood , $http401 ],
[ $httpReq , " " , $johnBad , $http401 ],
[ $httpReq , " " , $janeGood , $http401 ],
[ $httpReq , " " , $janeBad , $http401 ],
[ $httpReq , " " , $missingU , $http401 ],
[ $httpReq , " " , $missingP , $http401 ],
[ $noSess , null , $johnGood , $johnSess ],
[ $noSess , null , $johnBad , false ],
[ $noSess , null , $janeGood , $janeSess ],
[ $noSess , null , $janeBad , false ],
[ $noSess , null , $missingU , false ],
[ $noSess , null , $missingP , false ],
[ $noSess , $john , $johnGood , $johnSess ],
[ $noSess , $john , $johnBad , $johnSess ],
[ $noSess , $john , $janeGood , $johnSess ],
[ $noSess , $john , $janeBad , $johnSess ],
[ $noSess , $john , $missingU , $johnSess ],
[ $noSess , $john , $missingP , $johnSess ],
[ $noSess , $jane , $johnGood , $janeSess ],
[ $noSess , $jane , $johnBad , $janeSess ],
[ $noSess , $jane , $janeGood , $janeSess ],
[ $noSess , $jane , $janeBad , $janeSess ],
[ $noSess , $jane , $missingU , $janeSess ],
[ $noSess , $jane , $missingP , $janeSess ],
[ $noSess , " " , $johnGood , $http401 ],
[ $noSess , " " , $johnBad , $http401 ],
[ $noSess , " " , $janeGood , $http401 ],
[ $noSess , " " , $janeBad , $http401 ],
[ $noSess , " " , $missingU , $http401 ],
[ $noSess , " " , $missingP , $http401 ],
[ $fullHttp , null , $johnGood , $http401 ],
[ $fullHttp , null , $johnBad , $http401 ],
[ $fullHttp , null , $janeGood , $http401 ],
[ $fullHttp , null , $janeBad , $http401 ],
[ $fullHttp , null , $missingU , $http401 ],
[ $fullHttp , null , $missingP , $http401 ],
[ $fullHttp , $john , $johnGood , $johnSess ],
[ $fullHttp , $john , $johnBad , $johnSess ],
[ $fullHttp , $john , $janeGood , $johnSess ],
[ $fullHttp , $john , $janeBad , $johnSess ],
[ $fullHttp , $john , $missingU , $johnSess ],
[ $fullHttp , $john , $missingP , $johnSess ],
[ $fullHttp , $jane , $johnGood , $janeSess ],
[ $fullHttp , $jane , $johnBad , $janeSess ],
[ $fullHttp , $jane , $janeGood , $janeSess ],
[ $fullHttp , $jane , $janeBad , $janeSess ],
[ $fullHttp , $jane , $missingU , $janeSess ],
[ $fullHttp , $jane , $missingP , $janeSess ],
[ $fullHttp , " " , $johnGood , $http401 ],
[ $fullHttp , " " , $johnBad , $http401 ],
[ $fullHttp , " " , $janeGood , $http401 ],
[ $fullHttp , " " , $janeBad , $http401 ],
[ $fullHttp , " " , $missingU , $http401 ],
[ $fullHttp , " " , $missingP , $http401 ],
];
2019-01-11 10:38:06 -05:00
} elseif ( $type === " isLoggedIn " ) {
2018-10-26 14:40:20 -04:00
return [
// conf, user, session, result
[ $defaults , null , $sidJohn , $john ],
[ $defaults , null , $sidJane , $jane ],
[ $defaults , null , $sidBad , false ],
[ $defaults , $john , $sidJohn , $john ],
[ $defaults , $john , $sidJane , $jane ],
[ $defaults , $john , $sidBad , false ],
[ $defaults , $jane , $sidJohn , $john ],
[ $defaults , $jane , $sidJane , $jane ],
[ $defaults , $jane , $sidBad , false ],
[ $defaults , " " , $sidJohn , $http401 ],
[ $defaults , " " , $sidJane , $http401 ],
[ $defaults , " " , $sidBad , $http401 ],
[ $preAuth , null , $sidJohn , $http401 ],
[ $preAuth , null , $sidJane , $http401 ],
[ $preAuth , null , $sidBad , $http401 ],
[ $preAuth , $john , $sidJohn , $john ],
[ $preAuth , $john , $sidJane , $jane ],
[ $preAuth , $john , $sidBad , false ],
[ $preAuth , $jane , $sidJohn , $john ],
[ $preAuth , $jane , $sidJane , $jane ],
[ $preAuth , $jane , $sidBad , false ],
[ $preAuth , " " , $sidJohn , $http401 ],
[ $preAuth , " " , $sidJane , $http401 ],
[ $preAuth , " " , $sidBad , $http401 ],
[ $httpReq , null , $sidJohn , $http401 ],
[ $httpReq , null , $sidJane , $http401 ],
[ $httpReq , null , $sidBad , $http401 ],
[ $httpReq , $john , $sidJohn , $john ],
[ $httpReq , $john , $sidJane , $jane ],
[ $httpReq , $john , $sidBad , false ],
[ $httpReq , $jane , $sidJohn , $john ],
[ $httpReq , $jane , $sidJane , $jane ],
[ $httpReq , $jane , $sidBad , false ],
[ $httpReq , " " , $sidJohn , $http401 ],
[ $httpReq , " " , $sidJane , $http401 ],
[ $httpReq , " " , $sidBad , $http401 ],
[ $noSess , null , $sidJohn , $john ],
[ $noSess , null , $sidJane , $jane ],
[ $noSess , null , $sidBad , false ],
[ $noSess , $john , $sidJohn , $john ],
[ $noSess , $john , $sidJane , $john ],
[ $noSess , $john , $sidBad , $john ],
[ $noSess , $jane , $sidJohn , $jane ],
[ $noSess , $jane , $sidJane , $jane ],
[ $noSess , $jane , $sidBad , $jane ],
[ $noSess , " " , $sidJohn , $http401 ],
[ $noSess , " " , $sidJane , $http401 ],
[ $noSess , " " , $sidBad , $http401 ],
[ $fullHttp , null , $sidJohn , $http401 ],
[ $fullHttp , null , $sidJane , $http401 ],
[ $fullHttp , null , $sidBad , $http401 ],
[ $fullHttp , $john , $sidJohn , $john ],
[ $fullHttp , $john , $sidJane , $john ],
[ $fullHttp , $john , $sidBad , $john ],
[ $fullHttp , $jane , $sidJohn , $jane ],
[ $fullHttp , $jane , $sidJane , $jane ],
[ $fullHttp , $jane , $sidBad , $jane ],
[ $fullHttp , " " , $sidJohn , $http401 ],
[ $fullHttp , " " , $sidJane , $http401 ],
[ $fullHttp , " " , $sidBad , $http401 ],
];
}
}
2020-01-20 13:52:48 -05:00
public function testHandleGenericError () : void {
2021-02-27 15:24:45 -05:00
$this -> userMock -> auth -> throws ( new \JKingWeb\Arsse\Db\ExceptionTimeout ( " general " ));
2017-11-23 18:07:56 -05:00
$data = [
'op' => " login " ,
2021-02-27 15:24:45 -05:00
'user' => $this -> userId ,
2017-11-23 18:07:56 -05:00
'password' => " secret " ,
];
2018-01-03 23:13:08 -05:00
$exp = new EmptyResponse ( 500 );
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-11-23 18:07:56 -05:00
}
2020-01-20 13:52:48 -05:00
public function testLogOut () : void {
2021-02-27 15:24:45 -05:00
$this -> dbMock -> sessionDestroy -> returns ( true );
2017-09-25 10:08:37 -04:00
$data = [
'op' => " logout " ,
'sid' => " PriestsOfSyrinx " ,
];
$exp = $this -> respGood ([ 'status' => " OK " ]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2021-02-27 15:24:45 -05:00
$this -> dbMock -> sessionDestroy -> calledWith ( $this -> userId , " PriestsOfSyrinx " );
2017-09-25 10:08:37 -04:00
}
2020-01-20 13:52:48 -05:00
public function testHandleUnknownMethods () : void {
2017-11-23 18:07:56 -05:00
$exp = $this -> respErr ( " UNKNOWN_METHOD " , [ 'method' => " thisMethodDoesNotExist " ]);
$data = [
'op' => " thisMethodDoesNotExist " ,
'sid' => " PriestsOfSyrinx " ,
];
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-11-23 18:07:56 -05:00
}
2020-01-20 13:52:48 -05:00
public function testHandleMixedCaseMethods () : void {
2017-11-23 18:07:56 -05:00
$data = [
'op' => " isLoggedIn " ,
'sid' => " PriestsOfSyrinx " ,
];
$exp = $this -> respGood ([ 'status' => true ]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-11-23 18:07:56 -05:00
$data [ 'op' ] = " isloggedin " ;
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-11-23 18:07:56 -05:00
$data [ 'op' ] = " ISLOGGEDIN " ;
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-11-23 18:07:56 -05:00
$data [ 'op' ] = " iSlOgGeDiN " ;
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-11-23 18:07:56 -05:00
}
2020-01-20 13:52:48 -05:00
public function testRetrieveServerVersion () : void {
2017-09-25 10:08:37 -04:00
$data = [
'op' => " getVersion " ,
'sid' => " PriestsOfSyrinx " ,
];
$exp = $this -> respGood ([
2020-03-01 15:16:50 -05:00
'version' => \JKingWeb\Arsse\REST\TinyTinyRSS\API :: VERSION ,
2017-10-02 15:42:15 -04:00
'arsse_version' => Arsse :: VERSION ,
2017-09-25 10:08:37 -04:00
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-09-25 10:08:37 -04:00
}
2020-01-20 13:52:48 -05:00
public function testRetrieveProtocolLevel () : void {
2017-09-25 10:08:37 -04:00
$data = [
'op' => " getApiLevel " ,
'sid' => " PriestsOfSyrinx " ,
];
$exp = $this -> respGood ([ 'level' => \JKingWeb\Arsse\REST\TinyTinyRSS\API :: LEVEL ]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $data ));
2017-09-25 10:08:37 -04:00
}
2017-09-26 22:45:54 -04:00
2021-02-27 15:24:45 -05:00
/** @dataProvider provideCategoryAdditions */
public function testAddACategory ( array $in , array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " addCategory " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> folderAdd -> $action ( $out );
$this -> dbMock -> folderList -> with ( " ~ " , null , false ) -> returns ( new Result ( $this -> v ([
2017-09-26 22:45:54 -04:00
[ 'id' => 2 , 'name' => " Software " , 'parent' => null ],
[ 'id' => 1 , 'name' => " Politics " , 'parent' => null ],
2021-02-27 15:24:45 -05:00
])));
$this -> dbMock -> folderList -> with ( " ~ " , 1 , false ) -> returns ( new Result ( $this -> v ([
[ 'id' => 3 , 'name' => " Hardware " , 'parent' => 1 ],
])));
$this -> assertMessage ( $exp , $this -> req ( $in ));
$this -> dbMock -> folderAdd -> calledWith ( $this -> userId , $data );
if ( ! $out instanceof \Exception ) {
$this -> dbMock -> folderList -> never () -> called ();
}
2017-09-26 22:45:54 -04:00
}
2017-09-30 22:15:55 -04:00
2021-02-27 15:24:45 -05:00
public function provideCategoryAdditions () : iterable {
return [
[[], [ 'name' => null , 'parent' => null ], new ExceptionInput ( " missing " ), $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " " ], [ 'name' => " " , 'parent' => null ], new ExceptionInput ( " missing " ), $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " " ], [ 'name' => " " , 'parent' => null ], new ExceptionInput ( " whitespace " ), $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " Software " ], [ 'name' => " Software " , 'parent' => null ], 2 , $this -> respGood ( " 2 " )],
[[ 'caption' => " Hardware " , 'parent_id' => 1 ], [ 'name' => " Hardware " , 'parent' => 1 ], 3 , $this -> respGood ( " 3 " )],
[[ 'caption' => " Hardware " , 'parent_id' => 2112 ], [ 'name' => " Hardware " , 'parent' => 2112 ], new ExceptionInput ( " idMissing " ), $this -> respGood ( false )],
[[ 'caption' => " Software " ], [ 'name' => " Software " , 'parent' => null ], new ExceptionInput ( " constraintViolation " ), $this -> respGood ( " 2 " )],
[[ 'caption' => " Hardware " , 'parent_id' => 1 ], [ 'name' => " Hardware " , 'parent' => 1 ], new ExceptionInput ( " constraintViolation " ), $this -> respGood ( " 3 " )],
2017-09-30 22:15:55 -04:00
];
}
2021-02-27 15:24:45 -05:00
/** @dataProvider provideCategoryRemovals */
public function testRemoveACategory ( array $in , ? int $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " removeCategory " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> folderRemove -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $data > 0 ) {
$this -> dbMock -> folderRemove -> calledWith ( $this -> userId , ( int ) $data );
}
2017-09-30 22:15:55 -04:00
}
2021-02-27 15:24:45 -05:00
public function provideCategoryRemovals () : iterable {
return [
[[ 'category_id' => 42 ], 42 , true , $this -> respGood ()],
[[ 'category_id' => 2112 ], 2112 , new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => - 1 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-09-30 22:15:55 -04:00
];
}
2021-02-27 15:24:45 -05:00
/** @dataProvider provideCategoryMoves */
public function testMoveACategory ( array $in , array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " moveCategory " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> folderPropertiesSet -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> folderPropertiesSet -> calledWith ( ... $data );
} else {
$this -> dbMock -> folderPropertiesSet -> never () -> called ();
}
}
public function provideCategoryMoves () : iterable {
return [
[[ 'category_id' => 42 , 'parent_id' => 1 ], [ $this -> userId , 42 , [ 'parent' => 1 ]], true , $this -> respGood ()],
[[ 'category_id' => 2112 , 'parent_id' => 2 ], [ $this -> userId , 2112 , [ 'parent' => 2 ]], new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[ 'category_id' => 42 , 'parent_id' => 0 ], [ $this -> userId , 42 , [ 'parent' => 0 ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'category_id' => 42 , 'parent_id' => 47 ], [ $this -> userId , 42 , [ 'parent' => 47 ]], new ExceptionInput ( " idMissing " ), $this -> respGood ()],
[[ 'category_id' => - 1 , 'parent_id' => 1 ], [ $this -> userId , - 1 , [ 'parent' => 1 ]], null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => 42 , 'parent_id' => - 1 ], [ $this -> userId , 42 , [ 'parent' => - 1 ]], null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => 42 ], [ $this -> userId , 42 , [ 'parent' => 0 ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'parent_id' => - 1 ], [ $this -> userId , 0 , [ 'parent' => - 1 ]], null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], [ $this -> userId , 0 , [ 'parent' => 0 ]], null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-03 10:43:09 -04:00
];
2021-02-27 15:24:45 -05:00
}
/** @dataProvider provideCategoryRenamings */
public function testRenameACategory ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " renameCategory " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> folderPropertiesSet -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> folderPropertiesSet -> calledWith ( ... $data );
} else {
$this -> dbMock -> folderPropertiesSet -> never () -> called ();
}
}
public function provideCategoryRenamings () : iterable {
return [
[[ 'category_id' => 42 , 'caption' => " Ook " ], [ $this -> userId , 42 , [ 'name' => " Ook " ]], true , $this -> respGood ()],
[[ 'category_id' => 2112 , 'caption' => " Eek " ], [ $this -> userId , 2112 , [ 'name' => " Eek " ]], new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[ 'category_id' => 42 , 'caption' => " Eek " ], [ $this -> userId , 42 , [ 'name' => " Eek " ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'category_id' => 42 , 'caption' => " " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => 42 , 'caption' => " " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => - 1 , 'caption' => " Ook " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => 42 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " Ook " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-03 10:43:09 -04:00
];
2021-02-27 15:24:45 -05:00
}
2021-02-27 21:17:20 -05:00
/** @dataProvider provideFeedSubscriptions */
2021-02-27 15:24:45 -05:00
public function testAddASubscription ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " subscribeToFeed " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
2017-10-03 10:43:09 -04:00
$list = [
[ 'id' => 1 , 'url' => " http://localhost:8000/Feed/Discovery/Feed " ],
[ 'id' => 2 , 'url' => " http://example.com/0 " ],
[ 'id' => 3 , 'url' => " http://example.com/3 " ],
[ 'id' => 4 , 'url' => " http://example.com/9 " ],
];
2021-02-27 15:24:45 -05:00
$this -> dbMock -> subscriptionAdd -> $action ( $out );
$this -> dbMock -> folderPropertiesGet -> with ( $this -> userId , 42 ) -> returns ( $this -> v ([ 'id' => 42 ]));
$this -> dbMock -> folderPropertiesGet -> with ( $this -> userId , 47 ) -> returns ( $this -> v ([ 'id' => 47 ]));
$this -> dbMock -> folderPropertiesGet -> with ( $this -> userId , 2112 ) -> throws ( new ExceptionInput ( " subjectMissing " ));
$this -> dbMock -> subscriptionPropertiesSet -> with ( $this -> userId , " * " ) -> returns ( true );
$this -> dbMock -> subscriptionPropertiesSet -> with ( $this -> userId , 4 , " ~ " ) -> throws ( new ExceptionInput ( " idMissing " ));
$this -> dbMock -> subscriptionList -> with ( $this -> userId ) -> returns ( new Result ( $this -> v ( $list )));
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $data !== null ) {
$this -> dbMock -> subscriptionAdd -> calledWith ( ... $data );
} else {
$this -> dbMock -> subscriptionAdd -> never () -> called ();
2017-10-03 10:43:09 -04:00
}
2021-02-27 15:24:45 -05:00
$this -> dbMock -> subscriptionPropertiesSet -> never () -> calledWith ( $this -> userId , 4 , [ 'folder' => 1 ]);
}
2021-02-27 21:17:20 -05:00
public function provideFeedSubscriptions () : iterable {
2021-02-27 15:24:45 -05:00
return [
[[ 'feed_url' => " http://example.com/0 " ], [ $this -> userId , " http://example.com/0 " , " " , " " ], 2 , $this -> respGood ([ 'code' => 1 , 'feed_id' => 2 ])],
[[ 'feed_url' => " http://example.com/1 " , 'category_id' => 42 ], [ $this -> userId , " http://example.com/1 " , " " , " " ], new FeedException ( " unauthorized " ), $this -> respGood ([ 'code' => 5 , 'message' => ( new FeedException ( " unauthorized " )) -> getMessage ()])],
[[ 'feed_url' => " http://example.com/2 " , 'category_id' => 2112 ], null , null , $this -> respGood ([ 'code' => 1 , 'feed_id' => 0 ])],
[[ 'feed_url' => " http://example.com/3 " ], [ $this -> userId , " http://example.com/3 " , " " , " " ], new ExceptionInput ( " constraintViolation " ), $this -> respGood ([ 'code' => 0 , 'feed_id' => 3 ])],
[[ 'feed_url' => " http://localhost:8000/Feed/Discovery/Valid " ], [ $this -> userId , " http://localhost:8000/Feed/Discovery/Valid " , " " , " " ], new ExceptionInput ( " constraintViolation " ), $this -> respGood ([ 'code' => 0 , 'feed_id' => 1 ])],
[[ 'feed_url' => " http://localhost:8000/Feed/Discovery/Invalid " ], [ $this -> userId , " http://localhost:8000/Feed/Discovery/Invalid " , " " , " " ], new ExceptionInput ( " constraintViolation " ), $this -> respGood ([ 'code' => 3 , 'message' => ( new FeedException ( " subscriptionNotFound " , [ 'url' => " http://localhost:8000/Feed/Discovery/Invalid " ])) -> getMessage ()])],
[[ 'feed_url' => " http://example.com/6 " ], [ $this -> userId , " http://example.com/6 " , " " , " " ], new FeedException ( " invalidUrl " ), $this -> respGood ([ 'code' => 2 , 'message' => ( new FeedException ( " invalidUrl " )) -> getMessage ()])],
[[ 'feed_url' => " http://example.com/7 " ], [ $this -> userId , " http://example.com/7 " , " " , " " ], new FeedException ( " malformedXml " ), $this -> respGood ([ 'code' => 6 , 'message' => ( new FeedException ( " malformedXml " )) -> getMessage ()])],
[[ 'feed_url' => " http://example.com/8 " , 'category_id' => 47 ], [ $this -> userId , " http://example.com/8 " , " " , " " ], 4 , $this -> respGood ([ 'code' => 1 , 'feed_id' => 4 ])],
[[ 'feed_url' => " http://example.com/9 " , 'category_id' => 1 ], [ $this -> userId , " http://example.com/9 " , " " , " " ], new ExceptionInput ( " constraintViolation " ), $this -> respGood ([ 'code' => 0 , 'feed_id' => 4 ])],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_url' => " http://example.com/ " , 'login' => []], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_url' => " http://example.com/ " , 'login' => " " , 'password' => []], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_url' => " http://example.com/ " , 'category_id' => - 1 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
];
2017-10-03 10:43:09 -04:00
}
2021-02-27 21:17:20 -05:00
/** @dataProvider provideFeedUnsubscriptions */
public function testRemoveASubscription ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " unsubscribeFeed " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> subscriptionRemove -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> subscriptionRemove -> calledWith ( ... $data );
} else {
$this -> dbMock -> subscriptionRemove -> never () -> called ();
}
2017-09-30 22:15:55 -04:00
}
2021-02-27 21:17:20 -05:00
public function provideFeedUnsubscriptions () : iterable {
return [
[[ 'feed_id' => 42 ], [ $this -> userId , 42 ], true , $this -> respGood ([ 'status' => " OK " ])],
[[ 'feed_id' => 2112 ], [ $this -> userId , 2112 ], new ExceptionInput ( " subjectMissing " ), $this -> respErr ( " FEED_NOT_FOUND " )],
[[ 'feed_id' => - 1 ], [ $this -> userId , - 1 ], new ExceptionInput ( " typeViolation " ), $this -> respErr ( " FEED_NOT_FOUND " )],
[[], [ $this -> userId , 0 ], new ExceptionInput ( " typeViolation " ), $this -> respErr ( " FEED_NOT_FOUND " )],
2017-09-30 22:15:55 -04:00
];
2021-02-27 21:17:20 -05:00
}
/** @dataProvider provideFeedMoves */
public function testMoveAFeed ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " moveFeed " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> subscriptionPropertiesSet -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> subscriptionPropertiesSet -> calledWith ( ... $data );
} else {
$this -> dbMock -> subscriptionPropertiesSet -> never () -> called ();
}
}
public function provideFeedMoves () : iterable {
return [
[[ 'feed_id' => 42 , 'category_id' => 1 ], [ $this -> userId , 42 , [ 'folder' => 1 ]], true , $this -> respGood ()],
[[ 'feed_id' => 2112 , 'category_id' => 2 ], [ $this -> userId , 2112 , [ 'folder' => 2 ]], new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[ 'feed_id' => 42 , 'category_id' => 0 ], [ $this -> userId , 42 , [ 'folder' => 0 ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'feed_id' => 42 , 'category_id' => 47 ], [ $this -> userId , 42 , [ 'folder' => 47 ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'feed_id' => - 1 , 'category_id' => 1 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_id' => 42 , 'category_id' => - 1 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_id' => 42 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'category_id' => - 1 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-09-30 22:15:55 -04:00
];
2021-02-27 21:17:20 -05:00
}
/** @dataProvider provideFeedRenamings */
public function testRenameAFeed ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " renameFeed " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> subscriptionPropertiesSet -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> subscriptionPropertiesSet -> calledWith ( ... $data );
} else {
$this -> dbMock -> subscriptionPropertiesSet -> never () -> called ();
}
}
public function provideFeedRenamings () : iterable {
return [
[[ 'feed_id' => 42 , 'caption' => " Ook " ], [ $this -> userId , 42 , [ 'title' => " Ook " ]], true , $this -> respGood ()],
[[ 'feed_id' => 2112 , 'caption' => " Eek " ], [ $this -> userId , 2112 , [ 'title' => " Eek " ]], new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[ 'feed_id' => 42 , 'caption' => " Eek " ], [ $this -> userId , 42 , [ 'title' => " Eek " ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'feed_id' => 42 , 'caption' => " " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_id' => 42 , 'caption' => " " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_id' => - 1 , 'caption' => " Ook " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'feed_id' => 42 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " Ook " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-09-30 22:15:55 -04:00
];
}
2017-10-03 12:43:46 -04:00
2020-01-20 13:52:48 -05:00
public function testRetrieveTheGlobalUnreadCount () : void {
2017-10-03 12:43:46 -04:00
$in = [ 'op' => " getUnread " , 'sid' => " PriestsOfSyrinx " ];
2021-02-27 15:24:45 -05:00
$this -> dbMock -> subscriptionList -> returns ( new Result ( $this -> v ([
2017-10-03 12:43:46 -04:00
[ 'id' => 1 , 'unread' => 2112 ],
[ 'id' => 2 , 'unread' => 42 ],
[ 'id' => 3 , 'unread' => 47 ],
2017-12-31 13:58:37 -05:00
])));
2017-11-29 11:47:10 -05:00
$exp = $this -> respGood ([ 'unread' => ( string ) ( 2112 + 42 + 47 )]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $this -> req ( $in ));
2017-10-03 12:43:46 -04:00
}
2020-01-20 13:52:48 -05:00
public function testRetrieveTheServerConfiguration () : void {
2017-10-03 12:43:46 -04:00
$in = [ 'op' => " getConfig " , 'sid' => " PriestsOfSyrinx " ];
2019-01-20 22:40:49 -05:00
$interval = Arsse :: $conf -> serviceFrequency ;
2017-10-03 12:43:46 -04:00
$valid = ( new \DateTimeImmutable ( " now " , new \DateTimezone ( " UTC " ))) -> sub ( $interval );
$invalid = $valid -> sub ( $interval ) -> sub ( $interval );
2021-02-27 21:17:20 -05:00
$this -> dbMock -> metaGet -> with ( " service_last_checkin " ) -> returns ( Date :: transform ( $valid , " sql " ), Date :: transform ( $invalid , " sql " ));
2021-02-27 15:24:45 -05:00
$this -> dbMock -> subscriptionCount -> with ( $this -> userId ) -> returns ( 12 , 2 );
$this -> assertMessage ( $this -> respGood ([ 'icons_dir' => " feed-icons " , 'icons_url' => " feed-icons " , 'daemon_is_running' => true , 'num_feeds' => 12 ]), $this -> req ( $in ));
$this -> assertMessage ( $this -> respGood ([ 'icons_dir' => " feed-icons " , 'icons_url' => " feed-icons " , 'daemon_is_running' => false , 'num_feeds' => 2 ]), $this -> req ( $in ));
2017-10-03 12:43:46 -04:00
}
2017-10-03 16:14:37 -04:00
2021-02-27 21:17:20 -05:00
/** @dataProvider provideFeedUpdates */
public function testUpdateAFeed ( array $in , ? array $data , $out , ? int $id , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " updateFeed " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> subscriptionPropertiesGet -> $action ( $out );
$this -> dbMock -> feedUpdate -> returns ( true );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $data !== null ) {
$this -> dbMock -> subscriptionPropertiesGet -> calledWith ( ... $data );
} else {
$this -> dbMock -> subscriptionPropertiesGet -> never () -> called ();
}
if ( $id !== null ) {
$this -> dbMock -> feedUpdate -> calledWith ( $id );
} else {
$this -> dbMock -> feedUpdate -> never () -> called ();
}
2017-10-03 16:14:37 -04:00
}
2017-10-05 17:42:12 -04:00
2021-02-27 21:17:20 -05:00
public function provideFeedUpdates () : iterable {
return [
[[ 'feed_id' => 1 ], [ $this -> userId , 1 ], $this -> v ([ 'id' => 1 , 'feed' => 11 ]), 11 , $this -> respGood ([ 'status' => " OK " ])],
[[ 'feed_id' => 2 ], [ $this -> userId , 2 ], new ExceptionInput ( " subjectMissing " ), null , $this -> respErr ( " FEED_NOT_FOUND " )],
[[ 'feed_id' => - 1 ], null , null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-05 17:42:12 -04:00
];
2021-02-27 21:17:20 -05:00
}
/** @dataProvider provideLabelAdditions */
public function testAddALabel ( array $in , ? array $data1 , $out1 , ? array $data2 , $out2 , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " addLabel " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out1 instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> labelAdd -> $action ( $out1 );
$this -> dbMock -> labelPropertiesGet -> returns ( $out2 );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out1 !== null ) {
$this -> dbMock -> labelAdd -> calledWith ( ... $data1 );
} else {
$this -> dbMock -> labelAdd -> never () -> called ();
}
if ( $out2 !== null ) {
$this -> dbMock -> labelPropertiesGet -> calledWith ( ... $data2 );
} else {
$this -> dbMock -> labelPropertiesGet -> never () -> called ();
}
}
public function provideLabelAdditions () : iterable {
return [
[[ 'caption' => " Software " ], [ $this -> userId , [ 'name' => " Software " ]], 2 , null , null , $this -> respGood ( - 1026 )],
[[ 'caption' => " Hardware " ], [ $this -> userId , [ 'name' => " Hardware " ]], 3 , null , null , $this -> respGood ( - 1027 )],
[[ 'caption' => " Software " ], [ $this -> userId , [ 'name' => " Software " ]], new ExceptionInput ( " constraintViolation " ), [ $this -> userId , " Software " , true ], [ 'id' => 2 ], $this -> respGood ( - 1026 )],
[[ 'caption' => " Hardware " ], [ $this -> userId , [ 'name' => " Hardware " ]], new ExceptionInput ( " constraintViolation " ), [ $this -> userId , " Hardware " , true ], [ 'id' => 3 ], $this -> respGood ( - 1027 )],
[[], [ $this -> userId , [ 'name' => " " ]], new ExceptionInput ( " typeViolation " ), null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " " ], [ $this -> userId , [ 'name' => " " ]], new ExceptionInput ( " typeViolation " ), null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " " ], [ $this -> userId , [ 'name' => " " ]], new ExceptionInput ( " typeViolation " ), null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-05 17:42:12 -04:00
];
}
2021-07-05 21:47:44 -04:00
2021-02-28 12:07:53 -05:00
/** @dataProvider provideLabelRemovals */
public function testRemoveALabel ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " removeLabel " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> labelRemove -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> labelRemove -> calledWith ( ... $data );
} else {
$this -> dbMock -> labelRemove -> never () -> called ();
}
}
2017-10-05 17:42:12 -04:00
2021-02-28 12:07:53 -05:00
public function provideLabelRemovals () : iterable {
return [
[[ 'label_id' => - 1042 ], [ $this -> userId , 18 ], true , $this -> respGood ()],
[[ 'label_id' => - 2112 ], [ $this -> userId , 1088 ], new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[ 'label_id' => 1 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'label_id' => 0 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'label_id' => - 10 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-05 17:42:12 -04:00
];
}
2021-02-28 12:07:53 -05:00
/** @dataProvider provideLabelRenamings */
public function testRenameALabel ( array $in , ? array $data , $out , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " renameLabel " , 'sid' => " PriestsOfSyrinx " ], $in );
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> dbMock -> labelPropertiesSet -> $action ( $out );
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out !== null ) {
$this -> dbMock -> labelPropertiesSet -> calledWith ( ... $data );
} else {
$this -> dbMock -> labelPropertiesSet -> never () -> called ();
}
2017-10-05 17:42:12 -04:00
}
2017-10-07 12:46:05 -04:00
2021-02-28 12:07:53 -05:00
public function provideLabelRenamings () : iterable {
return [
[[ 'label_id' => - 1042 , 'caption' => " Ook " ], [ $this -> userId , 18 , [ 'name' => " Ook " ]], true , $this -> respGood ()],
[[ 'label_id' => - 2112 , 'caption' => " Eek " ], [ $this -> userId , 1088 , [ 'name' => " Eek " ]], new ExceptionInput ( " subjectMissing " ), $this -> respGood ()],
[[ 'label_id' => - 1042 , 'caption' => " Eek " ], [ $this -> userId , 18 , [ 'name' => " Eek " ]], new ExceptionInput ( " constraintViolation " ), $this -> respGood ()],
[[ 'label_id' => - 1042 , 'caption' => " " ], [ $this -> userId , 18 , [ 'name' => " " ]], new ExceptionInput ( " missing " ), $this -> respGood ()],
[[ 'label_id' => - 1042 , 'caption' => " " ], [ $this -> userId , 18 , [ 'name' => " " ]], new ExceptionInput ( " whitespace " ), $this -> respGood ()],
[[ 'label_id' => - 1042 ], [ $this -> userId , 18 , [ 'name' => " " ]], new ExceptionInput ( " missing " ), $this -> respGood ()],
2021-03-02 11:27:48 -05:00
[[ 'label_id' => - 1042 ], [ $this -> userId , 18 , [ 'name' => " " ]], new ExceptionInput ( " typeViolation " ), $this -> respErr ( " INCORRECT_USAGE " )],
2021-02-28 12:07:53 -05:00
[[ 'label_id' => - 1 , 'caption' => " Ook " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'caption' => " Ook " ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-07 12:46:05 -04:00
];
2021-02-28 12:07:53 -05:00
}
/** @dataProvider provideCategoryListings */
public function testRetrieveCategoryLists ( array $in , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " getCategories " , 'sid' => " PriestsOfSyrinx " ], $in );
$this -> dbMock -> folderList -> with ( " ~ " , null , true ) -> returns ( new Result ( $this -> v ( $this -> folders )));
$this -> dbMock -> folderList -> with ( " ~ " , null , false ) -> returns ( new Result ( $this -> v ( $this -> topFolders )));
$this -> dbMock -> subscriptionList -> returns ( new Result ( $this -> v ( $this -> subscriptions )));
$this -> dbMock -> labelList -> returns ( new Result ( $this -> v ( $this -> labels )));
2022-04-19 20:19:51 -04:00
$this -> dbMock -> articleCount -> with ( " ~ " , $this -> equalTo (( new Context ) -> hidden ( false ) -> unread ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , self :: NOW ), null ))) -> returns ( 7 );
2021-02-28 12:07:53 -05:00
$this -> dbMock -> articleStarred -> returns ( $this -> v ( $this -> starred ));
$this -> assertMessage ( $exp , $this -> req ( $in ));
}
public function provideCategoryListings () : iterable {
2017-10-07 12:46:05 -04:00
$exp = [
[
2017-11-29 11:47:10 -05:00
[ 'id' => " 5 " , 'title' => " Local " , 'unread' => 10 , 'order_id' => 1 ],
[ 'id' => " 6 " , 'title' => " National " , 'unread' => 18 , 'order_id' => 2 ],
[ 'id' => " 4 " , 'title' => " Photography " , 'unread' => 0 , 'order_id' => 3 ],
[ 'id' => " 3 " , 'title' => " Politics " , 'unread' => 0 , 'order_id' => 4 ],
[ 'id' => " 2 " , 'title' => " Rocketry " , 'unread' => 5 , 'order_id' => 5 ],
[ 'id' => " 1 " , 'title' => " Science " , 'unread' => 2 , 'order_id' => 6 ],
[ 'id' => 0 , 'title' => " Uncategorized " , 'unread' => 0 ],
[ 'id' => - 1 , 'title' => " Special " , 'unread' => 11 ],
[ 'id' => - 2 , 'title' => " Labels " , 'unread' => " 6 " ],
2017-10-07 12:46:05 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => " 5 " , 'title' => " Local " , 'unread' => 10 , 'order_id' => 1 ],
[ 'id' => " 6 " , 'title' => " National " , 'unread' => 18 , 'order_id' => 2 ],
[ 'id' => " 3 " , 'title' => " Politics " , 'unread' => 0 , 'order_id' => 4 ],
[ 'id' => " 2 " , 'title' => " Rocketry " , 'unread' => 5 , 'order_id' => 5 ],
[ 'id' => " 1 " , 'title' => " Science " , 'unread' => 2 , 'order_id' => 6 ],
[ 'id' => 0 , 'title' => " Uncategorized " , 'unread' => 0 ],
[ 'id' => - 1 , 'title' => " Special " , 'unread' => 11 ],
[ 'id' => - 2 , 'title' => " Labels " , 'unread' => " 6 " ],
2017-10-07 12:46:05 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => " 5 " , 'title' => " Local " , 'unread' => 10 , 'order_id' => 1 ],
[ 'id' => " 6 " , 'title' => " National " , 'unread' => 18 , 'order_id' => 2 ],
[ 'id' => " 2 " , 'title' => " Rocketry " , 'unread' => 5 , 'order_id' => 5 ],
[ 'id' => " 1 " , 'title' => " Science " , 'unread' => 2 , 'order_id' => 6 ],
[ 'id' => - 1 , 'title' => " Special " , 'unread' => 11 ],
[ 'id' => - 2 , 'title' => " Labels " , 'unread' => " 6 " ],
2017-10-07 12:46:05 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => " 4 " , 'title' => " Photography " , 'unread' => 0 , 'order_id' => 1 ],
[ 'id' => " 3 " , 'title' => " Politics " , 'unread' => 28 , 'order_id' => 2 ],
[ 'id' => " 1 " , 'title' => " Science " , 'unread' => 7 , 'order_id' => 3 ],
[ 'id' => 0 , 'title' => " Uncategorized " , 'unread' => 0 ],
[ 'id' => - 1 , 'title' => " Special " , 'unread' => 11 ],
[ 'id' => - 2 , 'title' => " Labels " , 'unread' => " 6 " ],
2017-10-07 12:46:05 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => " 3 " , 'title' => " Politics " , 'unread' => 28 , 'order_id' => 2 ],
[ 'id' => " 1 " , 'title' => " Science " , 'unread' => 7 , 'order_id' => 3 ],
[ 'id' => 0 , 'title' => " Uncategorized " , 'unread' => 0 ],
[ 'id' => - 1 , 'title' => " Special " , 'unread' => 11 ],
[ 'id' => - 2 , 'title' => " Labels " , 'unread' => " 6 " ],
2017-10-07 12:46:05 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => " 3 " , 'title' => " Politics " , 'unread' => 28 , 'order_id' => 2 ],
[ 'id' => " 1 " , 'title' => " Science " , 'unread' => 7 , 'order_id' => 3 ],
[ 'id' => - 1 , 'title' => " Special " , 'unread' => 11 ],
[ 'id' => - 2 , 'title' => " Labels " , 'unread' => " 6 " ],
2017-10-07 12:46:05 -04:00
],
];
2021-02-28 12:07:53 -05:00
return [
[[ 'include_empty' => true ], $this -> respGood ( $exp [ 0 ])],
[[], $this -> respGood ( $exp [ 1 ])],
[[ 'unread_only' => true ], $this -> respGood ( $exp [ 2 ])],
[[ 'enable_nested' => true , 'include_empty' => true ], $this -> respGood ( $exp [ 3 ])],
[[ 'enable_nested' => true ], $this -> respGood ( $exp [ 4 ])],
[[ 'enable_nested' => true , 'unread_only' => true ], $this -> respGood ( $exp [ 5 ])],
];
2017-10-07 12:46:05 -04:00
}
2017-10-11 12:55:50 -04:00
2020-01-20 13:52:48 -05:00
public function testRetrieveCounterList () : void {
2017-10-11 12:55:50 -04:00
$in = [ 'op' => " getCounters " , 'sid' => " PriestsOfSyrinx " ];
2021-02-27 15:24:45 -05:00
$this -> dbMock -> folderList -> returns ( new Result ( $this -> v ( $this -> folders )));
$this -> dbMock -> subscriptionList -> returns ( new Result ( $this -> v ( $this -> subscriptions )));
$this -> dbMock -> labelList -> with ( " ~ " , false ) -> returns ( new Result ( $this -> v ( $this -> usedLabels )));
$this -> dbMock -> articleCount -> returns ( 7 );
$this -> dbMock -> articleStarred -> returns ( $this -> v ( $this -> starred ));
2017-10-11 12:55:50 -04:00
$exp = [
[ 'id' => " global-unread " , 'counter' => 35 ],
[ 'id' => " subscribed-feeds " , 'counter' => 6 ],
[ 'id' => 0 , 'counter' => 0 , 'auxcounter' => 0 ],
[ 'id' => - 1 , 'counter' => 4 , 'auxcounter' => 10 ],
[ 'id' => - 2 , 'counter' => 0 , 'auxcounter' => 0 ],
[ 'id' => - 3 , 'counter' => 7 , 'auxcounter' => 0 ],
[ 'id' => - 4 , 'counter' => 35 , 'auxcounter' => 0 ],
[ 'id' => - 1027 , 'counter' => 6 , 'auxcounter' => 100 ],
2017-11-02 17:17:46 -04:00
[ 'id' => - 1025 , 'counter' => 0 , 'auxcounter' => 2 ],
2018-01-02 16:53:38 -05:00
[ 'id' => " 3 " , 'updated' => " 2016-05-23T06:40:02Z " , 'counter' => 2 , 'has_img' => 1 ],
[ 'id' => " 4 " , 'updated' => " 2017-10-09T15:58:34Z " , 'counter' => 6 , 'has_img' => 1 ],
[ 'id' => " 6 " , 'updated' => " 2010-02-12T20:08:47Z " , 'counter' => 0 , 'has_img' => 1 ],
[ 'id' => " 1 " , 'updated' => " 2017-09-15T22:54:16Z " , 'counter' => 5 , 'has_img' => 0 ],
[ 'id' => " 5 " , 'updated' => " 2017-07-07T17:07:17Z " , 'counter' => 12 , 'has_img' => 0 ],
[ 'id' => " 2 " , 'updated' => " 2011-11-11T11:11:11Z " , 'counter' => 10 , 'has_img' => 1 ],
2017-10-11 12:55:50 -04:00
[ 'id' => 5 , 'kind' => " cat " , 'counter' => 10 ],
[ 'id' => 6 , 'kind' => " cat " , 'counter' => 18 ],
2017-12-06 19:16:35 -05:00
[ 'id' => 4 , 'kind' => " cat " , 'counter' => 0 ],
2017-10-11 12:55:50 -04:00
[ 'id' => 3 , 'kind' => " cat " , 'counter' => 28 ],
[ 'id' => 2 , 'kind' => " cat " , 'counter' => 5 ],
[ 'id' => 1 , 'kind' => " cat " , 'counter' => 7 ],
2017-12-06 19:16:35 -05:00
[ 'id' => 0 , 'kind' => " cat " , 'counter' => 0 ],
2017-11-02 17:17:46 -04:00
[ 'id' => - 2 , 'kind' => " cat " , 'counter' => 6 ],
2017-10-11 12:55:50 -04:00
];
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $this -> respGood ( $exp ), $this -> req ( $in ));
2022-04-19 20:19:51 -04:00
$this -> dbMock -> articleCount -> calledWith ( $this -> userId , $this -> equalTo (( new Context ) -> hidden ( false ) -> unread ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , self :: NOW ), null )));
2017-10-11 12:55:50 -04:00
}
2017-10-13 17:05:06 -04:00
2021-02-28 12:07:53 -05:00
/** @dataProvider provideLabelListings */
public function testRetrieveTheLabelList ( array $in , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " getLabels " , 'sid' => " PriestsOfSyrinx " ], $in );
$this -> dbMock -> labelList -> returns ( new Result ( $this -> v ( $this -> labels )));
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 1 ) -> returns ( $this -> v ([ 1 , 3 ]));
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 2 ) -> returns ( $this -> v ([ 3 ]));
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 3 ) -> returns ([]);
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 4 ) -> throws ( new ExceptionInput ( " idMissing " ));
$this -> assertMessage ( $exp , $this -> req ( $in ));
}
public function provideLabelListings () : iterable {
2017-10-13 17:05:06 -04:00
$exp = [
[
[ 'id' => - 1027 , 'caption' => " Fascinating " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
[ 'id' => - 1029 , 'caption' => " Interesting " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-30 13:11:27 -04:00
[ 'id' => - 1025 , 'caption' => " Logical " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-13 17:05:06 -04:00
],
[
[ 'id' => - 1027 , 'caption' => " Fascinating " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => true ],
[ 'id' => - 1029 , 'caption' => " Interesting " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-30 13:11:27 -04:00
[ 'id' => - 1025 , 'caption' => " Logical " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => true ],
2017-10-13 17:05:06 -04:00
],
[
[ 'id' => - 1027 , 'caption' => " Fascinating " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => true ],
[ 'id' => - 1029 , 'caption' => " Interesting " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-30 13:11:27 -04:00
[ 'id' => - 1025 , 'caption' => " Logical " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-13 17:05:06 -04:00
],
[
[ 'id' => - 1027 , 'caption' => " Fascinating " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
[ 'id' => - 1029 , 'caption' => " Interesting " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-30 13:11:27 -04:00
[ 'id' => - 1025 , 'caption' => " Logical " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-13 17:05:06 -04:00
],
[
[ 'id' => - 1027 , 'caption' => " Fascinating " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
[ 'id' => - 1029 , 'caption' => " Interesting " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-30 13:11:27 -04:00
[ 'id' => - 1025 , 'caption' => " Logical " , 'fg_color' => " " , 'bg_color' => " " , 'checked' => false ],
2017-10-13 17:05:06 -04:00
],
];
2021-02-28 12:07:53 -05:00
return [
[[], $this -> respGood ( $exp [ 0 ])],
[[ 'article_id' => 1 ], $this -> respGood ( $exp [ 1 ])],
[[ 'article_id' => 2 ], $this -> respGood ( $exp [ 2 ])],
[[ 'article_id' => 3 ], $this -> respGood ( $exp [ 3 ])],
[[ 'article_id' => 4 ], $this -> respGood ( $exp [ 4 ])],
];
2017-10-13 17:05:06 -04:00
}
2017-10-28 10:52:38 -04:00
2021-03-01 08:00:30 -05:00
public function provideLabelAssignments () : iterable {
$ids = implode ( " , " , range ( 1 , 100 ));
return [
[[ 'label_id' => - 2112 , 'article_ids' => $ids ], 1088 , Database :: ASSOC_REMOVE , $this -> respGood ([ 'status' => " OK " , 'updated' => 89 ])],
[[ 'label_id' => - 2112 , 'article_ids' => $ids , 'assign' => true ], 1088 , Database :: ASSOC_ADD , $this -> respGood ([ 'status' => " OK " , 'updated' => 7 ])],
[[ 'label_id' => - 2112 ], null , null , $this -> respGood ([ 'status' => " OK " , 'updated' => 0 ])],
[[ 'label_id' => - 42 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'label_id' => 42 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'label_id' => 0 ], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
[[], null , null , $this -> respErr ( " INCORRECT_USAGE " )],
2017-10-28 10:52:38 -04:00
];
2021-03-01 08:00:30 -05:00
}
/** @dataProvider provideLabelAssignments */
public function testAssignArticlesToALabel ( array $in , ? int $label , ? int $operation , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " setArticleLabel " , 'sid' => " PriestsOfSyrinx " ], $in );
$this -> dbMock -> labelArticlesSet -> with ( $this -> userId , " ~ " , " ~ " , Database :: ASSOC_REMOVE ) -> returns ( 42 ) -> returns ( 47 );
$this -> dbMock -> labelArticlesSet -> with ( $this -> userId , " ~ " , " ~ " , Database :: ASSOC_ADD ) -> returns ( 5 ) -> returns ( 2 );
$this -> dbMock -> labelArticlesSet -> with ( $this -> userId , " ~ " , $this -> equalTo (( new Context ) -> articles ([])), " ~ " ) -> throws ( new ExceptionInput ( " tooShort " ));
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $label !== null ) {
$this -> dbMock -> labelArticlesSet -> calledWith ( $this -> userId , $label , $this -> equalTo (( new Context ) -> articles ( range ( 1 , 50 ))), $operation );
$this -> dbMock -> labelArticlesSet -> calledWith ( $this -> userId , $label , $this -> equalTo (( new Context ) -> articles ( range ( 51 , 100 ))), $operation );
}
2017-10-28 10:52:38 -04:00
}
2017-10-30 13:11:27 -04:00
2020-01-20 13:52:48 -05:00
public function testRetrieveFeedTree () : void {
2017-10-30 13:11:27 -04:00
$in = [
[ 'op' => " getFeedTree " , 'sid' => " PriestsOfSyrinx " , 'include_empty' => true ],
[ 'op' => " getFeedTree " , 'sid' => " PriestsOfSyrinx " ],
];
2021-02-27 15:24:45 -05:00
$this -> dbMock -> folderList -> with ( " ~ " , null , true ) -> returns ( new Result ( $this -> v ( $this -> folders )));
$this -> dbMock -> subscriptionList -> returns ( new Result ( $this -> v ( $this -> subscriptions )));
$this -> dbMock -> labelList -> with ( " ~ " , true ) -> returns ( new Result ( $this -> v ( $this -> labels )));
$this -> dbMock -> articleCount -> returns ( 7 );
$this -> dbMock -> articleStarred -> returns ( $this -> v ( $this -> starred ));
2017-10-30 13:11:27 -04:00
// the expectations are packed tightly since they're very verbose; one can use var_export() (or convert to JSON) to pretty-print them
2020-03-01 15:16:50 -05:00
$exp = [ 'categories' => [ 'identifier' => 'id' , 'label' => 'name' , 'items' => [[ 'name' => 'Special' , 'id' => 'CAT:-1' , 'bare_id' => - 1 , 'type' => 'category' , 'unread' => 0 , 'items' => [[ 'name' => 'All articles' , 'id' => 'FEED:-4' , 'bare_id' => - 4 , 'icon' => 'images/folder.png' , 'unread' => 35 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Fresh articles' , 'id' => 'FEED:-3' , 'bare_id' => - 3 , 'icon' => 'images/fresh.png' , 'unread' => 7 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Starred articles' , 'id' => 'FEED:-1' , 'bare_id' => - 1 , 'icon' => 'images/star.png' , 'unread' => 4 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Published articles' , 'id' => 'FEED:-2' , 'bare_id' => - 2 , 'icon' => 'images/feed.png' , 'unread' => 0 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Archived articles' , 'id' => 'FEED:0' , 'bare_id' => 0 , 'icon' => 'images/archive.png' , 'unread' => 0 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Recently read' , 'id' => 'FEED:-6' , 'bare_id' => - 6 , 'icon' => 'images/time.png' , 'unread' => 0 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ]]],[ 'name' => 'Labels' , 'id' => 'CAT:-2' , 'bare_id' => - 2 , 'type' => 'category' , 'unread' => 6 , 'items' => [[ 'name' => 'Fascinating' , 'id' => 'FEED:-1027' , 'bare_id' => - 1027 , 'unread' => 0 , 'icon' => 'images/label.png' , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' , 'fg_color' => '' , 'bg_color' => '' ],[ 'name' => 'Interesting' , 'id' => 'FEED:-1029' , 'bare_id' => - 1029 , 'unread' => 0 , 'icon' => 'images/label.png' , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' , 'fg_color' => '' , 'bg_color' => '' ],[ 'name' => 'Logical' , 'id' => 'FEED:-1025' , 'bare_id' => - 1025 , 'unread' => 0 , 'icon' => 'images/label.png' , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' , 'fg_color' => '' , 'bg_color' => '' ]]],[ 'name' => 'Photography' , 'id' => 'CAT:4' , 'bare_id' => 4 , 'parent_id' => null , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(0 feeds)' , 'items' => []],[ 'name' => 'Politics' , 'id' => 'CAT:3' , 'bare_id' => 3 , 'parent_id' => null , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(3 feeds)' , 'items' => [[ 'name' => 'Local' , 'id' => 'CAT:5' , 'bare_id' => 5 , 'parent_id' => 3 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(1 feed)' , 'items' => [[ 'name' => 'Toronto Star' , 'id' => 'FEED:2' , 'bare_id' => 2 , 'icon' => 'feed-icons/2.ico' , 'error' => 'oops' , 'param' => '2011-11-11T11:11:11Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]],[ 'name' => 'National' , 'id' => 'CAT:6' , 'bare_id' => 6 , 'parent_id' => 3 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(2 feeds)' , 'items' => [[ 'name' => 'CBC News' , 'id' => 'FEED:4' , 'bare_id' => 4 , 'icon' => 'feed-icons/4.ico' , 'error' => '' , 'param' => '2017-10-09T15:58:34Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ],[ 'name' => 'Ottawa Citizen' , 'id' => 'FEED:5' , 'bare_id' => 5 , 'icon' => false , 'error' => '' , 'param' => '2017-07-07T17:07:17Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]]]],[ 'name' => 'Science' , 'id' => 'CAT:1' , 'bare_id' => 1 , 'parent_id' => null , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(2 feeds)' , 'items' => [[ 'name' => 'Rocketry' , 'id' => 'CAT:2' , 'bare_id' => 2 , 'parent_id' => 1 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(1 feed)' , 'items' => [[ 'name' => 'NASA JPL' , 'id' => 'FEED:1' , 'bare_id' => 1 , 'icon' => false , 'error' => '' , 'param' => '2017-09-15T22:54:16Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]],[ 'name' => 'Ars Technica' , 'id' => 'FEED:3' , 'bare_id' => 3 , 'icon' => 'feed-icons/3.ico' , 'error' => 'argh' , 'param' => '2016-05-23T06:40:02Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]],[ 'name' => 'Uncategorized' , 'id'
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $this -> respGood ( $exp ), $this -> req ( $in [ 0 ]));
2020-03-01 15:16:50 -05:00
$exp = [ 'categories' => [ 'identifier' => 'id' , 'label' => 'name' , 'items' => [[ 'name' => 'Special' , 'id' => 'CAT:-1' , 'bare_id' => - 1 , 'type' => 'category' , 'unread' => 0 , 'items' => [[ 'name' => 'All articles' , 'id' => 'FEED:-4' , 'bare_id' => - 4 , 'icon' => 'images/folder.png' , 'unread' => 35 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Fresh articles' , 'id' => 'FEED:-3' , 'bare_id' => - 3 , 'icon' => 'images/fresh.png' , 'unread' => 7 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Starred articles' , 'id' => 'FEED:-1' , 'bare_id' => - 1 , 'icon' => 'images/star.png' , 'unread' => 4 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Published articles' , 'id' => 'FEED:-2' , 'bare_id' => - 2 , 'icon' => 'images/feed.png' , 'unread' => 0 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Archived articles' , 'id' => 'FEED:0' , 'bare_id' => 0 , 'icon' => 'images/archive.png' , 'unread' => 0 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ],[ 'name' => 'Recently read' , 'id' => 'FEED:-6' , 'bare_id' => - 6 , 'icon' => 'images/time.png' , 'unread' => 0 , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' ]]],[ 'name' => 'Labels' , 'id' => 'CAT:-2' , 'bare_id' => - 2 , 'type' => 'category' , 'unread' => 6 , 'items' => [[ 'name' => 'Fascinating' , 'id' => 'FEED:-1027' , 'bare_id' => - 1027 , 'unread' => 0 , 'icon' => 'images/label.png' , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' , 'fg_color' => '' , 'bg_color' => '' ],[ 'name' => 'Interesting' , 'id' => 'FEED:-1029' , 'bare_id' => - 1029 , 'unread' => 0 , 'icon' => 'images/label.png' , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' , 'fg_color' => '' , 'bg_color' => '' ],[ 'name' => 'Logical' , 'id' => 'FEED:-1025' , 'bare_id' => - 1025 , 'unread' => 0 , 'icon' => 'images/label.png' , 'type' => 'feed' , 'auxcounter' => 0 , 'error' => '' , 'updated' => '' , 'fg_color' => '' , 'bg_color' => '' ]]],[ 'name' => 'Politics' , 'id' => 'CAT:3' , 'bare_id' => 3 , 'parent_id' => null , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(3 feeds)' , 'items' => [[ 'name' => 'Local' , 'id' => 'CAT:5' , 'bare_id' => 5 , 'parent_id' => 3 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(1 feed)' , 'items' => [[ 'name' => 'Toronto Star' , 'id' => 'FEED:2' , 'bare_id' => 2 , 'icon' => 'feed-icons/2.ico' , 'error' => 'oops' , 'param' => '2011-11-11T11:11:11Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]],[ 'name' => 'National' , 'id' => 'CAT:6' , 'bare_id' => 6 , 'parent_id' => 3 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(2 feeds)' , 'items' => [[ 'name' => 'CBC News' , 'id' => 'FEED:4' , 'bare_id' => 4 , 'icon' => 'feed-icons/4.ico' , 'error' => '' , 'param' => '2017-10-09T15:58:34Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ],[ 'name' => 'Ottawa Citizen' , 'id' => 'FEED:5' , 'bare_id' => 5 , 'icon' => false , 'error' => '' , 'param' => '2017-07-07T17:07:17Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]]]],[ 'name' => 'Science' , 'id' => 'CAT:1' , 'bare_id' => 1 , 'parent_id' => null , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(2 feeds)' , 'items' => [[ 'name' => 'Rocketry' , 'id' => 'CAT:2' , 'bare_id' => 2 , 'parent_id' => 1 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'param' => '(1 feed)' , 'items' => [[ 'name' => 'NASA JPL' , 'id' => 'FEED:1' , 'bare_id' => 1 , 'icon' => false , 'error' => '' , 'param' => '2017-09-15T22:54:16Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]],[ 'name' => 'Ars Technica' , 'id' => 'FEED:3' , 'bare_id' => 3 , 'icon' => 'feed-icons/3.ico' , 'error' => 'argh' , 'param' => '2016-05-23T06:40:02Z' , 'unread' => 0 , 'auxcounter' => 0 , 'checkbox' => false ]]],[ 'name' => 'Uncategorized' , 'id' => 'CAT:0' , 'bare_id' => 0 , 'type' => 'category' , 'auxcounter' => 0 , 'unread' => 0 , 'child_unread' => 0 , 'checkbox' => false , 'parent_id' => null , 'param' => '(1 feed)' , 'items' => [[ 'name' => 'Eurogamer' , 'id' => ' FE
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $this -> respGood ( $exp ), $this -> req ( $in [ 1 ]));
2022-04-19 20:19:51 -04:00
$this -> dbMock -> articleCount -> twice () -> calledWith ( $this -> userId , $this -> equalTo (( new Context ) -> hidden ( false ) -> unread ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , self :: NOW ), null )));
2017-10-30 13:11:27 -04:00
}
2017-10-30 23:18:43 -04:00
2021-02-08 19:07:49 -05:00
/** @dataProvider provideMassMarkings */
public function testMarkFeedsAsRead ( array $in , ? Context $c ) : void {
$base = [ 'op' => " catchupFeed " , 'sid' => " PriestsOfSyrinx " ];
$in = array_merge ( $base , $in );
2021-02-27 15:24:45 -05:00
$this -> dbMock -> articleMark -> throws ( new ExceptionInput ( " typeViolation " ));
2021-02-08 19:07:49 -05:00
// create a mock-current time
2021-02-27 15:24:45 -05:00
$this -> objMock -> get -> with ( \DateTimeImmutable :: class ) -> returns ( new \DateTimeImmutable ( self :: NOW ));
2021-02-08 19:07:49 -05:00
// TT-RSS always responds the same regardless of success or failure
2021-02-08 19:14:11 -05:00
$this -> assertMessage ( $this -> respGood ([ 'status' => " OK " ]), $this -> req ( $in ));
2021-02-08 19:07:49 -05:00
if ( isset ( $c )) {
2021-02-28 12:07:53 -05:00
$this -> dbMock -> articleMark -> calledWith ( $this -> userId , [ 'read' => true ], $this -> equalTo ( $c ));
2021-02-08 19:07:49 -05:00
} else {
2021-02-27 15:24:45 -05:00
$this -> dbMock -> articleMark -> never () -> called ();
2017-10-30 23:18:43 -04:00
}
2021-02-08 19:07:49 -05:00
}
public function provideMassMarkings () : iterable {
$c = ( new Context ) -> hidden ( false );
return [
[[], null ],
[[ 'feed_id' => 0 ], null ],
[[ 'feed_id' => 0 , 'is_cat' => true ], ( clone $c ) -> folderShallow ( 0 )],
[[ 'feed_id' => 0 , 'is_cat' => true , 'mode' => " bogus " ], ( clone $c ) -> folderShallow ( 0 )],
[[ 'feed_id' => - 1 ], ( clone $c ) -> starred ( true )],
2021-02-09 09:37:31 -05:00
[[ 'feed_id' => - 1 , 'is_cat' => " t " ], null ],
2022-04-19 20:19:51 -04:00
[[ 'feed_id' => - 3 ], ( clone $c ) -> modifiedRange ( Date :: sub ( " PT24H " , self :: NOW ), null )],
[[ 'feed_id' => - 3 , 'mode' => " 1day " ], ( clone $c ) -> modifiedRange ( Date :: sub ( " PT24H " , self :: NOW ), Date :: sub ( " PT24H " , self :: NOW ))], // this is a nonsense query, but it's what TT-RSS appearsto do
2021-02-08 19:07:49 -05:00
[[ 'feed_id' => - 3 , 'is_cat' => true ], null ],
[[ 'feed_id' => - 2 ], null ],
[[ 'feed_id' => - 2 , 'is_cat' => true ], ( clone $c ) -> labelled ( true )],
[[ 'feed_id' => - 2 , 'is_cat' => true , 'mode' => " all " ], ( clone $c ) -> labelled ( true )],
[[ 'feed_id' => - 4 ], $c ],
[[ 'feed_id' => - 4 , 'is_cat' => true ], null ],
2021-02-09 09:37:31 -05:00
[[ 'feed_id' => - 6 , 'is_cat' => " f " ], null ],
2021-02-08 19:07:49 -05:00
[[ 'feed_id' => - 2112 ], ( clone $c ) -> label ( 1088 )],
[[ 'feed_id' => 42 , 'is_cat' => true ], ( clone $c ) -> folder ( 42 )],
2022-04-19 20:19:51 -04:00
[[ 'feed_id' => 42 , 'is_cat' => true , 'mode' => " 1week " ], ( clone $c ) -> folder ( 42 ) -> modifiedRange ( null , Date :: sub ( " P1W " , self :: NOW ))],
2021-02-08 19:07:49 -05:00
[[ 'feed_id' => 2112 ], ( clone $c ) -> subscription ( 2112 )],
2022-04-19 20:19:51 -04:00
[[ 'feed_id' => 2112 , 'mode' => " 2week " ], ( clone $c ) -> subscription ( 2112 ) -> modifiedRange ( null , Date :: sub ( " P2W " , self :: NOW ))],
2021-02-08 19:07:49 -05:00
];
2017-10-30 23:18:43 -04:00
}
2017-11-02 17:17:46 -04:00
2021-03-01 08:00:30 -05:00
/** @dataProvider provideFeedListings */
public function testRetrieveFeedList ( array $in , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " getFeeds " , 'sid' => " PriestsOfSyrinx " ], $in );
2017-11-02 17:17:46 -04:00
// statistical mocks
2021-03-01 08:00:30 -05:00
$this -> dbMock -> articleStarred -> returns ( $this -> v ( $this -> starred ));
2022-04-19 20:19:51 -04:00
$this -> dbMock -> articleCount -> with ( " ~ " , $this -> equalTo (( new Context ) -> unread ( true ) -> hidden ( false ) -> modifiedRange ( Date :: sub ( " PT24H " , self :: NOW ), null ))) -> returns ( 7 );
2021-03-01 08:00:30 -05:00
$this -> dbMock -> articleCount -> with ( " ~ " , $this -> equalTo (( new Context ) -> unread ( true ) -> hidden ( false ))) -> returns ( 35 );
2017-11-02 17:17:46 -04:00
// label mocks
2021-03-01 08:00:30 -05:00
$this -> dbMock -> labelList -> returns ( new Result ( $this -> v ( $this -> labels )));
$this -> dbMock -> labelList -> with ( " ~ " , false ) -> returns ( new Result ( $this -> v ( $this -> usedLabels )));
2017-11-02 17:17:46 -04:00
// subscription and folder list and unread count mocks
2021-03-01 08:00:30 -05:00
$this -> dbMock -> folderList -> throws ( new ExceptionInput ( " subjectMissing " ));
$this -> dbMock -> subscriptionList -> throws ( new ExceptionInput ( " subjectMissing " ));
$this -> dbMock -> folderList -> with ( " ~ " ) -> returns ( new Result ( $this -> v ( $this -> folders )));
$this -> dbMock -> subscriptionList -> with ( " ~ " , null , true ) -> returns ( new Result ( $this -> v ( $this -> subscriptions )));
$this -> dbMock -> subscriptionList -> with ( " ~ " , null , false ) -> returns ( new Result ( $this -> v ( $this -> filterSubs ( null ))));
$this -> dbMock -> folderList -> with ( " ~ " , null ) -> returns ( new Result ( $this -> v ( $this -> folders )));
$this -> dbMock -> folderList -> with ( " ~ " , null , false ) -> returns ( new Result ( $this -> v ( $this -> filterFolders ( null ))));
2017-11-02 17:17:46 -04:00
foreach ( $this -> folders as $f ) {
2021-03-01 08:00:30 -05:00
$this -> dbMock -> folderList -> with ( " ~ " , $f [ 'id' ], false ) -> returns ( new Result ( $this -> v ( $this -> filterFolders ( $f [ 'id' ]))));
$this -> dbMock -> articleCount -> with ( " ~ " , $this -> equalTo (( new Context ) -> unread ( true ) -> hidden ( false ) -> folder ( $f [ 'id' ]))) -> returns ( $this -> reduceFolders ( $f [ 'id' ]));
$this -> dbMock -> subscriptionList -> with ( " ~ " , $f [ 'id' ], false ) -> returns ( new Result ( $this -> v ( $this -> filterSubs ( $f [ 'id' ]))));
}
$this -> assertMessage ( $exp , $this -> req ( $in ));
}
protected function filterFolders ( int $id = null ) : array {
return array_filter ( $this -> folders , function ( $value ) use ( $id ) {
return $value [ 'parent' ] == $id ;
});
}
protected function filterSubs ( int $folder = null ) : array {
return array_filter ( $this -> subscriptions , function ( $value ) use ( $folder ) {
return $value [ 'folder' ] == $folder ;
});
}
protected function reduceFolders ( int $id = null ) : int {
$out = 0 ;
foreach ( $this -> filterFolders ( $id ) as $f ) {
$out += $this -> reduceFolders ( $f [ 'id' ]);
2017-11-02 17:17:46 -04:00
}
2021-03-01 08:00:30 -05:00
$out += array_reduce ( array_filter ( $this -> subscriptions , function ( $value ) use ( $id ) {
return $value [ 'folder' ] == $id ;
}), function ( $sum , $value ) {
return $sum + $value [ 'unread' ];
}, 0 );
return $out ;
}
public function provideFeedListings () : iterable {
2017-11-02 17:17:46 -04:00
$exp = [
[
[ 'id' => 6 , 'title' => 'Eurogamer' , 'unread' => 0 , 'cat_id' => 0 , 'feed_url' => " http://example.com/6 " , 'has_icon' => true , 'last_updated' => 1266005327 , 'order_id' => 1 ],
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => - 1 , 'title' => " Starred articles " , 'unread' => " 4 " , 'cat_id' => - 1 ],
[ 'id' => - 2 , 'title' => " Published articles " , 'unread' => " 0 " , 'cat_id' => - 1 ],
[ 'id' => - 3 , 'title' => " Fresh articles " , 'unread' => " 7 " , 'cat_id' => - 1 ],
[ 'id' => - 4 , 'title' => " All articles " , 'unread' => " 35 " , 'cat_id' => - 1 ],
[ 'id' => - 6 , 'title' => " Recently read " , 'unread' => 0 , 'cat_id' => - 1 ],
2020-03-01 15:16:50 -05:00
[ 'id' => 0 , 'title' => " Archived articles " , 'unread' => " 0 " , 'cat_id' => - 1 ],
2017-11-02 17:17:46 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => - 1 , 'title' => " Starred articles " , 'unread' => " 4 " , 'cat_id' => - 1 ],
[ 'id' => - 3 , 'title' => " Fresh articles " , 'unread' => " 7 " , 'cat_id' => - 1 ],
[ 'id' => - 4 , 'title' => " All articles " , 'unread' => " 35 " , 'cat_id' => - 1 ],
2017-11-02 17:17:46 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => - 1027 , 'title' => " Fascinating " , 'unread' => " 6 " , 'cat_id' => - 2 ],
[ 'id' => - 1025 , 'title' => " Logical " , 'unread' => " 0 " , 'cat_id' => - 2 ],
2017-11-02 17:17:46 -04:00
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => - 1027 , 'title' => " Fascinating " , 'unread' => " 6 " , 'cat_id' => - 2 ],
2017-11-02 17:17:46 -04:00
],
[
[ 'id' => 3 , 'title' => 'Ars Technica' , 'unread' => 2 , 'cat_id' => 1 , 'feed_url' => " http://example.com/3 " , 'has_icon' => true , 'last_updated' => 1463985602 , 'order_id' => 1 ],
[ 'id' => 4 , 'title' => 'CBC News' , 'unread' => 6 , 'cat_id' => 6 , 'feed_url' => " http://example.com/4 " , 'has_icon' => true , 'last_updated' => 1507564714 , 'order_id' => 2 ],
[ 'id' => 6 , 'title' => 'Eurogamer' , 'unread' => 0 , 'cat_id' => 0 , 'feed_url' => " http://example.com/6 " , 'has_icon' => true , 'last_updated' => 1266005327 , 'order_id' => 3 ],
[ 'id' => 1 , 'title' => 'NASA JPL' , 'unread' => 5 , 'cat_id' => 2 , 'feed_url' => " http://example.com/1 " , 'has_icon' => false , 'last_updated' => 1505516056 , 'order_id' => 4 ],
[ 'id' => 5 , 'title' => 'Ottawa Citizen' , 'unread' => 12 , 'cat_id' => 6 , 'feed_url' => " http://example.com/5 " , 'has_icon' => false , 'last_updated' => 1499447237 , 'order_id' => 5 ],
[ 'id' => 2 , 'title' => 'Toronto Star' , 'unread' => 10 , 'cat_id' => 5 , 'feed_url' => " http://example.com/2 " , 'has_icon' => true , 'last_updated' => 1321009871 , 'order_id' => 6 ],
],
[
[ 'id' => 3 , 'title' => 'Ars Technica' , 'unread' => 2 , 'cat_id' => 1 , 'feed_url' => " http://example.com/3 " , 'has_icon' => true , 'last_updated' => 1463985602 , 'order_id' => 1 ],
[ 'id' => 4 , 'title' => 'CBC News' , 'unread' => 6 , 'cat_id' => 6 , 'feed_url' => " http://example.com/4 " , 'has_icon' => true , 'last_updated' => 1507564714 , 'order_id' => 2 ],
[ 'id' => 1 , 'title' => 'NASA JPL' , 'unread' => 5 , 'cat_id' => 2 , 'feed_url' => " http://example.com/1 " , 'has_icon' => false , 'last_updated' => 1505516056 , 'order_id' => 4 ],
[ 'id' => 5 , 'title' => 'Ottawa Citizen' , 'unread' => 12 , 'cat_id' => 6 , 'feed_url' => " http://example.com/5 " , 'has_icon' => false , 'last_updated' => 1499447237 , 'order_id' => 5 ],
[ 'id' => 2 , 'title' => 'Toronto Star' , 'unread' => 10 , 'cat_id' => 5 , 'feed_url' => " http://example.com/2 " , 'has_icon' => true , 'last_updated' => 1321009871 , 'order_id' => 6 ],
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => - 1027 , 'title' => " Fascinating " , 'unread' => " 6 " , 'cat_id' => - 2 ],
[ 'id' => - 1025 , 'title' => " Logical " , 'unread' => " 0 " , 'cat_id' => - 2 ],
[ 'id' => - 1 , 'title' => " Starred articles " , 'unread' => " 4 " , 'cat_id' => - 1 ],
[ 'id' => - 2 , 'title' => " Published articles " , 'unread' => " 0 " , 'cat_id' => - 1 ],
[ 'id' => - 3 , 'title' => " Fresh articles " , 'unread' => " 7 " , 'cat_id' => - 1 ],
[ 'id' => - 4 , 'title' => " All articles " , 'unread' => " 35 " , 'cat_id' => - 1 ],
[ 'id' => - 6 , 'title' => " Recently read " , 'unread' => 0 , 'cat_id' => - 1 ],
2020-03-01 15:16:50 -05:00
[ 'id' => 0 , 'title' => " Archived articles " , 'unread' => " 0 " , 'cat_id' => - 1 ],
2017-11-02 17:17:46 -04:00
[ 'id' => 3 , 'title' => 'Ars Technica' , 'unread' => 2 , 'cat_id' => 1 , 'feed_url' => " http://example.com/3 " , 'has_icon' => true , 'last_updated' => 1463985602 , 'order_id' => 1 ],
[ 'id' => 4 , 'title' => 'CBC News' , 'unread' => 6 , 'cat_id' => 6 , 'feed_url' => " http://example.com/4 " , 'has_icon' => true , 'last_updated' => 1507564714 , 'order_id' => 2 ],
[ 'id' => 6 , 'title' => 'Eurogamer' , 'unread' => 0 , 'cat_id' => 0 , 'feed_url' => " http://example.com/6 " , 'has_icon' => true , 'last_updated' => 1266005327 , 'order_id' => 3 ],
[ 'id' => 1 , 'title' => 'NASA JPL' , 'unread' => 5 , 'cat_id' => 2 , 'feed_url' => " http://example.com/1 " , 'has_icon' => false , 'last_updated' => 1505516056 , 'order_id' => 4 ],
[ 'id' => 5 , 'title' => 'Ottawa Citizen' , 'unread' => 12 , 'cat_id' => 6 , 'feed_url' => " http://example.com/5 " , 'has_icon' => false , 'last_updated' => 1499447237 , 'order_id' => 5 ],
[ 'id' => 2 , 'title' => 'Toronto Star' , 'unread' => 10 , 'cat_id' => 5 , 'feed_url' => " http://example.com/2 " , 'has_icon' => true , 'last_updated' => 1321009871 , 'order_id' => 6 ],
],
[
2017-11-29 11:47:10 -05:00
[ 'id' => - 1027 , 'title' => " Fascinating " , 'unread' => " 6 " , 'cat_id' => - 2 ],
[ 'id' => - 1 , 'title' => " Starred articles " , 'unread' => " 4 " , 'cat_id' => - 1 ],
[ 'id' => - 3 , 'title' => " Fresh articles " , 'unread' => " 7 " , 'cat_id' => - 1 ],
[ 'id' => - 4 , 'title' => " All articles " , 'unread' => " 35 " , 'cat_id' => - 1 ],
2017-11-02 17:17:46 -04:00
[ 'id' => 3 , 'title' => 'Ars Technica' , 'unread' => 2 , 'cat_id' => 1 , 'feed_url' => " http://example.com/3 " , 'has_icon' => true , 'last_updated' => 1463985602 , 'order_id' => 1 ],
[ 'id' => 4 , 'title' => 'CBC News' , 'unread' => 6 , 'cat_id' => 6 , 'feed_url' => " http://example.com/4 " , 'has_icon' => true , 'last_updated' => 1507564714 , 'order_id' => 2 ],
[ 'id' => 1 , 'title' => 'NASA JPL' , 'unread' => 5 , 'cat_id' => 2 , 'feed_url' => " http://example.com/1 " , 'has_icon' => false , 'last_updated' => 1505516056 , 'order_id' => 4 ],
[ 'id' => 5 , 'title' => 'Ottawa Citizen' , 'unread' => 12 , 'cat_id' => 6 , 'feed_url' => " http://example.com/5 " , 'has_icon' => false , 'last_updated' => 1499447237 , 'order_id' => 5 ],
[ 'id' => 2 , 'title' => 'Toronto Star' , 'unread' => 10 , 'cat_id' => 5 , 'feed_url' => " http://example.com/2 " , 'has_icon' => true , 'last_updated' => 1321009871 , 'order_id' => 6 ],
],
[
[ 'id' => 4 , 'title' => 'CBC News' , 'unread' => 6 , 'cat_id' => 6 , 'feed_url' => " http://example.com/4 " , 'has_icon' => true , 'last_updated' => 1507564714 , 'order_id' => 1 ],
[ 'id' => 5 , 'title' => 'Ottawa Citizen' , 'unread' => 12 , 'cat_id' => 6 , 'feed_url' => " http://example.com/5 " , 'has_icon' => false , 'last_updated' => 1499447237 , 'order_id' => 2 ],
],
[
[ 'id' => 4 , 'title' => 'CBC News' , 'unread' => 6 , 'cat_id' => 6 , 'feed_url' => " http://example.com/4 " , 'has_icon' => true , 'last_updated' => 1507564714 , 'order_id' => 1 ],
],
[
[ 'id' => 5 , 'title' => 'Ottawa Citizen' , 'unread' => 12 , 'cat_id' => 6 , 'feed_url' => " http://example.com/5 " , 'has_icon' => false , 'last_updated' => 1499447237 , 'order_id' => 2 ],
],
[
[ 'id' => 3 , 'title' => 'Ars Technica' , 'unread' => 2 , 'cat_id' => 1 , 'feed_url' => " http://example.com/3 " , 'has_icon' => true , 'last_updated' => 1463985602 , 'order_id' => 1 ],
],
[
[ 'id' => 2 , 'title' => " Rocketry " , 'unread' => 5 , 'is_cat' => true , 'order_id' => 1 ],
[ 'id' => 3 , 'title' => 'Ars Technica' , 'unread' => 2 , 'cat_id' => 1 , 'feed_url' => " http://example.com/3 " , 'has_icon' => true , 'last_updated' => 1463985602 , 'order_id' => 1 ],
],
];
2021-03-01 08:00:30 -05:00
return [
[[], $this -> respGood ( $exp [ 0 ])],
[[ 'cat_id' => - 1 ], $this -> respGood ( $exp [ 1 ])],
[[ 'cat_id' => - 1 , 'unread_only' => true ], $this -> respGood ( $exp [ 2 ])],
[[ 'cat_id' => - 2 ], $this -> respGood ( $exp [ 3 ])],
[[ 'cat_id' => - 2 , 'unread_only' => true ], $this -> respGood ( $exp [ 4 ])],
[[ 'cat_id' => - 3 ], $this -> respGood ( $exp [ 5 ])],
[[ 'cat_id' => - 3 , 'unread_only' => true ], $this -> respGood ( $exp [ 6 ])],
[[ 'cat_id' => - 4 ], $this -> respGood ( $exp [ 7 ])],
[[ 'cat_id' => - 4 , 'unread_only' => true ], $this -> respGood ( $exp [ 8 ])],
[[ 'cat_id' => 6 ], $this -> respGood ( $exp [ 9 ])],
[[ 'cat_id' => 6 , 'limit' => 1 ], $this -> respGood ( $exp [ 10 ])],
[[ 'cat_id' => 6 , 'limit' => 1 , 'offset' => 1 ], $this -> respGood ( $exp [ 11 ])],
[[ 'cat_id' => 1 ], $this -> respGood ( $exp [ 12 ])],
[[ 'cat_id' => 1 , 'include_nested' => true ], $this -> respGood ( $exp [ 13 ])],
[[ 'cat_id' => 0 , 'unread_only' => true ], $this -> respGood ([])],
[[ 'cat_id' => 2112 ], $this -> respGood ([])],
[[ 'cat_id' => 2112 , 'include_nested' => true ], $this -> respGood ([])],
[[ 'cat_id' => 6 , 'limit' => - 42 ], $this -> respGood ([])],
[[ 'cat_id' => 6 , 'offset' => 2 ], $this -> respGood ([])],
];
2017-11-02 17:17:46 -04:00
}
2021-03-01 08:00:30 -05:00
/** @dataProvider provideArticleChanges */
public function testChangeArticles ( array $in , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " updateArticle " , 'sid' => " PriestsOfSyrinx " ], $in );
$this -> dbMock -> articleMark -> returns ( 1 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'starred' => false ], $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]))) -> returns ( 2 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'starred' => true ], $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]))) -> returns ( 4 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'starred' => false ], $this -> equalTo (( new Context ) -> articles ([ 42 ]))) -> returns ( 8 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'starred' => true ], $this -> equalTo (( new Context ) -> articles ([ 2112 ]))) -> returns ( 16 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'read' => true ], $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]))) -> returns ( 32 ); // false is read for TT-RSS
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'read' => false ], $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]))) -> returns ( 64 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'read' => true ], $this -> equalTo (( new Context ) -> articles ([ 42 ]))) -> returns ( 128 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'read' => false ], $this -> equalTo (( new Context ) -> articles ([ 2112 ]))) -> returns ( 256 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'note' => " " ], $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]))) -> returns ( 512 );
$this -> dbMock -> articleMark -> with ( $this -> userId , [ 'note' => " eh " ], $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]))) -> returns ( 1024 );
$this -> dbMock -> articleList -> with ( $this -> userId , $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]) -> starred ( true )), " ~ " ) -> returns ( new Result ( $this -> v ([[ 'id' => 42 ]])));
$this -> dbMock -> articleList -> with ( $this -> userId , $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]) -> starred ( false )), " ~ " ) -> returns ( new Result ( $this -> v ([[ 'id' => 2112 ]])));
$this -> dbMock -> articleList -> with ( $this -> userId , $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]) -> unread ( true )), " ~ " ) -> returns ( new Result ( $this -> v ([[ 'id' => 42 ]])));
$this -> dbMock -> articleList -> with ( $this -> userId , $this -> equalTo (( new Context ) -> articles ([ 42 , 2112 ]) -> unread ( false )), " ~ " ) -> returns ( new Result ( $this -> v ([[ 'id' => 2112 ]])));
$this -> assertMessage ( $exp , $this -> req ( $in ));
2017-11-02 17:17:46 -04:00
}
2021-03-01 08:00:30 -05:00
public function provideArticleChanges () : iterable {
return [
[[], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_ids' => " 42, 2112, -1 " ], $this -> respGood ([ 'status' => " OK " , 'updated' => 2 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 0 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 2 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 0 , 'mode' => 0 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 2 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 0 , 'mode' => 1 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 4 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 0 , 'mode' => 2 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 24 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 0 , 'mode' => 3 ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 1 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 0 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 1 , 'mode' => 0 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 0 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 1 , 'mode' => 1 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 0 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 1 , 'mode' => 2 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 0 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 1 , 'mode' => 3 ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 2 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 32 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 2 , 'mode' => 0 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 32 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 2 , 'mode' => 1 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 64 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 2 , 'mode' => 2 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 384 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 2 , 'mode' => 3 ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 3 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 512 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 3 , 'mode' => 0 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 512 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 3 , 'mode' => 1 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 512 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 3 , 'mode' => 2 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 512 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 3 , 'mode' => 3 ], $this -> respGood ([ 'status' => " OK " , 'updated' => 512 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 3 , 'data' => " eh " ], $this -> respGood ([ 'status' => " OK " , 'updated' => 1024 ])],
[[ 'article_ids' => " 42, 2112, -1 " , 'field' => 4 ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_ids' => " 0, -1 " , 'field' => 3 ], $this -> respErr ( " INCORRECT_USAGE " )],
];
2017-11-02 17:17:46 -04:00
}
2017-11-09 14:21:12 -05:00
2021-03-01 08:00:30 -05:00
/** @dataProvider provideArticleListings */
public function testListArticles ( array $in , ResponseInterface $exp ) : void {
$in = array_merge ([ 'op' => " getArticle " , 'sid' => " PriestsOfSyrinx " ], $in );
$this -> dbMock -> labelList -> with ( " ~ " ) -> returns ( new Result ( $this -> v ( $this -> labels )));
$this -> dbMock -> labelList -> with ( " ~ " , false ) -> returns ( new Result ( $this -> v ( $this -> usedLabels )));
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 101 ) -> returns ([]);
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 102 ) -> returns ( $this -> v ([ 1 , 3 ]));
$this -> dbMock -> articleList -> with ( " ~ " , $this -> equalTo (( new Context ) -> articles ([ 101 , 102 ])), " ~ " ) -> returns ( new Result ( $this -> v ( $this -> articles )));
$this -> dbMock -> articleList -> with ( " ~ " , $this -> equalTo (( new Context ) -> articles ([ 101 ])), " ~ " ) -> returns ( new Result ( $this -> v ([ $this -> articles [ 0 ]])));
$this -> dbMock -> articleList -> with ( " ~ " , $this -> equalTo (( new Context ) -> articles ([ 102 ])), " ~ " ) -> returns ( new Result ( $this -> v ([ $this -> articles [ 1 ]])));
$this -> assertMessage ( $exp , $this -> req ( $in ));
2017-11-09 14:21:12 -05:00
}
2017-11-15 15:38:49 -05:00
2021-03-01 08:00:30 -05:00
public function provideArticleListings () : iterable {
2017-11-15 15:38:49 -05:00
$exp = [
[
2020-03-01 15:16:50 -05:00
'id' => " 101 " ,
'guid' => null ,
'title' => 'Article title 1' ,
'link' => 'http://example.com/1' ,
'labels' => [],
'unread' => true ,
'marked' => false ,
'published' => false ,
'comments' => " " ,
'author' => '' ,
'updated' => strtotime ( '2000-01-01T00:00:01Z' ),
'feed_id' => " 8 " ,
'feed_title' => " Feed 11 " ,
2017-11-15 15:38:49 -05:00
'attachments' => [],
2020-03-01 15:16:50 -05:00
'score' => 0 ,
'note' => null ,
'lang' => " " ,
'content' => '<p>Article content 1</p>' ,
2017-11-15 15:38:49 -05:00
],
[
2020-03-01 15:16:50 -05:00
'id' => " 102 " ,
'guid' => " SHA256:5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7 " ,
'title' => 'Article title 2' ,
'link' => 'http://example.com/2' ,
2017-11-15 15:38:49 -05:00
'labels' => [
[ - 1025 , " Logical " , " " , " " ],
[ - 1027 , " Fascinating " , " " , " " ],
],
2020-03-01 15:16:50 -05:00
'unread' => false ,
'marked' => false ,
'published' => false ,
'comments' => " " ,
'author' => " J. King " ,
'updated' => strtotime ( '2000-01-02T00:00:02Z' ),
'feed_id' => " 8 " ,
'feed_title' => " Feed 11 " ,
2017-11-15 15:38:49 -05:00
'attachments' => [
[
2017-11-29 11:47:10 -05:00
'id' => " 0 " ,
2017-11-15 15:38:49 -05:00
'content_url' => " http://example.com/text " ,
'content_type' => " text/plain " ,
'title' => " " ,
'duration' => " " ,
'width' => " " ,
'height' => " " ,
2017-11-29 11:47:10 -05:00
'post_id' => " 102 " ,
2017-11-15 15:38:49 -05:00
],
],
2020-03-01 15:16:50 -05:00
'score' => 0 ,
'note' => " Note 2 " ,
'lang' => " " ,
2017-11-15 15:38:49 -05:00
'content' => '<p>Article content 2</p>' ,
],
];
2021-03-01 08:00:30 -05:00
return [
[[], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_id' => 0 ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_id' => - 1 ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_id' => " 0,-1 " ], $this -> respErr ( " INCORRECT_USAGE " )],
[[ 'article_id' => " 101,102 " ], $this -> respGood ( $exp )],
[[ 'article_id' => " 101 " ], $this -> respGood ([ $exp [ 0 ]])],
[[ 'article_id' => " 102 " ], $this -> respGood ([ $exp [ 1 ]])],
];
2017-11-15 15:38:49 -05:00
}
2017-11-20 00:09:20 -05:00
2020-12-21 21:49:57 -05:00
/** @dataProvider provideHeadlines */
public function testRetrieveHeadlines ( bool $full , array $in , $out , Context $c , array $fields , array $order , ResponseInterface $exp ) : void {
$base = [ 'op' => $full ? " getHeadlines " : " getCompactHeadlines " , 'sid' => " PriestsOfSyrinx " ];
$in = array_merge ( $base , $in );
2021-02-27 15:24:45 -05:00
$action = ( $out instanceof \Exception ) ? " throws " : " returns " ;
$this -> objMock -> get -> with ( \DateTimeImmutable :: class ) -> returns ( new \DateTimeImmutable ( self :: NOW ));
$this -> dbMock -> labelList -> returns ( new Result ( $this -> v ( $this -> labels )));
$this -> dbMock -> labelList -> with ( " ~ " , false ) -> returns ( new Result ( $this -> v ( $this -> usedLabels )));
$this -> dbMock -> articleLabelsGet -> returns ([]);
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 2112 ) -> returns ( $this -> v ([ 1 , 3 ]));
$this -> dbMock -> articleCategoriesGet -> returns ([]);
$this -> dbMock -> articleCategoriesGet -> with ( " ~ " , 2112 ) -> returns ([ " Boring " , " Illogical " ]);
$this -> dbMock -> articleCount -> returns ( 2 );
$this -> dbMock -> articleList -> $action ( $out );
2020-12-21 21:49:57 -05:00
$this -> assertMessage ( $exp , $this -> req ( $in ));
if ( $out ) {
2021-02-27 15:24:45 -05:00
$this -> dbMock -> articleList -> calledWith ( $this -> userId , $this -> equalTo ( $c ), $fields , $order );
2020-12-21 21:49:57 -05:00
} else {
2021-02-27 15:24:45 -05:00
$this -> dbMock -> articleList -> never () -> called ();
2017-11-22 20:18:16 -05:00
}
}
2020-12-21 21:49:57 -05:00
public function provideHeadlines () : iterable {
$t = Date :: normalize ( self :: NOW );
$c = ( new Context ) -> hidden ( false ) -> limit ( 200 );
$out = $this -> generateHeadlines ( 47 );
$gone = new ExceptionInput ( " idMissing " );
$comp = new Result ( $this -> v ([[ 'id' => 47 ], [ 'id' => 2112 ]]));
$expFull = $this -> outputHeadlines ( 47 );
$expComp = $this -> respGood ([[ 'id' => 47 ], [ 'id' => 2112 ]]);
$fields = [ " id " , " guid " , " title " , " author " , " url " , " unread " , " starred " , " edited_date " , " published_date " , " subscription " , " subscription_title " , " note " ];
$sort = [ " edited_date desc " ];
return [
2022-04-19 20:19:51 -04:00
[ true , [], null , $c , [], [], $this -> respErr ( " INCORRECT_USAGE " )],
[ true , [ 'feed_id' => 0 ], null , $c , [], [], $this -> respGood ([])],
[ true , [ 'feed_id' => - 1 ], $out , ( clone $c ) -> starred ( true ), $fields , [ " marked_date desc " ], $expFull ],
[ true , [ 'feed_id' => - 2 ], null , $c , [], [], $this -> respGood ([])],
[ true , [ 'feed_id' => - 4 ], $out , $c , $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 2112 ], $gone , ( clone $c ) -> subscription ( 2112 ), $fields , $sort , $this -> respGood ([])],
[ true , [ 'feed_id' => - 2112 ], $out , ( clone $c ) -> label ( 1088 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'view_mode' => " adaptive " ], $out , ( clone $c ) -> unread ( true ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'view_mode' => " published " ], null , $c , [], [], $this -> respGood ([])],
[ true , [ 'feed_id' => - 2112 , 'view_mode' => " adaptive " ], $out , ( clone $c ) -> label ( 1088 ) -> unread ( true ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 2112 , 'view_mode' => " unread " ], $out , ( clone $c ) -> label ( 1088 ) -> unread ( true ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 42 , 'view_mode' => " marked " ], $out , ( clone $c ) -> subscription ( 42 ) -> starred ( true ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 42 , 'view_mode' => " has_note " ], $out , ( clone $c ) -> subscription ( 42 ) -> annotated ( true ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 42 , 'view_mode' => " unread " , 'search' => " unread:false " ], null , $c , [], [], $this -> respGood ([])],
[ true , [ 'feed_id' => 42 , 'search' => " pub:true " ], null , $c , [], [], $this -> respGood ([])],
[ true , [ 'feed_id' => - 4 , 'limit' => 5 ], $out , ( clone $c ) -> limit ( 5 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'skip' => 2 ], $out , ( clone $c ) -> offset ( 2 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'limit' => 5 , 'skip' => 2 ], $out , ( clone $c ) -> limit ( 5 ) -> offset ( 2 ), $fields , $sort , $expFull ],
2022-04-19 22:53:36 -04:00
[ true , [ 'feed_id' => - 4 , 'since_id' => 47 ], $out , ( clone $c ) -> articleRange ( 48 , null ), $fields , $sort , $expFull ],
2022-04-19 20:19:51 -04:00
[ true , [ 'feed_id' => - 3 , 'is_cat' => true ], $out , $c , $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'is_cat' => true ], $out , $c , $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 2 , 'is_cat' => true ], $out , ( clone $c ) -> labelled ( true ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 1 , 'is_cat' => true ], null , $c , [], [], $this -> respGood ([])],
[ true , [ 'feed_id' => 0 , 'is_cat' => true ], $out , ( clone $c ) -> folderShallow ( 0 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 0 , 'is_cat' => true , 'include_nested' => true ], $out , ( clone $c ) -> folderShallow ( 0 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 42 , 'is_cat' => true ], $out , ( clone $c ) -> folderShallow ( 42 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => 42 , 'is_cat' => true , 'include_nested' => true ], $out , ( clone $c ) -> folder ( 42 ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'order_by' => " feed_dates " ], $out , $c , $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 4 , 'order_by' => " date_reverse " ], $out , $c , $fields , [ " edited_date " ], $expFull ],
[ true , [ 'feed_id' => 42 , 'search' => " interesting " ], $out , ( clone $c ) -> subscription ( 42 ) -> searchTerms ([ " interesting " ]), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 6 ], $out , ( clone $c ) -> unread ( false ) -> markedRange ( Date :: sub ( " PT24H " , $t ), null ), $fields , [ " marked_date desc " ], $expFull ],
[ true , [ 'feed_id' => - 6 , 'view_mode' => " unread " ], null , $c , $fields , $sort , $this -> respGood ([])],
[ true , [ 'feed_id' => - 3 ], $out , ( clone $c ) -> unread ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , $t ), null ), $fields , $sort , $expFull ],
[ true , [ 'feed_id' => - 3 , 'view_mode' => " marked " ], $out , ( clone $c ) -> unread ( true ) -> starred ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , $t ), null ), $fields , $sort , $expFull ],
[ false , [], null , ( clone $c ) -> limit ( null ), [], [], $this -> respErr ( " INCORRECT_USAGE " )],
[ false , [ 'feed_id' => 0 ], null , ( clone $c ) -> limit ( null ), [], [], $this -> respGood ([])],
[ false , [ 'feed_id' => - 1 ], $comp , ( clone $c ) -> limit ( null ) -> starred ( true ), [ " id " ], [ " marked_date desc " ], $expComp ],
[ false , [ 'feed_id' => - 2 ], null , ( clone $c ) -> limit ( null ), [], [], $this -> respGood ([])],
[ false , [ 'feed_id' => - 4 ], $comp , ( clone $c ) -> limit ( null ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => 2112 ], $gone , ( clone $c ) -> limit ( null ) -> subscription ( 2112 ), [ " id " ], $sort , $this -> respGood ([])],
[ false , [ 'feed_id' => - 2112 ], $comp , ( clone $c ) -> limit ( null ) -> label ( 1088 ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 4 , 'view_mode' => " adaptive " ], $comp , ( clone $c ) -> limit ( null ) -> unread ( true ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 4 , 'view_mode' => " published " ], null , ( clone $c ) -> limit ( null ), [], [], $this -> respGood ([])],
[ false , [ 'feed_id' => - 2112 , 'view_mode' => " adaptive " ], $comp , ( clone $c ) -> limit ( null ) -> label ( 1088 ) -> unread ( true ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 2112 , 'view_mode' => " unread " ], $comp , ( clone $c ) -> limit ( null ) -> label ( 1088 ) -> unread ( true ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => 42 , 'view_mode' => " marked " ], $comp , ( clone $c ) -> limit ( null ) -> subscription ( 42 ) -> starred ( true ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => 42 , 'view_mode' => " has_note " ], $comp , ( clone $c ) -> limit ( null ) -> subscription ( 42 ) -> annotated ( true ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 4 , 'limit' => 5 ], $comp , ( clone $c ) -> limit ( 5 ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 4 , 'skip' => 2 ], $comp , ( clone $c ) -> limit ( null ) -> offset ( 2 ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 4 , 'limit' => 5 , 'skip' => 2 ], $comp , ( clone $c ) -> limit ( 5 ) -> offset ( 2 ), [ " id " ], $sort , $expComp ],
2022-04-19 22:53:36 -04:00
[ false , [ 'feed_id' => - 4 , 'since_id' => 47 ], $comp , ( clone $c ) -> limit ( null ) -> articleRange ( 48 , null ), [ " id " ], $sort , $expComp ],
2022-04-19 20:19:51 -04:00
[ false , [ 'feed_id' => - 6 ], $comp , ( clone $c ) -> limit ( null ) -> unread ( false ) -> markedRange ( Date :: sub ( " PT24H " , $t ), null ), [ " id " ], [ " marked_date desc " ], $expComp ],
[ false , [ 'feed_id' => - 6 , 'view_mode' => " unread " ], null , ( clone $c ) -> limit ( null ), [ " id " ], $sort , $this -> respGood ([])],
[ false , [ 'feed_id' => - 3 ], $comp , ( clone $c ) -> limit ( null ) -> unread ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , $t ), null ), [ " id " ], $sort , $expComp ],
[ false , [ 'feed_id' => - 3 , 'view_mode' => " marked " ], $comp , ( clone $c ) -> limit ( null ) -> unread ( true ) -> starred ( true ) -> modifiedRange ( Date :: sub ( " PT24H " , $t ), null ), [ " id " ], $sort , $expComp ],
2020-12-21 21:49:57 -05:00
];
}
2020-01-20 13:52:48 -05:00
public function testRetrieveFullHeadlinesCheckingExtraFields () : void {
2017-11-22 20:18:16 -05:00
$in = [
// empty results
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 4 ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 4 , 'show_content' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 4 , 'include_attachments' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 4 , 'include_header' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 3 , 'is_cat' => true , 'include_header' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 1 , 'is_cat' => true , 'include_header' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => 2112 , 'include_header' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 4 , 'include_header' => true , 'order_by' => " date_reverse " ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => 42 , 'skip' => 47 , 'include_header' => true ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => 42 , 'skip' => 47 , 'include_header' => true , 'order_by' => " date_reverse " ],
[ 'op' => " getHeadlines " , 'sid' => " PriestsOfSyrinx " , 'feed_id' => - 4 , 'show_excerpt' => true ],
];
2021-03-01 09:37:56 -05:00
$this -> dbMock -> labelList -> with ( " ~ " ) -> returns ( new Result ( $this -> v ( $this -> labels )));
$this -> dbMock -> labelList -> with ( " ~ " , false ) -> returns ( new Result ( $this -> v ( $this -> usedLabels )));
$this -> dbMock -> articleLabelsGet -> returns ([]);
$this -> dbMock -> articleLabelsGet -> with ( " ~ " , 2112 ) -> returns ( $this -> v ([ 1 , 3 ]));
$this -> dbMock -> articleCategoriesGet -> returns ([]);
$this -> dbMock -> articleCategoriesGet -> with ( " ~ " , 2112 ) -> returns ([ " Boring " , " Illogical " ]);
$this -> dbMock -> articleList -> returns ( $this -> generateHeadlines ( 1 ));
$this -> dbMock -> articleCount -> returns ( 0 );
$this -> dbMock -> articleCount -> with ( " ~ " , $this -> equalTo (( new Context ) -> unread ( true ) -> hidden ( false ))) -> returns ( 1 );
2017-11-22 20:18:16 -05:00
// sanity check; this makes sure extra fields are not included in default situations
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 0 ]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $this -> outputHeadlines ( 1 ), $test );
2017-11-22 20:18:16 -05:00
// test 'show_content'
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 1 ]);
2018-01-03 23:13:08 -05:00
$this -> assertArrayHasKey ( " content " , $test -> getPayload ()[ 'content' ][ 0 ]);
$this -> assertArrayHasKey ( " content " , $test -> getPayload ()[ 'content' ][ 1 ]);
2017-11-22 20:18:16 -05:00
foreach ( $this -> generateHeadlines ( 1 ) as $key => $row ) {
2018-01-03 23:13:08 -05:00
$this -> assertSame ( $row [ 'content' ], $test -> getPayload ()[ 'content' ][ $key ][ 'content' ]);
2017-11-22 20:18:16 -05:00
}
// test 'include_attachments'
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 2 ]);
2017-11-22 20:18:16 -05:00
$exp = [
[
2017-11-29 11:47:10 -05:00
'id' => " 0 " ,
2017-11-22 20:18:16 -05:00
'content_url' => " http://example.com/text " ,
'content_type' => " text/plain " ,
'title' => " " ,
'duration' => " " ,
'width' => " " ,
'height' => " " ,
2017-11-29 11:47:10 -05:00
'post_id' => " 2112 " ,
2017-11-22 20:18:16 -05:00
],
];
2018-01-03 23:13:08 -05:00
$this -> assertArrayHasKey ( " attachments " , $test -> getPayload ()[ 'content' ][ 0 ]);
$this -> assertArrayHasKey ( " attachments " , $test -> getPayload ()[ 'content' ][ 1 ]);
$this -> assertSame ([], $test -> getPayload ()[ 'content' ][ 0 ][ 'attachments' ]);
$this -> assertSame ( $exp , $test -> getPayload ()[ 'content' ][ 1 ][ 'attachments' ]);
2017-11-22 20:18:16 -05:00
// test 'include_header'
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 3 ]);
2018-01-03 23:13:08 -05:00
$exp = $this -> respGood ([
2017-11-22 20:18:16 -05:00
[ 'id' => - 4 , 'is_cat' => false , 'first_id' => 1 ],
2018-01-03 23:13:08 -05:00
$this -> outputHeadlines ( 1 ) -> getPayload ()[ 'content' ],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'include_header' with a category
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 4 ]);
2018-01-03 23:13:08 -05:00
$exp = $this -> respGood ([
2017-11-22 20:18:16 -05:00
[ 'id' => - 3 , 'is_cat' => true , 'first_id' => 1 ],
2018-01-03 23:13:08 -05:00
$this -> outputHeadlines ( 1 ) -> getPayload ()[ 'content' ],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'include_header' with an empty result
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 5 ]);
2017-11-22 20:18:16 -05:00
$exp = $this -> respGood ([
[ 'id' => - 1 , 'is_cat' => true , 'first_id' => 0 ],
[],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'include_header' with an erroneous result
2021-03-01 09:37:56 -05:00
$this -> dbMock -> articleList -> with ( " ~ " , $this -> equalTo (( new Context ) -> limit ( 200 ) -> subscription ( 2112 ) -> hidden ( false )), " ~ " , [ " edited_date desc " ]) -> throws ( new ExceptionInput ( " subjectMissing " ));
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 6 ]);
2017-11-22 20:18:16 -05:00
$exp = $this -> respGood ([
[ 'id' => 2112 , 'is_cat' => false , 'first_id' => 0 ],
[],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'include_header' with ascending order
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 7 ]);
2018-01-03 23:13:08 -05:00
$exp = $this -> respGood ([
2017-11-22 20:18:16 -05:00
[ 'id' => - 4 , 'is_cat' => false , 'first_id' => 0 ],
2018-01-03 23:13:08 -05:00
$this -> outputHeadlines ( 1 ) -> getPayload ()[ 'content' ],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'include_header' with skip
2021-03-01 09:37:56 -05:00
$this -> dbMock -> articleList -> with ( " ~ " , $this -> equalTo (( new Context ) -> limit ( 1 ) -> subscription ( 42 ) -> hidden ( false )), " ~ " , [ " edited_date desc " ]) -> returns ( $this -> generateHeadlines ( 1867 ));
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 8 ]);
2018-01-03 23:13:08 -05:00
$exp = $this -> respGood ([
2017-11-22 20:18:16 -05:00
[ 'id' => 42 , 'is_cat' => false , 'first_id' => 1867 ],
2018-01-03 23:13:08 -05:00
$this -> outputHeadlines ( 1 ) -> getPayload ()[ 'content' ],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'include_header' with skip and ascending order
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 9 ]);
2018-01-03 23:13:08 -05:00
$exp = $this -> respGood ([
2017-11-22 20:18:16 -05:00
[ 'id' => 42 , 'is_cat' => false , 'first_id' => 0 ],
2018-01-03 23:13:08 -05:00
$this -> outputHeadlines ( 1 ) -> getPayload ()[ 'content' ],
]);
2018-01-11 11:09:25 -05:00
$this -> assertMessage ( $exp , $test );
2017-11-22 20:18:16 -05:00
// test 'show_excerpt'
$exp1 = " “This & that, you know‽” " ;
$exp2 = " Pour vous faire mieux connaitre d’ ou \ u { 300} vient l’ erreur de ceux qui bla \ u { 302}ment la volupte \ u { 301}, et qui louent en… " ;
2017-11-23 14:05:26 -05:00
$test = $this -> req ( $in [ 10 ]);
2018-01-03 23:13:08 -05:00
$this -> assertArrayHasKey ( " excerpt " , $test -> getPayload ()[ 'content' ][ 0 ]);
$this -> assertArrayHasKey ( " excerpt " , $test -> getPayload ()[ 'content' ][ 1 ]);
$this -> assertSame ( $exp1 , $test -> getPayload ()[ 'content' ][ 0 ][ 'excerpt' ]);
$this -> assertSame ( $exp2 , $test -> getPayload ()[ 'content' ][ 1 ][ 'excerpt' ]);
2017-11-22 20:18:16 -05:00
}
protected function generateHeadlines ( int $id ) : Result {
2017-12-31 13:58:37 -05:00
return new Result ( $this -> v ([
2017-11-22 20:18:16 -05:00
[
2020-03-01 15:16:50 -05:00
'id' => $id ,
'url' => 'http://example.com/1' ,
'title' => 'Article title 1' ,
2017-11-22 20:18:16 -05:00
'subscription_title' => " Feed 2112 " ,
2020-03-01 15:16:50 -05:00
'author' => '' ,
'content' => '<p>“This & that, you know‽”</p>' ,
'guid' => null ,
'published_date' => '2000-01-01 00:00:00' ,
'edited_date' => '2000-01-01 00:00:00' ,
'modified_date' => '2000-01-01 01:00:00' ,
'unread' => 0 ,
'starred' => 0 ,
'edition' => 101 ,
'subscription' => 12 ,
'fingerprint' => 'f5cb8bfc1c7396dc9816af212a3e2ac5221585c2a00bf7ccb6aabd95dcfcd6a6:fb0bc8f8cb08913dc5a497db700e327f1d34e4987402687d494a5891f24714d4:18fdd4fa93d693128c43b004399e5c9cea6c261ddfa002518d3669f55d8c2207' ,
'media_url' => null ,
'media_type' => null ,
'note' => " " ,
2017-11-22 20:18:16 -05:00
],
[
2020-03-01 15:16:50 -05:00
'id' => 2112 ,
'url' => 'http://example.com/2' ,
'title' => 'Article title 2' ,
2017-11-22 20:18:16 -05:00
'subscription_title' => " Feed 11 " ,
2020-03-01 15:16:50 -05:00
'author' => 'J. King' ,
'content' => $this -> richContent ,
'guid' => '5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7' ,
'published_date' => '2000-01-02 00:00:00' ,
'edited_date' => '2000-01-02 00:00:02' ,
'modified_date' => '2000-01-02 02:00:00' ,
'unread' => 1 ,
'starred' => 1 ,
'edition' => 202 ,
'subscription' => 8 ,
'fingerprint' => '0e86d2de822a174fe3c44a466953e63ca1f1a58a19cbf475fce0855d4e3d5153:13075894189c47ffcfafd1dfe7fbb539f7c74a69d35a399b3abf8518952714f9:2abd0a8cba83b8214a66c8f0293ba63e467d720540e29ff8ddcdab069d4f1c9e' ,
'media_url' => " http://example.com/text " ,
'media_type' => " text/plain " ,
'note' => " Note 2 " ,
2017-11-22 20:18:16 -05:00
],
2017-12-31 13:58:37 -05:00
]));
2017-11-22 20:18:16 -05:00
}
protected function outputHeadlines ( int $id ) : Response {
return $this -> respGood ([
[
2020-03-01 15:16:50 -05:00
'id' => $id ,
'guid' => '' ,
'title' => 'Article title 1' ,
'link' => 'http://example.com/1' ,
'labels' => [],
'unread' => false ,
'marked' => false ,
'published' => false ,
'author' => '' ,
'updated' => strtotime ( '2000-01-01T00:00:00Z' ),
'is_updated' => false ,
'feed_id' => " 12 " ,
'feed_title' => " Feed 2112 " ,
'score' => 0 ,
'note' => null ,
'lang' => " " ,
'tags' => [],
'comments_count' => 0 ,
'comments_link' => " " ,
2017-11-29 12:15:37 -05:00
'always_display_attachments' => false ,
2017-11-22 20:18:16 -05:00
],
[
2020-03-01 15:16:50 -05:00
'id' => 2112 ,
'guid' => " SHA256:5be8a5a46ecd52ed132191c8d27fb1af6b3d4edc00234c5d9f8f0e10562ed3b7 " ,
'title' => 'Article title 2' ,
'link' => 'http://example.com/2' ,
2017-11-22 20:18:16 -05:00
'labels' => [
[ - 1025 , " Logical " , " " , " " ],
[ - 1027 , " Fascinating " , " " , " " ],
],
2020-03-01 15:16:50 -05:00
'unread' => true ,
'marked' => true ,
'published' => false ,
'author' => " J. King " ,
'updated' => strtotime ( '2000-01-02T00:00:02Z' ),
'is_updated' => true ,
'feed_id' => " 8 " ,
'feed_title' => " Feed 11 " ,
'score' => 0 ,
'note' => " Note 2 " ,
'lang' => " " ,
'tags' => [ " Boring " , " Illogical " ],
'comments_count' => 0 ,
'comments_link' => " " ,
2017-11-29 12:15:37 -05:00
'always_display_attachments' => false ,
2017-11-22 20:18:16 -05:00
],
]);
}
2017-09-28 10:16:24 -04:00
}