2017-04-01 15:42:10 -04:00
< ? php
2017-11-16 20:23:18 -05:00
/** @ license MIT
* Copyright 2017 J . King , Dustin Wilson et al .
* See LICENSE and AUTHORS files for details */
2017-04-01 15:42:10 -04:00
declare ( strict_types = 1 );
namespace JKingWeb\Arsse\REST\NextCloudNews ;
2017-08-29 10:50:31 -04:00
2017-07-17 07:47:57 -04:00
use JKingWeb\Arsse\Arsse ;
2017-11-17 18:12:00 -05:00
use JKingWeb\Arsse\Database ;
2017-05-18 23:03:33 -04:00
use JKingWeb\Arsse\User ;
2017-08-02 18:27:04 -04:00
use JKingWeb\Arsse\Service ;
2017-06-18 10:23:37 -04:00
use JKingWeb\Arsse\Misc\Context ;
2017-09-27 22:25:45 -04:00
use JKingWeb\Arsse\Misc\ValueInfo ;
2017-04-02 12:14:15 -04:00
use JKingWeb\Arsse\AbstractException ;
2017-04-02 21:34:30 -04:00
use JKingWeb\Arsse\Db\ExceptionInput ;
2017-05-18 23:03:33 -04:00
use JKingWeb\Arsse\Feed\Exception as FeedException ;
2017-04-02 21:34:30 -04:00
use JKingWeb\Arsse\REST\Response ;
2017-04-01 15:42:10 -04:00
class V1_2 extends \JKingWeb\Arsse\REST\AbstractHandler {
2017-06-03 17:34:37 -04:00
const REALM = " NextCloud News API v1-2 " ;
2017-07-15 16:44:06 -04:00
const VERSION = " 11.0.5 " ;
2017-07-07 08:13:03 -04:00
2017-07-07 15:25:47 -04:00
protected $dateFormat = " unix " ;
2017-07-07 08:13:03 -04:00
protected $validInput = [
2017-10-19 20:35:45 -04:00
'name' => ValueInfo :: T_STRING ,
'url' => ValueInfo :: T_STRING ,
'folderId' => ValueInfo :: T_INT ,
'feedTitle' => ValueInfo :: T_STRING ,
'userId' => ValueInfo :: T_STRING ,
'feedId' => ValueInfo :: T_INT ,
'newestItemId' => ValueInfo :: T_INT ,
'batchSize' => ValueInfo :: T_INT ,
'offset' => ValueInfo :: T_INT ,
'type' => ValueInfo :: T_INT ,
'id' => ValueInfo :: T_INT ,
'getRead' => ValueInfo :: T_BOOL ,
'oldestFirst' => ValueInfo :: T_BOOL ,
'lastModified' => ValueInfo :: T_DATE ,
'items' => ValueInfo :: T_MIXED | ValueInfo :: M_ARRAY ,
2017-07-07 08:13:03 -04:00
];
2017-11-28 13:46:07 -05:00
protected $paths = [
'folders' => [ 'GET' => " folderList " , 'POST' => " folderAdd " ],
'folders/1' => [ 'PUT' => " folderRename " , 'DELETE' => " folderRemove " ],
'folders/1/read' => [ 'PUT' => " folderMarkRead " ],
'feeds' => [ 'GET' => " subscriptionList " , 'POST' => " subscriptionAdd " ],
'feeds/1' => [ 'DELETE' => " subscriptionRemove " ],
'feeds/1/move' => [ 'PUT' => " subscriptionMove " ],
'feeds/1/rename' => [ 'PUT' => " subscriptionRename " ],
'feeds/1/read' => [ 'PUT' => " subscriptionMarkRead " ],
'feeds/all' => [ 'GET' => " feedListStale " ],
'feeds/update' => [ 'GET' => " feedUpdate " ],
'items' => [ 'GET' => " articleList " ],
'items/updated' => [ 'GET' => " articleList " ],
'items/read' => [ 'PUT' => " articleMarkReadAll " ],
'items/1/read' => [ 'PUT' => " articleMarkRead " ],
'items/1/unread' => [ 'PUT' => " articleMarkRead " ],
'items/read/multiple' => [ 'PUT' => " articleMarkReadMulti " ],
'items/unread/multiple' => [ 'PUT' => " articleMarkReadMulti " ],
'items/1/1/star' => [ 'PUT' => " articleMarkStarred " ],
'items/1/1/unstar' => [ 'PUT' => " articleMarkStarred " ],
'items/star/multiple' => [ 'PUT' => " articleMarkStarredMulti " ],
'items/unstar/multiple' => [ 'PUT' => " articleMarkStarredMulti " ],
'cleanup/before-update' => [ 'GET' => " cleanupBefore " ],
'cleanup/after-update' => [ 'GET' => " cleanupAfter " ],
'version' => [ 'GET' => " serverVersion " ],
'status' => [ 'GET' => " serverStatus " ],
'user' => [ 'GET' => " userStatus " ],
];
2017-06-03 17:34:37 -04:00
2017-08-29 10:50:31 -04:00
public function __construct () {
2017-04-02 21:34:30 -04:00
}
2017-08-29 10:50:31 -04:00
public function dispatch ( \JKingWeb\Arsse\REST\Request $req ) : Response {
2017-04-02 21:34:30 -04:00
// try to authenticate
2017-08-29 10:50:31 -04:00
if ( ! Arsse :: $user -> authHTTP ()) {
2017-07-20 22:40:09 -04:00
return new Response ( 401 , " " , " " , [ 'WWW-Authenticate: Basic realm="' . self :: REALM . '"' ]);
}
2017-11-29 15:28:33 -05:00
// handle HTTP OPTIONS requests
if ( $req -> method == " OPTIONS " ) {
return $this -> handleHTTPOptions ( $req -> paths );
}
2017-04-02 21:34:30 -04:00
// normalize the input
2017-08-29 10:50:31 -04:00
if ( $req -> body ) {
2017-04-02 21:34:30 -04:00
// if the entity body is not JSON according to content type, return "415 Unsupported Media Type"
2017-08-29 10:50:31 -04:00
if ( ! preg_match ( " <^application/json \ b|^ $ > " , $req -> type )) {
2017-07-20 22:40:09 -04:00
return new Response ( 415 , " " , " " , [ 'Accept: application/json' ]);
}
2017-07-07 21:06:38 -04:00
$data = @ json_decode ( $req -> body , true );
2017-08-29 10:50:31 -04:00
if ( json_last_error () != \JSON_ERROR_NONE ) {
2017-04-02 21:34:30 -04:00
// if the body could not be parsed as JSON, return "400 Bad Request"
return new Response ( 400 );
}
} else {
$data = [];
}
// FIXME: Do query parameters take precedence in NextCloud? Is there a conflict error when values differ?
2017-10-19 20:35:45 -04:00
$data = $this -> normalizeInput ( array_merge ( $data , $req -> query ), $this -> validInput , " unix " );
2017-05-19 23:52:26 -04:00
// check to make sure the requested function is implemented
try {
$func = $this -> chooseCall ( $req -> paths , $req -> method );
2017-11-29 13:41:26 -05:00
} catch ( Exception404 $e ) {
return new Response ( 404 );
2017-08-29 10:50:31 -04:00
} catch ( Exception405 $e ) {
2017-05-19 23:52:26 -04:00
return new Response ( 405 , " " , " " , [ " Allow: " . $e -> getMessage ()]);
}
2017-08-29 10:50:31 -04:00
if ( ! method_exists ( $this , $func )) {
2017-09-05 19:35:14 -04:00
return new Response ( 501 ); // @codeCoverageIgnore
2017-07-20 22:40:09 -04:00
}
2017-05-19 23:52:26 -04:00
// dispatch
try {
return $this -> $func ( $req -> paths , $data );
2017-09-28 09:01:43 -04:00
// @codeCoverageIgnoreStart
2017-08-29 10:50:31 -04:00
} catch ( Exception $e ) {
2017-05-19 23:52:26 -04:00
// if there was a REST exception return 400
return new Response ( 400 );
2017-08-29 10:50:31 -04:00
} catch ( AbstractException $e ) {
2017-05-19 23:52:26 -04:00
// if there was any other Arsse exception return 500
return new Response ( 500 );
}
2017-09-05 19:35:14 -04:00
// @codeCoverageIgnoreEnd
2017-05-19 23:52:26 -04:00
}
2017-11-28 13:46:07 -05:00
protected function normalizePath ( array $url ) : string {
2017-09-27 22:25:45 -04:00
// any URL components which are database IDs (integers greater than zero) should be replaced with "1", for easier comparison (we don't care about the specific ID)
2017-08-29 10:50:31 -04:00
for ( $a = 0 ; $a < sizeof ( $url ); $a ++ ) {
2017-09-27 22:25:45 -04:00
if ( ValueInfo :: id ( $url [ $a ])) {
$url [ $a ] = " 1 " ;
2017-07-20 22:40:09 -04:00
}
2017-05-19 23:52:26 -04:00
}
2017-11-28 13:46:07 -05:00
return implode ( " / " , $url );
}
protected function chooseCall ( array $url , string $method ) : string {
// normalize the URL path
$url = $this -> normalizePath ( $url );
2017-05-19 23:52:26 -04:00
// normalize the HTTP method to uppercase
$method = strtoupper ( $method );
// we now evaluate the supplied URL against every supported path for the selected scope
// the URL is evaluated as an array so as to avoid decoded escapes turning invalid URLs into valid ones
2017-11-29 13:41:26 -05:00
if ( isset ( $this -> paths [ $url ])) {
// if the path is supported, make sure the method is allowed
if ( isset ( $this -> paths [ $url ][ $method ])) {
// if it is allowed, return the object method to run
return $this -> paths [ $url ][ $method ];
} else {
// otherwise return 405
throw new Exception405 ( implode ( " , " , array_keys ( $this -> paths [ $url ])));
2017-11-29 22:42:50 -05:00
}
2017-11-29 13:41:26 -05:00
} else {
2017-11-29 15:28:33 -05:00
// if the path is not supported, return 404
2017-11-29 22:42:50 -05:00
throw new Exception404 ();
2017-04-02 21:34:30 -04:00
}
}
2017-07-07 08:13:03 -04:00
2017-11-30 14:47:39 -05:00
protected function folderTranslate ( array $folder ) : array {
// map fields to proper names
$folder = $this -> fieldMapNames ( $folder , [
'id' => " id " ,
'name' => " name " ,
]);
// cast values
$folder = $this -> fieldMapTypes ( $folder , [
'id' => " int " ,
'name' => " string " ,
], $this -> dateFormat );
return $folder ;
}
2017-07-07 08:13:03 -04:00
protected function feedTranslate ( array $feed ) : array {
// map fields to proper names
$feed = $this -> fieldMapNames ( $feed , [
'id' => " id " ,
'url' => " url " ,
'title' => " title " ,
'added' => " added " ,
'pinned' => " pinned " ,
'link' => " source " ,
'faviconLink' => " favicon " ,
'folderId' => " top_folder " ,
'unreadCount' => " unread " ,
'ordering' => " order_type " ,
'updateErrorCount' => " err_count " ,
'lastUpdateError' => " err_msg " ,
]);
// cast values
$feed = $this -> fieldMapTypes ( $feed , [
2017-11-30 14:47:39 -05:00
'id' => " int " ,
'url' => " string " ,
'title' => " string " ,
'added' => " datetime " ,
'pinned' => " bool " ,
'link' => " string " ,
'faviconLink' => " string " ,
'folderId' => " int " ,
'unreadCount' => " int " ,
'ordering' => " int " ,
'updateErrorCount' => " int " ,
'lastUpdateError' => " string " ,
2017-07-07 15:25:47 -04:00
], $this -> dateFormat );
2017-07-07 08:13:03 -04:00
return $feed ;
}
protected function articleTranslate ( array $article ) : array {
// map fields to proper names
$article = $this -> fieldMapNames ( $article , [
'id' => " edition " ,
'guid' => " guid " ,
'guidHash' => " id " ,
'url' => " url " ,
'title' => " title " ,
'author' => " author " ,
'pubDate' => " edited_date " ,
'body' => " content " ,
2017-07-09 17:57:18 -04:00
'enclosureMime' => " media_type " ,
2017-07-07 08:13:03 -04:00
'enclosureLink' => " media_url " ,
2017-07-09 17:57:18 -04:00
'feedId' => " subscription " ,
2017-07-07 08:13:03 -04:00
'unread' => " unread " ,
'starred' => " starred " ,
'lastModified' => " modified_date " ,
'fingerprint' => " fingerprint " ,
]);
// cast values
$article = $this -> fieldMapTypes ( $article , [
2017-11-30 14:47:39 -05:00
'id' => " int " ,
'guid' => " string " ,
'guidHash' => " string " ,
'url' => " string " ,
'title' => " string " ,
'author' => " string " ,
'pubDate' => " datetime " ,
'body' => " string " ,
'enclosureMime' => " string " ,
'enclosureLink' => " string " ,
'feedId' => " int " ,
'unread' => " bool " ,
'starred' => " bool " ,
'lastModified' => " datetime " ,
'fingerprint' => " string " ,
2017-07-07 15:25:47 -04:00
], $this -> dateFormat );
2017-07-07 08:13:03 -04:00
return $article ;
}
2017-11-29 15:28:33 -05:00
protected function handleHTTPOptions ( array $url ) : Response {
// normalize the URL path
$url = $this -> normalizePath ( $url );
if ( isset ( $this -> paths [ $url ])) {
// if the path is supported, respond with the allowed methods and other metadata
$allowed = array_keys ( $this -> paths [ $url ]);
// if GET is allowed, so is HEAD
if ( in_array ( " GET " , $allowed )) {
array_unshift ( $allowed , " HEAD " );
}
return new Response ( 204 , " " , " " , [
" Allow: " . implode ( " , " , $allowed ),
" Accept: application/json " ,
]);
} else {
// if the path is not supported, return 404
2017-11-29 22:42:50 -05:00
return new Response ( 404 );
2017-11-29 15:28:33 -05:00
}
}
2017-05-18 23:03:33 -04:00
2017-04-02 21:34:30 -04:00
// list folders
2017-05-19 23:52:26 -04:00
protected function folderList ( array $url , array $data ) : Response {
2017-11-30 14:47:39 -05:00
$folders = [];
foreach ( Arsse :: $db -> folderList ( Arsse :: $user -> id , null , false ) as $folder ) {
$folders [] = $this -> folderTranslate ( $folder );
}
2017-04-02 21:34:30 -04:00
return new Response ( 200 , [ 'folders' => $folders ]);
}
2017-04-01 15:42:10 -04:00
2017-04-02 21:34:30 -04:00
// create a folder
2017-05-19 23:52:26 -04:00
protected function folderAdd ( array $url , array $data ) : Response {
2017-04-02 21:34:30 -04:00
try {
2017-10-19 20:35:45 -04:00
$folder = Arsse :: $db -> folderAdd ( Arsse :: $user -> id , [ 'name' => $data [ 'name' ]]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
switch ( $e -> getCode ()) {
2017-04-02 21:34:30 -04:00
// folder already exists
case 10236 : return new Response ( 409 );
// folder name not acceptable
case 10231 :
case 10232 : return new Response ( 422 );
// other errors related to input
2017-07-24 08:15:37 -04:00
default : return new Response ( 400 ); // @codeCoverageIgnore
2017-04-02 21:34:30 -04:00
}
}
2017-11-30 14:47:39 -05:00
$folder = $this -> folderTranslate ( Arsse :: $db -> folderPropertiesGet ( Arsse :: $user -> id , $folder ));
2017-04-02 21:34:30 -04:00
return new Response ( 200 , [ 'folders' => [ $folder ]]);
}
2017-04-01 23:06:52 -04:00
2017-04-02 21:34:30 -04:00
// delete a folder
2017-05-19 23:52:26 -04:00
protected function folderRemove ( array $url , array $data ) : Response {
2017-04-02 21:34:30 -04:00
// perform the deletion
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> folderRemove ( Arsse :: $user -> id , ( int ) $url [ 1 ]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-04-02 21:34:30 -04:00
// folder does not exist
return new Response ( 404 );
}
return new Response ( 204 );
}
2017-04-02 12:14:15 -04:00
2017-04-02 21:34:30 -04:00
// rename a folder (also supports moving nesting folders, but this is not a feature of the API)
2017-05-19 23:52:26 -04:00
protected function folderRename ( array $url , array $data ) : Response {
2017-04-02 21:34:30 -04:00
try {
2017-10-19 20:35:45 -04:00
Arsse :: $db -> folderPropertiesSet ( Arsse :: $user -> id , ( int ) $url [ 1 ], [ 'name' => $data [ 'name' ]]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
switch ( $e -> getCode ()) {
2017-04-02 21:34:30 -04:00
// folder does not exist
2017-05-21 10:10:36 -04:00
case 10239 : return new Response ( 404 );
2017-04-02 21:34:30 -04:00
// folder already exists
case 10236 : return new Response ( 409 );
// folder name not acceptable
case 10231 :
case 10232 : return new Response ( 422 );
// other errors related to input
2017-07-24 08:15:37 -04:00
default : return new Response ( 400 ); // @codeCoverageIgnore
2017-04-02 21:34:30 -04:00
}
}
return new Response ( 204 );
}
2017-04-02 21:49:37 -04:00
2017-07-09 17:57:18 -04:00
// mark all articles associated with a folder as read
2017-07-07 08:13:03 -04:00
protected function folderMarkRead ( array $url , array $data ) : Response {
2017-10-19 20:35:45 -04:00
if ( ! ValueInfo :: id ( $data [ 'newestItemId' ])) {
// if the item ID is invalid (i.e. not a positive integer), this is an error
2017-07-07 08:13:03 -04:00
return new Response ( 422 );
}
2017-10-19 20:35:45 -04:00
// build the context
$c = new Context ;
$c -> latestEdition (( int ) $data [ 'newestItemId' ]);
2017-07-07 08:13:03 -04:00
$c -> folder (( int ) $url [ 1 ]);
// perform the operation
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'read' => true ], $c );
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
// folder does not exist
return new Response ( 404 );
}
return new Response ( 204 );
}
2017-05-19 23:52:26 -04:00
2017-05-18 23:03:33 -04:00
// return list of feeds which should be refreshed
2017-05-19 23:52:26 -04:00
protected function feedListStale ( array $url , array $data ) : Response {
// function requires admin rights per spec
2017-08-29 10:50:31 -04:00
if ( Arsse :: $user -> rightsGet ( Arsse :: $user -> id ) == User :: RIGHTS_NONE ) {
2017-07-20 22:40:09 -04:00
return new Response ( 403 );
}
2017-05-19 23:52:26 -04:00
// list stale feeds which should be checked for updates
2017-07-17 07:47:57 -04:00
$feeds = Arsse :: $db -> feedListStale ();
2017-05-19 23:52:26 -04:00
$out = [];
2017-08-29 10:50:31 -04:00
foreach ( $feeds as $feed ) {
2017-05-19 23:52:26 -04:00
// since in our implementation feeds don't belong the users, the 'userId' field will always be an empty string
$out [] = [ 'id' => $feed , 'userId' => " " ];
2017-05-18 23:03:33 -04:00
}
2017-05-19 23:52:26 -04:00
return new Response ( 200 , [ 'feeds' => $out ]);
}
// refresh a feed
protected function feedUpdate ( array $url , array $data ) : Response {
2017-06-03 17:34:37 -04:00
// function requires admin rights per spec
2017-08-29 10:50:31 -04:00
if ( Arsse :: $user -> rightsGet ( Arsse :: $user -> id ) == User :: RIGHTS_NONE ) {
2017-07-20 22:40:09 -04:00
return new Response ( 403 );
}
2017-05-19 23:52:26 -04:00
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> feedUpdate ( $data [ 'feedId' ]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-09-27 22:25:45 -04:00
switch ( $e -> getCode ()) {
case 10239 : // feed does not exist
return new Response ( 404 );
2017-09-28 09:01:43 -04:00
case 10237 : // feed ID invalid
2017-09-27 22:25:45 -04:00
return new Response ( 422 );
default : // other errors related to input
return new Response ( 400 ); // @codeCoverageIgnore
}
2017-05-19 23:52:26 -04:00
}
2017-06-03 17:34:37 -04:00
return new Response ( 204 );
2017-05-18 23:03:33 -04:00
}
// add a new feed
2017-05-19 23:52:26 -04:00
protected function subscriptionAdd ( array $url , array $data ) : Response {
2017-05-18 23:03:33 -04:00
// try to add the feed
2017-07-17 07:47:57 -04:00
$tr = Arsse :: $db -> begin ();
2017-05-18 23:03:33 -04:00
try {
2017-10-19 20:35:45 -04:00
$id = Arsse :: $db -> subscriptionAdd ( Arsse :: $user -> id , ( string ) $data [ 'url' ]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-05-18 23:03:33 -04:00
// feed already exists
return new Response ( 409 );
2017-08-29 10:50:31 -04:00
} catch ( FeedException $e ) {
2017-05-18 23:03:33 -04:00
// feed could not be retrieved
return new Response ( 422 );
}
// if a folder was specified, move the feed to the correct folder; silently ignore errors
2017-10-19 20:35:45 -04:00
if ( $data [ 'folderId' ]) {
2017-05-18 23:03:33 -04:00
try {
2017-10-19 20:35:45 -04:00
Arsse :: $db -> subscriptionPropertiesSet ( Arsse :: $user -> id , $id , [ 'folder' => $data [ 'folderId' ]]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
}
2017-05-18 23:03:33 -04:00
}
$tr -> commit ();
// fetch the feed's metadata and format it appropriately
2017-07-17 07:47:57 -04:00
$feed = Arsse :: $db -> subscriptionPropertiesGet ( Arsse :: $user -> id , $id );
2017-05-18 23:03:33 -04:00
$feed = $this -> feedTranslate ( $feed );
$out = [ 'feeds' => [ $feed ]];
2017-07-17 07:47:57 -04:00
$newest = Arsse :: $db -> editionLatest ( Arsse :: $user -> id , ( new Context ) -> subscription ( $id ));
2017-08-29 10:50:31 -04:00
if ( $newest ) {
2017-07-20 22:40:09 -04:00
$out [ 'newestItemId' ] = $newest ;
}
2017-05-18 23:03:33 -04:00
return new Response ( 200 , $out );
}
2017-07-07 08:13:03 -04:00
// return list of feeds for the logged-in user
protected function subscriptionList ( array $url , array $data ) : Response {
2017-07-17 07:47:57 -04:00
$subs = Arsse :: $db -> subscriptionList ( Arsse :: $user -> id );
2017-07-07 08:13:03 -04:00
$out = [];
2017-08-29 10:50:31 -04:00
foreach ( $subs as $sub ) {
2017-07-07 08:13:03 -04:00
$out [] = $this -> feedTranslate ( $sub );
}
$out = [ 'feeds' => $out ];
2017-10-11 12:55:50 -04:00
$out [ 'starredCount' ] = Arsse :: $db -> articleStarred ( Arsse :: $user -> id )[ 'total' ];
2017-07-17 07:47:57 -04:00
$newest = Arsse :: $db -> editionLatest ( Arsse :: $user -> id );
2017-08-29 10:50:31 -04:00
if ( $newest ) {
2017-07-20 22:40:09 -04:00
$out [ 'newestItemId' ] = $newest ;
}
2017-07-07 08:13:03 -04:00
return new Response ( 200 , $out );
}
2017-05-18 23:03:33 -04:00
// delete a feed
2017-05-19 23:52:26 -04:00
protected function subscriptionRemove ( array $url , array $data ) : Response {
2017-05-18 23:03:33 -04:00
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> subscriptionRemove ( Arsse :: $user -> id , ( int ) $url [ 1 ]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-05-18 23:03:33 -04:00
// feed does not exist
return new Response ( 404 );
}
return new Response ( 204 );
}
// rename a feed
2017-05-19 23:52:26 -04:00
protected function subscriptionRename ( array $url , array $data ) : Response {
try {
2017-10-19 20:35:45 -04:00
Arsse :: $db -> subscriptionPropertiesSet ( Arsse :: $user -> id , ( int ) $url [ 1 ], [ 'title' => ( string ) $data [ 'feedTitle' ]]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
switch ( $e -> getCode ()) {
2017-05-21 10:10:36 -04:00
// subscription does not exist
case 10239 : return new Response ( 404 );
// name is invalid
case 10231 :
case 10232 : return new Response ( 422 );
// other errors related to input
2017-07-24 08:15:37 -04:00
default : return new Response ( 400 ); // @codeCoverageIgnore
2017-05-21 10:10:36 -04:00
}
2017-05-18 23:03:33 -04:00
}
2017-05-19 23:52:26 -04:00
return new Response ( 204 );
}
// move a feed to a folder
protected function subscriptionMove ( array $url , array $data ) : Response {
2017-10-19 20:35:45 -04:00
// if no folder is specified this is an error
if ( ! isset ( $data [ 'folderId' ])) {
2017-05-19 23:52:26 -04:00
return new Response ( 422 );
2017-05-18 23:03:33 -04:00
}
2017-05-19 23:52:26 -04:00
// perform the move
try {
2017-10-19 20:35:45 -04:00
Arsse :: $db -> subscriptionPropertiesSet ( Arsse :: $user -> id , ( int ) $url [ 1 ], [ 'folder' => $data [ 'folderId' ]]);
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
switch ( $e -> getCode ()) {
2017-09-27 22:25:45 -04:00
case 10239 : // subscription does not exist
return new Response ( 404 );
case 10235 : // folder does not exist
case 10237 : // folder ID is invalid
return new Response ( 422 );
default : // other errors related to input
return new Response ( 400 ); // @codeCoverageIgnore
2017-05-21 10:10:36 -04:00
}
2017-05-18 23:03:33 -04:00
}
return new Response ( 204 );
}
2017-07-07 08:13:03 -04:00
2017-07-09 17:57:18 -04:00
// mark all articles associated with a subscription as read
2017-07-07 08:13:03 -04:00
protected function subscriptionMarkRead ( array $url , array $data ) : Response {
2017-10-19 20:35:45 -04:00
if ( ! ValueInfo :: id ( $data [ 'newestItemId' ])) {
// if the item ID is invalid (i.e. not a positive integer), this is an error
2017-07-07 08:13:03 -04:00
return new Response ( 422 );
}
2017-10-19 20:35:45 -04:00
// build the context
$c = new Context ;
$c -> latestEdition (( int ) $data [ 'newestItemId' ]);
2017-07-07 08:13:03 -04:00
$c -> subscription (( int ) $url [ 1 ]);
// perform the operation
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'read' => true ], $c );
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
// subscription does not exist
return new Response ( 404 );
}
return new Response ( 204 );
}
2017-07-09 17:57:18 -04:00
// list articles and their properties
2017-07-07 08:13:03 -04:00
protected function articleList ( array $url , array $data ) : Response {
// set the context options supplied by the client
$c = new Context ;
// set the batch size
2017-10-19 20:35:45 -04:00
if ( $data [ 'batchSize' ] > 0 ) {
2017-07-20 22:40:09 -04:00
$c -> limit ( $data [ 'batchSize' ]);
}
2017-07-07 08:13:03 -04:00
// set the order of returned items
2017-10-19 20:35:45 -04:00
if ( $data [ 'oldestFirst' ]) {
2017-07-07 08:13:03 -04:00
$c -> reverse ( false );
} else {
$c -> reverse ( true );
}
// set the edition mark-off; the database uses an or-equal comparison for internal consistency, but the protocol does not, so we must adjust by one
2017-10-19 20:35:45 -04:00
if ( $data [ 'offset' ] > 0 ) {
2017-08-29 10:50:31 -04:00
if ( $c -> reverse ) {
2017-07-07 08:13:03 -04:00
$c -> latestEdition ( $data [ 'offset' ] - 1 );
} else {
$c -> oldestEdition ( $data [ 'offset' ] + 1 );
}
}
// set whether to only return unread
2017-10-19 20:35:45 -04:00
if ( ! ValueInfo :: bool ( $data [ 'getRead' ], true )) {
2017-07-20 22:40:09 -04:00
$c -> unread ( true );
}
2017-07-07 08:13:03 -04:00
// if no type is specified assume 3 (All)
2017-10-19 20:35:45 -04:00
$data [ 'type' ] = $data [ 'type' ] ? ? 3 ;
2017-08-29 10:50:31 -04:00
switch ( $data [ 'type' ]) {
2017-07-07 08:13:03 -04:00
case 0 : // feed
2017-08-29 10:50:31 -04:00
if ( isset ( $data [ 'id' ])) {
2017-07-20 22:40:09 -04:00
$c -> subscription ( $data [ 'id' ]);
}
2017-07-07 08:13:03 -04:00
break ;
case 1 : // folder
2017-08-29 10:50:31 -04:00
if ( isset ( $data [ 'id' ])) {
2017-07-20 22:40:09 -04:00
$c -> folder ( $data [ 'id' ]);
}
2017-07-07 08:13:03 -04:00
break ;
case 2 : // starred
$c -> starred ( true );
break ;
2017-07-24 08:15:37 -04:00
default : // @codeCoverageIgnore
2017-07-07 08:13:03 -04:00
// return all items
}
// whether to return only updated items
2017-10-19 20:35:45 -04:00
if ( $data [ 'lastModified' ]) {
2017-11-17 17:52:00 -05:00
$c -> markedSince ( $data [ 'lastModified' ]);
2017-07-20 22:40:09 -04:00
}
2017-07-07 08:13:03 -04:00
// perform the fetch
try {
2017-11-17 18:12:00 -05:00
$items = Arsse :: $db -> articleList ( Arsse :: $user -> id , $c , Database :: LIST_TYPICAL );
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
// ID of subscription or folder is not valid
return new Response ( 422 );
}
$out = [];
2017-08-29 10:50:31 -04:00
foreach ( $items as $item ) {
2017-07-07 08:13:03 -04:00
$out [] = $this -> articleTranslate ( $item );
}
$out = [ 'items' => $out ];
return new Response ( 200 , $out );
}
2017-07-09 17:57:18 -04:00
// mark all articles as read
2017-07-07 08:13:03 -04:00
protected function articleMarkReadAll ( array $url , array $data ) : Response {
2017-10-19 20:35:45 -04:00
if ( ! ValueInfo :: id ( $data [ 'newestItemId' ])) {
// if the item ID is invalid (i.e. not a positive integer), this is an error
2017-07-07 08:13:03 -04:00
return new Response ( 422 );
}
2017-10-19 20:35:45 -04:00
// build the context
$c = new Context ;
$c -> latestEdition (( int ) $data [ 'newestItemId' ]);
2017-07-07 08:13:03 -04:00
// perform the operation
2017-07-17 07:47:57 -04:00
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'read' => true ], $c );
2017-07-07 08:13:03 -04:00
return new Response ( 204 );
}
2017-07-09 17:57:18 -04:00
// mark a single article as read
2017-07-07 08:13:03 -04:00
protected function articleMarkRead ( array $url , array $data ) : Response {
// initialize the matching context
$c = new Context ;
$c -> edition (( int ) $url [ 1 ]);
// determine whether to mark read or unread
$set = ( $url [ 2 ] == " read " );
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'read' => $set ], $c );
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
// ID is not valid
return new Response ( 404 );
}
return new Response ( 204 );
}
2017-07-09 17:57:18 -04:00
// mark a single article as read
2017-07-07 08:13:03 -04:00
protected function articleMarkStarred ( array $url , array $data ) : Response {
// initialize the matching context
$c = new Context ;
$c -> article (( int ) $url [ 2 ]);
// determine whether to mark read or unread
$set = ( $url [ 3 ] == " star " );
try {
2017-07-17 07:47:57 -04:00
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'starred' => $set ], $c );
2017-08-29 10:50:31 -04:00
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
// ID is not valid
return new Response ( 404 );
}
return new Response ( 204 );
}
2017-07-09 17:57:18 -04:00
// mark an array of articles as read
2017-07-07 08:13:03 -04:00
protected function articleMarkReadMulti ( array $url , array $data ) : Response {
// determine whether to mark read or unread
$set = ( $url [ 1 ] == " read " );
2017-11-07 10:00:31 -05:00
// initialize the matching context
$c = new Context ;
$c -> editions ( $data [ 'items' ] ? ? []);
try {
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'read' => $set ], $c );
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
}
return new Response ( 204 );
}
2017-07-09 17:57:18 -04:00
// mark an array of articles as starred
2017-07-07 08:13:03 -04:00
protected function articleMarkStarredMulti ( array $url , array $data ) : Response {
2017-07-09 17:57:18 -04:00
// determine whether to mark starred or unstarred
2017-07-07 08:13:03 -04:00
$set = ( $url [ 1 ] == " star " );
2017-11-07 10:00:31 -05:00
// initialize the matching context
$c = new Context ;
$c -> articles ( array_column ( $data [ 'items' ] ? ? [], " guidHash " ));
try {
Arsse :: $db -> articleMark ( Arsse :: $user -> id , [ 'starred' => $set ], $c );
} catch ( ExceptionInput $e ) {
2017-07-07 08:13:03 -04:00
}
return new Response ( 204 );
}
2017-07-15 16:44:06 -04:00
protected function userStatus ( array $url , array $data ) : Response {
2017-08-19 23:56:32 -04:00
$data = Arsse :: $user -> propertiesGet ( Arsse :: $user -> id , true );
2017-07-19 18:07:36 -04:00
// construct the avatar structure, if an image is available
2017-08-29 10:50:31 -04:00
if ( isset ( $data [ 'avatar' ])) {
2017-07-19 18:07:36 -04:00
$avatar = [
'data' => base64_encode ( $data [ 'avatar' ][ 'data' ]),
2017-11-30 14:47:39 -05:00
'mime' => ( string ) $data [ 'avatar' ][ 'type' ],
2017-07-19 18:07:36 -04:00
];
} else {
$avatar = null ;
}
// construct the rest of the structure
2017-07-15 16:44:06 -04:00
$out = [
2017-11-30 14:47:39 -05:00
'userId' => ( string ) Arsse :: $user -> id ,
'displayName' => ( string ) ( $data [ 'name' ] ? ? Arsse :: $user -> id ),
2017-07-15 16:44:06 -04:00
'lastLoginTimestamp' => time (),
2017-07-19 18:07:36 -04:00
'avatar' => $avatar ,
2017-07-15 16:44:06 -04:00
];
return new Response ( 200 , $out );
}
protected function cleanupBefore ( array $url , array $data ) : Response {
// function requires admin rights per spec
2017-08-29 10:50:31 -04:00
if ( Arsse :: $user -> rightsGet ( Arsse :: $user -> id ) == User :: RIGHTS_NONE ) {
2017-07-20 22:40:09 -04:00
return new Response ( 403 );
}
2017-08-02 18:27:04 -04:00
Service :: cleanupPre ();
2017-07-15 16:44:06 -04:00
return new Response ( 204 );
}
protected function cleanupAfter ( array $url , array $data ) : Response {
// function requires admin rights per spec
2017-08-29 10:50:31 -04:00
if ( Arsse :: $user -> rightsGet ( Arsse :: $user -> id ) == User :: RIGHTS_NONE ) {
2017-07-20 22:40:09 -04:00
return new Response ( 403 );
}
2017-08-17 22:36:15 -04:00
Service :: cleanupPost ();
2017-07-15 16:44:06 -04:00
return new Response ( 204 );
}
// return the server version
protected function serverVersion ( array $url , array $data ) : Response {
return new Response ( 200 , [
'version' => self :: VERSION ,
2017-10-01 09:33:49 -04:00
'arsse_version' => Arsse :: VERSION ,
2017-07-15 16:44:06 -04:00
]);
}
protected function serverStatus ( array $url , array $data ) : Response {
return new Response ( 200 , [
'version' => self :: VERSION ,
2017-10-01 09:33:49 -04:00
'arsse_version' => Arsse :: VERSION ,
2017-07-15 16:44:06 -04:00
'warnings' => [
2017-08-02 18:27:04 -04:00
'improperlyConfiguredCron' => ! Service :: hasCheckedIn (),
2017-11-29 18:14:59 -05:00
'incorrectDbCharset' => ! Arsse :: $db -> driverCharsetAcceptable (),
2017-07-15 16:44:06 -04:00
]
]);
}
2017-08-29 10:50:31 -04:00
}