2019-03-18 22:49:47 -04:00
< ? php
/** @ license MIT
* Copyright 2017 J . King , Dustin Wilson et al .
* See LICENSE and AUTHORS files for details */
declare ( strict_types = 1 );
namespace JKingWeb\Arsse\TestCase\REST\Fever ;
use JKingWeb\Arsse\Arsse ;
use JKingWeb\Arsse\Conf ;
use JKingWeb\Arsse\User ;
use JKingWeb\Arsse\Database ;
use JKingWeb\Arsse\Service ;
use JKingWeb\Arsse\REST\Request ;
use JKingWeb\Arsse\Test\Result ;
use JKingWeb\Arsse\Misc\Date ;
use JKingWeb\Arsse\Context\Context ;
use JKingWeb\Arsse\Db\ExceptionInput ;
2019-03-20 22:24:35 -04:00
use JKingWeb\Arsse\User\Exception as UserException ;
2019-03-18 22:49:47 -04:00
use JKingWeb\Arsse\Db\Transaction ;
use JKingWeb\Arsse\REST\Fever\API ;
use Psr\Http\Message\ResponseInterface ;
use Zend\Diactoros\ServerRequest ;
use Zend\Diactoros\Response\JsonResponse ;
use Zend\Diactoros\Response\EmptyResponse ;
/** @covers \JKingWeb\Arsse\REST\Fever\API<extended> */
class TestAPI extends \JKingWeb\Arsse\Test\AbstractTest {
protected function v ( $value ) {
return $value ;
}
2019-03-27 15:09:04 -04:00
protected function req ( $dataGet , $dataPost = " " , string $method = " POST " , string $type = null , string $url = " " , string $user = null ) : ResponseInterface {
2019-03-18 22:49:47 -04:00
$url = " /fever/ " . $url ;
$server = [
'REQUEST_METHOD' => $method ,
'REQUEST_URI' => $url ,
'HTTP_CONTENT_TYPE' => $type ? ? " application/x-www-form-urlencoded " ,
];
$req = new ServerRequest ( $server , [], $url , $method , " php://memory " );
2019-03-27 15:09:04 -04:00
if ( ! is_array ( $dataGet )) {
parse_str ( $dataGet , $dataGet );
2019-03-18 22:49:47 -04:00
}
2019-03-27 15:09:04 -04:00
$req = $req -> withRequestTarget ( $url ) -> withQueryParams ( $dataGet );
2019-03-18 22:49:47 -04:00
if ( is_array ( $dataPost )) {
$req = $req -> withParsedBody ( $dataPost );
} else {
$body = $req -> getBody ();
2019-03-20 22:24:35 -04:00
$body -> write ( $dataPost );
2019-03-18 22:49:47 -04:00
$req = $req -> withBody ( $body );
}
if ( isset ( $user )) {
if ( strlen ( $user )) {
$req = $req -> withAttribute ( " authenticated " , true ) -> withAttribute ( " authenticatedUser " , $user );
} else {
$req = $req -> withAttribute ( " authenticationFailed " , true );
}
}
return $this -> h -> dispatch ( $req );
}
public function setUp () {
self :: clearData ();
self :: setConf ();
// create a mock user manager
2019-03-27 11:54:47 -04:00
Arsse :: $user = \Phake :: mock ( User :: class );
\Phake :: when ( Arsse :: $user ) -> auth -> thenReturn ( true );
2019-03-18 22:49:47 -04:00
Arsse :: $user -> id = " john.doe@example.com " ;
// create a mock database interface
2019-03-27 11:54:47 -04:00
Arsse :: $db = \Phake :: mock ( Database :: class );
\Phake :: when ( Arsse :: $db ) -> begin -> thenReturn ( \Phake :: mock ( Transaction :: class ));
\Phake :: when ( Arsse :: $db ) -> tokenLookup -> thenReturn ([ 'user' => " john.doe@example.com " ]);
2019-03-27 15:09:04 -04:00
// instantiate the handler as a partial mock to simplify testing
$this -> h = \Phake :: partialMock ( API :: class );
\Phake :: when ( $this -> h ) -> baseResponse -> thenReturn ([]);
2019-03-18 22:49:47 -04:00
}
public function tearDown () {
self :: clearData ();
}
2019-03-24 15:05:21 -04:00
/** @dataProvider provideTokenAuthenticationRequests */
2019-03-21 11:00:07 -04:00
public function testAuthenticateAUserToken ( bool $httpRequired , bool $tokenEnforced , string $httpUser = null , array $dataPost , array $dataGet , ResponseInterface $exp ) {
2019-03-18 22:49:47 -04:00
self :: setConf ([
'userHTTPAuthRequired' => $httpRequired ,
'userSessionEnforced' => $tokenEnforced ,
], true );
2019-03-19 23:37:08 -04:00
Arsse :: $user -> id = null ;
2019-03-18 22:49:47 -04:00
\Phake :: when ( Arsse :: $db ) -> tokenLookup -> thenThrow ( new ExceptionInput ( " subjectMissing " ));
\Phake :: when ( Arsse :: $db ) -> tokenLookup ( " fever.login " , " validtoken " ) -> thenReturn ([ 'user' => " jane.doe@example.com " ]);
2019-03-27 15:09:04 -04:00
// test only the authentication process
\Phake :: when ( $this -> h ) -> baseResponse -> thenReturnCallback ( function ( bool $authenticated ) {
return [ 'auth' => ( int ) $authenticated ];
});
2019-03-27 11:54:47 -04:00
\Phake :: when ( $this -> h ) -> processRequest -> thenReturnCallback ( function ( $out , $G , $P ) {
return $out ;
});
2019-03-18 22:49:47 -04:00
$act = $this -> req ( $dataGet , $dataPost , " POST " , null , " " , $httpUser );
$this -> assertMessage ( $exp , $act );
}
2019-03-24 15:05:21 -04:00
public function provideTokenAuthenticationRequests () {
2019-03-27 15:09:04 -04:00
$success = new JsonResponse ([ 'auth' => 1 ]);
$failure = new JsonResponse ([ 'auth' => 0 ]);
2019-03-19 23:37:08 -04:00
$denied = new EmptyResponse ( 401 );
2019-03-18 22:49:47 -04:00
return [
2019-03-19 23:37:08 -04:00
[ false , true , null , [], [ 'api' => null ], $failure ],
[ false , false , null , [], [ 'api' => null ], $failure ],
[ true , true , null , [], [ 'api' => null ], $denied ],
[ true , false , null , [], [ 'api' => null ], $denied ],
[ false , true , " " , [], [ 'api' => null ], $denied ],
[ false , false , " " , [], [ 'api' => null ], $denied ],
[ true , true , " " , [], [ 'api' => null ], $denied ],
[ true , false , " " , [], [ 'api' => null ], $denied ],
[ false , true , null , [], [ 'api' => null , 'api_key' => " validToken " ], $failure ],
[ false , false , null , [], [ 'api' => null , 'api_key' => " validToken " ], $failure ],
[ true , true , null , [], [ 'api' => null , 'api_key' => " validToken " ], $denied ],
[ true , false , null , [], [ 'api' => null , 'api_key' => " validToken " ], $denied ],
2019-03-21 13:49:55 -04:00
[ false , true , " " , [], [ 'api' => null , 'api_key' => " validToken " ], $denied ],
[ false , false , " " , [], [ 'api' => null , 'api_key' => " validToken " ], $denied ],
[ true , true , " " , [], [ 'api' => null , 'api_key' => " validToken " ], $denied ],
[ true , false , " " , [], [ 'api' => null , 'api_key' => " validToken " ], $denied ],
[ false , true , " validUser " , [], [ 'api' => null , 'api_key' => " validToken " ], $failure ],
[ false , false , " validUser " , [], [ 'api' => null , 'api_key' => " validToken " ], $success ],
[ true , true , " validUser " , [], [ 'api' => null , 'api_key' => " validToken " ], $failure ],
[ true , false , " validUser " , [], [ 'api' => null , 'api_key' => " validToken " ], $success ],
2019-03-19 23:37:08 -04:00
[ false , true , null , [ 'api_key' => " validToken " ], [ 'api' => null ], $success ],
[ false , false , null , [ 'api_key' => " validToken " ], [ 'api' => null ], $success ],
[ true , true , null , [ 'api_key' => " validToken " ], [ 'api' => null ], $denied ],
[ true , false , null , [ 'api_key' => " validToken " ], [ 'api' => null ], $denied ],
[ false , true , " " , [ 'api_key' => " validToken " ], [ 'api' => null ], $denied ],
[ false , false , " " , [ 'api_key' => " validToken " ], [ 'api' => null ], $denied ],
[ true , true , " " , [ 'api_key' => " validToken " ], [ 'api' => null ], $denied ],
[ true , false , " " , [ 'api_key' => " validToken " ], [ 'api' => null ], $denied ],
[ false , true , " validUser " , [ 'api_key' => " validToken " ], [ 'api' => null ], $success ],
[ false , false , " validUser " , [ 'api_key' => " validToken " ], [ 'api' => null ], $success ],
[ true , true , " validUser " , [ 'api_key' => " validToken " ], [ 'api' => null ], $success ],
[ true , false , " validUser " , [ 'api_key' => " validToken " ], [ 'api' => null ], $success ],
[ false , true , null , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $failure ],
[ false , false , null , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $failure ],
[ true , true , null , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $denied ],
[ true , false , null , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $denied ],
[ false , true , " " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $denied ],
[ false , false , " " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $denied ],
[ true , true , " " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $denied ],
[ true , false , " " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $denied ],
[ false , true , " validUser " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $failure ],
[ false , false , " validUser " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $success ],
[ true , true , " validUser " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $failure ],
[ true , false , " validUser " , [ 'api_key' => " invalidToken " ], [ 'api' => null ], $success ],
2019-03-18 22:49:47 -04:00
];
}
2019-03-27 15:09:04 -04:00
public function testListGroups () {
\Phake :: when ( Arsse :: $db ) -> tagList ( Arsse :: $user -> id ) -> thenReturn ( new Result ([
[ 'id' => 1 , 'name' => " Fascinating " , 'subscriptions' => 2 ],
[ 'id' => 2 , 'name' => " Interesting " , 'subscriptions' => 2 ],
[ 'id' => 3 , 'name' => " Boring " , 'subscriptions' => 0 ],
]));
\Phake :: when ( Arsse :: $db ) -> tagSummarize ( Arsse :: $user -> id ) -> thenReturn ( new Result ([
[ 'id' => 1 , 'name' => " Fascinating " , 'subscription' => 1 ],
[ 'id' => 1 , 'name' => " Fascinating " , 'subscription' => 2 ],
[ 'id' => 2 , 'name' => " Interesting " , 'subscription' => 1 ],
[ 'id' => 2 , 'name' => " Interesting " , 'subscription' => 3 ],
]));
$exp = new JsonResponse ([
'groups' => [
[ 'id' => 1 , 'title' => " Fascinating " ],
[ 'id' => 2 , 'title' => " Interesting " ],
[ 'id' => 3 , 'title' => " Boring " ],
],
'feeds_groups' => [
[ 'group_id' => 1 , 'feed_ids' => " 1,2 " ],
[ 'group_id' => 2 , 'feed_ids' => " 1,3 " ],
],
]);
$act = $this -> req ( " api&groups " );
$this -> assertMessage ( $exp , $act );
}
public function testListFeeds () {
\Phake :: when ( Arsse :: $db ) -> subscriptionList ( Arsse :: $user -> id ) -> thenReturn ( new Result ([
[ 'id' => 1 , 'feed' => 5 , 'title' => " Ankh-Morpork News " , 'url' => " http://example.com/feed " , 'source' => " http://example.com/ " , 'edited' => " 2019-01-01 21:12:00 " , 'favicon' => " http://example.com/favicon.ico " ],
[ 'id' => 2 , 'feed' => 9 , 'title' => " Ook, Ook Eek Ook! " , 'url' => " http://example.net/feed " , 'source' => " http://example.net/ " , 'edited' => " 1988-06-24 12:21:00 " , 'favicon' => " " ],
[ 'id' => 3 , 'feed' => 1 , 'title' => " The Last Soul " , 'url' => " http://example.org/feed " , 'source' => " http://example.org/ " , 'edited' => " 1991-08-12 03:22:00 " , 'favicon' => " http://example.org/favicon.ico " ],
]));
\Phake :: when ( Arsse :: $db ) -> tagSummarize ( Arsse :: $user -> id ) -> thenReturn ( new Result ([
[ 'id' => 1 , 'name' => " Fascinating " , 'subscription' => 1 ],
[ 'id' => 1 , 'name' => " Fascinating " , 'subscription' => 2 ],
[ 'id' => 2 , 'name' => " Interesting " , 'subscription' => 1 ],
[ 'id' => 2 , 'name' => " Interesting " , 'subscription' => 3 ],
]));
$exp = new JsonResponse ([
'feeds' => [
[ 'id' => 1 , 'favicon_id' => 5 , 'title' => " Ankh-Morpork News " , 'url' => " http://example.com/feed " , 'site_url' => " http://example.com/ " , 'is_spark' => 0 , 'last_updated_on_time' => strtotime ( " 2019-01-01T21:12:00Z " )],
[ 'id' => 2 , 'favicon_id' => 0 , 'title' => " Ook, Ook Eek Ook! " , 'url' => " http://example.net/feed " , 'site_url' => " http://example.net/ " , 'is_spark' => 0 , 'last_updated_on_time' => strtotime ( " 1988-06-24T12:21:00Z " )],
[ 'id' => 3 , 'favicon_id' => 1 , 'title' => " The Last Soul " , 'url' => " http://example.org/feed " , 'site_url' => " http://example.org/ " , 'is_spark' => 0 , 'last_updated_on_time' => strtotime ( " 1991-08-12T03:22:00Z " )],
],
'feeds_groups' => [
[ 'group_id' => 1 , 'feed_ids' => " 1,2 " ],
[ 'group_id' => 2 , 'feed_ids' => " 1,3 " ],
],
]);
$act = $this -> req ( " api&feeds " );
$this -> assertMessage ( $exp , $act );
}
2019-03-18 22:49:47 -04:00
}