2016-09-25 16:50:08 +00:00
< ? php
/**
* Friendica implementation of statusnet / twitter API
*
2017-12-24 02:20:50 +00:00
* @ file include / api . php
2016-09-25 16:50:08 +00:00
* @ todo Automatically detect if incoming data is HTML or BBCode
*/
2018-01-25 02:08:45 +00:00
2017-04-30 04:07:00 +00:00
use Friendica\App ;
2018-01-10 03:42:04 +00:00
use Friendica\Content\ContactSelector ;
2017-12-04 14:04:36 +00:00
use Friendica\Content\Feature ;
2018-01-27 01:01:32 +00:00
use Friendica\Content\Text\BBCode ;
2018-03-07 21:24:13 +00:00
use Friendica\Content\Text\HTML ;
2018-01-17 18:42:40 +00:00
use Friendica\Core\Addon ;
2018-10-17 12:19:58 +00:00
use Friendica\Core\Authentication ;
2017-04-30 04:01:26 +00:00
use Friendica\Core\Config ;
2018-01-21 16:38:01 +00:00
use Friendica\Core\L10n ;
2018-10-29 21:20:46 +00:00
use Friendica\Core\Logger ;
2018-01-25 02:08:45 +00:00
use Friendica\Core\NotificationsManager ;
2017-12-18 19:39:35 +00:00
use Friendica\Core\PConfig ;
2018-07-01 18:42:38 +00:00
use Friendica\Core\Protocol ;
2018-01-25 02:08:45 +00:00
use Friendica\Core\System ;
2017-11-05 12:15:53 +00:00
use Friendica\Core\Worker ;
2018-07-20 12:19:26 +00:00
use Friendica\Database\DBA ;
2017-12-07 14:04:24 +00:00
use Friendica\Model\Contact ;
2017-12-09 18:45:17 +00:00
use Friendica\Model\Group ;
2018-01-25 02:08:45 +00:00
use Friendica\Model\Item ;
2018-01-15 17:14:09 +00:00
use Friendica\Model\Mail ;
2017-12-07 13:56:11 +00:00
use Friendica\Model\Photo ;
2017-11-26 19:46:08 +00:00
use Friendica\Model\User ;
2017-12-04 23:30:18 +00:00
use Friendica\Network\FKOAuth1 ;
2017-11-24 04:48:15 +00:00
use Friendica\Network\HTTPException ;
use Friendica\Network\HTTPException\BadRequestException ;
use Friendica\Network\HTTPException\ForbiddenException ;
use Friendica\Network\HTTPException\InternalServerErrorException ;
use Friendica\Network\HTTPException\MethodNotAllowedException ;
use Friendica\Network\HTTPException\NotFoundException ;
use Friendica\Network\HTTPException\NotImplementedException ;
use Friendica\Network\HTTPException\TooManyRequestsException ;
2018-01-25 02:08:45 +00:00
use Friendica\Network\HTTPException\UnauthorizedException ;
2017-12-07 13:56:11 +00:00
use Friendica\Object\Image ;
2017-11-08 13:34:48 +00:00
use Friendica\Protocol\Diaspora ;
2018-01-27 02:38:34 +00:00
use Friendica\Util\DateTimeFormat ;
2018-01-27 04:09:48 +00:00
use Friendica\Util\Network ;
2018-07-31 02:06:22 +00:00
use Friendica\Util\Proxy as ProxyUtils ;
2018-11-08 13:45:46 +00:00
use Friendica\Util\Strings ;
2017-11-10 12:45:33 +00:00
use Friendica\Util\XML ;
2017-01-17 19:21:46 +00:00
2017-04-05 20:07:55 +00:00
require_once 'mod/share.php' ;
require_once 'mod/item.php' ;
require_once 'mod/wall_upload.php' ;
define ( 'API_METHOD_ANY' , '*' );
define ( 'API_METHOD_GET' , 'GET' );
define ( 'API_METHOD_POST' , 'POST,PUT' );
define ( 'API_METHOD_DELETE' , 'POST,DELETE' );
2018-01-15 13:05:12 +00:00
$API = [];
2018-04-24 14:16:57 +00:00
$called_api = [];
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
/**
* It is not sufficient to use local_user () to check whether someone is allowed to use the API ,
* because this will open CSRF holes ( just embed an image with src = friendicasite . com / api / statuses / update ? status = CSRF
* into a page , and visitors will post something without noticing it ) .
2017-12-24 02:20:50 +00:00
*
* @ brief Auth API user
2017-11-10 05:00:50 +00:00
*/
function api_user ()
{
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_SESSION [ 'allow_api' ])) {
2017-11-10 05:00:50 +00:00
return local_user ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return false ;
}
/**
* Clients can send 'source' parameter to be show in post metadata
* as " sent via <source> " .
* Some clients doesn ' t send a source param , we support ones we know
* ( only Twidere , atm )
*
2017-12-24 02:20:50 +00:00
* @ brief Get source name from API client
*
2017-11-10 05:00:50 +00:00
* @ return string
* Client source name , default to " api " if unset / unknown
*/
function api_source ()
{
if ( requestdata ( 'source' )) {
return requestdata ( 'source' );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// Support for known clients that doesn't send a source name
2018-07-01 04:15:11 +00:00
if ( ! empty ( $_SERVER [ 'HTTP_USER_AGENT' ])) {
if ( strpos ( $_SERVER [ 'HTTP_USER_AGENT' ], " Twidere " ) !== false ) {
return " Twidere " ;
}
2016-09-25 16:50:08 +00:00
2018-10-30 13:58:45 +00:00
Logger :: log ( " Unrecognized user-agent " . $_SERVER [ 'HTTP_USER_AGENT' ], Logger :: DEBUG );
2018-07-04 21:33:09 +00:00
} else {
2018-10-30 13:58:45 +00:00
Logger :: log ( " Empty user-agent " , Logger :: DEBUG );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return " api " ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Format date for API
*
* @ param string $str Source date , as UTC
* @ return string Date in UTC formatted as " D M d H:i:s +0000 Y "
*/
function api_date ( $str )
{
// Wed May 23 06:01:13 +0000 2007
2018-01-27 02:38:34 +00:00
return DateTimeFormat :: utc ( $str , " D M d H:i:s +0000 Y " );
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Register a function to be the endpoint for defined API path .
2017-11-10 05:00:50 +00:00
*
2017-12-24 02:20:50 +00:00
* @ brief Register API endpoint
2017-11-10 05:00:50 +00:00
*
* @ param string $path API URL path , relative to System :: baseUrl ()
* @ param string $func Function name to call on path request
* @ param bool $auth API need logged user
* @ param string $method HTTP method reqiured to call this endpoint .
* One of API_METHOD_ANY , API_METHOD_GET , API_METHOD_POST .
* Default to API_METHOD_ANY
*/
function api_register_func ( $path , $func , $auth = false , $method = API_METHOD_ANY )
{
global $API ;
2018-01-15 13:05:12 +00:00
$API [ $path ] = [
2017-11-10 05:00:50 +00:00
'func' => $func ,
'auth' => $auth ,
'method' => $method ,
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
// Workaround for hotot
$path = str_replace ( " api/ " , " api/1.1/ " , $path );
2018-01-15 13:05:12 +00:00
$API [ $path ] = [
2017-11-10 05:00:50 +00:00
'func' => $func ,
'auth' => $auth ,
'method' => $method ,
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* Log in user via OAuth1 or Simple HTTP Auth .
* Simple Auth allow username in form of < pre > user @ server </ pre > , ignoring server part
*
2017-12-24 02:20:50 +00:00
* @ brief Login API user
*
2017-11-10 05:00:50 +00:00
* @ param object $a App
* @ hook 'authenticate'
* array $addon_auth
* 'username' => username from login form
* 'password' => password from login form
* 'authenticated' => return status ,
* 'user_record' => return authenticated user record
* @ hook 'logged_in'
* array $user logged user record
*/
function api_login ( App $a )
{
2017-12-05 02:10:54 +00:00
$oauth1 = new FKOAuth1 ();
2017-11-10 05:00:50 +00:00
// login with oauth
try {
2018-07-10 12:27:56 +00:00
$request = OAuthRequest :: from_request ();
list ( $consumer , $token ) = $oauth1 -> verify_request ( $request );
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $token )) {
2017-12-05 02:10:54 +00:00
$oauth1 -> loginUser ( $token -> uid );
2018-01-17 18:42:40 +00:00
Addon :: callHooks ( 'logged_in' , $a -> user );
2017-11-10 05:00:50 +00:00
return ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
echo __FILE__ . __LINE__ . __FUNCTION__ . " <pre> " ;
var_dump ( $consumer , $token );
die ();
} catch ( Exception $e ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( $e );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// workaround for HTTP-auth in CGI mode
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_SERVER [ 'REDIRECT_REMOTE_USER' ])) {
$userpass = base64_decode ( substr ( $_SERVER [ " REDIRECT_REMOTE_USER " ], 6 ));
2017-11-10 05:00:50 +00:00
if ( strlen ( $userpass )) {
list ( $name , $password ) = explode ( ':' , $userpass );
$_SERVER [ 'PHP_AUTH_USER' ] = $name ;
$_SERVER [ 'PHP_AUTH_PW' ] = $password ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( empty ( $_SERVER [ 'PHP_AUTH_USER' ])) {
2018-10-30 13:58:45 +00:00
Logger :: log ( 'API_login: ' . print_r ( $_SERVER , true ), Logger :: DEBUG );
2017-11-10 05:00:50 +00:00
header ( 'WWW-Authenticate: Basic realm="Friendica"' );
throw new UnauthorizedException ( " This API requires login " );
}
2016-09-25 16:50:08 +00:00
2018-07-01 04:15:11 +00:00
$user = defaults ( $_SERVER , 'PHP_AUTH_USER' , '' );
$password = defaults ( $_SERVER , 'PHP_AUTH_PW' , '' );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// allow "user@server" login (but ignore 'server' part)
$at = strstr ( $user , " @ " , true );
if ( $at ) {
$user = $at ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// next code from mod/auth.php. needs better solution
$record = null ;
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$addon_auth = [
2017-11-10 05:00:50 +00:00
'username' => trim ( $user ),
'password' => trim ( $password ),
'authenticated' => 0 ,
'user_record' => null ,
2018-01-15 13:05:12 +00:00
];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/*
2018-01-17 19:22:38 +00:00
* An addon indicates successful login by setting 'authenticated' to non - zero value and returning a user record
* Addons should never set 'authenticated' except to indicate success - as hooks may be chained
* and later addons should not interfere with an earlier one that succeeded .
*/
2018-01-17 18:42:40 +00:00
Addon :: callHooks ( 'authenticate' , $addon_auth );
2017-11-10 05:00:50 +00:00
2017-12-22 23:10:32 +00:00
if ( $addon_auth [ 'authenticated' ] && count ( $addon_auth [ 'user_record' ])) {
2017-11-10 05:00:50 +00:00
$record = $addon_auth [ 'user_record' ];
} else {
2017-11-26 19:46:08 +00:00
$user_id = User :: authenticate ( trim ( $user ), trim ( $password ));
2018-04-09 19:34:53 +00:00
if ( $user_id !== false ) {
2018-07-20 12:19:26 +00:00
$record = DBA :: selectFirst ( 'user' , [], [ 'uid' => $user_id ]);
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $record )) {
2018-10-30 13:58:45 +00:00
Logger :: log ( 'API_login failure: ' . print_r ( $_SERVER , true ), Logger :: DEBUG );
2017-11-10 05:00:50 +00:00
header ( 'WWW-Authenticate: Basic realm="Friendica"' );
//header('HTTP/1.0 401 Unauthorized');
//die('This api requires login');
throw new UnauthorizedException ( " This API requires login " );
}
2016-09-25 16:50:08 +00:00
2018-10-17 19:30:41 +00:00
Authentication :: setAuthenticatedSessionForUser ( $record );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$_SESSION [ " allow_api " ] = true ;
2016-09-25 16:50:08 +00:00
2018-01-17 18:42:40 +00:00
Addon :: callHooks ( 'logged_in' , $a -> user );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* API endpoints can define which HTTP method to accept when called .
* This function check the current HTTP method agains endpoint
* registered method .
*
2017-12-24 02:20:50 +00:00
* @ brief Check HTTP method of called API
*
2017-11-10 05:00:50 +00:00
* @ param string $method Required methods , uppercase , separated by comma
* @ return bool
*/
function api_check_method ( $method )
{
if ( $method == " * " ) {
return true ;
2016-09-25 16:50:08 +00:00
}
2018-07-01 04:15:11 +00:00
return ( stripos ( $method , defaults ( $_SERVER , 'REQUEST_METHOD' , 'GET' )) !== false );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* Authenticate user , call registered API function , set HTTP headers
*
2017-12-24 02:20:50 +00:00
* @ brief Main API entry point
*
2017-11-10 05:00:50 +00:00
* @ param object $a App
2018-04-09 17:34:02 +00:00
* @ return string | array API call result
2017-11-10 05:00:50 +00:00
*/
function api_call ( App $a )
{
global $API , $called_api ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$type = " json " ;
if ( strpos ( $a -> query_string , " .xml " ) > 0 ) {
$type = " xml " ;
}
if ( strpos ( $a -> query_string , " .json " ) > 0 ) {
2017-04-05 20:07:55 +00:00
$type = " json " ;
2017-11-10 05:00:50 +00:00
}
if ( strpos ( $a -> query_string , " .rss " ) > 0 ) {
$type = " rss " ;
}
if ( strpos ( $a -> query_string , " .atom " ) > 0 ) {
$type = " atom " ;
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
try {
foreach ( $API as $p => $info ) {
if ( strpos ( $a -> query_string , $p ) === 0 ) {
if ( ! api_check_method ( $info [ 'method' ])) {
throw new MethodNotAllowedException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$called_api = explode ( " / " , $p );
//unset($_SERVER['PHP_AUTH_USER']);
2017-04-05 20:17:15 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO should be "true ==[=] $info['auth']", if you miss only one = character, you assign a variable (only with ==). Let's make all this even.
2018-07-01 04:15:11 +00:00
if ( ! empty ( $info [ 'auth' ]) && api_user () === false ) {
2017-11-10 05:00:50 +00:00
api_login ( $a );
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'API call for ' . $a -> user [ 'username' ] . ': ' . $a -> query_string );
Logger :: log ( 'API parameters: ' . print_r ( $_REQUEST , true ));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$stamp = microtime ( true );
2018-01-04 01:54:35 +00:00
$return = call_user_func ( $info [ 'func' ], $type );
2017-11-10 05:00:50 +00:00
$duration = ( float ) ( microtime ( true ) - $stamp );
2018-10-30 13:58:45 +00:00
Logger :: log ( " API call duration: " . round ( $duration , 2 ) . " \t " . $a -> query_string , Logger :: DEBUG );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( Config :: get ( " system " , " profiler " )) {
$duration = microtime ( true ) - $a -> performance [ " start " ];
2016-11-12 13:17:28 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO round() really everywhere?
2018-10-29 21:20:46 +00:00
Logger :: log (
2017-11-10 05:00:50 +00:00
parse_url ( $a -> query_string , PHP_URL_PATH ) . " : " . sprintf (
2018-03-01 06:25:39 +00:00
" Database: %s/%s, Cache %s/%s, Network: %s, I/O: %s, Other: %s, Total: %s " ,
2016-11-04 22:45:20 +00:00
round ( $a -> performance [ " database " ] - $a -> performance [ " database_write " ], 3 ),
round ( $a -> performance [ " database_write " ], 3 ),
2018-03-01 06:25:39 +00:00
round ( $a -> performance [ " cache " ], 3 ),
round ( $a -> performance [ " cache_write " ], 3 ),
2016-11-04 22:45:20 +00:00
round ( $a -> performance [ " network " ], 2 ),
round ( $a -> performance [ " file " ], 2 ),
2018-03-01 06:25:39 +00:00
round ( $duration - ( $a -> performance [ " database " ]
+ $a -> performance [ " cache " ] + $a -> performance [ " cache_write " ]
+ $a -> performance [ " network " ] + $a -> performance [ " file " ]), 2 ),
2017-11-10 05:00:50 +00:00
round ( $duration , 2 )
),
2018-10-30 13:58:45 +00:00
Logger :: DEBUG
2017-11-10 05:00:50 +00:00
);
if ( Config :: get ( " rendertime " , " callstack " )) {
$o = " Database Read: \n " ;
foreach ( $a -> callstack [ " database " ] as $func => $time ) {
$time = round ( $time , 3 );
if ( $time > 0 ) {
$o .= $func . " : " . $time . " \n " ;
2016-11-04 22:45:20 +00:00
}
2017-11-10 05:00:50 +00:00
}
$o .= " \n Database Write: \n " ;
foreach ( $a -> callstack [ " database_write " ] as $func => $time ) {
$time = round ( $time , 3 );
if ( $time > 0 ) {
$o .= $func . " : " . $time . " \n " ;
2016-11-04 22:45:20 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-11-04 22:45:20 +00:00
2018-03-01 06:25:39 +00:00
$o = " Cache Read: \n " ;
foreach ( $a -> callstack [ " cache " ] as $func => $time ) {
$time = round ( $time , 3 );
if ( $time > 0 ) {
$o .= $func . " : " . $time . " \n " ;
}
}
$o .= " \n Cache Write: \n " ;
foreach ( $a -> callstack [ " cache_write " ] as $func => $time ) {
$time = round ( $time , 3 );
if ( $time > 0 ) {
$o .= $func . " : " . $time . " \n " ;
}
}
2017-11-10 05:00:50 +00:00
$o .= " \n Network: \n " ;
foreach ( $a -> callstack [ " network " ] as $func => $time ) {
$time = round ( $time , 3 );
if ( $time > 0 ) {
$o .= $func . " : " . $time . " \n " ;
2016-11-04 22:45:20 +00:00
}
}
2018-10-30 13:58:45 +00:00
Logger :: log ( $o , Logger :: DEBUG );
2016-11-04 22:45:20 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-11-04 22:45:20 +00:00
2018-01-04 01:54:35 +00:00
if ( false === $return ) {
2017-11-10 05:00:50 +00:00
/*
* api function returned false withour throw an
* exception . This should not happend , throw a 500
*/
throw new InternalServerErrorException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " xml " :
header ( " Content-Type: text/xml " );
break ;
case " json " :
header ( " Content-Type: application/json " );
2018-04-09 19:34:53 +00:00
$json = json_encode ( end ( $return ));
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_GET [ 'callback' ])) {
2017-12-23 23:27:45 +00:00
$json = $_GET [ 'callback' ] . " ( " . $json . " ) " ;
}
2018-01-04 01:54:35 +00:00
$return = $json ;
2017-11-10 05:00:50 +00:00
break ;
case " rss " :
header ( " Content-Type: application/rss+xml " );
2018-01-04 01:54:35 +00:00
$return = '<?xml version="1.0" encoding="UTF-8"?>' . " \n " . $return ;
2017-11-10 05:00:50 +00:00
break ;
case " atom " :
header ( " Content-Type: application/atom+xml " );
2018-01-04 01:54:35 +00:00
$return = '<?xml version="1.0" encoding="UTF-8"?>' . " \n " . $return ;
2017-11-10 05:00:50 +00:00
break ;
2016-09-25 16:50:08 +00:00
}
2018-01-04 01:54:35 +00:00
return $return ;
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'API call not implemented: ' . $a -> query_string );
2017-11-10 05:00:50 +00:00
throw new NotImplementedException ();
} catch ( HTTPException $e ) {
header ( " HTTP/1.1 { $e -> httpcode } { $e -> httpdesc } " );
return api_error ( $type , $e );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Format API error string
*
* @ param string $type Return type ( xml , json , rss , as )
* @ param object $e HTTPException Error object
2018-04-09 17:34:02 +00:00
* @ return string | array error message formatted as $type
2017-11-10 05:00:50 +00:00
*/
function api_error ( $type , $e )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
$error = ( $e -> getMessage () !== " " ? $e -> getMessage () : $e -> httpdesc );
/// @TODO: https://dev.twitter.com/overview/api/response-codes
2018-01-15 13:05:12 +00:00
$error = [ " error " => $error ,
2017-11-10 05:00:50 +00:00
" code " => $e -> httpcode . " " . $e -> httpdesc ,
2018-01-15 13:05:12 +00:00
" request " => $a -> query_string ];
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$return = api_format_data ( 'status' , $type , [ 'status' => $error ]);
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " xml " :
header ( " Content-Type: text/xml " );
break ;
case " json " :
header ( " Content-Type: application/json " );
2018-01-04 01:54:35 +00:00
$return = json_encode ( $return );
2017-11-10 05:00:50 +00:00
break ;
case " rss " :
header ( " Content-Type: application/rss+xml " );
break ;
case " atom " :
header ( " Content-Type: application/atom+xml " );
break ;
}
2018-01-04 01:54:35 +00:00
return $return ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Set values for RSS template
*
* @ param App $a
* @ param array $arr Array to be passed to template
* @ param array $user_info User info
* @ return array
* @ todo find proper type - hints
*/
function api_rss_extra ( App $a , $arr , $user_info )
{
if ( is_null ( $user_info )) {
$user_info = api_get_user ( $a );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$arr [ '$user' ] = $user_info ;
2018-01-15 13:05:12 +00:00
$arr [ '$rss' ] = [
2017-11-10 05:00:50 +00:00
'alternate' => $user_info [ 'url' ],
'self' => System :: baseUrl () . " / " . $a -> query_string ,
'base' => System :: baseUrl (),
'updated' => api_date ( null ),
2018-01-27 02:38:34 +00:00
'atom_updated' => DateTimeFormat :: utcNow ( DateTimeFormat :: ATOM ),
2018-07-04 21:33:09 +00:00
'language' => $user_info [ 'lang' ],
2017-11-10 05:00:50 +00:00
'logo' => System :: baseUrl () . " /images/friendica-32.png " ,
2018-01-15 13:05:12 +00:00
];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $arr ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Unique contact to contact url .
*
* @ param int $id Contact id
* @ return bool | string
* Contact url or False if contact id is unknown
*/
2017-12-17 11:11:28 +00:00
function api_unique_id_to_nurl ( $id )
2017-11-10 05:00:50 +00:00
{
2018-07-20 12:19:26 +00:00
$r = DBA :: selectFirst ( 'contact' , [ 'nurl' ], [ 'id' => $id ]);
2017-11-10 05:00:50 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2017-12-17 11:11:28 +00:00
return $r [ " nurl " ];
2017-11-10 05:00:50 +00:00
} else {
return false ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Get user info array .
*
* @ param object $a App
* @ param int | string $contact_id Contact ID or URL
*/
2017-12-16 15:16:25 +00:00
function api_get_user ( App $a , $contact_id = null )
2017-11-10 05:00:50 +00:00
{
global $called_api ;
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
$user = null ;
$extra_query = " " ;
$url = " " ;
2018-10-30 13:58:45 +00:00
Logger :: log ( " api_get_user: Fetching user data for user " . $contact_id , Logger :: DEBUG );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Searching for contact URL
if ( ! is_null ( $contact_id ) && ( intval ( $contact_id ) == 0 )) {
2018-11-08 16:28:29 +00:00
$user = DBA :: escape ( Strings :: normaliseLink ( $contact_id ));
2017-11-10 05:00:50 +00:00
$url = $user ;
$extra_query = " AND `contact`.`nurl` = '%s' " ;
if ( api_user () !== false ) {
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// Searching for contact id with uid = 0
if ( ! is_null ( $contact_id ) && ( intval ( $contact_id ) != 0 )) {
2018-07-21 13:10:13 +00:00
$user = DBA :: escape ( api_unique_id_to_nurl ( intval ( $contact_id )));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $user == " " ) {
2018-06-03 09:55:41 +00:00
throw new BadRequestException ( " User ID " . $contact_id . " not found. " );
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
$url = $user ;
$extra_query = " AND `contact`.`nurl` = '%s' " ;
if ( api_user () !== false ) {
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
2017-05-15 20:11:33 +00:00
}
2016-09-25 16:50:08 +00:00
}
2018-11-30 14:06:22 +00:00
if ( is_null ( $user ) && ! empty ( $_GET [ 'user_id' ])) {
2018-07-21 13:10:13 +00:00
$user = DBA :: escape ( api_unique_id_to_nurl ( $_GET [ 'user_id' ]));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $user == " " ) {
2018-06-03 09:55:41 +00:00
throw new BadRequestException ( " User ID " . $_GET [ 'user_id' ] . " not found. " );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$url = $user ;
$extra_query = " AND `contact`.`nurl` = '%s' " ;
if ( api_user () !== false ) {
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
}
}
2018-11-30 14:06:22 +00:00
if ( is_null ( $user ) && ! empty ( $_GET [ 'screen_name' ])) {
2018-07-21 13:10:13 +00:00
$user = DBA :: escape ( $_GET [ 'screen_name' ]);
2017-11-10 05:00:50 +00:00
$extra_query = " AND `contact`.`nick` = '%s' " ;
if ( api_user () !== false ) {
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
}
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( is_null ( $user ) && ! empty ( $_GET [ 'profileurl' ])) {
2018-11-08 16:28:29 +00:00
$user = DBA :: escape ( Strings :: normaliseLink ( $_GET [ 'profileurl' ]));
2017-11-10 05:00:50 +00:00
$extra_query = " AND `contact`.`nurl` = '%s' " ;
if ( api_user () !== false ) {
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-07-01 04:16:32 +00:00
// $called_api is the API path exploded on / and is expected to have at least 2 elements
2017-11-10 05:00:50 +00:00
if ( is_null ( $user ) && ( $a -> argc > ( count ( $called_api ) - 1 )) && ( count ( $called_api ) > 0 )) {
$argid = count ( $called_api );
2018-07-08 09:37:05 +00:00
if ( ! empty ( $a -> argv [ $argid ])) {
2018-09-02 08:01:13 +00:00
$data = explode ( " . " , $a -> argv [ $argid ]);
if ( count ( $data ) > 1 ) {
list ( $user , $null ) = $data ;
}
2018-07-08 09:37:05 +00:00
}
2017-11-10 05:00:50 +00:00
if ( is_numeric ( $user )) {
2018-07-21 13:10:13 +00:00
$user = DBA :: escape ( api_unique_id_to_nurl ( intval ( $user )));
2016-09-25 16:50:08 +00:00
2018-04-10 08:46:10 +00:00
if ( $user != " " ) {
$url = $user ;
$extra_query = " AND `contact`.`nurl` = '%s' " ;
if ( api_user () !== false ) {
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
}
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
} else {
2018-07-21 13:10:13 +00:00
$user = DBA :: escape ( $user );
2016-09-25 16:50:08 +00:00
$extra_query = " AND `contact`.`nick` = '%s' " ;
2017-04-05 20:07:55 +00:00
if ( api_user () !== false ) {
2017-11-10 05:00:50 +00:00
$extra_query .= " AND `contact`.`uid`= " . intval ( api_user ());
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-10-30 13:58:45 +00:00
Logger :: log ( " api_get_user: user " . $user , Logger :: DEBUG );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $user ) {
if ( api_user () === false ) {
api_login ( $a );
return false ;
} else {
$user = $_SESSION [ 'uid' ];
$extra_query = " AND `contact`.`uid` = %d AND `contact`.`self` " ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'api_user: ' . $extra_query . ', user: ' . $user );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// user info
$uinfo = q (
" SELECT *, `contact`.`id` AS `cid` FROM `contact`
WHERE 1
$extra_query " ,
$user
);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Selecting the id by priority, friendica first
2018-04-09 19:34:53 +00:00
if ( is_array ( $uinfo )) {
api_best_nickname ( $uinfo );
}
2016-11-04 22:45:20 +00:00
2017-11-10 05:00:50 +00:00
// if the contact wasn't found, fetch it from the contacts with uid = 0
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $uinfo )) {
2018-08-19 12:46:11 +00:00
if ( $url == " " ) {
throw new BadRequestException ( " User not found. " );
2016-09-25 16:50:08 +00:00
}
2018-11-08 16:28:29 +00:00
$contact = DBA :: selectFirst ( 'contact' , [], [ 'uid' => 0 , 'nurl' => Strings :: normaliseLink ( $url )]);
2018-08-19 12:46:11 +00:00
if ( DBA :: isResult ( $contact )) {
2017-11-10 05:00:50 +00:00
// If no nick where given, extract it from the address
2018-08-19 12:46:11 +00:00
if (( $contact [ 'nick' ] == " " ) || ( $contact [ 'name' ] == $contact [ 'nick' ])) {
$contact [ 'nick' ] = api_get_nick ( $contact [ " url " ]);
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$ret = [
2018-08-19 12:46:11 +00:00
'id' => $contact [ " id " ],
'id_str' => ( string ) $contact [ " id " ],
'name' => $contact [ " name " ],
'screen_name' => (( $contact [ 'nick' ]) ? $contact [ 'nick' ] : $contact [ 'name' ]),
2018-12-26 16:02:28 +00:00
'location' => ( $contact [ " location " ] != " " ) ? $contact [ " location " ] : ContactSelector :: networkToName ( $contact [ 'network' ], $contact [ 'url' ]),
2018-08-19 12:46:11 +00:00
'description' => $contact [ " about " ],
'profile_image_url' => $contact [ " micro " ],
'profile_image_url_https' => $contact [ " micro " ],
'profile_image_url_profile_size' => $contact [ " thumb " ],
'profile_image_url_large' => $contact [ " photo " ],
'url' => $contact [ " url " ],
2017-11-10 05:00:50 +00:00
'protected' => false ,
'followers_count' => 0 ,
'friends_count' => 0 ,
'listed_count' => 0 ,
2018-08-19 12:46:11 +00:00
'created_at' => api_date ( $contact [ " created " ]),
2017-11-10 05:00:50 +00:00
'favourites_count' => 0 ,
'utc_offset' => 0 ,
'time_zone' => 'UTC' ,
'geo_enabled' => false ,
'verified' => false ,
'statuses_count' => 0 ,
'lang' => '' ,
'contributors_enabled' => false ,
'is_translator' => false ,
'is_translation_enabled' => false ,
'following' => false ,
'follow_request_sent' => false ,
'statusnet_blocking' => false ,
'notifications' => false ,
2018-08-19 12:46:11 +00:00
'statusnet_profile_url' => $contact [ " url " ],
2017-11-10 05:00:50 +00:00
'uid' => 0 ,
2018-08-19 12:46:11 +00:00
'cid' => Contact :: getIdForURL ( $contact [ " url " ], api_user (), true ),
'pid' => Contact :: getIdForURL ( $contact [ " url " ], 0 , true ),
2017-11-10 05:00:50 +00:00
'self' => 0 ,
2018-08-19 12:46:11 +00:00
'network' => $contact [ " network " ],
2018-01-15 13:05:12 +00:00
];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $ret ;
2016-11-04 22:45:20 +00:00
} else {
2018-06-03 09:55:41 +00:00
throw new BadRequestException ( " User " . $url . " not found. " );
2016-11-04 22:45:20 +00:00
}
2017-11-10 05:00:50 +00:00
}
if ( $uinfo [ 0 ][ 'self' ]) {
if ( $uinfo [ 0 ][ 'network' ] == " " ) {
2018-08-11 20:40:44 +00:00
$uinfo [ 0 ][ 'network' ] = Protocol :: DFRN ;
2017-11-10 05:00:50 +00:00
}
2018-07-20 12:19:26 +00:00
$usr = DBA :: selectFirst ( 'user' , [ 'default-location' ], [ 'uid' => api_user ()]);
$profile = DBA :: selectFirst ( 'profile' , [ 'about' ], [ 'uid' => api_user (), 'is-default' => true ]);
2017-11-10 05:00:50 +00:00
}
2018-06-19 17:11:59 +00:00
$countitems = 0 ;
2017-11-10 05:00:50 +00:00
$countfriends = 0 ;
$countfollowers = 0 ;
$starred = 0 ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Add a nick if it isn't present there
if (( $uinfo [ 0 ][ 'nick' ] == " " ) || ( $uinfo [ 0 ][ 'name' ] == $uinfo [ 0 ][ 'nick' ])) {
$uinfo [ 0 ][ 'nick' ] = api_get_nick ( $uinfo [ 0 ][ " url " ]);
2016-09-25 16:50:08 +00:00
}
2017-11-19 22:03:39 +00:00
$pcontact_id = Contact :: getIdForURL ( $uinfo [ 0 ][ 'url' ], 0 , true );
2017-11-10 05:00:50 +00:00
2018-06-09 16:56:37 +00:00
if ( ! empty ( $profile [ 'about' ])) {
$description = $profile [ 'about' ];
2017-12-24 11:51:38 +00:00
} else {
$description = $uinfo [ 0 ][ " about " ];
}
2018-06-09 16:56:37 +00:00
if ( ! empty ( $usr [ 'default-location' ])) {
$location = $usr [ 'default-location' ];
2017-12-24 11:51:38 +00:00
} elseif ( ! empty ( $uinfo [ 0 ][ " location " ])) {
$location = $uinfo [ 0 ][ " location " ];
} else {
2018-12-26 16:02:28 +00:00
$location = ContactSelector :: networkToName ( $uinfo [ 0 ][ 'network' ], $uinfo [ 0 ][ 'url' ]);
2017-12-24 11:51:38 +00:00
}
2018-01-15 13:05:12 +00:00
$ret = [
2017-11-10 05:00:50 +00:00
'id' => intval ( $pcontact_id ),
'id_str' => ( string ) intval ( $pcontact_id ),
'name' => (( $uinfo [ 0 ][ 'name' ]) ? $uinfo [ 0 ][ 'name' ] : $uinfo [ 0 ][ 'nick' ]),
'screen_name' => (( $uinfo [ 0 ][ 'nick' ]) ? $uinfo [ 0 ][ 'nick' ] : $uinfo [ 0 ][ 'name' ]),
2017-12-24 11:51:38 +00:00
'location' => $location ,
'description' => $description ,
2017-11-10 05:00:50 +00:00
'profile_image_url' => $uinfo [ 0 ][ 'micro' ],
'profile_image_url_https' => $uinfo [ 0 ][ 'micro' ],
2018-04-07 11:13:31 +00:00
'profile_image_url_profile_size' => $uinfo [ 0 ][ " thumb " ],
'profile_image_url_large' => $uinfo [ 0 ][ " photo " ],
2017-11-10 05:00:50 +00:00
'url' => $uinfo [ 0 ][ 'url' ],
'protected' => false ,
'followers_count' => intval ( $countfollowers ),
'friends_count' => intval ( $countfriends ),
'listed_count' => 0 ,
'created_at' => api_date ( $uinfo [ 0 ][ 'created' ]),
'favourites_count' => intval ( $starred ),
'utc_offset' => " 0 " ,
'time_zone' => 'UTC' ,
'geo_enabled' => false ,
'verified' => true ,
2018-06-19 17:11:59 +00:00
'statuses_count' => intval ( $countitems ),
2017-11-10 05:00:50 +00:00
'lang' => '' ,
'contributors_enabled' => false ,
'is_translator' => false ,
'is_translation_enabled' => false ,
2018-07-25 02:53:46 +00:00
'following' => (( $uinfo [ 0 ][ 'rel' ] == Contact :: FOLLOWER ) || ( $uinfo [ 0 ][ 'rel' ] == Contact :: FRIEND )),
2017-11-10 05:00:50 +00:00
'follow_request_sent' => false ,
'statusnet_blocking' => false ,
'notifications' => false ,
/// @TODO old way?
2018-10-13 11:29:56 +00:00
//'statusnet_profile_url' => System::baseUrl()."/contact/".$uinfo[0]['cid'],
2017-11-10 05:00:50 +00:00
'statusnet_profile_url' => $uinfo [ 0 ][ 'url' ],
'uid' => intval ( $uinfo [ 0 ][ 'uid' ]),
'cid' => intval ( $uinfo [ 0 ][ 'cid' ]),
2018-06-03 07:22:01 +00:00
'pid' => Contact :: getIdForURL ( $uinfo [ 0 ][ " url " ], 0 , true ),
2017-11-10 05:00:50 +00:00
'self' => $uinfo [ 0 ][ 'self' ],
'network' => $uinfo [ 0 ][ 'network' ],
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
2017-12-18 19:39:35 +00:00
// If this is a local user and it uses Frio, we can get its color preferences.
if ( $ret [ 'self' ]) {
2018-07-20 12:19:26 +00:00
$theme_info = DBA :: selectFirst ( 'user' , [ 'theme' ], [ 'uid' => $ret [ 'uid' ]]);
2017-12-18 19:39:35 +00:00
if ( $theme_info [ 'theme' ] === 'frio' ) {
$schema = PConfig :: get ( $ret [ 'uid' ], 'frio' , 'schema' );
2018-07-08 06:32:51 +00:00
2017-12-19 10:33:59 +00:00
if ( $schema && ( $schema != '---' )) {
2017-12-18 19:39:35 +00:00
if ( file_exists ( 'view/theme/frio/schema/' . $schema . '.php' )) {
$schemefile = 'view/theme/frio/schema/' . $schema . '.php' ;
2017-12-19 10:33:59 +00:00
require_once $schemefile ;
2017-12-18 19:39:35 +00:00
}
} else {
$nav_bg = PConfig :: get ( $ret [ 'uid' ], 'frio' , 'nav_bg' );
$link_color = PConfig :: get ( $ret [ 'uid' ], 'frio' , 'link_color' );
$bgcolor = PConfig :: get ( $ret [ 'uid' ], 'frio' , 'background_color' );
}
2018-07-01 04:15:11 +00:00
if ( empty ( $nav_bg )) {
2017-12-18 19:39:35 +00:00
$nav_bg = " #708fa0 " ;
}
2018-07-01 04:15:11 +00:00
if ( empty ( $link_color )) {
2017-12-18 19:39:35 +00:00
$link_color = " #6fdbe8 " ;
}
2018-07-01 04:15:11 +00:00
if ( empty ( $bgcolor )) {
2017-12-18 19:39:35 +00:00
$bgcolor = " #ededed " ;
}
$ret [ 'profile_sidebar_fill_color' ] = str_replace ( '#' , '' , $nav_bg );
$ret [ 'profile_link_color' ] = str_replace ( '#' , '' , $link_color );
$ret [ 'profile_background_color' ] = str_replace ( '#' , '' , $bgcolor );
}
}
2017-11-10 05:00:50 +00:00
return $ret ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief return api - formatted array for item ' s author and owner
*
* @ param object $a App
* @ param array $item item from db
* @ return array ( array : author , array : owner )
*/
function api_item_get_user ( App $a , $item )
{
2018-07-01 04:15:11 +00:00
$status_user = api_get_user ( $a , defaults ( $item , 'author-id' , null ));
2017-11-10 05:00:50 +00:00
2018-07-01 04:15:11 +00:00
$status_user [ " protected " ] = defaults ( $item , 'private' , 0 );
2017-11-10 05:00:50 +00:00
2018-07-01 04:15:11 +00:00
if ( defaults ( $item , 'thr-parent' , '' ) == defaults ( $item , 'uri' , '' )) {
2018-10-01 17:36:23 +00:00
$owner_user = api_get_user ( $a , defaults ( $item , 'owner-id' , null ));
2017-11-10 05:00:50 +00:00
} else {
$owner_user = $status_user ;
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return ([ $status_user , $owner_user ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief walks recursively through an array with the possibility to change value and key
*
* @ param array $array The array to walk through
* @ param string $callback The callback function
*
* @ return array the transformed array
*/
function api_walk_recursive ( array & $array , callable $callback )
{
2018-01-15 13:05:12 +00:00
$new_array = [];
2017-11-10 05:00:50 +00:00
foreach ( $array as $k => $v ) {
if ( is_array ( $v )) {
if ( $callback ( $v , $k )) {
$new_array [ $k ] = api_walk_recursive ( $v , $callback );
}
2017-04-17 14:38:13 +00:00
} else {
2017-11-10 05:00:50 +00:00
if ( $callback ( $v , $k )) {
$new_array [ $k ] = $v ;
}
2017-04-17 14:38:13 +00:00
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$array = $new_array ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $array ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Callback function to transform the array in an array that can be transformed in a XML file
*
* @ param mixed $item Array item value
* @ param string $key Array key
*
* @ return boolean Should the array item be deleted ?
*/
function api_reformat_xml ( & $item , & $key )
{
if ( is_bool ( $item )) {
$item = ( $item ? " true " : " false " );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( substr ( $key , 0 , 10 ) == " statusnet_ " ) {
$key = " statusnet: " . substr ( $key , 10 );
} elseif ( substr ( $key , 0 , 10 ) == " friendica_ " ) {
$key = " friendica: " . substr ( $key , 10 );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
/// @TODO old-lost code?
//else
// $key = "default:".$key;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return true ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Creates the XML from a JSON style array
*
* @ param array $data JSON style array
* @ param string $root_element Name of the root element
*
* @ return string The XML data
*/
2018-05-10 11:19:50 +00:00
function api_create_xml ( array $data , $root_element )
2017-11-10 05:00:50 +00:00
{
$childname = key ( $data );
$data2 = array_pop ( $data );
2018-01-15 13:05:12 +00:00
$namespaces = [ " " => " http://api.twitter.com " ,
2017-11-10 05:00:50 +00:00
" statusnet " => " http://status.net/schema/api/1/ " ,
" friendica " => " http://friendi.ca/schema/api/1/ " ,
2018-01-15 13:05:12 +00:00
" georss " => " http://www.georss.org/georss " ];
2017-11-10 05:00:50 +00:00
/// @todo Auto detection of needed namespaces
2018-01-15 13:05:12 +00:00
if ( in_array ( $root_element , [ " ok " , " hash " , " config " , " version " , " ids " , " notes " , " photos " ])) {
$namespaces = [];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( is_array ( $data2 )) {
2018-07-01 18:40:56 +00:00
$key = key ( $data2 );
2017-11-10 05:00:50 +00:00
api_walk_recursive ( $data2 , " api_reformat_xml " );
2016-09-25 16:50:08 +00:00
2018-07-01 18:40:56 +00:00
if ( $key == " 0 " ) {
$data4 = [];
$i = 1 ;
2016-09-25 16:50:08 +00:00
2018-07-01 18:40:56 +00:00
foreach ( $data2 as $item ) {
$data4 [ $i ++ . " : " . $childname ] = $item ;
}
2016-09-25 16:50:08 +00:00
2018-07-01 18:40:56 +00:00
$data2 = $data4 ;
}
2016-09-25 16:50:08 +00:00
}
2018-01-15 13:05:12 +00:00
$data3 = [ $root_element => $data2 ];
2016-09-25 16:50:08 +00:00
2017-11-20 17:56:31 +00:00
$ret = XML :: fromArray ( $data3 , $xml , false , $namespaces );
2017-11-10 05:00:50 +00:00
return $ret ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Formats the data according to the data type
*
* @ param string $root_element Name of the root element
* @ param string $type Return type ( atom , rss , xml , json )
* @ param array $data JSON style array
*
2018-04-09 17:19:00 +00:00
* @ return ( string | array ) XML data or JSON data
2017-11-10 05:00:50 +00:00
*/
function api_format_data ( $root_element , $type , $data )
{
switch ( $type ) {
case " atom " :
case " rss " :
case " xml " :
$ret = api_create_xml ( $data , $root_element );
break ;
case " json " :
2018-04-09 19:34:53 +00:00
default :
2017-11-10 05:00:50 +00:00
$ret = $data ;
break ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
return $ret ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* TWITTER API
*/
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* Returns an HTTP 200 OK response code and a representation of the requesting user if authentication was successful ;
* returns a 401 status code and an error message if not .
2017-12-24 02:20:50 +00:00
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / manage - account - settings / api - reference / get - account - verify_credentials
*
* @ param string $type Return type ( atom , rss , xml , json )
2017-11-10 05:00:50 +00:00
*/
function api_account_verify_credentials ( $type )
{
2016-09-25 16:50:08 +00:00
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " user_id " ]);
unset ( $_GET [ " user_id " ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " screen_name " ]);
unset ( $_GET [ " screen_name " ]);
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
$skip_status = defaults ( $_REQUEST , 'skip_status' , false );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// "verified" isn't used here in the standard
unset ( $user_info [ " verified " ]);
// - Adding last status
if ( ! $skip_status ) {
$user_info [ " status " ] = api_status_show ( " raw " );
if ( ! count ( $user_info [ " status " ])) {
unset ( $user_info [ " status " ]);
} else {
unset ( $user_info [ " status " ][ " user " ]);
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// "uid" and "self" are only needed for some internal stuff, so remove it from here
unset ( $user_info [ " uid " ]);
unset ( $user_info [ " self " ]);
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( " user " , $type , [ 'user' => $user_info ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/account/verify_credentials' , 'api_account_verify_credentials' , true );
/**
* Get data from $_POST or $_GET
2017-12-24 02:20:50 +00:00
*
* @ param string $k
2017-11-10 05:00:50 +00:00
*/
function requestdata ( $k )
{
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_POST [ $k ])) {
2017-11-10 05:00:50 +00:00
return $_POST [ $k ];
}
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_GET [ $k ])) {
2017-11-10 05:00:50 +00:00
return $_GET [ $k ];
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
return null ;
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
2018-06-05 05:42:26 +00:00
* Deprecated function to upload media .
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_statuses_mediap ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( 'api_statuses_update: no user' );
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
$user_info = api_get_user ( $a );
$_REQUEST [ 'profile_uid' ] = api_user ();
$_REQUEST [ 'api_source' ] = true ;
$txt = requestdata ( 'status' );
/// @TODO old-lost code?
//$txt = urldecode(requestdata('status'));
if (( strpos ( $txt , '<' ) !== false ) || ( strpos ( $txt , '>' ) !== false )) {
2018-11-06 11:34:32 +00:00
$txt = HTML :: toBBCodeVideo ( $txt );
2017-11-10 05:00:50 +00:00
$config = HTMLPurifier_Config :: createDefault ();
$config -> set ( 'Cache.DefinitionImpl' , null );
$purifier = new HTMLPurifier ( $config );
$txt = $purifier -> purify ( $txt );
2016-09-25 16:50:08 +00:00
}
2018-03-07 21:24:13 +00:00
$txt = HTML :: toBBCode ( $txt );
2016-09-25 16:50:08 +00:00
2018-06-05 05:42:26 +00:00
$a -> argv [ 1 ] = $user_info [ 'screen_name' ]; //should be set to username?
2016-09-25 16:50:08 +00:00
2018-06-05 05:42:26 +00:00
$picture = wall_upload_post ( $a , false );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// now that we have the img url in bbcode we can add it to the status and insert the wall item.
2018-06-05 05:42:26 +00:00
$_REQUEST [ 'body' ] = $txt . " \n \n " . '[url=' . $picture [ " albumpage " ] . '][img]' . $picture [ " preview " ] . " [/img][/url] " ;
2018-09-02 07:20:04 +00:00
$item_id = item_post ( $a );
2016-09-25 16:50:08 +00:00
2018-09-02 07:20:04 +00:00
// output the post that we just posted.
return api_status_show ( $type , $item_id );
2017-11-10 05:00:50 +00:00
}
/// @TODO move this to top of file or somewhere better!
api_register_func ( 'api/statuses/mediap' , 'api_statuses_mediap' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Updates the user’ s current status .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / tweets / post - and - engage / api - reference / post - statuses - update
*/
2017-11-10 05:00:50 +00:00
function api_statuses_update ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( 'api_statuses_update: no user' );
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2018-01-04 18:26:09 +00:00
api_get_user ( $a );
2017-11-10 05:00:50 +00:00
// convert $_POST array items to the form we use for web posts.
if ( requestdata ( 'htmlstatus' )) {
$txt = requestdata ( 'htmlstatus' );
if (( strpos ( $txt , '<' ) !== false ) || ( strpos ( $txt , '>' ) !== false )) {
2018-11-06 11:34:32 +00:00
$txt = HTML :: toBBCodeVideo ( $txt );
2017-11-10 05:00:50 +00:00
2016-09-25 16:50:08 +00:00
$config = HTMLPurifier_Config :: createDefault ();
$config -> set ( 'Cache.DefinitionImpl' , null );
2017-11-10 05:00:50 +00:00
2016-09-25 16:50:08 +00:00
$purifier = new HTMLPurifier ( $config );
$txt = $purifier -> purify ( $txt );
2018-03-07 21:24:13 +00:00
$_REQUEST [ 'body' ] = HTML :: toBBCode ( $txt );
2017-11-10 05:00:50 +00:00
}
} else {
$_REQUEST [ 'body' ] = requestdata ( 'status' );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$_REQUEST [ 'title' ] = requestdata ( 'title' );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$parent = requestdata ( 'in_reply_to_status_id' );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Twidere sends "-1" if it is no reply ...
if ( $parent == - 1 ) {
$parent = " " ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
if ( ctype_digit ( $parent )) {
$_REQUEST [ 'parent' ] = $parent ;
} else {
$_REQUEST [ 'parent_uri' ] = $parent ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( requestdata ( 'lat' ) && requestdata ( 'long' )) {
$_REQUEST [ 'coord' ] = sprintf ( " %s %s " , requestdata ( 'lat' ), requestdata ( 'long' ));
}
$_REQUEST [ 'profile_uid' ] = api_user ();
2018-07-19 13:52:05 +00:00
if ( ! $parent ) {
2017-11-10 05:00:50 +00:00
// Check for throttling (maximum posts per day, week and month)
$throttle_day = Config :: get ( 'system' , 'throttle_limit_day' );
if ( $throttle_day > 0 ) {
2018-01-27 02:38:34 +00:00
$datefrom = date ( DateTimeFormat :: MYSQL , time () - 24 * 60 * 60 );
2017-11-10 05:00:50 +00:00
2018-09-02 07:35:12 +00:00
$condition = [ " `uid` = ? AND `wall` AND `created` > ? " , api_user (), $datefrom ];
$posts_day = DBA :: count ( 'thread' , $condition );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $posts_day > $throttle_day ) {
2018-10-30 13:58:45 +00:00
Logger :: log ( 'Daily posting limit reached for user ' . api_user (), Logger :: DEBUG );
2018-01-24 02:59:16 +00:00
// die(api_error($type, L10n::t("Daily posting limit of %d posts reached. The post was rejected.", $throttle_day));
2018-01-24 03:38:47 +00:00
throw new TooManyRequestsException ( L10n :: tt ( " Daily posting limit of %d post reached. The post was rejected. " , " Daily posting limit of %d posts reached. The post was rejected. " , $throttle_day ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$throttle_week = Config :: get ( 'system' , 'throttle_limit_week' );
if ( $throttle_week > 0 ) {
2018-01-27 02:38:34 +00:00
$datefrom = date ( DateTimeFormat :: MYSQL , time () - 24 * 60 * 60 * 7 );
2016-09-25 16:50:08 +00:00
2018-09-02 07:35:12 +00:00
$condition = [ " `uid` = ? AND `wall` AND `created` > ? " , api_user (), $datefrom ];
$posts_week = DBA :: count ( 'thread' , $condition );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $posts_week > $throttle_week ) {
2018-10-30 13:58:45 +00:00
Logger :: log ( 'Weekly posting limit reached for user ' . api_user (), Logger :: DEBUG );
2018-01-24 02:59:16 +00:00
// die(api_error($type, L10n::t("Weekly posting limit of %d posts reached. The post was rejected.", $throttle_week)));
2018-01-24 03:38:47 +00:00
throw new TooManyRequestsException ( L10n :: tt ( " Weekly posting limit of %d post reached. The post was rejected. " , " Weekly posting limit of %d posts reached. The post was rejected. " , $throttle_week ));
2017-11-10 05:00:50 +00:00
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$throttle_month = Config :: get ( 'system' , 'throttle_limit_month' );
if ( $throttle_month > 0 ) {
2018-01-27 02:38:34 +00:00
$datefrom = date ( DateTimeFormat :: MYSQL , time () - 24 * 60 * 60 * 30 );
2016-09-25 16:50:08 +00:00
2018-09-02 07:35:12 +00:00
$condition = [ " `uid` = ? AND `wall` AND `created` > ? " , api_user (), $datefrom ];
$posts_month = DBA :: count ( 'thread' , $condition );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $posts_month > $throttle_month ) {
2018-10-30 13:58:45 +00:00
Logger :: log ( 'Monthly posting limit reached for user ' . api_user (), Logger :: DEBUG );
2018-01-24 02:59:16 +00:00
// die(api_error($type, L10n::t("Monthly posting limit of %d posts reached. The post was rejected.", $throttle_month));
2018-01-24 03:43:22 +00:00
throw new TooManyRequestsException ( L10n :: t ( " Monthly posting limit of %d post reached. The post was rejected. " , " Monthly posting limit of %d posts reached. The post was rejected. " , $throttle_month ));
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_FILES [ 'media' ])) {
2017-11-10 05:00:50 +00:00
// upload the image if we have one
2018-06-05 05:42:26 +00:00
$picture = wall_upload_post ( $a , false );
2018-06-05 05:44:04 +00:00
if ( is_array ( $picture )) {
2018-06-05 05:42:26 +00:00
$_REQUEST [ 'body' ] .= " \n \n " . '[url=' . $picture [ " albumpage " ] . '][img]' . $picture [ " preview " ] . " [/img][/url] " ;
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// To-Do: Multiple IDs
if ( requestdata ( 'media_ids' )) {
$r = q (
" SELECT `resource-id`, `scale`, `nickname`, `type` FROM `photo` INNER JOIN `user` ON `user`.`uid` = `photo`.`uid` WHERE `resource-id` IN (SELECT `resource-id` FROM `photo` WHERE `id` = %d) AND `scale` > 0 AND `photo`.`uid` = %d ORDER BY `photo`.`width` DESC LIMIT 1 " ,
intval ( requestdata ( 'media_ids' )),
api_user ()
);
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2017-12-07 13:56:11 +00:00
$phototypes = Image :: supportedTypes ();
2017-11-10 05:00:50 +00:00
$ext = $phototypes [ $r [ 0 ][ 'type' ]];
$_REQUEST [ 'body' ] .= " \n \n " . '[url=' . System :: baseUrl () . '/photos/' . $r [ 0 ][ 'nickname' ] . '/image/' . $r [ 0 ][ 'resource-id' ] . ']' ;
$_REQUEST [ 'body' ] .= '[img]' . System :: baseUrl () . '/photo/' . $r [ 0 ][ 'resource-id' ] . '-' . $r [ 0 ][ 'scale' ] . '.' . $ext . '[/img][/url]' ;
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// set this so that the item_post() function is quiet and doesn't redirect or emit json
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$_REQUEST [ 'api_source' ] = true ;
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( empty ( $_REQUEST [ 'source' ])) {
2017-11-10 05:00:50 +00:00
$_REQUEST [ " source " ] = api_source ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// call out normal post function
2018-09-02 07:20:04 +00:00
$item_id = item_post ( $a );
2016-09-25 16:50:08 +00:00
2018-09-02 07:20:04 +00:00
// output the post that we just posted.
return api_status_show ( $type , $item_id );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/statuses/update' , 'api_statuses_update' , true , API_METHOD_POST );
api_register_func ( 'api/statuses/update_with_media' , 'api_statuses_update' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Uploads an image to Friendica .
*
* @ return array
* @ see https :// developer . twitter . com / en / docs / media / upload - media / api - reference / post - media - upload
*/
2018-01-04 01:54:35 +00:00
function api_media_upload ()
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( 'no user' );
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2018-01-04 18:26:09 +00:00
api_get_user ( $a );
2018-11-30 14:06:22 +00:00
if ( empty ( $_FILES [ 'media' ])) {
2017-11-10 05:00:50 +00:00
// Output error
throw new BadRequestException ( " No media. " );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$media = wall_upload_post ( $a , false );
if ( ! $media ) {
// Output error
throw new InternalServerErrorException ();
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$returndata = [];
2017-11-10 05:00:50 +00:00
$returndata [ " media_id " ] = $media [ " id " ];
$returndata [ " media_id_string " ] = ( string ) $media [ " id " ];
$returndata [ " size " ] = $media [ " size " ];
2018-01-15 13:05:12 +00:00
$returndata [ " image " ] = [ " w " => $media [ " width " ],
2017-11-10 05:00:50 +00:00
" h " => $media [ " height " ],
2018-01-15 13:05:12 +00:00
" image_type " => $media [ " type " ]];
2016-09-25 16:50:08 +00:00
2018-10-30 13:58:45 +00:00
Logger :: log ( " Media uploaded: " . print_r ( $returndata , true ), Logger :: DEBUG );
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return [ " media " => $returndata ];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/media/upload' , 'api_media_upload' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2018-09-02 07:20:04 +00:00
function api_status_show ( $type , $item_id = 0 )
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-10-30 13:58:45 +00:00
Logger :: log ( 'api_status_show: user_info: ' . print_r ( $user_info , true ), Logger :: DEBUG );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " raw " ) {
2018-06-03 07:22:01 +00:00
$privacy_sql = " AND NOT `private` " ;
2017-11-10 05:00:50 +00:00
} else {
$privacy_sql = " " ;
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2018-09-02 07:20:04 +00:00
if ( ! empty ( $item_id )) {
// Get the item with the given id
$condition = [ 'id' => $item_id ];
} else {
// get last public wall message
$condition = [ 'owner-id' => $user_info [ 'pid' ], 'uid' => api_user (),
'gravity' => [ GRAVITY_PARENT , GRAVITY_COMMENT ]];
}
2018-07-07 18:14:16 +00:00
$lastwall = Item :: selectFirst ( Item :: ITEM_FIELDLIST , $condition , [ 'order' => [ 'id' => true ]]);
2017-11-10 05:00:50 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $lastwall )) {
2017-11-10 05:00:50 +00:00
$in_reply_to = api_in_reply_to ( $lastwall );
$converted = api_convert_item ( $lastwall );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
$geo = " georss:point " ;
} else {
$geo = " geo " ;
}
2018-01-15 13:05:12 +00:00
$status_info = [
2017-11-10 05:00:50 +00:00
'created_at' => api_date ( $lastwall [ 'created' ]),
'id' => intval ( $lastwall [ 'id' ]),
'id_str' => ( string ) $lastwall [ 'id' ],
'text' => $converted [ " text " ],
'source' => (( $lastwall [ 'app' ]) ? $lastwall [ 'app' ] : 'web' ),
'truncated' => false ,
'in_reply_to_status_id' => $in_reply_to [ 'status_id' ],
'in_reply_to_status_id_str' => $in_reply_to [ 'status_id_str' ],
'in_reply_to_user_id' => $in_reply_to [ 'user_id' ],
'in_reply_to_user_id_str' => $in_reply_to [ 'user_id_str' ],
'in_reply_to_screen_name' => $in_reply_to [ 'screen_name' ],
'user' => $user_info ,
$geo => null ,
2018-07-04 21:33:09 +00:00
'coordinates' => '' ,
'place' => '' ,
'contributors' => '' ,
2017-11-10 05:00:50 +00:00
'is_quote_status' => false ,
'retweet_count' => 0 ,
'favorite_count' => 0 ,
'favorited' => $lastwall [ 'starred' ] ? true : false ,
'retweeted' => false ,
'possibly_sensitive' => false ,
2018-07-04 21:33:09 +00:00
'lang' => '' ,
2017-12-17 15:16:18 +00:00
'statusnet_html' => $converted [ " html " ],
'statusnet_conversation_id' => $lastwall [ 'parent' ],
2018-07-04 21:33:09 +00:00
'external_url' => System :: baseUrl () . '/display/' . $lastwall [ 'guid' ],
2018-01-15 13:05:12 +00:00
];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( count ( $converted [ " attachments " ]) > 0 ) {
$status_info [ " attachments " ] = $converted [ " attachments " ];
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
if ( count ( $converted [ " entities " ]) > 0 ) {
$status_info [ " entities " ] = $converted [ " entities " ];
2016-09-25 16:50:08 +00:00
}
2018-06-03 07:22:01 +00:00
if ( $status_info [ " source " ] == 'web' ) {
2018-10-22 14:15:14 +00:00
$status_info [ " source " ] = ContactSelector :: networkToName ( $lastwall [ 'network' ], $lastwall [ 'author-link' ]);
} elseif ( ContactSelector :: networkToName ( $lastwall [ 'network' ], $lastwall [ 'author-link' ]) != $status_info [ " source " ]) {
$status_info [ " source " ] = trim ( $status_info [ " source " ] . ' (' . ContactSelector :: networkToName ( $lastwall [ 'network' ], $lastwall [ 'author-link' ]) . ')' );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// "uid" and "self" are only needed for some internal stuff, so remove it from here
unset ( $status_info [ " user " ][ " uid " ]);
unset ( $status_info [ " user " ][ " self " ]);
2016-09-25 16:50:08 +00:00
2018-10-30 13:58:45 +00:00
Logger :: log ( 'status_info: ' . print_r ( $status_info , true ), Logger :: DEBUG );
2016-09-25 16:50:08 +00:00
2018-04-09 19:34:53 +00:00
if ( $type == " raw " ) {
return $status_info ;
}
2017-04-05 20:07:55 +00:00
2018-04-09 19:34:53 +00:00
return api_format_data ( " statuses " , $type , [ 'status' => $status_info ]);
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* Returns extended information of a given user , specified by ID or screen name as per the required id parameter .
* The author ' s most recent status will be returned inline .
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / get - users - show
2017-11-10 05:00:50 +00:00
*/
function api_users_show ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$user_info = api_get_user ( $a );
2018-06-27 18:09:33 +00:00
$condition = [ 'owner-id' => $user_info [ 'pid' ], 'uid' => api_user (),
'gravity' => [ GRAVITY_PARENT , GRAVITY_COMMENT ], 'private' => false ];
2018-07-07 18:14:16 +00:00
$lastwall = Item :: selectFirst ( Item :: ITEM_FIELDLIST , $condition , [ 'order' => [ 'id' => true ]]);
2017-11-10 05:00:50 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $lastwall )) {
2017-11-10 05:00:50 +00:00
$in_reply_to = api_in_reply_to ( $lastwall );
$converted = api_convert_item ( $lastwall );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
$geo = " georss:point " ;
} else {
$geo = " geo " ;
}
2018-01-15 13:05:12 +00:00
$user_info [ 'status' ] = [
2017-11-10 05:00:50 +00:00
'text' => $converted [ " text " ],
'truncated' => false ,
'created_at' => api_date ( $lastwall [ 'created' ]),
'in_reply_to_status_id' => $in_reply_to [ 'status_id' ],
'in_reply_to_status_id_str' => $in_reply_to [ 'status_id_str' ],
'source' => (( $lastwall [ 'app' ]) ? $lastwall [ 'app' ] : 'web' ),
'id' => intval ( $lastwall [ 'contact-id' ]),
'id_str' => ( string ) $lastwall [ 'contact-id' ],
'in_reply_to_user_id' => $in_reply_to [ 'user_id' ],
'in_reply_to_user_id_str' => $in_reply_to [ 'user_id_str' ],
'in_reply_to_screen_name' => $in_reply_to [ 'screen_name' ],
$geo => null ,
'favorited' => $lastwall [ 'starred' ] ? true : false ,
'statusnet_html' => $converted [ " html " ],
2017-12-17 15:16:18 +00:00
'statusnet_conversation_id' => $lastwall [ 'parent' ],
'external_url' => System :: baseUrl () . " /display/ " . $lastwall [ 'guid' ],
2018-01-15 13:05:12 +00:00
];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( count ( $converted [ " attachments " ]) > 0 ) {
$user_info [ " status " ][ " attachments " ] = $converted [ " attachments " ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( count ( $converted [ " entities " ]) > 0 ) {
$user_info [ " status " ][ " entities " ] = $converted [ " entities " ];
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-03 07:22:01 +00:00
if ( $user_info [ " status " ][ " source " ] == 'web' ) {
2018-10-22 14:15:14 +00:00
$user_info [ " status " ][ " source " ] = ContactSelector :: networkToName ( $lastwall [ 'network' ], $lastwall [ 'author-link' ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-03 07:22:01 +00:00
if ( ContactSelector :: networkToName ( $lastwall [ 'network' ], $user_info [ 'url' ]) != $user_info [ " status " ][ " source " ]) {
2018-10-22 14:15:14 +00:00
$user_info [ " status " ][ " source " ] = trim ( $user_info [ " status " ][ " source " ] . ' (' . ContactSelector :: networkToName ( $lastwall [ 'network' ], $lastwall [ 'author-link' ]) . ')' );
2017-11-10 05:00:50 +00:00
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// "uid" and "self" are only needed for some internal stuff, so remove it from here
unset ( $user_info [ " uid " ]);
unset ( $user_info [ " self " ]);
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( " user " , $type , [ 'user' => $user_info ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/users/show' , 'api_users_show' );
api_register_func ( 'api/externalprofile/show' , 'api_users_show' );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Search a public user account .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / get - users - search
*/
2017-11-10 05:00:50 +00:00
function api_users_search ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$userlist = [];
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_GET [ 'q' ])) {
2018-07-21 13:10:13 +00:00
$r = q ( " SELECT id FROM `contact` WHERE `uid` = 0 AND `name` = '%s' " , DBA :: escape ( $_GET [ " q " ]));
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2018-07-21 13:10:13 +00:00
$r = q ( " SELECT `id` FROM `contact` WHERE `uid` = 0 AND `nick` = '%s' " , DBA :: escape ( $_GET [ " q " ]));
2016-09-25 16:50:08 +00:00
}
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
$k = 0 ;
foreach ( $r as $user ) {
2017-12-23 00:27:17 +00:00
$user_info = api_get_user ( $a , $user [ " id " ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
$userlist [ $k ++. " :user " ] = $user_info ;
} else {
$userlist [] = $user_info ;
}
}
2018-01-15 13:05:12 +00:00
$userlist = [ " users " => $userlist ];
2017-11-10 05:00:50 +00:00
} else {
2018-06-03 09:55:41 +00:00
throw new BadRequestException ( " User " . $_GET [ " q " ] . " not found. " );
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
} else {
2018-06-03 09:55:41 +00:00
throw new BadRequestException ( " No user specified. " );
2016-09-25 16:50:08 +00:00
}
2018-01-04 01:54:35 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " users " , $type , $userlist );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/users/search' , 'api_users_search' );
2016-09-25 16:50:08 +00:00
2017-12-18 22:51:03 +00:00
/**
* Return user objects
*
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / get - users - lookup
*
* @ param string $type Return format : json or xml
*
* @ return array | string
2017-12-19 13:52:46 +00:00
* @ throws NotFoundException if the results are empty .
2017-12-18 22:51:03 +00:00
*/
function api_users_lookup ( $type )
{
2018-01-15 13:05:12 +00:00
$users = [];
2017-12-19 10:37:55 +00:00
2018-07-01 08:03:57 +00:00
if ( ! empty ( $_REQUEST [ 'user_id' ])) {
2017-12-19 10:37:55 +00:00
foreach ( explode ( ',' , $_REQUEST [ 'user_id' ]) as $id ) {
if ( ! empty ( $id )) {
$users [] = api_get_user ( get_app (), $id );
}
2017-12-18 22:51:03 +00:00
}
}
2017-12-19 10:37:55 +00:00
if ( empty ( $users )) {
throw new NotFoundException ;
}
2018-01-15 13:05:12 +00:00
return api_format_data ( " users " , $type , [ 'users' => $users ]);
2017-12-18 22:51:03 +00:00
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/users/lookup' , 'api_users_lookup' , true );
2017-12-18 01:59:11 +00:00
/**
* Returns statuses that match a specified query .
*
* @ see https :// developer . twitter . com / en / docs / tweets / search / api - reference / get - search - tweets
*
* @ param string $type Return format : json , xml , atom , rss
*
* @ return array | string
2017-12-19 13:52:46 +00:00
* @ throws BadRequestException if the " q " parameter is missing .
2017-12-18 01:59:11 +00:00
*/
function api_search ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
if ( api_user () === false || $user_info === false ) {
throw new ForbiddenException ();
}
2018-01-15 13:05:12 +00:00
$data = [];
2017-12-18 01:59:11 +00:00
2018-11-30 14:06:22 +00:00
if ( empty ( $_REQUEST [ 'q' ])) {
2017-12-18 12:35:36 +00:00
throw new BadRequestException ( " q parameter is required. " );
}
2017-12-18 01:59:11 +00:00
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_REQUEST [ 'rpp' ])) {
2017-12-18 12:35:36 +00:00
$count = $_REQUEST [ 'rpp' ];
2018-11-30 14:06:22 +00:00
} elseif ( ! empty ( $_REQUEST [ 'count' ])) {
2017-12-18 12:35:36 +00:00
$count = $_REQUEST [ 'count' ];
} else {
$count = 15 ;
}
2017-12-18 01:59:11 +00:00
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$page = ( ! empty ( $_REQUEST [ 'page' ]) ? $_REQUEST [ 'page' ] - 1 : 0 );
2017-12-18 01:59:11 +00:00
2017-12-18 12:35:36 +00:00
$start = $page * $count ;
2017-12-18 01:59:11 +00:00
2018-06-27 18:09:33 +00:00
$condition = [ " `gravity` IN (?, ?) AND `item`.`id` > ?
2018-06-09 16:56:37 +00:00
AND ( `item` . `uid` = 0 OR ( `item` . `uid` = ? AND NOT `item` . `global` ))
AND `item` . `body` LIKE CONCAT ( '%' , ? , '%' ) " ,
2018-06-27 18:09:33 +00:00
GRAVITY_PARENT , GRAVITY_COMMENT , $since_id , api_user (), $_REQUEST [ 'q' ]];
2017-12-18 01:59:11 +00:00
2018-06-09 19:12:13 +00:00
if ( $max_id > 0 ) {
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
}
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2017-12-18 12:35:36 +00:00
2018-06-21 15:14:01 +00:00
$data [ 'status' ] = api_format_items ( Item :: inArray ( $statuses ), $user_info );
2017-12-18 12:35:36 +00:00
2017-12-18 01:59:11 +00:00
return api_format_data ( " statuses " , $type , $data );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/search/tweets' , 'api_search' , true );
api_register_func ( 'api/search' , 'api_search' , true );
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns the most recent statuses posted by the user and the users they follow .
*
* @ see https :// developer . twitter . com / en / docs / tweets / timelines / api - reference / get - statuses - home_timeline
2017-11-10 05:00:50 +00:00
*
2017-12-24 02:20:50 +00:00
* @ param string $type Return type ( atom , rss , xml , json )
2017-11-10 05:00:50 +00:00
*
2017-12-24 02:20:50 +00:00
* @ todo Optional parameters
* @ todo Add reply info
2017-11-10 05:00:50 +00:00
*/
function api_statuses_home_timeline ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " user_id " ]);
unset ( $_GET [ " user_id " ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " screen_name " ]);
unset ( $_GET [ " screen_name " ]);
2016-09-25 16:50:08 +00:00
2018-04-07 20:37:57 +00:00
// get last network messages
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// params
2018-11-30 14:06:22 +00:00
$count = defaults ( $_REQUEST , 'count' , 20 );
$page = ( ! empty ( $_REQUEST [ 'page' ]) ? $_REQUEST [ 'page' ] - 1 : 0 );
2017-11-10 05:00:50 +00:00
if ( $page < 0 ) {
$page = 0 ;
}
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$exclude_replies = ! empty ( $_REQUEST [ 'exclude_replies' ]);
$conversation_id = defaults ( $_REQUEST , 'conversation_id' , 0 );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$start = $page * $count ;
2016-09-25 16:50:08 +00:00
2018-06-27 18:09:33 +00:00
$condition = [ " `uid` = ? AND `gravity` IN (?, ?) AND `item`.`id` > ? " ,
api_user (), GRAVITY_PARENT , GRAVITY_COMMENT , $since_id ];
2018-06-09 16:56:37 +00:00
2017-11-10 05:00:50 +00:00
if ( $max_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
2017-11-10 05:00:50 +00:00
}
2018-11-30 14:06:22 +00:00
if ( $exclude_replies ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= ' AND `item`.`parent` = `item`.`id`' ;
2017-11-10 05:00:50 +00:00
}
if ( $conversation_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`parent` = ? " ;
$condition [] = $conversation_id ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2017-11-10 05:00:50 +00:00
2018-06-21 15:14:01 +00:00
$items = Item :: inArray ( $statuses );
2018-06-09 20:08:15 +00:00
$ret = api_format_items ( $items , $user_info , false , $type );
2017-11-10 05:00:50 +00:00
// Set all posts from the query above to seen
2018-01-15 13:05:12 +00:00
$idarray = [];
2018-06-09 20:08:15 +00:00
foreach ( $items as $item ) {
2017-11-10 05:00:50 +00:00
$idarray [] = intval ( $item [ " id " ]);
}
2016-09-25 16:50:08 +00:00
2018-05-02 19:26:15 +00:00
if ( ! empty ( $idarray )) {
2018-06-27 19:37:13 +00:00
$unseen = Item :: exists ([ 'unseen' => true , 'id' => $idarray ]);
2017-11-10 05:00:50 +00:00
if ( $unseen ) {
2018-05-02 19:26:15 +00:00
Item :: update ([ 'unseen' => false ], [ 'unseen' => true , 'id' => $idarray ]);
2016-09-25 16:50:08 +00:00
}
}
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
break ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " statuses " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/home_timeline' , 'api_statuses_home_timeline' , true );
api_register_func ( 'api/statuses/friends_timeline' , 'api_statuses_home_timeline' , true );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Returns the most recent statuses from public users .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_statuses_public_timeline ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2017-04-05 20:07:55 +00:00
2018-04-07 20:37:57 +00:00
// get last network messages
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// params
2018-11-30 14:06:22 +00:00
$count = defaults ( $_REQUEST , 'count' , 20 );
$page = ( ! empty ( $_REQUEST [ 'page' ]) ? $_REQUEST [ 'page' ] - 1 : 0 );
2017-11-10 05:00:50 +00:00
if ( $page < 0 ) {
$page = 0 ;
2016-09-25 16:50:08 +00:00
}
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$exclude_replies = ( ! empty ( $_REQUEST [ 'exclude_replies' ]) ? 1 : 0 );
$conversation_id = defaults ( $_REQUEST , 'conversation_id' , 0 );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$start = $page * $count ;
2016-09-25 16:50:08 +00:00
2017-12-17 17:47:15 +00:00
if ( $exclude_replies && ! $conversation_id ) {
2018-06-27 18:09:33 +00:00
$condition = [ " `gravity` IN (?, ?) AND `iid` > ? AND NOT `private` AND `wall` AND NOT `user`.`hidewall` " ,
GRAVITY_PARENT , GRAVITY_COMMENT , $since_id ];
2016-09-25 16:50:08 +00:00
2018-06-09 19:12:13 +00:00
if ( $max_id > 0 ) {
$condition [ 0 ] .= " AND `thread`.`iid` <= ? " ;
$condition [] = $max_id ;
}
2018-06-09 16:56:37 +00:00
2018-06-09 19:12:13 +00:00
$params = [ 'order' => [ 'iid' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectThreadForUser ( api_user (), Item :: DISPLAY_FIELDLIST , $condition , $params );
2017-12-17 17:47:15 +00:00
2018-06-21 15:14:01 +00:00
$r = Item :: inArray ( $statuses );
2017-12-17 17:47:15 +00:00
} else {
2018-06-27 18:09:33 +00:00
$condition = [ " `gravity` IN (?, ?) AND `id` > ? AND NOT `private` AND `wall` AND NOT `user`.`hidewall` AND `item`.`origin` " ,
GRAVITY_PARENT , GRAVITY_COMMENT , $since_id ];
2017-12-17 17:47:15 +00:00
2018-06-09 19:12:13 +00:00
if ( $max_id > 0 ) {
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
}
if ( $conversation_id > 0 ) {
$condition [ 0 ] .= " AND `item`.`parent` = ? " ;
$condition [] = $conversation_id ;
}
2018-06-09 16:56:37 +00:00
2018-06-09 19:12:13 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2017-12-17 17:47:15 +00:00
2018-06-21 15:14:01 +00:00
$r = Item :: inArray ( $statuses );
2017-12-17 17:47:15 +00:00
}
2017-11-10 05:00:50 +00:00
$ret = api_format_items ( $r , $user_info , false , $type );
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
break ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " statuses " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/public_timeline' , 'api_statuses_public_timeline' , true );
2016-09-25 16:50:08 +00:00
2017-12-16 20:41:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns the most recent statuses posted by users this node knows about .
*
2017-12-16 20:41:50 +00:00
* @ brief Returns the list of public federated posts this node knows about
*
* @ param string $type Return format : json , xml , atom , rss
* @ return array | string
* @ throws ForbiddenException
*/
function api_statuses_networkpublic_timeline ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2017-12-16 20:41:50 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-12-16 20:41:50 +00:00
throw new ForbiddenException ();
}
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
2017-12-16 20:41:50 +00:00
// pagination
2018-11-30 14:06:22 +00:00
$count = defaults ( $_REQUEST , 'count' , 20 );
$page = defaults ( $_REQUEST , 'page' , 1 );
2017-12-16 20:41:50 +00:00
if ( $page < 1 ) {
$page = 1 ;
}
$start = ( $page - 1 ) * $count ;
2018-06-27 18:09:33 +00:00
$condition = [ " `uid` = 0 AND `gravity` IN (?, ?) AND `thread`.`iid` > ? AND NOT `private` " ,
GRAVITY_PARENT , GRAVITY_COMMENT , $since_id ];
2017-12-16 20:41:50 +00:00
2018-06-09 19:12:13 +00:00
if ( $max_id > 0 ) {
$condition [ 0 ] .= " AND `thread`.`iid` <= ? " ;
$condition [] = $max_id ;
}
2017-12-16 20:41:50 +00:00
2018-06-09 19:12:13 +00:00
$params = [ 'order' => [ 'iid' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectThreadForUser ( api_user (), Item :: DISPLAY_FIELDLIST , $condition , $params );
2018-06-09 16:56:37 +00:00
2018-06-21 15:14:01 +00:00
$ret = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2017-12-16 20:41:50 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-12-16 20:41:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-12-16 20:41:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
break ;
}
return api_format_data ( " statuses " , $type , $data );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/networkpublic_timeline' , 'api_statuses_networkpublic_timeline' , true );
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns a single status .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ see https :// developer . twitter . com / en / docs / tweets / post - and - engage / api - reference / get - statuses - show - id
2017-11-10 05:00:50 +00:00
*/
function api_statuses_show ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// params
2018-07-01 09:17:59 +00:00
$id = intval ( defaults ( $a -> argv , 3 , 0 ));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $id == 0 ) {
2018-07-01 09:17:59 +00:00
$id = intval ( defaults ( $_REQUEST , 'id' , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Hotot workaround
if ( $id == 0 ) {
2018-07-01 09:17:59 +00:00
$id = intval ( defaults ( $a -> argv , 4 , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'API: api_statuses_show: ' . $id );
2016-09-25 16:50:08 +00:00
2018-07-01 09:17:59 +00:00
$conversation = ! empty ( $_REQUEST [ 'conversation' ]);
2016-09-25 16:50:08 +00:00
2018-05-10 10:13:48 +00:00
// try to fetch the item for the local user - or the public item, if there is no local one
2018-07-07 18:14:16 +00:00
$uri_item = Item :: selectFirst ([ 'uri' ], [ 'id' => $id ]);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $uri_item )) {
2018-05-10 10:13:48 +00:00
throw new BadRequestException ( " There is no status with this id. " );
}
2018-07-07 18:14:16 +00:00
$item = Item :: selectFirst ([ 'id' ], [ 'uri' => $uri_item [ 'uri' ], 'uid' => [ 0 , api_user ()]], [ 'order' => [ 'uid' => true ]]);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $item )) {
2018-05-10 10:13:48 +00:00
throw new BadRequestException ( " There is no status with this id. " );
}
$id = $item [ 'id' ];
2018-06-09 16:56:37 +00:00
if ( $conversation ) {
2018-06-27 18:09:33 +00:00
$condition = [ 'parent' => $id , 'gravity' => [ GRAVITY_PARENT , GRAVITY_COMMENT ]];
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ]];
} else {
2018-06-27 18:09:33 +00:00
$condition = [ 'id' => $id , 'gravity' => [ GRAVITY_PARENT , GRAVITY_COMMENT ]];
2018-06-09 16:56:37 +00:00
$params = [];
}
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2017-11-10 05:00:50 +00:00
/// @TODO How about copying this to above methods which don't check $r ?
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $statuses )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " There is no status with this id. " );
}
2016-09-25 16:50:08 +00:00
2018-06-21 15:14:01 +00:00
$ret = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $conversation ) {
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-04-05 20:07:55 +00:00
return api_format_data ( " statuses " , $type , $data );
2017-11-10 05:00:50 +00:00
} else {
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret [ 0 ]];
2017-11-10 05:00:50 +00:00
return api_format_data ( " status " , $type , $data );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/show' , 'api_statuses_show' , true );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ todo nothing to say ?
2017-11-10 05:00:50 +00:00
*/
function api_conversation_show ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// params
2018-07-01 09:17:59 +00:00
$id = intval ( defaults ( $a -> argv , 3 , 0 ));
$since_id = intval ( defaults ( $_REQUEST , 'since_id' , 0 ));
$max_id = intval ( defaults ( $_REQUEST , 'max_id' , 0 ));
$count = intval ( defaults ( $_REQUEST , 'count' , 20 ));
$page = intval ( defaults ( $_REQUEST , 'page' , 1 )) - 1 ;
2017-11-10 05:00:50 +00:00
if ( $page < 0 ) {
$page = 0 ;
}
2016-09-25 16:50:08 +00:00
2018-07-01 09:17:59 +00:00
$start = $page * $count ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $id == 0 ) {
2018-07-01 09:17:59 +00:00
$id = intval ( defaults ( $_REQUEST , 'id' , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Hotot workaround
if ( $id == 0 ) {
2018-07-01 09:17:59 +00:00
$id = intval ( defaults ( $a -> argv , 4 , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'API: api_conversation_show: ' . $id );
2016-09-25 16:50:08 +00:00
2018-05-10 10:13:48 +00:00
// try to fetch the item for the local user - or the public item, if there is no local one
2018-07-07 18:14:16 +00:00
$item = Item :: selectFirst ([ 'parent-uri' ], [ 'id' => $id ]);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $item )) {
2018-05-10 10:13:48 +00:00
throw new BadRequestException ( " There is no status with this id. " );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-07-07 18:14:16 +00:00
$parent = Item :: selectFirst ([ 'id' ], [ 'uri' => $item [ 'parent-uri' ], 'uid' => [ 0 , api_user ()]], [ 'order' => [ 'uid' => true ]]);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $parent )) {
2018-05-10 10:13:48 +00:00
throw new BadRequestException ( " There is no status with this id. " );
}
$id = $parent [ 'id' ];
2018-06-27 18:09:33 +00:00
$condition = [ " `parent` = ? AND `uid` IN (0, ?) AND `gravity` IN (?, ?) AND `item`.`id` > ? " ,
$id , api_user (), GRAVITY_PARENT , GRAVITY_COMMENT , $since_id ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $max_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
}
2017-11-10 05:00:50 +00:00
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2018-06-09 16:56:37 +00:00
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $statuses )) {
2018-06-09 16:56:37 +00:00
throw new BadRequestException ( " There is no status with id $id . " );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-21 15:14:01 +00:00
$ret = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
return api_format_data ( " statuses " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/conversation/show' , 'api_conversation_show' , true );
api_register_func ( 'api/statusnet/conversation' , 'api_conversation_show' , true );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Repeats a status .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ see https :// developer . twitter . com / en / docs / tweets / post - and - engage / api - reference / post - statuses - retweet - id
2017-11-10 05:00:50 +00:00
*/
function api_statuses_repeat ( $type )
{
global $called_api ;
2016-09-25 16:50:08 +00:00
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2018-01-04 18:26:09 +00:00
api_get_user ( $a );
2017-11-10 05:00:50 +00:00
// params
2018-07-01 18:42:38 +00:00
$id = intval ( defaults ( $a -> argv , 3 , 0 ));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $id == 0 ) {
2018-07-01 18:42:38 +00:00
$id = intval ( defaults ( $_REQUEST , 'id' , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Hotot workaround
if ( $id == 0 ) {
2018-07-01 18:42:38 +00:00
$id = intval ( defaults ( $a -> argv , 4 , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'API: api_statuses_repeat: ' . $id );
2017-11-10 05:00:50 +00:00
2018-06-09 16:56:37 +00:00
$fields = [ 'body' , 'author-name' , 'author-link' , 'author-avatar' , 'guid' , 'created' , 'plink' ];
2018-06-17 17:05:17 +00:00
$item = Item :: selectFirst ( $fields , [ 'id' => $id , 'private' => false ]);
2017-11-10 05:00:50 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $item ) && $item [ 'body' ] != " " ) {
2018-06-09 16:56:37 +00:00
if ( strpos ( $item [ 'body' ], " [/share] " ) !== false ) {
$pos = strpos ( $item [ 'body' ], " [share " );
$post = substr ( $item [ 'body' ], $pos );
2017-11-10 05:00:50 +00:00
} else {
2018-06-09 16:56:37 +00:00
$post = share_header ( $item [ 'author-name' ], $item [ 'author-link' ], $item [ 'author-avatar' ], $item [ 'guid' ], $item [ 'created' ], $item [ 'plink' ]);
2016-09-25 16:50:08 +00:00
2018-06-09 16:56:37 +00:00
$post .= $item [ 'body' ];
2017-11-10 05:00:50 +00:00
$post .= " [/share] " ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$_REQUEST [ 'body' ] = $post ;
$_REQUEST [ 'profile_uid' ] = api_user ();
$_REQUEST [ 'api_source' ] = true ;
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( empty ( $_REQUEST [ 'source' ])) {
2017-11-10 05:00:50 +00:00
$_REQUEST [ " source " ] = api_source ();
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
2018-09-02 07:20:04 +00:00
$item_id = item_post ( $a );
2017-11-10 05:00:50 +00:00
} else {
throw new ForbiddenException ();
2016-09-25 16:50:08 +00:00
}
2018-09-02 07:35:12 +00:00
// output the post that we just posted.
2018-04-24 14:16:57 +00:00
$called_api = [];
2018-09-02 07:20:04 +00:00
return api_status_show ( $type , $item_id );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/retweet' , 'api_statuses_repeat' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Destroys a specific status .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ see https :// developer . twitter . com / en / docs / tweets / post - and - engage / api - reference / post - statuses - destroy - id
2017-11-10 05:00:50 +00:00
*/
function api_statuses_destroy ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2018-01-04 18:26:09 +00:00
api_get_user ( $a );
2017-11-10 05:00:50 +00:00
// params
2018-07-01 18:42:38 +00:00
$id = intval ( defaults ( $a -> argv , 3 , 0 ));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $id == 0 ) {
2018-07-01 18:42:38 +00:00
$id = intval ( defaults ( $_REQUEST , 'id' , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Hotot workaround
if ( $id == 0 ) {
2018-07-01 18:42:38 +00:00
$id = intval ( defaults ( $a -> argv , 4 , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log ( 'API: api_statuses_destroy: ' . $id );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$ret = api_statuses_show ( $type );
2016-09-25 16:50:08 +00:00
2018-05-29 05:22:57 +00:00
Item :: deleteForUser ([ 'id' => $id ], api_user ());
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $ret ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/destroy' , 'api_statuses_destroy' , true , API_METHOD_DELETE );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns the most recent mentions .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ see http :// developer . twitter . com / doc / get / statuses / mentions
2017-11-10 05:00:50 +00:00
*/
function api_statuses_mentions ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " user_id " ]);
unset ( $_GET [ " user_id " ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " screen_name " ]);
unset ( $_GET [ " screen_name " ]);
2016-09-25 16:50:08 +00:00
2018-04-07 20:37:57 +00:00
// get last network messages
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// params
2018-01-04 01:54:35 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$count = defaults ( $_REQUEST , 'count' , 20 );
$page = defaults ( $_REQUEST , 'page' , 1 );
if ( $page < 1 ) {
$page = 1 ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-01-04 01:54:35 +00:00
$start = ( $page - 1 ) * $count ;
2016-09-25 16:50:08 +00:00
2018-06-27 18:09:33 +00:00
$condition = [ " `uid` = ? AND `gravity` IN (?, ?) AND `item`.`id` > ? AND `author-id` != ?
2018-06-24 10:48:29 +00:00
AND `item` . `parent` IN ( SELECT `iid` FROM `thread` WHERE `thread` . `uid` = ? AND `thread` . `mention` AND NOT `thread` . `ignored` ) " ,
2018-06-27 18:09:33 +00:00
api_user (), GRAVITY_PARENT , GRAVITY_COMMENT , $since_id , $user_info [ 'pid' ], api_user ()];
2018-02-11 21:13:29 +00:00
2017-11-10 05:00:50 +00:00
if ( $max_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2017-11-10 05:00:50 +00:00
2018-06-21 15:14:01 +00:00
$ret = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
break ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " statuses " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/mentions' , 'api_statuses_mentions' , true );
api_register_func ( 'api/statuses/replies' , 'api_statuses_mentions' , true );
2016-09-25 16:50:08 +00:00
2017-12-16 16:34:44 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns the most recent statuses posted by the user .
*
2017-12-16 16:34:44 +00:00
* @ brief Returns a user ' s public timeline
*
* @ param string $type Either " json " or " xml "
* @ return string | array
* @ throws ForbiddenException
2017-12-24 02:20:50 +00:00
* @ see https :// developer . twitter . com / en / docs / tweets / timelines / api - reference / get - statuses - user_timeline
2017-12-16 16:34:44 +00:00
*/
2017-11-10 05:00:50 +00:00
function api_statuses_user_timeline ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2018-10-29 21:20:46 +00:00
Logger :: log (
2017-11-10 05:00:50 +00:00
" api_statuses_user_timeline: api_user: " . api_user () .
" \n user_info: " . print_r ( $user_info , true ) .
" \n _REQUEST: " . print_r ( $_REQUEST , true ),
2018-10-30 13:58:45 +00:00
Logger :: DEBUG
2017-11-10 05:00:50 +00:00
);
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$exclude_replies = ! empty ( $_REQUEST [ 'exclude_replies' ]);
$conversation_id = defaults ( $_REQUEST , 'conversation_id' , 0 );
2016-09-25 16:50:08 +00:00
2017-12-16 16:34:44 +00:00
// pagination
2018-11-30 14:06:22 +00:00
$count = defaults ( $_REQUEST , 'count' , 20 );
$page = defaults ( $_REQUEST , 'page' , 1 );
2017-12-16 16:34:44 +00:00
if ( $page < 1 ) {
$page = 1 ;
}
$start = ( $page - 1 ) * $count ;
2016-09-25 16:50:08 +00:00
2018-06-27 18:09:33 +00:00
$condition = [ " `uid` = ? AND `gravity` IN (?, ?) AND `item`.`id` > ? AND `item`.`contact-id` = ? " ,
api_user (), GRAVITY_PARENT , GRAVITY_COMMENT , $since_id , $user_info [ 'cid' ]];
2018-06-09 16:56:37 +00:00
2017-11-10 05:00:50 +00:00
if ( $user_info [ 'self' ] == 1 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= ' AND `item`.`wall` ' ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( $exclude_replies ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= ' AND `item`.`parent` = `item`.`id`' ;
2017-11-10 05:00:50 +00:00
}
2017-12-16 16:34:44 +00:00
2017-11-10 05:00:50 +00:00
if ( $conversation_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`parent` = ? " ;
$condition [] = $conversation_id ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-16 16:34:44 +00:00
if ( $max_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
2017-12-16 16:34:44 +00:00
}
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2017-11-10 05:00:50 +00:00
2018-06-21 15:14:01 +00:00
$ret = api_format_items ( Item :: inArray ( $statuses ), $user_info , true , $type );
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
break ;
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " statuses " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-12-23 23:27:45 +00:00
api_register_func ( 'api/statuses/user_timeline' , 'api_statuses_user_timeline' , true );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Star / unstar an item .
2017-11-10 05:00:50 +00:00
* param : id : id of the item
*
2017-12-24 02:20:50 +00:00
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ see https :// web . archive . org / web / 20131019055350 / https :// dev . twitter . com / docs / api / 1 / post / favorites / create /% 3 Aid
2017-11-10 05:00:50 +00:00
*/
function api_favorites_create_destroy ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// for versioned api.
/// @TODO We need a better global soluton
$action_argv_id = 2 ;
2018-07-01 04:15:11 +00:00
if ( count ( $a -> argv ) > 1 && $a -> argv [ 1 ] == " 1.1 " ) {
2017-11-10 05:00:50 +00:00
$action_argv_id = 3 ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $a -> argc <= $action_argv_id ) {
throw new BadRequestException ( " Invalid request. " );
}
$action = str_replace ( " . " . $type , " " , $a -> argv [ $action_argv_id ]);
if ( $a -> argc == $action_argv_id + 2 ) {
2018-07-01 18:42:38 +00:00
$itemid = intval ( defaults ( $a -> argv , $action_argv_id + 1 , 0 ));
2017-11-10 05:00:50 +00:00
} else {
2018-07-01 18:42:38 +00:00
$itemid = intval ( defaults ( $_REQUEST , 'id' , 0 ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-17 17:05:17 +00:00
$item = Item :: selectFirstForUser ( api_user (), [], [ 'id' => $itemid , 'uid' => api_user ()]);
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $item )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " Invalid item. " );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
switch ( $action ) {
case " create " :
2018-06-09 16:56:37 +00:00
$item [ 'starred' ] = 1 ;
2017-11-10 05:00:50 +00:00
break ;
case " destroy " :
2018-06-09 16:56:37 +00:00
$item [ 'starred' ] = 0 ;
2017-11-10 05:00:50 +00:00
break ;
default :
throw new BadRequestException ( " Invalid action " . $action );
}
2016-09-25 16:50:08 +00:00
2018-06-09 16:56:37 +00:00
$r = Item :: update ([ 'starred' => $item [ 'starred' ]], [ 'id' => $itemid ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $r === false ) {
throw new InternalServerErrorException ( " DB error " );
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$user_info = api_get_user ( $a );
2018-06-09 16:56:37 +00:00
$rets = api_format_items ([ $item ], $user_info , false , $type );
2017-11-10 05:00:50 +00:00
$ret = $rets [ 0 ];
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " status " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/favorites/create' , 'api_favorites_create_destroy' , true , API_METHOD_POST );
api_register_func ( 'api/favorites/destroy' , 'api_favorites_create_destroy' , true , API_METHOD_DELETE );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Returns the most recent favorite statuses .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return string | array
*/
2017-11-10 05:00:50 +00:00
function api_favorites ( $type )
{
global $called_api ;
2016-09-25 16:50:08 +00:00
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2017-11-10 05:00:50 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
}
2018-01-15 13:05:12 +00:00
$called_api = [];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// in friendica starred item are private
// return favorites only for self
2018-10-29 21:20:46 +00:00
Logger :: log ( 'api_favorites: self:' . $user_info [ 'self' ]);
2017-11-10 05:00:50 +00:00
if ( $user_info [ 'self' ] == 0 ) {
2018-01-15 13:05:12 +00:00
$ret = [];
2017-11-10 05:00:50 +00:00
} else {
2016-09-25 16:50:08 +00:00
// params
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$count = defaults ( $_GET , 'count' , 20 );
$page = ( ! empty ( $_REQUEST [ 'page' ]) ? $_REQUEST [ 'page' ] - 1 : 0 );
2017-04-05 20:07:55 +00:00
if ( $page < 0 ) {
$page = 0 ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$start = $page * $count ;
2016-09-25 16:50:08 +00:00
2018-06-27 18:09:33 +00:00
$condition = [ " `uid` = ? AND `gravity` IN (?, ?) AND `id` > ? AND `starred` " ,
api_user (), GRAVITY_PARENT , GRAVITY_COMMENT , $since_id ];
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2017-04-05 20:07:55 +00:00
if ( $max_id > 0 ) {
2018-06-09 16:56:37 +00:00
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2016-09-25 16:50:08 +00:00
2018-06-21 15:14:01 +00:00
$ret = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " statuses " , $type , $data );
}
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/favorites' , 'api_favorites' , true );
2017-12-24 02:20:50 +00:00
/**
*
* @ param array $item
* @ param array $recipient
* @ param array $sender
*
* @ return array
*/
2017-11-10 05:00:50 +00:00
function api_format_messages ( $item , $recipient , $sender )
{
// standard meta information
2018-01-15 13:05:12 +00:00
$ret = [
2018-07-01 18:42:38 +00:00
'id' => $item [ 'id' ],
2018-10-01 17:36:23 +00:00
'sender_id' => $sender [ 'id' ],
2018-07-01 18:42:38 +00:00
'text' => " " ,
'recipient_id' => $recipient [ 'id' ],
'created_at' => api_date ( defaults ( $item , 'created' , DateTimeFormat :: utcNow ())),
'sender_screen_name' => $sender [ 'screen_name' ],
'recipient_screen_name' => $recipient [ 'screen_name' ],
'sender' => $sender ,
'recipient' => $recipient ,
'title' => " " ,
'friendica_seen' => defaults ( $item , 'seen' , 0 ),
'friendica_parent_uri' => defaults ( $item , 'parent-uri' , '' ),
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
// "uid" and "self" are only needed for some internal stuff, so remove it from here
2018-07-01 18:42:38 +00:00
if ( isset ( $ret [ 'sender' ][ 'uid' ])) {
unset ( $ret [ 'sender' ][ 'uid' ]);
}
if ( isset ( $ret [ 'sender' ][ 'self' ])) {
unset ( $ret [ 'sender' ][ 'self' ]);
}
if ( isset ( $ret [ 'recipient' ][ 'uid' ])) {
unset ( $ret [ 'recipient' ][ 'uid' ]);
}
if ( isset ( $ret [ 'recipient' ][ 'self' ])) {
unset ( $ret [ 'recipient' ][ 'self' ]);
}
2017-11-10 05:00:50 +00:00
//don't send title to regular StatusNET requests to avoid confusing these apps
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_GET [ 'getText' ])) {
2017-11-10 05:00:50 +00:00
$ret [ 'title' ] = $item [ 'title' ];
if ( $_GET [ 'getText' ] == 'html' ) {
2018-02-15 02:33:55 +00:00
$ret [ 'text' ] = BBCode :: convert ( $item [ 'body' ], false );
2017-11-10 05:00:50 +00:00
} elseif ( $_GET [ 'getText' ] == 'plain' ) {
2018-03-07 21:29:44 +00:00
$ret [ 'text' ] = trim ( HTML :: toPlaintext ( BBCode :: convert ( api_clean_plain_items ( $item [ 'body' ]), false , 2 , true ), 0 ));
2017-11-10 05:00:50 +00:00
}
} else {
2018-03-07 21:29:44 +00:00
$ret [ 'text' ] = $item [ 'title' ] . " \n " . HTML :: toPlaintext ( BBCode :: convert ( api_clean_plain_items ( $item [ 'body' ]), false , 2 , true ), 0 );
2017-11-10 05:00:50 +00:00
}
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_GET [ 'getUserObjects' ]) && $_GET [ 'getUserObjects' ] == 'false' ) {
2017-11-10 05:00:50 +00:00
unset ( $ret [ 'sender' ]);
unset ( $ret [ 'recipient' ]);
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $ret ;
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param array $item
*
* @ return array
*/
2017-11-10 05:00:50 +00:00
function api_convert_item ( $item )
{
$body = $item [ 'body' ];
$attachments = api_get_attachments ( $body );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Workaround for ostatus messages where the title is identically to the body
2018-02-15 02:33:55 +00:00
$html = BBCode :: convert ( api_clean_plain_items ( $body ), false , 2 , true );
2018-03-07 21:29:44 +00:00
$statusbody = trim ( HTML :: toPlaintext ( $html , 0 ));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// handle data: images
$statusbody = api_format_items_embeded_images ( $item , $statusbody );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$statustitle = trim ( $item [ 'title' ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if (( $statustitle != '' ) && ( strpos ( $statusbody , $statustitle ) !== false )) {
$statustext = trim ( $statusbody );
} else {
$statustext = trim ( $statustitle . " \n \n " . $statusbody );
}
2016-09-25 16:50:08 +00:00
2018-07-01 18:42:38 +00:00
if (( defaults ( $item , 'network' , Protocol :: PHANTOM ) == Protocol :: FEED ) && ( strlen ( $statustext ) > 1000 )) {
$statustext = substr ( $statustext , 0 , 1000 ) . " ... \n " . defaults ( $item , 'plink' , '' );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-02-15 02:33:55 +00:00
$statushtml = BBCode :: convert ( api_clean_attachments ( $body ), false );
2017-11-10 05:00:50 +00:00
// Workaround for clients with limited HTML parser functionality
2018-01-15 13:05:12 +00:00
$search = [ " <br> " , " <blockquote> " , " </blockquote> " ,
2017-11-10 05:00:50 +00:00
" <h1> " , " </h1> " , " <h2> " , " </h2> " ,
" <h3> " , " </h3> " , " <h4> " , " </h4> " ,
2018-01-15 13:05:12 +00:00
" <h5> " , " </h5> " , " <h6> " , " </h6> " ];
$replace = [ " <br> " , " <br><blockquote> " , " </blockquote><br> " ,
2017-11-10 05:00:50 +00:00
" <br><h1> " , " </h1><br> " , " <br><h2> " , " </h2><br> " ,
" <br><h3> " , " </h3><br> " , " <br><h4> " , " </h4><br> " ,
2018-01-15 13:05:12 +00:00
" <br><h5> " , " </h5><br> " , " <br><h6> " , " </h6><br> " ];
2017-11-10 05:00:50 +00:00
$statushtml = str_replace ( $search , $replace , $statushtml );
if ( $item [ 'title' ] != " " ) {
2018-02-15 02:33:55 +00:00
$statushtml = " <br><h4> " . BBCode :: convert ( $item [ 'title' ]) . " </h4><br> " . $statushtml ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
do {
$oldtext = $statushtml ;
$statushtml = str_replace ( " <br><br> " , " <br> " , $statushtml );
} while ( $oldtext != $statushtml );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( substr ( $statushtml , 0 , 4 ) == '<br>' ) {
$statushtml = substr ( $statushtml , 4 );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( substr ( $statushtml , 0 , - 4 ) == '<br>' ) {
$statushtml = substr ( $statushtml , - 4 );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// feeds without body should contain the link
2018-07-01 18:42:38 +00:00
if (( defaults ( $item , 'network' , Protocol :: PHANTOM ) == Protocol :: FEED ) && ( strlen ( $item [ 'body' ]) == 0 )) {
2018-02-15 02:33:55 +00:00
$statushtml .= BBCode :: convert ( $item [ 'plink' ]);
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$entities = api_get_entitities ( $statustext , $body );
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return [
2017-11-10 05:00:50 +00:00
" text " => $statustext ,
" html " => $statushtml ,
" attachments " => $attachments ,
" entities " => $entities
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $body
*
2018-04-24 14:22:29 +00:00
* @ return array
2017-12-24 02:20:50 +00:00
*/
2017-11-10 05:00:50 +00:00
function api_get_attachments ( & $body )
{
$text = $body ;
$text = preg_replace ( " / \ [img \ =([0-9]*)x([0-9]*) \ ](.*?) \ [ \ /img \ ]/ism " , '[img]$3[/img]' , $text );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$URLSearchString = " ^ \ [ \ ] " ;
$ret = preg_match_all ( " / \ [img \ ]([ $URLSearchString ]*) \ [ \ /img \ ]/ism " , $text , $images );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $ret ) {
2018-04-24 14:22:29 +00:00
return [];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$attachments = [];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
foreach ( $images [ 1 ] as $image ) {
2017-12-07 13:56:11 +00:00
$imagedata = Image :: getInfoFromURL ( $image );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $imagedata ) {
2018-01-15 13:05:12 +00:00
$attachments [] = [ " url " => $image , " mimetype " => $imagedata [ " mime " ], " size " => $imagedata [ " size " ]];
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-07-01 18:42:38 +00:00
if ( strstr ( defaults ( $_SERVER , 'HTTP_USER_AGENT' , '' ), " AndStatus " )) {
2017-11-10 05:00:50 +00:00
foreach ( $images [ 0 ] as $orig ) {
$body = str_replace ( $orig , " " , $body );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
return $attachments ;
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $text
* @ param string $bbcode
*
* @ return array
* @ todo Links at the first character of the post
*/
2017-11-10 05:00:50 +00:00
function api_get_entitities ( & $text , $bbcode )
{
2018-11-30 14:06:22 +00:00
$include_entities = strtolower ( defaults ( $_REQUEST , 'include_entities' , " false " ));
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $include_entities != " true " ) {
preg_match_all ( " / \ [img](.*?) \ [ \ /img \ ]/ism " , $bbcode , $images );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
foreach ( $images [ 1 ] as $image ) {
2018-07-31 02:06:22 +00:00
$replace = ProxyUtils :: proxifyUrl ( $image );
2017-11-10 05:00:50 +00:00
$text = str_replace ( $image , $replace , $text );
2016-09-25 16:50:08 +00:00
}
2018-01-15 13:05:12 +00:00
return [];
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2018-02-05 00:23:49 +00:00
$bbcode = BBCode :: cleanPictureLinks ( $bbcode );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Change pure links in text to bbcode uris
$bbcode = preg_replace ( " /([^ \ ] \ =' " . '"' . " ]|^)(https? \ : \ / \ /[a-zA-Z0-9 \ : \ / \ - \ ? \ & \ ; \ . \ = \ _ \ ~ \ # \ % \$ \ ! \ + \ ,]+)/ism " , '$1[url=$2]$2[/url]' , $bbcode );
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$entities = [];
$entities [ " hashtags " ] = [];
$entities [ " symbols " ] = [];
$entities [ " urls " ] = [];
$entities [ " user_mentions " ] = [];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$URLSearchString = " ^ \ [ \ ] " ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$bbcode = preg_replace ( " /# \ [url \ =([ $URLSearchString ]*) \ ](.*?) \ [ \ /url \ ]/ism " , '#$2' , $bbcode );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$bbcode = preg_replace ( " / \ [bookmark \ =([ $URLSearchString ]*) \ ](.*?) \ [ \ /bookmark \ ]/ism " , '[url=$1]$2[/url]' , $bbcode );
//$bbcode = preg_replace("/\[url\](.*?)\[\/url\]/ism",'[url=$1]$1[/url]',$bbcode);
$bbcode = preg_replace ( " / \ [video \ ](.*?) \ [ \ /video \ ]/ism " , '[url=$1]$1[/url]' , $bbcode );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$bbcode = preg_replace (
" / \ [youtube \ ]([A-Za-z0-9 \ -_=]+)(.*?) \ [ \ /youtube \ ]/ism " ,
'[url=https://www.youtube.com/watch?v=$1]https://www.youtube.com/watch?v=$1[/url]' ,
$bbcode
);
$bbcode = preg_replace ( " / \ [youtube \ ](.*?) \ [ \ /youtube \ ]/ism " , '[url=$1]$1[/url]' , $bbcode );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$bbcode = preg_replace (
" / \ [vimeo \ ]([0-9]+)(.*?) \ [ \ /vimeo \ ]/ism " ,
'[url=https://vimeo.com/$1]https://vimeo.com/$1[/url]' ,
$bbcode
);
$bbcode = preg_replace ( " / \ [vimeo \ ](.*?) \ [ \ /vimeo \ ]/ism " , '[url=$1]$1[/url]' , $bbcode );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$bbcode = preg_replace ( " / \ [img \ =([0-9]*)x([0-9]*) \ ](.*?) \ [ \ /img \ ]/ism " , '[img]$3[/img]' , $bbcode );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
//preg_match_all("/\[url\]([$URLSearchString]*)\[\/url\]/ism", $bbcode, $urls1);
preg_match_all ( " / \ [url \ =([ $URLSearchString ]*) \ ](.*?) \ [ \ /url \ ]/ism " , $bbcode , $urls );
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$ordered_urls = [];
2017-11-10 05:00:50 +00:00
foreach ( $urls [ 1 ] as $id => $url ) {
//$start = strpos($text, $url, $offset);
$start = iconv_strpos ( $text , $url , 0 , " UTF-8 " );
if ( ! ( $start === false )) {
2018-01-15 13:05:12 +00:00
$ordered_urls [ $start ] = [ " url " => $url , " title " => $urls [ 2 ][ $id ]];
2017-11-10 05:00:50 +00:00
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
ksort ( $ordered_urls );
$offset = 0 ;
//foreach ($urls[1] AS $id=>$url) {
foreach ( $ordered_urls as $url ) {
if (( substr ( $url [ " title " ], 0 , 7 ) != " http:// " ) && ( substr ( $url [ " title " ], 0 , 8 ) != " https:// " )
&& ! strpos ( $url [ " title " ], " http:// " ) && ! strpos ( $url [ " title " ], " https:// " )
2017-12-22 23:10:32 +00:00
) {
2017-11-10 05:00:50 +00:00
$display_url = $url [ " title " ];
2017-12-22 23:10:32 +00:00
} else {
2018-01-15 13:05:12 +00:00
$display_url = str_replace ([ " http://www. " , " https://www. " ], [ " " , " " ], $url [ " url " ]);
$display_url = str_replace ([ " http:// " , " https:// " ], [ " " , " " ], $display_url );
2016-09-25 16:50:08 +00:00
2017-12-22 23:10:32 +00:00
if ( strlen ( $display_url ) > 26 ) {
2017-11-10 05:00:50 +00:00
$display_url = substr ( $display_url , 0 , 25 ) . " … " ;
2017-12-22 23:10:32 +00:00
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
//$start = strpos($text, $url, $offset);
$start = iconv_strpos ( $text , $url [ " url " ], $offset , " UTF-8 " );
if ( ! ( $start === false )) {
2018-01-15 13:05:12 +00:00
$entities [ " urls " ][] = [ " url " => $url [ " url " ],
2017-11-10 05:00:50 +00:00
" expanded_url " => $url [ " url " ],
" display_url " => $display_url ,
2018-01-15 13:05:12 +00:00
" indices " => [ $start , $start + strlen ( $url [ " url " ])]];
2017-11-10 05:00:50 +00:00
$offset = $start + 1 ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
preg_match_all ( " / \ [img](.*?) \ [ \ /img \ ]/ism " , $bbcode , $images );
2018-01-15 13:05:12 +00:00
$ordered_images = [];
2017-11-10 05:00:50 +00:00
foreach ( $images [ 1 ] as $image ) {
//$start = strpos($text, $url, $offset);
$start = iconv_strpos ( $text , $image , 0 , " UTF-8 " );
2017-12-23 23:27:45 +00:00
if ( ! ( $start === false )) {
2017-11-10 05:00:50 +00:00
$ordered_images [ $start ] = $image ;
2017-12-23 23:27:45 +00:00
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
//$entities["media"] = array();
$offset = 0 ;
foreach ( $ordered_images as $url ) {
2018-01-15 13:05:12 +00:00
$display_url = str_replace ([ " http://www. " , " https://www. " ], [ " " , " " ], $url );
$display_url = str_replace ([ " http:// " , " https:// " ], [ " " , " " ], $display_url );
2017-11-10 05:00:50 +00:00
2017-12-23 23:27:45 +00:00
if ( strlen ( $display_url ) > 26 ) {
2017-11-10 05:00:50 +00:00
$display_url = substr ( $display_url , 0 , 25 ) . " … " ;
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
$start = iconv_strpos ( $text , $url , $offset , " UTF-8 " );
if ( ! ( $start === false )) {
2017-12-07 13:56:11 +00:00
$image = Image :: getInfoFromURL ( $url );
2017-11-10 05:00:50 +00:00
if ( $image ) {
// If image cache is activated, then use the following sizes:
// thumb (150), small (340), medium (600) and large (1024)
if ( ! Config :: get ( " system " , " proxy_disabled " )) {
2018-07-31 02:06:22 +00:00
$media_url = ProxyUtils :: proxifyUrl ( $url );
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$sizes = [];
2017-12-07 13:56:11 +00:00
$scale = Image :: getScalingDimensions ( $image [ 0 ], $image [ 1 ], 150 );
2018-01-15 13:05:12 +00:00
$sizes [ " thumb " ] = [ " w " => $scale [ " width " ], " h " => $scale [ " height " ], " resize " => " fit " ];
2017-11-10 05:00:50 +00:00
if (( $image [ 0 ] > 150 ) || ( $image [ 1 ] > 150 )) {
2017-12-07 13:56:11 +00:00
$scale = Image :: getScalingDimensions ( $image [ 0 ], $image [ 1 ], 340 );
2018-01-15 13:05:12 +00:00
$sizes [ " small " ] = [ " w " => $scale [ " width " ], " h " => $scale [ " height " ], " resize " => " fit " ];
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
2017-12-07 13:56:11 +00:00
$scale = Image :: getScalingDimensions ( $image [ 0 ], $image [ 1 ], 600 );
2018-01-15 13:05:12 +00:00
$sizes [ " medium " ] = [ " w " => $scale [ " width " ], " h " => $scale [ " height " ], " resize " => " fit " ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if (( $image [ 0 ] > 600 ) || ( $image [ 1 ] > 600 )) {
2017-12-07 13:56:11 +00:00
$scale = Image :: getScalingDimensions ( $image [ 0 ], $image [ 1 ], 1024 );
2018-01-15 13:05:12 +00:00
$sizes [ " large " ] = [ " w " => $scale [ " width " ], " h " => $scale [ " height " ], " resize " => " fit " ];
2017-11-10 05:00:50 +00:00
}
} else {
$media_url = $url ;
2018-01-15 13:05:12 +00:00
$sizes [ " medium " ] = [ " w " => $image [ 0 ], " h " => $image [ 1 ], " resize " => " fit " ];
2017-11-10 05:00:50 +00:00
}
2018-01-15 13:05:12 +00:00
$entities [ " media " ][] = [
2017-11-10 05:00:50 +00:00
" id " => $start + 1 ,
" id_str " => ( string ) $start + 1 ,
2018-01-15 13:05:12 +00:00
" indices " => [ $start , $start + strlen ( $url )],
2018-11-08 16:28:29 +00:00
" media_url " => Strings :: normaliseLink ( $media_url ),
2017-11-10 05:00:50 +00:00
" media_url_https " => $media_url ,
" url " => $url ,
" display_url " => $display_url ,
" expanded_url " => $url ,
" type " => " photo " ,
2018-01-15 13:05:12 +00:00
" sizes " => $sizes ];
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$offset = $start + 1 ;
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
return $entities ;
}
2017-12-24 02:20:50 +00:00
/**
*
* @ param array $item
* @ param string $text
*
* @ return string
*/
2018-01-04 01:54:35 +00:00
function api_format_items_embeded_images ( $item , $text )
2017-11-10 05:00:50 +00:00
{
$text = preg_replace_callback (
2018-01-04 01:54:35 +00:00
'|data:image/([^;]+)[^=]+=*|m' ,
function () use ( $item ) {
return System :: baseUrl () . '/display/' . $item [ 'guid' ];
2017-11-10 05:00:50 +00:00
},
$text
);
return $text ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief return < a href = 'url' > name </ a > as array
*
* @ param string $txt text
* @ return array
2017-12-24 02:20:50 +00:00
* 'name' => 'name' ,
2017-11-10 05:00:50 +00:00
* 'url => ' url '
*/
function api_contactlink_to_array ( $txt )
{
2018-01-15 13:05:12 +00:00
$match = [];
2017-11-10 05:00:50 +00:00
$r = preg_match_all ( '|<a href="([^"]*)">([^<]*)</a>|' , $txt , $match );
if ( $r && count ( $match ) == 3 ) {
2018-01-15 13:05:12 +00:00
$res = [
2017-11-10 05:00:50 +00:00
'name' => $match [ 2 ],
'url' => $match [ 1 ]
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
} else {
2018-01-15 13:05:12 +00:00
$res = [
2018-02-11 21:13:29 +00:00
'name' => $txt ,
2017-11-10 05:00:50 +00:00
'url' => " "
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
}
return $res ;
}
2017-09-14 05:16:23 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief return likes , dislikes and attend status for item
*
* @ param array $item array
2017-12-24 02:20:50 +00:00
* @ param string $type Return type ( atom , rss , xml , json )
*
2017-11-10 05:00:50 +00:00
* @ return array
2017-12-24 02:20:50 +00:00
* likes => int count ,
2017-11-10 05:00:50 +00:00
* dislikes => int count
*/
2018-10-01 18:16:19 +00:00
function api_format_items_activities ( $item , $type = " json " )
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$activities = [
'like' => [],
'dislike' => [],
'attendyes' => [],
'attendno' => [],
'attendmaybe' => [],
];
2017-11-10 05:00:50 +00:00
2018-06-18 20:36:34 +00:00
$condition = [ 'uid' => $item [ 'uid' ], 'thr-parent' => $item [ 'uri' ]];
$ret = Item :: selectForUser ( $item [ 'uid' ], [ 'author-id' , 'verb' ], $condition );
2017-11-10 05:00:50 +00:00
2018-10-01 18:16:19 +00:00
while ( $parent_item = Item :: fetch ( $ret )) {
2017-11-10 05:00:50 +00:00
// not used as result should be structured like other user data
//builtin_activity_puller($i, $activities);
// get user data and add it to the array of the activity
2018-10-01 18:16:19 +00:00
$user = api_get_user ( $a , $parent_item [ 'author-id' ]);
switch ( $parent_item [ 'verb' ]) {
2017-11-10 05:00:50 +00:00
case ACTIVITY_LIKE :
$activities [ 'like' ][] = $user ;
break ;
case ACTIVITY_DISLIKE :
$activities [ 'dislike' ][] = $user ;
break ;
case ACTIVITY_ATTEND :
$activities [ 'attendyes' ][] = $user ;
break ;
case ACTIVITY_ATTENDNO :
$activities [ 'attendno' ][] = $user ;
break ;
case ACTIVITY_ATTENDMAYBE :
$activities [ 'attendmaybe' ][] = $user ;
break ;
default :
break ;
2017-09-14 05:16:23 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-09-14 05:16:23 +00:00
2018-07-20 12:19:26 +00:00
DBA :: close ( $ret );
2018-06-18 20:36:34 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
2018-01-15 13:05:12 +00:00
$xml_activities = [];
2017-11-10 05:00:50 +00:00
foreach ( $activities as $k => $v ) {
// change xml element from "like" to "friendica:like"
$xml_activities [ " friendica: " . $k ] = $v ;
// add user data into xml output
$k_user = 0 ;
2017-12-23 23:27:45 +00:00
foreach ( $v as $user ) {
2017-11-10 05:00:50 +00:00
$xml_activities [ " friendica: " . $k ][ $k_user ++. " :user " ] = $user ;
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
}
$activities = $xml_activities ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $activities ;
}
2017-06-08 02:00:59 +00:00
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief return data from profiles
*
2018-01-04 01:54:35 +00:00
* @ param array $profile_row array containing data from db table 'profile'
2017-11-10 05:00:50 +00:00
* @ return array
*/
2018-01-04 01:54:35 +00:00
function api_format_items_profiles ( $profile_row )
2017-11-10 05:00:50 +00:00
{
2018-01-15 13:05:12 +00:00
$profile = [
2018-01-04 01:54:35 +00:00
'profile_id' => $profile_row [ 'id' ],
'profile_name' => $profile_row [ 'profile-name' ],
'is_default' => $profile_row [ 'is-default' ] ? true : false ,
'hide_friends' => $profile_row [ 'hide-friends' ] ? true : false ,
'profile_photo' => $profile_row [ 'photo' ],
'profile_thumb' => $profile_row [ 'thumb' ],
'publish' => $profile_row [ 'publish' ] ? true : false ,
'net_publish' => $profile_row [ 'net-publish' ] ? true : false ,
'description' => $profile_row [ 'pdesc' ],
'date_of_birth' => $profile_row [ 'dob' ],
'address' => $profile_row [ 'address' ],
'city' => $profile_row [ 'locality' ],
'region' => $profile_row [ 'region' ],
'postal_code' => $profile_row [ 'postal-code' ],
'country' => $profile_row [ 'country-name' ],
'hometown' => $profile_row [ 'hometown' ],
'gender' => $profile_row [ 'gender' ],
'marital' => $profile_row [ 'marital' ],
'marital_with' => $profile_row [ 'with' ],
'marital_since' => $profile_row [ 'howlong' ],
'sexual' => $profile_row [ 'sexual' ],
'politic' => $profile_row [ 'politic' ],
'religion' => $profile_row [ 'religion' ],
'public_keywords' => $profile_row [ 'pub_keywords' ],
'private_keywords' => $profile_row [ 'prv_keywords' ],
2018-02-15 02:33:55 +00:00
'likes' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'likes' ]) , false , 2 ),
'dislikes' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'dislikes' ]) , false , 2 ),
'about' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'about' ]) , false , 2 ),
'music' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'music' ]) , false , 2 ),
'book' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'book' ]) , false , 2 ),
'tv' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'tv' ]) , false , 2 ),
'film' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'film' ]) , false , 2 ),
'interest' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'interest' ]) , false , 2 ),
'romance' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'romance' ]) , false , 2 ),
'work' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'work' ]) , false , 2 ),
'education' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'education' ]), false , 2 ),
'social_networks' => BBCode :: convert ( api_clean_plain_items ( $profile_row [ 'contact' ]) , false , 2 ),
2018-01-04 01:54:35 +00:00
'homepage' => $profile_row [ 'homepage' ],
'users' => null
2018-01-15 13:05:12 +00:00
];
2018-01-04 01:54:35 +00:00
return $profile ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief format items to be returned by api
*
2017-12-24 02:20:50 +00:00
* @ param array $r array of items
* @ param array $user_info
* @ param bool $filter_user filter items by $user_info
* @ param string $type Return type ( atom , rss , xml , json )
2017-11-10 05:00:50 +00:00
*/
function api_format_items ( $r , $user_info , $filter_user = false , $type = " json " )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$ret = [];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
foreach ( $r as $item ) {
localize_item ( $item );
list ( $status_user , $owner_user ) = api_item_get_user ( $a , $item );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// Look if the posts are matching if they should be filtered by user id
if ( $filter_user && ( $status_user [ " id " ] != $user_info [ " id " ])) {
continue ;
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$in_reply_to = api_in_reply_to ( $item );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$converted = api_convert_item ( $item );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
$geo = " georss:point " ;
} else {
$geo = " geo " ;
}
2018-01-15 13:05:12 +00:00
$status = [
2017-11-10 05:00:50 +00:00
'text' => $converted [ " text " ],
'truncated' => false ,
'created_at' => api_date ( $item [ 'created' ]),
'in_reply_to_status_id' => $in_reply_to [ 'status_id' ],
'in_reply_to_status_id_str' => $in_reply_to [ 'status_id_str' ],
'source' => (( $item [ 'app' ]) ? $item [ 'app' ] : 'web' ),
'id' => intval ( $item [ 'id' ]),
'id_str' => ( string ) intval ( $item [ 'id' ]),
'in_reply_to_user_id' => $in_reply_to [ 'user_id' ],
'in_reply_to_user_id_str' => $in_reply_to [ 'user_id_str' ],
'in_reply_to_screen_name' => $in_reply_to [ 'screen_name' ],
$geo => null ,
'favorited' => $item [ 'starred' ] ? true : false ,
2018-10-01 17:36:23 +00:00
'user' => $status_user ,
2017-11-10 05:00:50 +00:00
'friendica_owner' => $owner_user ,
2018-08-23 14:09:50 +00:00
'friendica_private' => $item [ 'private' ] == 1 ,
2017-11-10 05:00:50 +00:00
//'entities' => NULL,
2017-12-17 15:16:18 +00:00
'statusnet_html' => $converted [ " html " ],
'statusnet_conversation_id' => $item [ 'parent' ],
'external_url' => System :: baseUrl () . " /display/ " . $item [ 'guid' ],
2017-11-10 05:00:50 +00:00
'friendica_activities' => api_format_items_activities ( $item , $type ),
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
if ( count ( $converted [ " attachments " ]) > 0 ) {
$status [ " attachments " ] = $converted [ " attachments " ];
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
if ( count ( $converted [ " entities " ]) > 0 ) {
$status [ " entities " ] = $converted [ " entities " ];
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2018-06-03 07:22:01 +00:00
if ( $status [ " source " ] == 'web' ) {
2018-10-22 14:15:14 +00:00
$status [ " source " ] = ContactSelector :: networkToName ( $item [ 'network' ], $item [ 'author-link' ]);
} elseif ( ContactSelector :: networkToName ( $item [ 'network' ], $item [ 'author-link' ]) != $status [ " source " ]) {
$status [ " source " ] = trim ( $status [ " source " ] . ' (' . ContactSelector :: networkToName ( $item [ 'network' ], $item [ 'author-link' ]) . ')' );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $item [ " id " ] == $item [ " parent " ]) {
$retweeted_item = api_share_as_retweet ( $item );
if ( $retweeted_item !== false ) {
$retweeted_status = $status ;
2018-11-11 19:05:57 +00:00
$status [ 'user' ] = $status [ 'friendica_owner' ];
2017-11-10 05:00:50 +00:00
try {
2018-06-03 07:22:01 +00:00
$retweeted_status [ " user " ] = api_get_user ( $a , $retweeted_item [ " author-id " ]);
2017-11-10 05:00:50 +00:00
} catch ( BadRequestException $e ) {
// user not found. should be found?
/// @todo check if the user should be always found
2018-01-15 13:05:12 +00:00
$retweeted_status [ " user " ] = [];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$rt_converted = api_convert_item ( $retweeted_item );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$retweeted_status [ 'text' ] = $rt_converted [ " text " ];
$retweeted_status [ 'statusnet_html' ] = $rt_converted [ " html " ];
$retweeted_status [ 'friendica_activities' ] = api_format_items_activities ( $retweeted_item , $type );
$retweeted_status [ 'created_at' ] = api_date ( $retweeted_item [ 'created' ]);
$status [ 'retweeted_status' ] = $retweeted_status ;
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
// "uid" and "self" are only needed for some internal stuff, so remove it from here
unset ( $status [ " user " ][ " uid " ]);
unset ( $status [ " user " ][ " self " ]);
if ( $item [ " coord " ] != " " ) {
$coords = explode ( ' ' , $item [ " coord " ]);
if ( count ( $coords ) == 2 ) {
2017-12-22 23:10:32 +00:00
if ( $type == " json " ) {
2018-01-15 13:05:12 +00:00
$status [ " geo " ] = [ 'type' => 'Point' ,
'coordinates' => [( float ) $coords [ 0 ],
( float ) $coords [ 1 ]]];
2017-12-22 23:10:32 +00:00
} else { // Not sure if this is the official format - if someone founds a documentation we can check
2017-11-10 05:00:50 +00:00
$status [ " georss:point " ] = $item [ " coord " ];
2017-12-22 23:10:32 +00:00
}
2017-11-10 05:00:50 +00:00
}
}
$ret [] = $status ;
};
return $ret ;
}
2017-12-24 02:20:50 +00:00
/**
* Returns the remaining number of API requests available to the user before the API limit is reached .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_account_rate_limit_status ( $type )
{
if ( $type == " xml " ) {
2018-01-15 13:05:12 +00:00
$hash = [
2017-11-10 05:00:50 +00:00
'remaining-hits' => '150' ,
2018-01-15 13:05:12 +00:00
'@attributes' => [ " type " => " integer " ],
2017-11-10 05:00:50 +00:00
'hourly-limit' => '150' ,
2018-01-15 13:05:12 +00:00
'@attributes2' => [ " type " => " integer " ],
2018-01-27 02:38:34 +00:00
'reset-time' => DateTimeFormat :: utc ( 'now + 1 hour' , DateTimeFormat :: ATOM ),
2018-01-15 13:05:12 +00:00
'@attributes3' => [ " type " => " datetime " ],
2017-11-10 05:00:50 +00:00
'reset_time_in_seconds' => strtotime ( 'now + 1 hour' ),
2018-01-15 13:05:12 +00:00
'@attributes4' => [ " type " => " integer " ],
];
2017-11-10 05:00:50 +00:00
} else {
2018-01-15 13:05:12 +00:00
$hash = [
2017-11-10 05:00:50 +00:00
'reset_time_in_seconds' => strtotime ( 'now + 1 hour' ),
'remaining_hits' => '150' ,
'hourly_limit' => '150' ,
2018-01-27 02:38:34 +00:00
'reset_time' => api_date ( DateTimeFormat :: utc ( 'now + 1 hour' , DateTimeFormat :: ATOM )),
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( 'hash' , $type , [ 'hash' => $hash ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/account/rate_limit_status' , 'api_account_rate_limit_status' , true );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Returns the string " ok " in the requested format with a 200 OK HTTP status code .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_help_test ( $type )
{
if ( $type == 'xml' ) {
$ok = " true " ;
} else {
$ok = " ok " ;
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( 'ok' , $type , [ " ok " => $ok ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/help/test' , 'api_help_test' , false );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
2018-04-07 13:54:26 +00:00
* Returns all lists the user subscribes to .
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
2018-04-07 13:54:26 +00:00
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / create - manage - lists / api - reference / get - lists - list
2017-12-24 02:20:50 +00:00
*/
2018-04-07 13:54:26 +00:00
function api_lists_list ( $type )
2017-11-10 05:00:50 +00:00
{
2018-01-15 13:05:12 +00:00
$ret = [];
2017-11-10 05:00:50 +00:00
/// @TODO $ret is not filled here?
2018-01-15 13:05:12 +00:00
return api_format_data ( 'lists' , $type , [ " lists_list " => $ret ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2018-04-07 13:54:26 +00:00
api_register_func ( 'api/lists/list' , 'api_lists_list' , true );
api_register_func ( 'api/lists/subscriptions' , 'api_lists_list' , true );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
2018-04-07 13:54:26 +00:00
* Returns all groups the user owns .
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
2018-04-07 13:54:26 +00:00
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / create - manage - lists / api - reference / get - lists - ownerships
2017-12-24 02:20:50 +00:00
*/
2018-04-07 13:54:26 +00:00
function api_lists_ownerships ( $type )
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-07 13:54:26 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// params
$user_info = api_get_user ( $a );
$uid = $user_info [ 'uid' ];
2018-07-20 12:19:26 +00:00
$groups = DBA :: select ( 'group' , [], [ 'deleted' => 0 , 'uid' => $uid ]);
2018-04-07 13:54:26 +00:00
// loop through all groups
2018-04-07 17:55:41 +00:00
$lists = [];
foreach ( $groups as $group ) {
if ( $group [ 'visible' ]) {
2018-04-07 13:54:26 +00:00
$mode = 'public' ;
} else {
$mode = 'private' ;
}
2018-04-07 17:55:41 +00:00
$lists [] = [
'name' => $group [ 'name' ],
'id' => intval ( $group [ 'id' ]),
'id_str' => ( string ) $group [ 'id' ],
2018-04-07 13:54:26 +00:00
'user' => $user_info ,
'mode' => $mode
];
}
2018-04-07 17:55:41 +00:00
return api_format_data ( " lists " , $type , [ 'lists' => [ 'lists' => $lists ]]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2018-04-07 13:54:26 +00:00
api_register_func ( 'api/lists/ownerships' , 'api_lists_ownerships' , true );
/**
* Returns recent statuses from users in the specified group .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / create - manage - lists / api - reference / get - lists - ownerships
*/
function api_lists_statuses ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-07 13:54:26 +00:00
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
if ( api_user () === false || $user_info === false ) {
2018-04-07 13:54:26 +00:00
throw new ForbiddenException ();
}
unset ( $_REQUEST [ " user_id " ]);
unset ( $_GET [ " user_id " ]);
unset ( $_REQUEST [ " screen_name " ]);
unset ( $_GET [ " screen_name " ]);
2018-04-07 22:55:05 +00:00
if ( empty ( $_REQUEST [ 'list_id' ])) {
2018-04-07 20:36:54 +00:00
throw new BadRequestException ( 'list_id not specified' );
}
2018-04-07 13:54:26 +00:00
// params
2018-11-30 14:06:22 +00:00
$count = defaults ( $_REQUEST , 'count' , 20 );
$page = ( ! empty ( $_REQUEST [ 'page' ]) ? $_REQUEST [ 'page' ] - 1 : 0 );
2018-04-07 13:54:26 +00:00
if ( $page < 0 ) {
$page = 0 ;
}
2018-11-30 14:06:22 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
$exclude_replies = ( ! empty ( $_REQUEST [ 'exclude_replies' ]) ? 1 : 0 );
$conversation_id = defaults ( $_REQUEST , 'conversation_id' , 0 );
2018-04-07 13:54:26 +00:00
$start = $page * $count ;
2018-06-27 18:09:33 +00:00
$condition = [ " `uid` = ? AND `gravity` IN (?, ?) AND `id` > ? AND `group_member`.`gid` = ? " ,
api_user (), GRAVITY_PARENT , GRAVITY_COMMENT , $since_id , $_REQUEST [ 'list_id' ]];
2018-04-07 13:54:26 +00:00
2018-06-09 19:12:13 +00:00
if ( $max_id > 0 ) {
$condition [ 0 ] .= " AND `item`.`id` <= ? " ;
$condition [] = $max_id ;
}
if ( $exclude_replies > 0 ) {
$condition [ 0 ] .= ' AND `item`.`parent` = `item`.`id`' ;
}
if ( $conversation_id > 0 ) {
$condition [ 0 ] .= " AND `item`.`parent` = ? " ;
$condition [] = $conversation_id ;
}
2018-06-09 16:56:37 +00:00
$params = [ 'order' => [ 'id' => true ], 'limit' => [ $start , $count ]];
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition , $params );
2018-04-07 13:54:26 +00:00
2018-06-21 15:14:01 +00:00
$items = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2018-04-07 13:54:26 +00:00
2018-04-07 17:55:41 +00:00
$data = [ 'status' => $items ];
2018-04-07 13:54:26 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2018-04-07 13:54:26 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
break ;
}
return api_format_data ( " statuses " , $type , $data );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/lists/statuses' , 'api_lists_statuses' , true );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Considers friends and followers lists to be private and won ' t return
2017-12-16 15:16:25 +00:00
* anything if any user_id parameter is passed .
*
2017-12-24 02:20:50 +00:00
* @ brief Returns either the friends of the follower list
*
2017-12-16 15:16:25 +00:00
* @ param string $qtype Either " friends " or " followers "
* @ return boolean | array
* @ throws ForbiddenException
2017-11-10 05:00:50 +00:00
*/
2017-12-16 15:16:25 +00:00
function api_statuses_f ( $qtype )
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-12-16 15:16:25 +00:00
// pagination
2018-11-30 14:06:22 +00:00
$count = defaults ( $_GET , 'count' , 20 );
$page = defaults ( $_GET , 'page' , 1 );
2017-12-16 15:16:25 +00:00
if ( $page < 1 ) {
$page = 1 ;
}
$start = ( $page - 1 ) * $count ;
2017-11-10 05:00:50 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_GET [ 'cursor' ]) && $_GET [ 'cursor' ] == 'undefined' ) {
2017-11-10 05:00:50 +00:00
/* this is to stop Hotot to load friends multiple times
* I 'm not sure if I' m missing return something or
* is a bug in hotot . Workaround , meantime
*/
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/* $ret = Array ();
return array ( '$users' => $ret ); */
return false ;
}
2016-09-25 16:50:08 +00:00
2017-12-23 00:30:50 +00:00
$sql_extra = '' ;
2017-11-10 05:00:50 +00:00
if ( $qtype == 'friends' ) {
2018-07-25 02:53:46 +00:00
$sql_extra = sprintf ( " AND ( `rel` = %d OR `rel` = %d ) " , intval ( Contact :: SHARING ), intval ( Contact :: FRIEND ));
2017-12-23 00:30:50 +00:00
} elseif ( $qtype == 'followers' ) {
2018-07-25 02:53:46 +00:00
$sql_extra = sprintf ( " AND ( `rel` = %d OR `rel` = %d ) " , intval ( Contact :: FOLLOWER ), intval ( Contact :: FRIEND ));
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// friends and followers only for self
if ( $user_info [ 'self' ] == 0 ) {
$sql_extra = " AND false " ;
}
2016-09-25 16:50:08 +00:00
2017-12-18 14:18:31 +00:00
if ( $qtype == 'blocks' ) {
2017-12-18 22:51:03 +00:00
$sql_filter = 'AND `blocked` AND NOT `pending`' ;
} elseif ( $qtype == 'incoming' ) {
$sql_filter = 'AND `pending`' ;
2017-12-18 14:18:31 +00:00
} else {
2017-12-18 22:51:03 +00:00
$sql_filter = 'AND (NOT `blocked` OR `pending`)' ;
2017-12-18 14:18:31 +00:00
}
2017-11-10 05:00:50 +00:00
$r = q (
2017-12-16 15:16:25 +00:00
" SELECT `nurl`
FROM `contact`
WHERE `uid` = % d
AND NOT `self`
2017-12-18 22:51:03 +00:00
$sql_filter
2017-12-16 15:16:25 +00:00
$sql_extra
ORDER BY `nick`
LIMIT % d , % d " ,
intval ( api_user ()),
intval ( $start ),
intval ( $count )
2017-11-10 05:00:50 +00:00
);
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$ret = [];
2017-11-10 05:00:50 +00:00
foreach ( $r as $cid ) {
$user = api_get_user ( $a , $cid [ 'nurl' ]);
// "uid" and "self" are only needed for some internal stuff, so remove it from here
unset ( $user [ " uid " ]);
unset ( $user [ " self " ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $user ) {
$ret [] = $user ;
2016-09-25 16:50:08 +00:00
}
}
2018-01-15 13:05:12 +00:00
return [ 'user' => $ret ];
2017-11-10 05:00:50 +00:00
}
2017-01-16 20:59:16 +00:00
2017-12-16 15:16:25 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns the user ' s friends .
*
2017-12-16 15:16:25 +00:00
* @ brief Returns the list of friends of the provided user
*
* @ deprecated By Twitter API in favor of friends / list
*
* @ param string $type Either " json " or " xml "
* @ return boolean | string | array
*/
2017-11-10 05:00:50 +00:00
function api_statuses_friends ( $type )
{
2017-12-16 15:16:25 +00:00
$data = api_statuses_f ( " friends " );
2017-11-10 05:00:50 +00:00
if ( $data === false ) {
return false ;
}
return api_format_data ( " users " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-12-16 15:16:25 +00:00
/**
2017-12-24 02:20:50 +00:00
* Returns the user ' s followers .
*
* @ brief Returns the list of followers of the provided user
2017-12-16 15:16:25 +00:00
*
* @ deprecated By Twitter API in favor of friends / list
*
* @ param string $type Either " json " or " xml "
* @ return boolean | string | array
*/
2017-11-10 05:00:50 +00:00
function api_statuses_followers ( $type )
{
2017-12-16 15:16:25 +00:00
$data = api_statuses_f ( " followers " );
2017-11-10 05:00:50 +00:00
if ( $data === false ) {
return false ;
}
return api_format_data ( " users " , $type , $data );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/statuses/friends' , 'api_statuses_friends' , true );
api_register_func ( 'api/statuses/followers' , 'api_statuses_followers' , true );
2017-12-18 14:18:31 +00:00
/**
* Returns the list of blocked users
*
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / mute - block - report - users / api - reference / get - blocks - list
*
* @ param string $type Either " json " or " xml "
*
* @ return boolean | string | array
*/
function api_blocks_list ( $type )
{
$data = api_statuses_f ( 'blocks' );
if ( $data === false ) {
return false ;
}
return api_format_data ( " users " , $type , $data );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/blocks/list' , 'api_blocks_list' , true );
2017-12-18 22:51:03 +00:00
/**
* Returns the list of pending users IDs
*
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / get - friendships - incoming
*
* @ param string $type Either " json " or " xml "
*
* @ return boolean | string | array
*/
function api_friendships_incoming ( $type )
{
$data = api_statuses_f ( 'incoming' );
if ( $data === false ) {
return false ;
}
2018-01-15 13:05:12 +00:00
$ids = [];
2017-12-18 22:51:03 +00:00
foreach ( $data [ 'user' ] as $user ) {
$ids [] = $user [ 'id' ];
}
2018-01-15 13:05:12 +00:00
return api_format_data ( " ids " , $type , [ 'id' => $ids ]);
2017-12-18 22:51:03 +00:00
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/friendships/incoming' , 'api_friendships_incoming' , true );
2017-12-24 02:20:50 +00:00
/**
* Returns the instance ' s configuration information .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_statusnet_config ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
2018-07-01 18:42:58 +00:00
$name = Config :: get ( 'config' , 'sitename' );
2018-10-09 17:58:58 +00:00
$server = $a -> getHostName ();
2018-07-01 18:42:58 +00:00
$logo = System :: baseUrl () . '/images/friendica-64.png' ;
$email = Config :: get ( 'config' , 'admin_email' );
2018-07-15 19:04:48 +00:00
$closed = intval ( Config :: get ( 'config' , 'register_policy' )) === REGISTER_CLOSED ? 'true' : 'false' ;
2018-07-01 18:42:58 +00:00
$private = Config :: get ( 'system' , 'block_public' ) ? 'true' : 'false' ;
$textlimit = ( string ) Config :: get ( 'config' , 'api_import_size' , Config :: get ( 'config' , 'max_import_size' , 200000 ));
$ssl = Config :: get ( 'system' , 'have_ssl' ) ? 'true' : 'false' ;
$sslserver = Config :: get ( 'system' , 'have_ssl' ) ? str_replace ( 'http:' , 'https:' , System :: baseUrl ()) : '' ;
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$config = [
'site' => [ 'name' => $name , 'server' => $server , 'theme' => 'default' , 'path' => '' ,
2017-11-10 05:00:50 +00:00
'logo' => $logo , 'fancy' => true , 'language' => 'en' , 'email' => $email , 'broughtby' => '' ,
'broughtbyurl' => '' , 'timezone' => 'UTC' , 'closed' => $closed , 'inviteonly' => false ,
'private' => $private , 'textlimit' => $textlimit , 'sslserver' => $sslserver , 'ssl' => $ssl ,
'shorturllength' => '30' ,
2018-01-15 13:05:12 +00:00
'friendica' => [
2017-11-10 05:00:50 +00:00
'FRIENDICA_PLATFORM' => FRIENDICA_PLATFORM ,
'FRIENDICA_VERSION' => FRIENDICA_VERSION ,
'DFRN_PROTOCOL_VERSION' => DFRN_PROTOCOL_VERSION ,
'DB_UPDATE_VERSION' => DB_UPDATE_VERSION
2018-01-15 13:05:12 +00:00
]
],
];
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( 'config' , $type , [ 'config' => $config ]);
2017-11-10 05:00:50 +00:00
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/gnusocial/config' , 'api_statusnet_config' , false );
api_register_func ( 'api/statusnet/config' , 'api_statusnet_config' , false );
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_statusnet_version ( $type )
{
// liar
$fake_statusnet_version = " 0.9.7 " ;
2018-01-15 13:05:12 +00:00
return api_format_data ( 'version' , $type , [ 'version' => $fake_statusnet_version ]);
2017-11-10 05:00:50 +00:00
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/gnusocial/version' , 'api_statusnet_version' , false );
api_register_func ( 'api/statusnet/version' , 'api_statusnet_version' , false );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
*
2017-11-10 05:00:50 +00:00
* @ todo use api_format_data () to return data
*/
2018-01-04 01:54:35 +00:00
function api_ff_ids ( $type )
2017-11-10 05:00:50 +00:00
{
2018-10-01 17:36:23 +00:00
if ( ! api_user ()) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
2016-09-25 16:50:08 +00:00
}
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-02-11 21:13:29 +00:00
2018-01-04 18:26:09 +00:00
api_get_user ( $a );
2018-01-04 01:54:35 +00:00
$stringify_ids = defaults ( $_REQUEST , 'stringify_ids' , false );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$r = q (
" SELECT `pcontact`.`id` FROM `contact`
INNER JOIN `contact` AS `pcontact` ON `contact` . `nurl` = `pcontact` . `nurl` AND `pcontact` . `uid` = 0
WHERE `contact` . `uid` = % s AND NOT `contact` . `self` " ,
intval ( api_user ())
);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
return ;
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$ids = [];
2017-11-10 05:00:50 +00:00
foreach ( $r as $rr ) {
if ( $stringify_ids ) {
$ids [] = $rr [ 'id' ];
} else {
$ids [] = intval ( $rr [ 'id' ]);
}
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( " ids " , $type , [ 'id' => $ids ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Returns the ID of every user the user is following .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / get - friends - ids
*/
2017-11-10 05:00:50 +00:00
function api_friends_ids ( $type )
{
2018-01-04 01:54:35 +00:00
return api_ff_ids ( $type );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Returns the ID of every user following the user .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / get - followers - ids
*/
2017-11-10 05:00:50 +00:00
function api_followers_ids ( $type )
{
2018-01-04 01:54:35 +00:00
return api_ff_ids ( $type );
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/friends/ids' , 'api_friends_ids' , true );
api_register_func ( 'api/followers/ids' , 'api_followers_ids' , true );
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Sends a new direct message .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / direct - messages / sending - and - receiving / api - reference / new - message
*/
2017-11-10 05:00:50 +00:00
function api_direct_messages_new ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2018-07-01 18:42:38 +00:00
if ( empty ( $_POST [ " text " ]) || empty ( $_POST [ " screen_name " ]) && empty ( $_POST [ " user_id " ])) {
2017-12-23 23:27:45 +00:00
return ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$sender = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-07-01 18:42:38 +00:00
$recipient = null ;
if ( ! empty ( $_POST [ 'screen_name' ])) {
2017-11-10 05:00:50 +00:00
$r = q (
" SELECT `id`, `nurl`, `network` FROM `contact` WHERE `uid`=%d AND `nick`='%s' " ,
intval ( api_user ()),
2018-07-21 13:10:13 +00:00
DBA :: escape ( $_POST [ 'screen_name' ])
2017-11-10 05:00:50 +00:00
);
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2018-07-01 18:43:22 +00:00
// Selecting the id by priority, friendica first
api_best_nickname ( $r );
2016-09-25 16:50:08 +00:00
2018-07-01 18:43:22 +00:00
$recipient = api_get_user ( $a , $r [ 0 ][ 'nurl' ]);
}
2017-11-10 05:00:50 +00:00
} else {
$recipient = api_get_user ( $a , $_POST [ 'user_id' ]);
2016-09-25 16:50:08 +00:00
}
2018-07-01 18:43:22 +00:00
if ( empty ( $recipient )) {
throw new NotFoundException ( 'Recipient not found' );
}
2017-11-10 05:00:50 +00:00
$replyto = '' ;
$sub = '' ;
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_REQUEST [ 'replyto' ])) {
2017-11-10 05:00:50 +00:00
$r = q (
'SELECT `parent-uri`, `title` FROM `mail` WHERE `uid`=%d AND `id`=%d' ,
intval ( api_user ()),
intval ( $_REQUEST [ 'replyto' ])
);
$replyto = $r [ 0 ][ 'parent-uri' ];
$sub = $r [ 0 ][ 'title' ];
} else {
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_REQUEST [ 'title' ])) {
2017-11-10 05:00:50 +00:00
$sub = $_REQUEST [ 'title' ];
2017-04-05 20:07:55 +00:00
} else {
2017-11-10 05:00:50 +00:00
$sub = (( strlen ( $_POST [ 'text' ]) > 10 ) ? substr ( $_POST [ 'text' ], 0 , 10 ) . " ... " : $_POST [ 'text' ]);
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2018-01-15 17:14:09 +00:00
$id = Mail :: send ( $recipient [ 'cid' ], $_POST [ 'text' ], $sub , $replyto );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $id > - 1 ) {
$r = q ( " SELECT * FROM `mail` WHERE id=%d " , intval ( $id ));
$ret = api_format_messages ( $r [ 0 ], $recipient , $sender );
} else {
2018-01-15 13:05:12 +00:00
$ret = [ " error " => $id ];
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'direct_message' => $ret ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
2018-02-11 21:13:29 +00:00
$data = api_rss_extra ( $a , $data , $sender );
2018-11-13 22:23:16 +00:00
break ;
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " direct-messages " , $type , $data );
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/direct_messages/new' , 'api_direct_messages_new' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Destroys a direct message .
*
2017-11-10 05:00:50 +00:00
* @ brief delete a direct_message from mail table through api
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-12-24 02:20:50 +00:00
* @ see https :// developer . twitter . com / en / docs / direct - messages / sending - and - receiving / api - reference / delete - message
2017-11-10 05:00:50 +00:00
*/
function api_direct_messages_destroy ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
//required
2018-11-30 14:06:22 +00:00
$id = defaults ( $_REQUEST , 'id' , 0 );
2017-11-10 05:00:50 +00:00
// optional
2018-11-30 14:06:22 +00:00
$parenturi = defaults ( $_REQUEST , 'friendica_parenturi' , " " );
$verbose = ( ! empty ( $_GET [ 'friendica_verbose' ]) ? strtolower ( $_GET [ 'friendica_verbose' ]) : " false " );
2017-11-10 05:00:50 +00:00
/// @todo optional parameter 'include_entities' from Twitter API not yet implemented
$uid = $user_info [ 'uid' ];
// error if no id or parenturi specified (for clients posting parent-uri as well)
if ( $verbose == " true " && ( $id == 0 || $parenturi == " " )) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'message id or parenturi not specified' ];
return api_format_data ( " direct_messages_delete " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// BadRequestException if no id specified (for clients using Twitter API)
if ( $id == 0 ) {
throw new BadRequestException ( 'Message id not specified' );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// add parent-uri to sql command if specified by calling app
2018-07-21 13:10:13 +00:00
$sql_extra = ( $parenturi != " " ? " AND `parent-uri` = ' " . DBA :: escape ( $parenturi ) . " ' " : " " );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// get data of the specified message id
$r = q (
" SELECT `id` FROM `mail` WHERE `uid` = %d AND `id` = %d " . $sql_extra ,
intval ( $uid ),
intval ( $id )
);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// error message if specified id is not in database
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
if ( $verbose == " true " ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'message id not in database' ];
return api_format_data ( " direct_messages_delete " , $type , [ '$result' => $answer ]);
2017-04-05 20:17:15 +00:00
}
2017-11-10 05:00:50 +00:00
/// @todo BadRequestException ok for Twitter API clients?
throw new BadRequestException ( 'message id not in database' );
}
// delete message
$result = q (
" DELETE FROM `mail` WHERE `uid` = %d AND `id` = %d " . $sql_extra ,
intval ( $uid ),
intval ( $id )
);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $verbose == " true " ) {
if ( $result ) {
// return success
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'ok' , 'message' => 'message deleted' ];
return api_format_data ( " direct_message_delete " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
} else {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'unknown error' ];
return api_format_data ( " direct_messages_delete " , $type , [ '$result' => $answer ]);
2017-04-05 20:17:15 +00:00
}
2017-11-10 05:00:50 +00:00
}
/// @todo return JSON data like Twitter API not yet implemented
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/direct_messages/destroy' , 'api_direct_messages_destroy' , true , API_METHOD_DELETE );
2016-09-25 16:50:08 +00:00
2018-09-14 17:35:24 +00:00
/**
* Unfollow Contact
*
2018-11-30 15:48:47 +00:00
* @ brief unfollow contact
2018-09-14 17:35:24 +00:00
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
* @ return string | array
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / follow - search - get - users / api - reference / post - friendships - destroy . html
*/
2018-09-13 21:23:53 +00:00
function api_friendships_destroy ( $type )
{
2018-09-15 09:06:55 +00:00
$uid = api_user ();
2018-09-14 07:28:14 +00:00
2018-09-15 09:06:55 +00:00
if ( $uid === false ) {
2018-09-14 07:28:14 +00:00
throw new ForbiddenException ();
}
2018-09-13 21:23:53 +00:00
2018-09-14 07:28:14 +00:00
$contact_id = defaults ( $_REQUEST , 'user_id' );
2018-09-13 21:23:53 +00:00
2018-09-15 09:06:55 +00:00
if ( empty ( $contact_id )) {
2018-10-30 13:58:45 +00:00
Logger :: log ( " No user_id specified " , Logger :: DEBUG );
2018-09-14 07:28:14 +00:00
throw new BadRequestException ( " no user_id specified " );
}
2018-09-13 21:23:53 +00:00
2018-09-13 21:32:26 +00:00
// Get Contact by given id
2018-09-14 07:28:14 +00:00
$contact = DBA :: selectFirst ( 'contact' , [ 'url' ], [ 'id' => $contact_id , 'uid' => 0 , 'self' => false ]);
2018-09-13 21:23:53 +00:00
if ( ! DBA :: isResult ( $contact )) {
2018-10-30 13:58:45 +00:00
Logger :: log ( " No contact found for ID " . $contact_id , Logger :: DEBUG );
2018-09-14 17:35:24 +00:00
throw new NotFoundException ( " no contact found to given ID " );
2018-09-13 21:23:53 +00:00
}
2018-09-14 07:28:14 +00:00
$url = $contact [ " url " ];
2018-09-13 21:23:53 +00:00
$condition = [ " `uid` = ? AND (`rel` = ? OR `rel` = ?) AND (`nurl` = ? OR `alias` = ? OR `alias` = ?) " ,
2018-11-08 16:28:29 +00:00
$uid , Contact :: SHARING , Contact :: FRIEND , Strings :: normaliseLink ( $url ),
Strings :: normaliseLink ( $url ), $url ];
2018-09-13 21:23:53 +00:00
$contact = DBA :: selectFirst ( 'contact' , [], $condition );
if ( ! DBA :: isResult ( $contact )) {
2018-10-30 13:58:45 +00:00
Logger :: log ( " Not following Contact " , Logger :: DEBUG );
2018-09-14 17:35:24 +00:00
throw new NotFoundException ( " Not following Contact " );
2018-09-13 21:23:53 +00:00
}
if ( ! in_array ( $contact [ 'network' ], Protocol :: NATIVE_SUPPORT )) {
2018-10-30 13:58:45 +00:00
Logger :: log ( " Not supported " , Logger :: DEBUG );
2018-09-14 07:28:14 +00:00
throw new ExpectationFailedException ( " Not supported " );
2018-09-13 21:23:53 +00:00
}
$dissolve = ( $contact [ 'rel' ] == Contact :: SHARING );
$owner = User :: getOwnerDataById ( $uid );
if ( $owner ) {
Contact :: terminateFriendship ( $owner , $contact , $dissolve );
}
2018-09-14 07:28:14 +00:00
else {
2018-10-30 13:58:45 +00:00
Logger :: log ( " No owner found " , Logger :: DEBUG );
2018-09-14 17:35:24 +00:00
throw new NotFoundException ( " Error Processing Request " );
2018-09-14 07:28:14 +00:00
}
2018-09-13 21:23:53 +00:00
// Sharing-only contacts get deleted as there no relationship any more
if ( $dissolve ) {
Contact :: remove ( $contact [ 'id' ]);
} else {
DBA :: update ( 'contact' , [ 'rel' => Contact :: FOLLOWER ], [ 'id' => $contact [ 'id' ]]);
}
2018-09-16 17:36:25 +00:00
// "uid" and "self" are only needed for some internal stuff, so remove it from here
unset ( $contact [ " uid " ]);
unset ( $contact [ " self " ]);
// Set screen_name since Twidere requests it
$contact [ " screen_name " ] = $contact [ " nick " ];
return api_format_data ( " friendships-destroy " , $type , [ 'user' => $contact ]);
2018-09-13 21:23:53 +00:00
}
api_register_func ( 'api/friendships/destroy' , 'api_friendships_destroy' , true , API_METHOD_POST );
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type Return type ( atom , rss , xml , json )
* @ param string $box
* @ param string $verbose
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_direct_messages_box ( $type , $box , $verbose )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-06-18 09:55:02 +00:00
if ( api_user () === false ) {
2017-11-10 05:00:50 +00:00
throw new ForbiddenException ();
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// params
2018-07-01 18:42:38 +00:00
$count = defaults ( $_GET , 'count' , 20 );
$page = defaults ( $_REQUEST , 'page' , 1 ) - 1 ;
2017-11-10 05:00:50 +00:00
if ( $page < 0 ) {
$page = 0 ;
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:17:15 +00:00
2018-07-01 18:42:38 +00:00
$since_id = defaults ( $_REQUEST , 'since_id' , 0 );
$max_id = defaults ( $_REQUEST , 'max_id' , 0 );
2017-04-05 20:17:15 +00:00
2018-07-01 18:42:38 +00:00
$user_id = defaults ( $_REQUEST , 'user_id' , '' );
$screen_name = defaults ( $_REQUEST , 'screen_name' , '' );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// caller user info
unset ( $_REQUEST [ " user_id " ]);
unset ( $_GET [ " user_id " ]);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
unset ( $_REQUEST [ " screen_name " ]);
unset ( $_GET [ " screen_name " ]);
2016-09-25 16:50:08 +00:00
2018-06-17 17:57:32 +00:00
$user_info = api_get_user ( $a );
2018-06-18 09:55:02 +00:00
if ( $user_info === false ) {
2018-06-17 17:57:32 +00:00
throw new ForbiddenException ();
}
2017-11-10 05:00:50 +00:00
$profile_url = $user_info [ " url " ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// pagination
$start = $page * $count ;
2016-09-25 16:50:08 +00:00
2018-02-11 21:13:29 +00:00
$sql_extra = " " ;
2017-11-10 05:00:50 +00:00
// filters
if ( $box == " sentbox " ) {
2018-07-21 13:10:13 +00:00
$sql_extra = " `mail`.`from-url`=' " . DBA :: escape ( $profile_url ) . " ' " ;
2017-11-10 05:00:50 +00:00
} elseif ( $box == " conversation " ) {
2018-07-21 13:10:13 +00:00
$sql_extra = " `mail`.`parent-uri`=' " . DBA :: escape ( defaults ( $_GET , 'uri' , '' )) . " ' " ;
2017-11-10 05:00:50 +00:00
} elseif ( $box == " all " ) {
$sql_extra = " true " ;
} elseif ( $box == " inbox " ) {
2018-07-21 13:10:13 +00:00
$sql_extra = " `mail`.`from-url`!=' " . DBA :: escape ( $profile_url ) . " ' " ;
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:17:15 +00:00
2017-11-10 05:00:50 +00:00
if ( $max_id > 0 ) {
$sql_extra .= ' AND `mail`.`id` <= ' . intval ( $max_id );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $user_id != " " ) {
$sql_extra .= ' AND `mail`.`contact-id` = ' . intval ( $user_id );
} elseif ( $screen_name != " " ) {
2018-07-21 13:10:13 +00:00
$sql_extra .= " AND `contact`.`nick` = ' " . DBA :: escape ( $screen_name ) . " ' " ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$r = q (
" SELECT `mail`.*, `contact`.`nurl` AS `contact-url` FROM `mail`,`contact` WHERE `mail`.`contact-id` = `contact`.`id` AND `mail`.`uid`=%d AND $sql_extra AND `mail`.`id` > %d ORDER BY `mail`.`id` DESC LIMIT %d,%d " ,
intval ( api_user ()),
intval ( $since_id ),
intval ( $start ),
intval ( $count )
);
2018-07-21 12:46:04 +00:00
if ( $verbose == " true " && ! DBA :: isResult ( $r )) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'no mails available' ];
return api_format_data ( " direct_messages_all " , $type , [ '$result' => $answer ]);
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:17:15 +00:00
2018-01-15 13:05:12 +00:00
$ret = [];
2017-11-10 05:00:50 +00:00
foreach ( $r as $item ) {
if ( $box == " inbox " || $item [ 'from-url' ] != $profile_url ) {
$recipient = $user_info ;
2018-11-08 16:28:29 +00:00
$sender = api_get_user ( $a , Strings :: normaliseLink ( $item [ 'contact-url' ]));
2017-11-10 05:00:50 +00:00
} elseif ( $box == " sentbox " || $item [ 'from-url' ] == $profile_url ) {
2018-11-08 16:28:29 +00:00
$recipient = api_get_user ( $a , Strings :: normaliseLink ( $item [ 'contact-url' ]));
2017-11-10 05:00:50 +00:00
$sender = $user_info ;
}
2016-09-25 16:50:08 +00:00
2018-04-09 19:34:53 +00:00
if ( isset ( $recipient ) && isset ( $sender )) {
$ret [] = api_format_messages ( $item , $recipient , $sender );
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$data = [ 'direct_message' => $ret ];
2017-11-10 05:00:50 +00:00
switch ( $type ) {
case " atom " :
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
case " rss " :
$data = api_rss_extra ( $a , $data , $user_info );
2018-11-13 22:23:16 +00:00
break ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " direct-messages " , $type , $data );
}
2017-12-24 02:20:50 +00:00
/**
* Returns the most recent direct messages sent by the user .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / direct - messages / sending - and - receiving / api - reference / get - sent - message
*/
2017-11-10 05:00:50 +00:00
function api_direct_messages_sentbox ( $type )
{
2018-11-30 14:06:22 +00:00
$verbose = ! empty ( $_GET [ 'friendica_verbose' ]) ? strtolower ( $_GET [ 'friendica_verbose' ]) : " false " ;
2017-11-10 05:00:50 +00:00
return api_direct_messages_box ( $type , " sentbox " , $verbose );
}
2017-12-24 02:20:50 +00:00
/**
* Returns the most recent direct messages sent to the user .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / direct - messages / sending - and - receiving / api - reference / get - messages
*/
2017-11-10 05:00:50 +00:00
function api_direct_messages_inbox ( $type )
{
2018-11-30 14:06:22 +00:00
$verbose = ! empty ( $_GET [ 'friendica_verbose' ]) ? strtolower ( $_GET [ 'friendica_verbose' ]) : " false " ;
2017-11-10 05:00:50 +00:00
return api_direct_messages_box ( $type , " inbox " , $verbose );
}
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_direct_messages_all ( $type )
{
2018-11-30 14:06:22 +00:00
$verbose = ! empty ( $_GET [ 'friendica_verbose' ]) ? strtolower ( $_GET [ 'friendica_verbose' ]) : " false " ;
2017-11-10 05:00:50 +00:00
return api_direct_messages_box ( $type , " all " , $verbose );
}
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_direct_messages_conversation ( $type )
{
2018-11-30 14:06:22 +00:00
$verbose = ! empty ( $_GET [ 'friendica_verbose' ]) ? strtolower ( $_GET [ 'friendica_verbose' ]) : " false " ;
2017-11-10 05:00:50 +00:00
return api_direct_messages_box ( $type , " conversation " , $verbose );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/direct_messages/conversation' , 'api_direct_messages_conversation' , true );
api_register_func ( 'api/direct_messages/all' , 'api_direct_messages_all' , true );
api_register_func ( 'api/direct_messages/sent' , 'api_direct_messages_sentbox' , true );
api_register_func ( 'api/direct_messages' , 'api_direct_messages_inbox' , true );
2017-12-24 02:20:50 +00:00
/**
* Returns an OAuth Request Token .
*
* @ see https :// oauth . net / core / 1.0 / #auth_step1
*/
2018-01-04 01:54:35 +00:00
function api_oauth_request_token ()
2017-11-10 05:00:50 +00:00
{
2017-12-05 02:10:54 +00:00
$oauth1 = new FKOAuth1 ();
2017-11-10 05:00:50 +00:00
try {
2017-12-05 02:10:54 +00:00
$r = $oauth1 -> fetch_request_token ( OAuthRequest :: from_request ());
2017-11-10 05:00:50 +00:00
} catch ( Exception $e ) {
echo " error= " . OAuthUtil :: urlencode_rfc3986 ( $e -> getMessage ());
killme ();
}
echo $r ;
killme ();
}
2017-12-24 02:20:50 +00:00
/**
* Returns an OAuth Access Token .
*
* @ return array | string
* @ see https :// oauth . net / core / 1.0 / #auth_step3
*/
2018-01-04 14:51:05 +00:00
function api_oauth_access_token ()
2017-11-10 05:00:50 +00:00
{
2017-12-05 02:10:54 +00:00
$oauth1 = new FKOAuth1 ();
2017-11-10 05:00:50 +00:00
try {
2017-12-05 02:10:54 +00:00
$r = $oauth1 -> fetch_access_token ( OAuthRequest :: from_request ());
2017-11-10 05:00:50 +00:00
} catch ( Exception $e ) {
echo " error= " . OAuthUtil :: urlencode_rfc3986 ( $e -> getMessage ());
killme ();
}
echo $r ;
killme ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/oauth/request_token' , 'api_oauth_request_token' , false );
api_register_func ( 'api/oauth/access_token' , 'api_oauth_access_token' , false );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief delete a complete photoalbum with all containing photos from database through api
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_fr_photoalbum_delete ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// input params
2018-11-30 14:06:22 +00:00
$album = defaults ( $_REQUEST , 'album' , " " );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// we do not allow calls without album string
if ( $album == " " ) {
throw new BadRequestException ( " no albumname specified " );
}
// check if album is existing
$r = q (
" SELECT DISTINCT `resource-id` FROM `photo` WHERE `uid` = %d AND `album` = '%s' " ,
intval ( api_user ()),
2018-07-21 13:10:13 +00:00
DBA :: escape ( $album )
2017-11-10 05:00:50 +00:00
);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " album not available " );
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
// function for setting the items to "deleted = 1" which ensures that comments, likes etc. are not shown anymore
// to the user and the contacts of the users (drop_items() performs the federation of the deletion to other networks
foreach ( $r as $rr ) {
2018-06-18 20:36:34 +00:00
$condition = [ 'uid' => local_user (), 'resource-id' => $rr [ 'resource-id' ], 'type' => 'photo' ];
$photo_item = Item :: selectFirstForUser ( local_user (), [ 'id' ], $condition );
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $photo_item )) {
2017-11-10 05:00:50 +00:00
throw new InternalServerErrorException ( " problem with deleting items occured " );
2017-04-19 21:03:44 +00:00
}
2018-06-18 20:36:34 +00:00
Item :: deleteForUser ([ 'id' => $photo_item [ 'id' ]], api_user ());
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// now let's delete all photos from the album
2018-12-11 19:03:29 +00:00
$result = Photo :: delete ([ 'uid' => api_user (), 'album' => $album ]);
2017-11-10 05:00:50 +00:00
// return success of deletion or error message
if ( $result ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'deleted' , 'message' => 'album `' . $album . '` with all containing photos has been deleted.' ];
return api_format_data ( " photoalbum_delete " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
} else {
throw new InternalServerErrorException ( " unknown error - deleting from database failed " );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-04-05 20:17:15 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief update the name of the album for all photos of an album
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_fr_photoalbum_update ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// input params
2018-11-30 14:06:22 +00:00
$album = defaults ( $_REQUEST , 'album' , " " );
$album_new = defaults ( $_REQUEST , 'album_new' , " " );
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// we do not allow calls without album string
if ( $album == " " ) {
throw new BadRequestException ( " no albumname specified " );
}
if ( $album_new == " " ) {
throw new BadRequestException ( " no new albumname specified " );
}
// check if album is existing
2018-12-11 19:03:29 +00:00
if ( ! Photo :: exists ( null , [ 'uid' => api_user (), 'album' => $album ])) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " album not available " );
}
// now let's update all photos to the albumname
2018-12-11 19:03:29 +00:00
$result = Photo :: update ([ 'album' => $album_new ], [ 'uid' => api_user (), 'album' => $album ]);
2017-11-10 05:00:50 +00:00
// return success of updating or error message
if ( $result ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'updated' , 'message' => 'album `' . $album . '` with all containing photos has been renamed to `' . $album_new . '`.' ];
return api_format_data ( " photoalbum_update " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
} else {
throw new InternalServerErrorException ( " unknown error - updating in database failed " );
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief list all photos of the authenticated user
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_fr_photos_list ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
}
$r = q (
" SELECT `resource-id`, MAX(scale) AS `scale`, `album`, `filename`, `type`, MAX(`created`) AS `created`,
MAX ( `edited` ) AS `edited` , MAX ( `desc` ) AS `desc` FROM `photo`
WHERE `uid` = % d AND `album` != 'Contact Photos' GROUP BY `resource-id` " ,
intval ( local_user ())
);
2018-01-15 13:05:12 +00:00
$typetoext = [
2017-11-10 05:00:50 +00:00
'image/jpeg' => 'jpg' ,
'image/png' => 'png' ,
'image/gif' => 'gif'
2018-01-15 13:05:12 +00:00
];
$data = [ 'photo' => []];
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
foreach ( $r as $rr ) {
2018-01-15 13:05:12 +00:00
$photo = [];
2017-11-10 05:00:50 +00:00
$photo [ 'id' ] = $rr [ 'resource-id' ];
$photo [ 'album' ] = $rr [ 'album' ];
$photo [ 'filename' ] = $rr [ 'filename' ];
$photo [ 'type' ] = $rr [ 'type' ];
$thumb = System :: baseUrl () . " /photo/ " . $rr [ 'resource-id' ] . " - " . $rr [ 'scale' ] . " . " . $typetoext [ $rr [ 'type' ]];
$photo [ 'created' ] = $rr [ 'created' ];
$photo [ 'edited' ] = $rr [ 'edited' ];
$photo [ 'desc' ] = $rr [ 'desc' ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
2018-01-15 13:05:12 +00:00
$data [ 'photo' ][] = [ " @attributes " => $photo , " 1 " => $thumb ];
2017-04-05 20:07:55 +00:00
} else {
2017-11-10 05:00:50 +00:00
$photo [ 'thumb' ] = $thumb ;
$data [ 'photo' ][] = $photo ;
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
}
return api_format_data ( " photos " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief upload a new photo or change an existing photo
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_fr_photo_create_update ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// input params
2018-11-30 14:06:22 +00:00
$photo_id = defaults ( $_REQUEST , 'photo_id' , null );
$desc = defaults ( $_REQUEST , 'desc' , ( array_key_exists ( 'desc' , $_REQUEST ) ? " " : null )) ; // extra check necessary to distinguish between 'not provided' and 'empty string'
$album = defaults ( $_REQUEST , 'album' , null );
$album_new = defaults ( $_REQUEST , 'album_new' , null );
$allow_cid = defaults ( $_REQUEST , 'allow_cid' , ( array_key_exists ( 'allow_cid' , $_REQUEST ) ? " " : null ));
$deny_cid = defaults ( $_REQUEST , 'deny_cid' , ( array_key_exists ( 'deny_cid' , $_REQUEST ) ? " " : null ));
$allow_gid = defaults ( $_REQUEST , 'allow_gid' , ( array_key_exists ( 'allow_gid' , $_REQUEST ) ? " " : null ));
$deny_gid = defaults ( $_REQUEST , 'deny_gid' , ( array_key_exists ( 'deny_gid' , $_REQUEST ) ? " " : null ));
2018-11-30 15:48:47 +00:00
$visibility = ! empty ( $_REQUEST [ 'visibility' ]) && $_REQUEST [ 'visibility' ] !== " false " ;
2017-11-10 05:00:50 +00:00
// do several checks on input parameters
// we do not allow calls without album string
if ( $album == null ) {
throw new BadRequestException ( " no albumname specified " );
}
// if photo_id == null --> we are uploading a new photo
if ( $photo_id == null ) {
$mode = " create " ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// error if no media posted in create-mode
2018-11-30 14:06:22 +00:00
if ( empty ( $_FILES [ 'media' ])) {
2017-11-10 05:00:50 +00:00
// Output error
throw new BadRequestException ( " no media data submitted " );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// album_new will be ignored in create-mode
$album_new = " " ;
} else {
$mode = " update " ;
2016-09-25 16:50:08 +00:00
2018-12-11 19:03:29 +00:00
// check if photo is existing in databasei
if ( ! Photo :: exists ( $photo_id , [ 'uid' => api_user (), 'album' => $album ]) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " photo not available " );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// checks on acl strings provided by clients
$acl_input_error = false ;
$acl_input_error |= check_acl_input ( $allow_cid );
$acl_input_error |= check_acl_input ( $deny_cid );
$acl_input_error |= check_acl_input ( $allow_gid );
$acl_input_error |= check_acl_input ( $deny_gid );
if ( $acl_input_error ) {
throw new BadRequestException ( " acl data invalid " );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// now let's upload the new media in create-mode
if ( $mode == " create " ) {
$media = $_FILES [ 'media' ];
$data = save_media_to_database ( " photo " , $media , $type , $album , trim ( $allow_cid ), trim ( $deny_cid ), trim ( $allow_gid ), trim ( $deny_gid ), $desc , $visibility );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// return success of updating or error message
if ( ! is_null ( $data )) {
return api_format_data ( " photo_create " , $type , $data );
} else {
throw new InternalServerErrorException ( " unknown error - uploading photo failed, see Friendica log for more information " );
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// now let's do the changes in update-mode
if ( $mode == " update " ) {
2018-12-11 19:03:29 +00:00
$updated_fields = [];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $desc )) {
2018-12-11 19:03:29 +00:00
$updated_fields [ 'desc' ] = $desc ;
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $album_new )) {
2018-12-11 19:03:29 +00:00
$updated_fields [ 'album' ] = $album_new ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $allow_cid )) {
$allow_cid = trim ( $allow_cid );
2018-12-11 19:03:29 +00:00
$updated_fields [ 'allow_cid' ] = $allow_cid ;
2017-04-05 20:07:55 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $deny_cid )) {
$deny_cid = trim ( $deny_cid );
2018-12-11 19:03:29 +00:00
$updated_fields [ 'deny_cid' ] = $deny_cid ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $allow_gid )) {
$allow_gid = trim ( $allow_gid );
2018-12-11 19:03:29 +00:00
$updated_fields [ 'allow_gid' ] = $allow_gid ;
2017-11-10 05:00:50 +00:00
}
2017-01-09 12:09:01 +00:00
2017-11-10 05:00:50 +00:00
if ( ! is_null ( $deny_gid )) {
$deny_gid = trim ( $deny_gid );
2018-12-11 19:03:29 +00:00
$updated_fields [ 'deny_gid' ] = $deny_gid ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
$result = false ;
2018-12-11 19:03:29 +00:00
if ( count ( $updated_fields ) > 0 ) {
2017-11-10 05:00:50 +00:00
$nothingtodo = false ;
2018-12-11 19:03:29 +00:00
$result = Photo :: update ( $updated_fields , [ 'uid' => api_user (), 'resource-id' => $photo_id , 'album' => $album ]);
2017-11-10 05:00:50 +00:00
} else {
$nothingtodo = true ;
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_FILES [ 'media' ])) {
2017-11-10 05:00:50 +00:00
$nothingtodo = false ;
$media = $_FILES [ 'media' ];
$data = save_media_to_database ( " photo " , $media , $type , $album , $allow_cid , $deny_cid , $allow_gid , $deny_gid , $desc , 0 , $visibility , $photo_id );
if ( ! is_null ( $data )) {
return api_format_data ( " photo_update " , $type , $data );
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
// return success of updating or error message
if ( $result ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'updated' , 'message' => 'Image id `' . $photo_id . '` has been updated.' ];
return api_format_data ( " photo_update " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
} else {
if ( $nothingtodo ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'cancelled' , 'message' => 'Nothing to update for image id `' . $photo_id . '`.' ];
return api_format_data ( " photo_update " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
}
throw new InternalServerErrorException ( " unknown error - update photo entry in database failed " );
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
throw new InternalServerErrorException ( " unknown error - this error on uploading or updating a photo should never happen " );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief delete a single photo from the database through api
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_fr_photo_delete ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// input params
2018-11-30 14:06:22 +00:00
$photo_id = defaults ( $_REQUEST , 'photo_id' , null );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// do several checks on input parameters
// we do not allow calls without photo id
if ( $photo_id == null ) {
throw new BadRequestException ( " no photo_id specified " );
}
// check if photo is existing in database
2018-12-11 19:03:29 +00:00
$r = Photo :: exists ( $photo_id , [ 'uid' => api_user ()]);
if ( ! $r ) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " photo not available " );
}
// now we can perform on the deletion of the photo
2018-12-11 19:03:29 +00:00
$result = Photo :: delete ([ 'uid' => api_user (), 'resource-id' => $photo_id ]);
2017-11-10 05:00:50 +00:00
// return success of deletion or error message
if ( $result ) {
// retrieve the id of the parent element (the photo element)
2018-06-18 20:36:34 +00:00
$condition = [ 'uid' => local_user (), 'resource-id' => $photo_id , 'type' => 'photo' ];
$photo_item = Item :: selectFirstForUser ( local_user (), [ 'id' ], $condition );
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $photo_item )) {
2017-11-10 05:00:50 +00:00
throw new InternalServerErrorException ( " problem with deleting items occured " );
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
// function for setting the items to "deleted = 1" which ensures that comments, likes etc. are not shown anymore
// to the user and the contacts of the users (drop_items() do all the necessary magic to avoid orphans in database and federate deletion)
2018-06-18 20:36:34 +00:00
Item :: deleteForUser ([ 'id' => $photo_item [ 'id' ]], api_user ());
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'deleted' , 'message' => 'photo with id `' . $photo_id . '` has been deleted from server.' ];
return api_format_data ( " photo_delete " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
} else {
throw new InternalServerErrorException ( " unknown error on deleting photo from database table " );
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief returns the details of a specified photo id , if scale is given , returns the photo data in base 64
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2017-12-25 20:14:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_fr_photo_detail ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2018-11-30 14:06:22 +00:00
if ( empty ( $_REQUEST [ 'photo_id' ])) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " No photo id. " );
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
$scale = ( ! empty ( $_REQUEST [ 'scale' ]) ? intval ( $_REQUEST [ 'scale' ]) : false );
2017-11-10 05:00:50 +00:00
$photo_id = $_REQUEST [ 'photo_id' ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// prepare json/xml output with data from database for the requested photo
$data = prepare_photo_data ( $type , $scale , $photo_id );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return api_format_data ( " photo_detail " , $type , $data );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
2017-12-24 02:20:50 +00:00
* Updates the user’ s profile image .
*
2017-11-10 05:00:50 +00:00
* @ brief updates the profile image for the user ( either a specified profile or the default profile )
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2017-12-24 02:20:50 +00:00
*
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-12-24 02:20:50 +00:00
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / manage - account - settings / api - reference / post - account - update_profile_image
2017-11-10 05:00:50 +00:00
*/
function api_account_update_profile_image ( $type )
{
if ( api_user () === false ) {
throw new ForbiddenException ();
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// input params
2018-05-04 21:43:29 +00:00
$profile_id = defaults ( $_REQUEST , 'profile_id' , 0 );
2017-04-05 20:17:15 +00:00
2017-11-10 05:00:50 +00:00
// error if image data is missing
2018-11-30 14:06:22 +00:00
if ( empty ( $_FILES [ 'image' ])) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " no media data submitted " );
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// check if specified profile id is valid
2018-05-04 21:43:29 +00:00
if ( $profile_id != 0 ) {
2018-07-20 12:19:26 +00:00
$profile = DBA :: selectFirst ( 'profile' , [ 'is-default' ], [ 'uid' => api_user (), 'id' => $profile_id ]);
2017-11-10 05:00:50 +00:00
// error message if specified profile id is not in database
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $profile )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " profile_id not available " );
2016-09-25 16:50:08 +00:00
}
2018-05-04 06:33:36 +00:00
$is_default_profile = $profile [ 'is-default' ];
2017-11-10 05:00:50 +00:00
} else {
$is_default_profile = 1 ;
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// get mediadata from image or media (Twitter call api/account/update_profile_image provides image)
$media = null ;
2018-11-30 14:06:22 +00:00
if ( ! empty ( $_FILES [ 'image' ])) {
2017-11-10 05:00:50 +00:00
$media = $_FILES [ 'image' ];
2018-11-30 14:06:22 +00:00
} elseif ( ! empty ( $_FILES [ 'media' ])) {
2017-11-10 05:00:50 +00:00
$media = $_FILES [ 'media' ];
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// save new profile image
2018-01-21 16:38:01 +00:00
$data = save_media_to_database ( " profileimage " , $media , $type , L10n :: t ( 'Profile Photos' ), " " , " " , " " , " " , " " , $is_default_profile );
2017-11-10 05:00:50 +00:00
// get filetype
if ( is_array ( $media [ 'type' ])) {
$filetype = $media [ 'type' ][ 0 ];
} else {
$filetype = $media [ 'type' ];
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
if ( $filetype == " image/jpeg " ) {
$fileext = " jpg " ;
} elseif ( $filetype == " image/png " ) {
$fileext = " png " ;
2018-04-09 19:34:53 +00:00
} else {
throw new InternalServerErrorException ( 'Unsupported filetype' );
2017-05-01 20:19:48 +00:00
}
2018-05-13 14:46:58 +00:00
2017-11-10 05:00:50 +00:00
// change specified profile or all profiles to the new resource-id
if ( $is_default_profile ) {
2018-05-04 21:43:29 +00:00
$condition = [ " `profile` AND `resource-id` != ? AND `uid` = ? " , $data [ 'photo' ][ 'id' ], api_user ()];
2018-12-11 19:03:29 +00:00
Photo :: update ([ 'profile' => false ], $condition );
2017-11-10 05:00:50 +00:00
} else {
2018-05-04 21:43:29 +00:00
$fields = [ 'photo' => System :: baseUrl () . '/photo/' . $data [ 'photo' ][ 'id' ] . '-4.' . $filetype ,
'thumb' => System :: baseUrl () . '/photo/' . $data [ 'photo' ][ 'id' ] . '-5.' . $filetype ];
2018-07-20 12:19:26 +00:00
DBA :: update ( 'profile' , $fields , [ 'id' => $_REQUEST [ 'profile' ], 'uid' => api_user ()]);
2016-09-25 16:50:08 +00:00
}
2018-05-04 06:33:36 +00:00
Contact :: updateSelfFromUserID ( api_user (), true );
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
// Update global directory in background
2018-12-28 00:22:35 +00:00
$url = System :: baseUrl () . '/profile/' . \get_app () -> user [ 'nickname' ];
2017-11-10 05:00:50 +00:00
if ( $url && strlen ( Config :: get ( 'system' , 'directory' ))) {
2017-11-18 07:59:30 +00:00
Worker :: add ( PRIORITY_LOW , " Directory " , $url );
2017-05-01 20:19:48 +00:00
}
2017-11-19 16:59:37 +00:00
Worker :: add ( PRIORITY_LOW , 'ProfileUpdate' , api_user ());
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
// output for client
if ( $data ) {
return api_account_verify_credentials ( $type );
} else {
// SaveMediaToDatabase failed for some reason
throw new InternalServerErrorException ( " image upload failed " );
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
}
// place api-register for photoalbum calls before 'api/friendica/photo', otherwise this function is never reached
api_register_func ( 'api/friendica/photoalbum/delete' , 'api_fr_photoalbum_delete' , true , API_METHOD_DELETE );
api_register_func ( 'api/friendica/photoalbum/update' , 'api_fr_photoalbum_update' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/photos/list' , 'api_fr_photos_list' , true );
api_register_func ( 'api/friendica/photo/create' , 'api_fr_photo_create_update' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/photo/update' , 'api_fr_photo_create_update' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/photo/delete' , 'api_fr_photo_delete' , true , API_METHOD_DELETE );
api_register_func ( 'api/friendica/photo' , 'api_fr_photo_detail' , true );
api_register_func ( 'api/account/update_profile_image' , 'api_account_update_profile_image' , true , API_METHOD_POST );
2017-12-22 23:46:01 +00:00
/**
* Update user profile
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
*
* @ return array | string
*/
function api_account_update_profile ( $type )
{
2017-12-26 00:05:00 +00:00
$local_user = api_user ();
2017-12-22 23:46:01 +00:00
$api_user = api_get_user ( get_app ());
2017-12-26 00:05:00 +00:00
if ( ! empty ( $_POST [ 'name' ])) {
2018-07-20 12:19:26 +00:00
DBA :: update ( 'profile' , [ 'name' => $_POST [ 'name' ]], [ 'uid' => $local_user ]);
DBA :: update ( 'user' , [ 'username' => $_POST [ 'name' ]], [ 'uid' => $local_user ]);
DBA :: update ( 'contact' , [ 'name' => $_POST [ 'name' ]], [ 'uid' => $local_user , 'self' => 1 ]);
DBA :: update ( 'contact' , [ 'name' => $_POST [ 'name' ]], [ 'id' => $api_user [ 'id' ]]);
2017-12-22 23:46:01 +00:00
}
2017-12-26 00:05:00 +00:00
if ( isset ( $_POST [ 'description' ])) {
2018-07-20 12:19:26 +00:00
DBA :: update ( 'profile' , [ 'about' => $_POST [ 'description' ]], [ 'uid' => $local_user ]);
DBA :: update ( 'contact' , [ 'about' => $_POST [ 'description' ]], [ 'uid' => $local_user , 'self' => 1 ]);
DBA :: update ( 'contact' , [ 'about' => $_POST [ 'description' ]], [ 'id' => $api_user [ 'id' ]]);
2017-12-22 23:46:01 +00:00
}
2017-12-26 20:49:43 +00:00
Worker :: add ( PRIORITY_LOW , 'ProfileUpdate' , $local_user );
2017-12-26 00:08:51 +00:00
// Update global directory in background
2017-12-26 20:49:43 +00:00
if ( $api_user [ 'url' ] && strlen ( Config :: get ( 'system' , 'directory' ))) {
Worker :: add ( PRIORITY_LOW , " Directory " , $api_user [ 'url' ]);
2017-12-26 00:08:51 +00:00
}
2017-12-22 23:46:01 +00:00
return api_account_verify_credentials ( $type );
}
/// @TODO move to top of file or somewhere better
api_register_func ( 'api/account/update_profile' , 'api_account_update_profile' , true , API_METHOD_POST );
2017-11-10 05:00:50 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $acl_string
*/
2017-11-10 05:00:50 +00:00
function check_acl_input ( $acl_string )
{
if ( $acl_string == null || $acl_string == " " ) {
return false ;
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
$contact_not_found = false ;
// split <x><y><z> into array of cid's
preg_match_all ( " /<[A-Za-z0-9]+>/ " , $acl_string , $array );
// check for each cid if it is available on server
$cid_array = $array [ 0 ];
foreach ( $cid_array as $cid ) {
$cid = str_replace ( " < " , " " , $cid );
$cid = str_replace ( " > " , " " , $cid );
2018-08-19 12:46:11 +00:00
$condition = [ 'id' => $cid , 'uid' => api_user ()];
$contact_not_found |= ! DBA :: exists ( 'contact' , $condition );
2017-11-10 05:00:50 +00:00
}
return $contact_not_found ;
}
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $mediatype
* @ param array $media
* @ param string $type
* @ param string $album
* @ param string $allow_cid
* @ param string $deny_cid
* @ param string $allow_gid
* @ param string $deny_gid
* @ param string $desc
* @ param integer $profile
* @ param boolean $visibility
* @ param string $photo_id
*/
2017-11-10 05:00:50 +00:00
function save_media_to_database ( $mediatype , $media , $type , $album , $allow_cid , $deny_cid , $allow_gid , $deny_gid , $desc , $profile = 0 , $visibility = false , $photo_id = null )
{
$visitor = 0 ;
$src = " " ;
$filetype = " " ;
$filename = " " ;
$filesize = 0 ;
if ( is_array ( $media )) {
if ( is_array ( $media [ 'tmp_name' ])) {
$src = $media [ 'tmp_name' ][ 0 ];
} else {
$src = $media [ 'tmp_name' ];
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
if ( is_array ( $media [ 'name' ])) {
$filename = basename ( $media [ 'name' ][ 0 ]);
2017-05-01 20:19:48 +00:00
} else {
2017-11-10 05:00:50 +00:00
$filename = basename ( $media [ 'name' ]);
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
if ( is_array ( $media [ 'size' ])) {
$filesize = intval ( $media [ 'size' ][ 0 ]);
} else {
$filesize = intval ( $media [ 'size' ]);
2017-05-01 20:19:48 +00:00
}
if ( is_array ( $media [ 'type' ])) {
$filetype = $media [ 'type' ][ 0 ];
} else {
$filetype = $media [ 'type' ];
}
}
2017-11-10 05:00:50 +00:00
if ( $filetype == " " ) {
2017-12-07 13:56:11 +00:00
$filetype = Image :: guessType ( $filename );
2017-11-10 05:00:50 +00:00
}
2018-09-02 08:01:13 +00:00
$imagedata = @ getimagesize ( $src );
2017-11-10 05:00:50 +00:00
if ( $imagedata ) {
$filetype = $imagedata [ 'mime' ];
}
2018-10-29 21:20:46 +00:00
Logger :: log (
2017-11-10 05:00:50 +00:00
" File upload src: " . $src . " - filename: " . $filename .
2017-12-23 23:27:45 +00:00
" - size: " . $filesize . " - type: " . $filetype ,
2018-10-30 13:58:45 +00:00
Logger :: DEBUG
2017-11-10 05:00:50 +00:00
);
// check if there was a php upload error
if ( $filesize == 0 && $media [ 'error' ] == 1 ) {
throw new InternalServerErrorException ( " image size exceeds PHP config settings, file was rejected by server " );
}
// check against max upload size within Friendica instance
$maximagesize = Config :: get ( 'system' , 'maximagesize' );
2017-12-22 23:10:32 +00:00
if ( $maximagesize && ( $filesize > $maximagesize )) {
2018-11-08 15:28:49 +00:00
$formattedBytes = Strings :: formatBytes ( $maximagesize );
2017-11-10 05:00:50 +00:00
throw new InternalServerErrorException ( " image size exceeds Friendica config setting (uploaded size: $formattedBytes ) " );
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
// create Photo instance with the data of the image
$imagedata = @ file_get_contents ( $src );
2017-12-07 13:56:11 +00:00
$Image = new Image ( $imagedata , $filetype );
2018-10-01 17:36:23 +00:00
if ( ! $Image -> isValid ()) {
2017-11-10 05:00:50 +00:00
throw new InternalServerErrorException ( " unable to process image data " );
}
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
// check orientation of image
2017-12-07 13:56:11 +00:00
$Image -> orient ( $src );
2017-11-10 05:00:50 +00:00
@ unlink ( $src );
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
// check max length of images on server
$max_length = Config :: get ( 'system' , 'max_image_length' );
2018-10-01 17:36:23 +00:00
if ( ! $max_length ) {
2017-11-10 05:00:50 +00:00
$max_length = MAX_IMAGE_LENGTH ;
}
if ( $max_length > 0 ) {
2017-12-07 13:56:11 +00:00
$Image -> scaleDown ( $max_length );
2018-10-30 13:58:45 +00:00
Logger :: log ( " File upload: Scaling picture to new size " . $max_length , Logger :: DEBUG );
2017-11-10 05:00:50 +00:00
}
2017-12-07 13:56:11 +00:00
$width = $Image -> getWidth ();
$height = $Image -> getHeight ();
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
// create a new resource-id if not already provided
2018-02-20 10:02:07 +00:00
$hash = ( $photo_id == null ) ? Photo :: newResource () : $photo_id ;
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
if ( $mediatype == " photo " ) {
// upload normal image (scales 0, 1, 2)
2018-10-30 13:58:45 +00:00
Logger :: log ( " photo upload: starting new photo upload " , Logger :: DEBUG );
2017-05-01 20:19:48 +00:00
2017-12-07 13:56:11 +00:00
$r = Photo :: store ( $Image , local_user (), $visitor , $hash , $filename , $album , 0 , 0 , $allow_cid , $allow_gid , $deny_cid , $deny_gid , $desc );
2018-10-01 17:36:23 +00:00
if ( ! $r ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( " photo upload: image upload with scale 0 (original size) failed " );
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
if ( $width > 640 || $height > 640 ) {
2017-12-07 13:56:11 +00:00
$Image -> scaleDown ( 640 );
$r = Photo :: store ( $Image , local_user (), $visitor , $hash , $filename , $album , 1 , 0 , $allow_cid , $allow_gid , $deny_cid , $deny_gid , $desc );
2018-10-01 17:36:23 +00:00
if ( ! $r ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( " photo upload: image upload with scale 1 (640x640) failed " );
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
if ( $width > 320 || $height > 320 ) {
2017-12-07 13:56:11 +00:00
$Image -> scaleDown ( 320 );
$r = Photo :: store ( $Image , local_user (), $visitor , $hash , $filename , $album , 2 , 0 , $allow_cid , $allow_gid , $deny_cid , $deny_gid , $desc );
2018-10-01 17:36:23 +00:00
if ( ! $r ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( " photo upload: image upload with scale 2 (320x320) failed " );
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
}
2018-10-30 13:58:45 +00:00
Logger :: log ( " photo upload: new photo upload ended " , Logger :: DEBUG );
2017-11-10 05:00:50 +00:00
} elseif ( $mediatype == " profileimage " ) {
// upload profile image (scales 4, 5, 6)
2018-10-30 13:58:45 +00:00
Logger :: log ( " photo upload: starting new profile image upload " , Logger :: DEBUG );
2017-05-01 20:19:48 +00:00
2018-10-23 14:36:57 +00:00
if ( $width > 300 || $height > 300 ) {
$Image -> scaleDown ( 300 );
2017-12-07 13:56:11 +00:00
$r = Photo :: store ( $Image , local_user (), $visitor , $hash , $filename , $album , 4 , $profile , $allow_cid , $allow_gid , $deny_cid , $deny_gid , $desc );
2018-10-01 17:36:23 +00:00
if ( ! $r ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( " photo upload: profile image upload with scale 4 (300x300) failed " );
2017-05-01 20:19:48 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-05-01 20:19:48 +00:00
2017-11-10 05:00:50 +00:00
if ( $width > 80 || $height > 80 ) {
2017-12-07 13:56:11 +00:00
$Image -> scaleDown ( 80 );
$r = Photo :: store ( $Image , local_user (), $visitor , $hash , $filename , $album , 5 , $profile , $allow_cid , $allow_gid , $deny_cid , $deny_gid , $desc );
2018-10-01 17:36:23 +00:00
if ( ! $r ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( " photo upload: profile image upload with scale 5 (80x80) failed " );
2017-05-01 20:19:48 +00:00
}
}
2017-11-10 05:00:50 +00:00
if ( $width > 48 || $height > 48 ) {
2017-12-07 13:56:11 +00:00
$Image -> scaleDown ( 48 );
$r = Photo :: store ( $Image , local_user (), $visitor , $hash , $filename , $album , 6 , $profile , $allow_cid , $allow_gid , $deny_cid , $deny_gid , $desc );
2018-10-01 17:36:23 +00:00
if ( ! $r ) {
2018-10-29 21:20:46 +00:00
Logger :: log ( " photo upload: profile image upload with scale 6 (48x48) failed " );
2017-05-01 20:19:48 +00:00
}
}
2017-12-07 13:56:11 +00:00
$Image -> __destruct ();
2018-10-30 13:58:45 +00:00
Logger :: log ( " photo upload: new profile image upload ended " , Logger :: DEBUG );
2017-05-01 20:19:48 +00:00
}
2018-04-09 19:34:53 +00:00
if ( isset ( $r ) && $r ) {
2017-11-10 05:00:50 +00:00
// create entry in 'item'-table on new uploads to enable users to comment/like/dislike the photo
if ( $photo_id == null && $mediatype == " photo " ) {
post_photo_item ( $hash , $allow_cid , $deny_cid , $allow_gid , $deny_gid , $filetype , $visibility );
}
// on success return image data in json/xml format (like /api/friendica/photo does when no scale is given)
return prepare_photo_data ( $type , false , $hash );
} else {
throw new InternalServerErrorException ( " image upload failed " );
}
}
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $hash
* @ param string $allow_cid
* @ param string $deny_cid
* @ param string $allow_gid
* @ param string $deny_gid
* @ param string $filetype
* @ param boolean $visibility
*/
2017-11-10 05:00:50 +00:00
function post_photo_item ( $hash , $allow_cid , $deny_cid , $allow_gid , $deny_gid , $filetype , $visibility = false )
{
// get data about the api authenticated user
2018-06-16 06:44:19 +00:00
$uri = Item :: newURI ( intval ( api_user ()));
2018-08-19 12:46:11 +00:00
$owner_record = DBA :: selectFirst ( 'contact' , [], [ 'uid' => api_user (), 'self' => true ]);
2017-11-10 05:00:50 +00:00
2018-01-15 13:05:12 +00:00
$arr = [];
2018-09-27 11:52:15 +00:00
$arr [ 'guid' ] = System :: createUUID ();
2017-11-10 05:00:50 +00:00
$arr [ 'uid' ] = intval ( api_user ());
$arr [ 'uri' ] = $uri ;
$arr [ 'parent-uri' ] = $uri ;
$arr [ 'type' ] = 'photo' ;
$arr [ 'wall' ] = 1 ;
$arr [ 'resource-id' ] = $hash ;
2018-08-19 12:46:11 +00:00
$arr [ 'contact-id' ] = $owner_record [ 'id' ];
$arr [ 'owner-name' ] = $owner_record [ 'name' ];
$arr [ 'owner-link' ] = $owner_record [ 'url' ];
$arr [ 'owner-avatar' ] = $owner_record [ 'thumb' ];
$arr [ 'author-name' ] = $owner_record [ 'name' ];
$arr [ 'author-link' ] = $owner_record [ 'url' ];
$arr [ 'author-avatar' ] = $owner_record [ 'thumb' ];
2017-11-10 05:00:50 +00:00
$arr [ 'title' ] = " " ;
$arr [ 'allow_cid' ] = $allow_cid ;
$arr [ 'allow_gid' ] = $allow_gid ;
$arr [ 'deny_cid' ] = $deny_cid ;
$arr [ 'deny_gid' ] = $deny_gid ;
$arr [ 'visible' ] = $visibility ;
$arr [ 'origin' ] = 1 ;
2018-01-15 13:05:12 +00:00
$typetoext = [
2017-04-05 20:07:55 +00:00
'image/jpeg' => 'jpg' ,
'image/png' => 'png' ,
'image/gif' => 'gif'
2018-01-15 13:05:12 +00:00
];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// adds link to the thumbnail scale photo
2018-08-19 12:46:11 +00:00
$arr [ 'body' ] = '[url=' . System :: baseUrl () . '/photos/' . $owner_record [ 'nick' ] . '/image/' . $hash . ']'
2017-11-10 05:00:50 +00:00
. '[img]' . System :: baseUrl () . '/photo/' . $hash . '-' . " 2 " . '.' . $typetoext [ $filetype ] . '[/img]'
. '[/url]' ;
// do the magic for storing the item in the database and trigger the federation to other contacts
2018-01-28 11:18:08 +00:00
Item :: insert ( $arr );
2017-11-10 05:00:50 +00:00
}
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type
* @ param int $scale
* @ param string $photo_id
*
* @ return array
*/
2017-11-10 05:00:50 +00:00
function prepare_photo_data ( $type , $scale , $photo_id )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
if ( $user_info === false ) {
throw new ForbiddenException ();
}
2017-11-10 05:00:50 +00:00
$scale_sql = ( $scale === false ? " " : sprintf ( " AND scale=%d " , intval ( $scale )));
$data_sql = ( $scale === false ? " " : " data, " );
// added allow_cid, allow_gid, deny_cid, deny_gid to output as string like stored in database
// clients needs to convert this in their way for further processing
$r = q (
" SELECT %s `resource-id`, `created`, `edited`, `title`, `desc`, `album`, `filename`,
`type` , `height` , `width` , `datasize` , `profile` , `allow_cid` , `deny_cid` , `allow_gid` , `deny_gid` ,
MIN ( `scale` ) AS `minscale` , MAX ( `scale` ) AS `maxscale`
FROM `photo` WHERE `uid` = % d AND `resource-id` = '%s' % s GROUP BY `resource-id` " ,
$data_sql ,
intval ( local_user ()),
2018-07-21 13:10:13 +00:00
DBA :: escape ( $photo_id ),
2017-11-10 05:00:50 +00:00
$scale_sql
);
2018-01-15 13:05:12 +00:00
$typetoext = [
2017-11-10 05:00:50 +00:00
'image/jpeg' => 'jpg' ,
'image/png' => 'png' ,
'image/gif' => 'gif'
2018-01-15 13:05:12 +00:00
];
2017-11-10 05:00:50 +00:00
// prepare output data for photo
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2018-01-15 13:05:12 +00:00
$data = [ 'photo' => $r [ 0 ]];
2017-11-10 05:00:50 +00:00
$data [ 'photo' ][ 'id' ] = $data [ 'photo' ][ 'resource-id' ];
if ( $scale !== false ) {
$data [ 'photo' ][ 'data' ] = base64_encode ( $data [ 'photo' ][ 'data' ]);
2016-09-25 16:50:08 +00:00
} else {
2017-11-10 05:00:50 +00:00
unset ( $data [ 'photo' ][ 'datasize' ]); //needed only with scale param
2016-09-25 16:50:08 +00:00
}
2017-05-01 20:19:48 +00:00
if ( $type == " xml " ) {
2018-01-15 13:05:12 +00:00
$data [ 'photo' ][ 'links' ] = [];
2017-11-10 05:00:50 +00:00
for ( $k = intval ( $data [ 'photo' ][ 'minscale' ]); $k <= intval ( $data [ 'photo' ][ 'maxscale' ]); $k ++ ) {
2018-01-15 13:05:12 +00:00
$data [ 'photo' ][ 'links' ][ $k . " :link " ][ " @attributes " ] = [ " type " => $data [ 'photo' ][ 'type' ],
2017-11-10 05:00:50 +00:00
" scale " => $k ,
2018-01-15 13:05:12 +00:00
" href " => System :: baseUrl () . " /photo/ " . $data [ 'photo' ][ 'resource-id' ] . " - " . $k . " . " . $typetoext [ $data [ 'photo' ][ 'type' ]]];
2017-05-01 20:19:48 +00:00
}
} else {
2018-01-15 13:05:12 +00:00
$data [ 'photo' ][ 'link' ] = [];
2017-11-10 05:00:50 +00:00
// when we have profile images we could have only scales from 4 to 6, but index of array always needs to start with 0
$i = 0 ;
for ( $k = intval ( $data [ 'photo' ][ 'minscale' ]); $k <= intval ( $data [ 'photo' ][ 'maxscale' ]); $k ++ ) {
$data [ 'photo' ][ 'link' ][ $i ] = System :: baseUrl () . " /photo/ " . $data [ 'photo' ][ 'resource-id' ] . " - " . $k . " . " . $typetoext [ $data [ 'photo' ][ 'type' ]];
$i ++ ;
2017-05-01 20:19:48 +00:00
}
}
2017-11-10 05:00:50 +00:00
unset ( $data [ 'photo' ][ 'resource-id' ]);
unset ( $data [ 'photo' ][ 'minscale' ]);
unset ( $data [ 'photo' ][ 'maxscale' ]);
} else {
throw new NotFoundException ();
}
// retrieve item element for getting activities (like, dislike etc.) related to photo
2018-06-18 20:36:34 +00:00
$condition = [ 'uid' => local_user (), 'resource-id' => $photo_id , 'type' => 'photo' ];
$item = Item :: selectFirstForUser ( local_user (), [ 'id' ], $condition );
$data [ 'photo' ][ 'friendica_activities' ] = api_format_items_activities ( $item , $type );
2017-11-10 05:00:50 +00:00
// retrieve comments on photo
2018-06-27 18:09:33 +00:00
$condition = [ " `parent` = ? AND `uid` = ? AND (`gravity` IN (?, ?) OR `type`='photo') " ,
$item [ 0 ][ 'parent' ], api_user (), GRAVITY_PARENT , GRAVITY_COMMENT ];
2018-06-09 16:56:37 +00:00
2018-06-17 17:05:17 +00:00
$statuses = Item :: selectForUser ( api_user (), [], $condition );
2017-11-10 05:00:50 +00:00
// prepare output of comments
2018-06-21 15:14:01 +00:00
$commentData = api_format_items ( Item :: inArray ( $statuses ), $user_info , false , $type );
2018-01-15 13:05:12 +00:00
$comments = [];
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
$k = 0 ;
foreach ( $commentData as $comment ) {
$comments [ $k ++ . " :comment " ] = $comment ;
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
} else {
foreach ( $commentData as $comment ) {
$comments [] = $comment ;
}
}
$data [ 'photo' ][ 'friendica_comments' ] = $comments ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// include info if rights on photo and rights on item are mismatching
$rights_mismatch = $data [ 'photo' ][ 'allow_cid' ] != $item [ 0 ][ 'allow_cid' ] ||
$data [ 'photo' ][ 'deny_cid' ] != $item [ 0 ][ 'deny_cid' ] ||
$data [ 'photo' ][ 'allow_gid' ] != $item [ 0 ][ 'allow_gid' ] ||
$data [ 'photo' ][ 'deny_cid' ] != $item [ 0 ][ 'deny_cid' ];
$data [ 'photo' ][ 'rights_mismatch' ] = $rights_mismatch ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $data ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* Similar as / mod / redir . php
* redirect to 'url' after dfrn auth
*
* Why this when there is mod / redir . php already ?
* This use api_user () and api_login ()
*
* params
* c_url : url of remote contact to auth to
* url : string , url to redirect after auth
*/
function api_friendica_remoteauth ()
{
2018-11-30 14:06:22 +00:00
$url = defaults ( $_GET , 'url' , '' );
$c_url = defaults ( $_GET , 'c_url' , '' );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $url === '' || $c_url === '' ) {
throw new BadRequestException ( " Wrong parameters. " );
}
2016-09-25 16:50:08 +00:00
2018-11-08 16:28:29 +00:00
$c_url = Strings :: normaliseLink ( $c_url );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// traditional DFRN
2016-09-25 16:50:08 +00:00
2018-07-20 12:19:26 +00:00
$contact = DBA :: selectFirst ( 'contact' , [], [ 'uid' => api_user (), 'nurl' => $c_url ]);
2016-09-25 16:50:08 +00:00
2018-08-11 20:40:44 +00:00
if ( ! DBA :: isResult ( $contact ) || ( $contact [ 'network' ] !== Protocol :: DFRN )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " Unknown contact " );
}
2018-01-11 08:26:30 +00:00
$cid = $contact [ 'id' ];
2017-11-10 05:00:50 +00:00
2018-01-11 08:26:30 +00:00
$dfrn_id = defaults ( $contact , 'issued-id' , $contact [ 'dfrn-id' ]);
2017-11-10 05:00:50 +00:00
2018-01-11 08:26:30 +00:00
if ( $contact [ 'duplex' ] && $contact [ 'issued-id' ]) {
$orig_id = $contact [ 'issued-id' ];
2017-11-10 05:00:50 +00:00
$dfrn_id = '1:' . $orig_id ;
}
2018-01-11 08:26:30 +00:00
if ( $contact [ 'duplex' ] && $contact [ 'dfrn-id' ]) {
$orig_id = $contact [ 'dfrn-id' ];
2017-11-10 05:00:50 +00:00
$dfrn_id = '0:' . $orig_id ;
}
2016-09-25 16:50:08 +00:00
2018-11-08 13:45:46 +00:00
$sec = Strings :: getRandomHex ();
2017-11-10 05:00:50 +00:00
2018-05-04 06:33:36 +00:00
$fields = [ 'uid' => api_user (), 'cid' => $cid , 'dfrn_id' => $dfrn_id ,
'sec' => $sec , 'expire' => time () + 45 ];
2018-07-20 12:19:26 +00:00
DBA :: insert ( 'profile_check' , $fields );
2017-11-10 05:00:50 +00:00
2018-10-30 13:58:45 +00:00
Logger :: log ( $contact [ 'name' ] . ' ' . $sec , Logger :: DEBUG );
2017-12-22 23:00:49 +00:00
$dest = ( $url ? '&destination_url=' . $url : '' );
2018-10-13 18:02:04 +00:00
2018-10-19 23:01:15 +00:00
System :: externalRedirect (
2018-01-11 08:26:30 +00:00
$contact [ 'poll' ] . '?dfrn_id=' . $dfrn_id
2017-11-10 05:00:50 +00:00
. '&dfrn_version=' . DFRN_PROTOCOL_VERSION
2018-02-11 21:13:29 +00:00
. '&type=profile&sec=' . $sec . $dest
2017-11-10 05:00:50 +00:00
);
}
api_register_func ( 'api/friendica/remoteauth' , 'api_friendica_remoteauth' , true );
/**
* @ brief Return the item shared , if the item contains only the [ share ] tag
*
* @ param array $item Sharer item
2017-12-23 00:36:55 +00:00
* @ return array | false Shared item or false if not a reshare
2017-11-10 05:00:50 +00:00
*/
function api_share_as_retweet ( & $item )
{
$body = trim ( $item [ " body " ]);
2018-10-02 03:41:38 +00:00
if ( Diaspora :: isReshare ( $body , false ) === false ) {
if ( $item [ 'author-id' ] == $item [ 'owner-id' ]) {
return false ;
} else {
// Reshares from OStatus, ActivityPub and Twitter
$reshared_item = $item ;
$reshared_item [ 'owner-id' ] = $reshared_item [ 'author-id' ];
$reshared_item [ 'owner-link' ] = $reshared_item [ 'author-link' ];
$reshared_item [ 'owner-name' ] = $reshared_item [ 'author-name' ];
$reshared_item [ 'owner-avatar' ] = $reshared_item [ 'author-avatar' ];
return $reshared_item ;
}
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
/// @TODO "$1" should maybe mean '$1' ?
$attributes = preg_replace ( " / \ [share(.*?) \ ] \ s?(.*?) \ s? \ [ \ /share \ ] \ s?/ism " , " $ 1 " , $body );
/*
2018-10-03 09:15:38 +00:00
* Skip if there is no shared message in there
* we already checked this in diaspora :: isReshare ()
* but better one more than one less ...
*/
if (( $body == $attributes ) || empty ( $attributes )) {
2017-11-10 05:00:50 +00:00
return false ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// build the fake reshared item
$reshared_item = $item ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$author = " " ;
preg_match ( " /author='(.*?)'/ism " , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$author = html_entity_decode ( $matches [ 1 ], ENT_QUOTES , 'UTF-8' );
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
preg_match ( '/author="(.*?)"/ism' , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$author = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$profile = " " ;
preg_match ( " /profile='(.*?)'/ism " , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$profile = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
preg_match ( '/profile="(.*?)"/ism' , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$profile = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$avatar = " " ;
preg_match ( " /avatar='(.*?)'/ism " , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$avatar = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
preg_match ( '/avatar="(.*?)"/ism' , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$avatar = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$link = " " ;
preg_match ( " /link='(.*?)'/ism " , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$link = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
preg_match ( '/link="(.*?)"/ism' , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$link = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$posted = " " ;
preg_match ( " /posted='(.*?)'/ism " , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$posted = $matches [ 1 ];
2017-12-23 23:27:45 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
preg_match ( '/posted="(.*?)"/ism' , $attributes , $matches );
2018-10-03 09:15:38 +00:00
if ( ! empty ( $matches [ 1 ])) {
2017-11-10 05:00:50 +00:00
$posted = $matches [ 1 ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$shared_body = preg_replace ( " / \ [share(.*?) \ ] \ s?(.*?) \ s? \ [ \ /share \ ] \ s?/ism " , " $ 2 " , $body );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if (( $shared_body == " " ) || ( $profile == " " ) || ( $author == " " ) || ( $avatar == " " ) || ( $posted == " " )) {
return false ;
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$reshared_item [ " body " ] = $shared_body ;
2018-12-21 22:55:36 +00:00
$reshared_item [ " author-id " ] = Contact :: getIdForURL ( $profile , 0 , true );
2017-11-10 05:00:50 +00:00
$reshared_item [ " author-name " ] = $author ;
$reshared_item [ " author-link " ] = $profile ;
$reshared_item [ " author-avatar " ] = $avatar ;
$reshared_item [ " plink " ] = $link ;
$reshared_item [ " created " ] = $posted ;
$reshared_item [ " edited " ] = $posted ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $reshared_item ;
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $profile
*
* @ return string | false
* @ todo remove trailing junk from profile url
* @ todo pump . io check has to check the website
*/
2017-11-10 05:00:50 +00:00
function api_get_nick ( $profile )
{
$nick = " " ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$r = q (
" SELECT `nick` FROM `contact` WHERE `uid` = 0 AND `nurl` = '%s' " ,
2018-11-08 16:28:29 +00:00
DBA :: escape ( Strings :: normaliseLink ( $profile ))
2017-11-10 05:00:50 +00:00
);
2016-09-25 16:50:08 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
$nick = $r [ 0 ][ " nick " ];
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $nick == " " ) {
$r = q (
" SELECT `nick` FROM `contact` WHERE `uid` = 0 AND `nurl` = '%s' " ,
2018-11-08 16:28:29 +00:00
DBA :: escape ( Strings :: normaliseLink ( $profile ))
2017-11-10 05:00:50 +00:00
);
2017-04-05 20:07:55 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $r )) {
2016-09-25 16:50:08 +00:00
$nick = $r [ 0 ][ " nick " ];
2017-04-05 20:07:55 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $nick == " " ) {
$friendica = preg_replace ( " =https?://(.*)/profile/(.*)=ism " , " $ 2 " , $profile );
if ( $friendica != $profile ) {
$nick = $friendica ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $nick == " " ) {
$diaspora = preg_replace ( " =https?://(.*)/u/(.*)=ism " , " $ 2 " , $profile );
if ( $diaspora != $profile ) {
$nick = $diaspora ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $nick == " " ) {
$twitter = preg_replace ( " =https?://twitter.com/(.*)=ism " , " $ 1 " , $profile );
if ( $twitter != $profile ) {
$nick = $twitter ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( ! $nick == " " ) {
$StatusnetHost = preg_replace ( " =https?://(.*)/user/(.*)=ism " , " $ 1 " , $profile );
if ( $StatusnetHost != $profile ) {
$StatusnetUser = preg_replace ( " =https?://(.*)/user/(.*)=ism " , " $ 2 " , $profile );
if ( $StatusnetUser != $profile ) {
2018-01-27 16:13:41 +00:00
$UserData = Network :: fetchUrl ( " http:// " . $StatusnetHost . " /api/users/show.json?user_id= " . $StatusnetUser );
2017-11-10 05:00:50 +00:00
$user = json_decode ( $UserData );
if ( $user ) {
$nick = $user -> screen_name ;
2016-09-25 16:50:08 +00:00
}
}
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// To-Do: look at the page if its really a pumpio site
//if (!$nick == "") {
// $pumpio = preg_replace("=https?://(.*)/(.*)/=ism", "$2", $profile."/");
// if ($pumpio != $profile)
// $nick = $pumpio;
// <div class="media" id="profile-block" data-profile-id="acct:kabniel@microca.st">
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
//}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $nick != " " ) {
return $nick ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
return false ;
}
2016-11-04 22:45:20 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param array $item
*
* @ return array
*/
2017-11-10 05:00:50 +00:00
function api_in_reply_to ( $item )
{
2018-01-15 13:05:12 +00:00
$in_reply_to = [];
2016-11-04 22:45:20 +00:00
2017-11-10 05:00:50 +00:00
$in_reply_to [ 'status_id' ] = null ;
$in_reply_to [ 'user_id' ] = null ;
$in_reply_to [ 'status_id_str' ] = null ;
$in_reply_to [ 'user_id_str' ] = null ;
$in_reply_to [ 'screen_name' ] = null ;
2016-11-04 22:45:20 +00:00
2017-11-10 05:00:50 +00:00
if (( $item [ 'thr-parent' ] != $item [ 'uri' ]) && ( intval ( $item [ 'parent' ]) != intval ( $item [ 'id' ]))) {
2018-06-18 20:36:34 +00:00
$parent = Item :: selectFirst ([ 'id' ], [ 'uid' => $item [ 'uid' ], 'uri' => $item [ 'thr-parent' ]]);
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $parent )) {
2018-06-18 20:36:34 +00:00
$in_reply_to [ 'status_id' ] = intval ( $parent [ 'id' ]);
2017-11-10 05:00:50 +00:00
} else {
$in_reply_to [ 'status_id' ] = intval ( $item [ 'parent' ]);
}
2016-11-04 22:45:20 +00:00
2017-11-10 05:00:50 +00:00
$in_reply_to [ 'status_id_str' ] = ( string ) intval ( $in_reply_to [ 'status_id' ]);
2016-11-04 22:45:20 +00:00
2018-06-18 20:36:34 +00:00
$fields = [ 'author-nick' , 'author-name' , 'author-id' , 'author-link' ];
$parent = Item :: selectFirst ( $fields , [ 'id' => $in_reply_to [ 'status_id' ]]);
2016-11-04 22:45:20 +00:00
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $parent )) {
2018-06-18 20:36:34 +00:00
if ( $parent [ 'author-nick' ] == " " ) {
$parent [ 'author-nick' ] = api_get_nick ( $parent [ 'author-link' ]);
2016-11-04 22:45:20 +00:00
}
2018-06-18 20:36:34 +00:00
$in_reply_to [ 'screen_name' ] = (( $parent [ 'author-nick' ]) ? $parent [ 'author-nick' ] : $parent [ 'author-name' ]);
$in_reply_to [ 'user_id' ] = intval ( $parent [ 'author-id' ]);
$in_reply_to [ 'user_id_str' ] = ( string ) intval ( $parent [ 'author-id' ]);
2016-11-04 22:45:20 +00:00
}
2017-11-10 05:00:50 +00:00
// There seems to be situation, where both fields are identical:
// https://github.com/friendica/friendica/issues/1010
// This is a bugfix for that.
if ( intval ( $in_reply_to [ 'status_id' ]) == intval ( $item [ 'id' ])) {
2018-10-30 13:58:45 +00:00
Logger :: log ( 'this message should never appear: id: ' . $item [ 'id' ] . ' similar to reply-to: ' . $in_reply_to [ 'status_id' ], Logger :: DEBUG );
2017-11-10 05:00:50 +00:00
$in_reply_to [ 'status_id' ] = null ;
$in_reply_to [ 'user_id' ] = null ;
$in_reply_to [ 'status_id_str' ] = null ;
$in_reply_to [ 'user_id_str' ] = null ;
$in_reply_to [ 'screen_name' ] = null ;
}
2016-11-04 22:45:20 +00:00
}
2017-11-10 05:00:50 +00:00
return $in_reply_to ;
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
2018-05-10 13:13:08 +00:00
* @ param string $text
2017-12-24 02:20:50 +00:00
*
* @ return string
*/
2018-05-10 13:13:08 +00:00
function api_clean_plain_items ( $text )
2017-11-10 05:00:50 +00:00
{
2018-11-30 14:06:22 +00:00
$include_entities = strtolower ( defaults ( $_REQUEST , 'include_entities' , " false " ));
2016-09-25 16:50:08 +00:00
2018-05-10 13:13:08 +00:00
$text = BBCode :: cleanPictureLinks ( $text );
2017-11-10 05:00:50 +00:00
$URLSearchString = " ^ \ [ \ ] " ;
2016-09-25 16:50:08 +00:00
2018-05-10 13:13:08 +00:00
$text = preg_replace ( " /([!#@]) \ [url \ =([ $URLSearchString ]*) \ ](.*?) \ [ \ /url \ ]/ism " , '$1$3' , $text );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $include_entities == " true " ) {
2018-05-10 13:13:08 +00:00
$text = preg_replace ( " / \ [url \ =([ $URLSearchString ]*) \ ](.*?) \ [ \ /url \ ]/ism " , '[url=$1]$1[/url]' , $text );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// Simplify "attachment" element
2018-05-10 13:13:08 +00:00
$text = api_clean_attachments ( $text );
2016-09-25 16:50:08 +00:00
2018-05-10 13:13:08 +00:00
return $text ;
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Removes most sharing information for API text export
*
* @ param string $body The original body
*
* @ return string Cleaned body
*/
function api_clean_attachments ( $body )
{
2018-01-27 01:01:32 +00:00
$data = BBCode :: getAttachmentData ( $body );
2016-09-25 16:50:08 +00:00
2018-04-09 19:34:53 +00:00
if ( empty ( $data )) {
2016-09-25 16:50:08 +00:00
return $body ;
2017-12-22 23:10:32 +00:00
}
2017-11-10 05:00:50 +00:00
$body = " " ;
2016-09-25 16:50:08 +00:00
2017-12-22 23:10:32 +00:00
if ( isset ( $data [ " text " ])) {
2017-11-10 05:00:50 +00:00
$body = $data [ " text " ];
2017-12-22 23:10:32 +00:00
}
if (( $body == " " ) && isset ( $data [ " title " ])) {
2017-11-10 05:00:50 +00:00
$body = $data [ " title " ];
2017-12-22 23:10:32 +00:00
}
if ( isset ( $data [ " url " ])) {
2017-11-10 05:00:50 +00:00
$body .= " \n " . $data [ " url " ];
2017-12-22 23:10:32 +00:00
}
2017-11-10 05:00:50 +00:00
$body .= $data [ " after " ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
return $body ;
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
*
* @ param array $contacts
*
* @ return array
*/
2017-11-10 05:00:50 +00:00
function api_best_nickname ( & $contacts )
{
2018-01-15 13:05:12 +00:00
$best_contact = [];
2016-09-25 16:50:08 +00:00
2018-02-11 21:13:29 +00:00
if ( count ( $contacts ) == 0 ) {
2017-11-10 05:00:50 +00:00
return ;
2017-12-23 23:27:45 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
foreach ( $contacts as $contact ) {
2017-11-10 05:00:50 +00:00
if ( $contact [ " network " ] == " " ) {
$contact [ " network " ] = " dfrn " ;
2018-01-15 13:05:12 +00:00
$best_contact = [ $contact ];
2017-11-10 05:00:50 +00:00
}
2017-12-23 23:27:45 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( sizeof ( $best_contact ) == 0 ) {
foreach ( $contacts as $contact ) {
if ( $contact [ " network " ] == " dfrn " ) {
2018-01-15 13:05:12 +00:00
$best_contact = [ $contact ];
2017-12-23 23:27:45 +00:00
}
}
}
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( sizeof ( $best_contact ) == 0 ) {
foreach ( $contacts as $contact ) {
if ( $contact [ " network " ] == " dspr " ) {
2018-01-15 13:05:12 +00:00
$best_contact = [ $contact ];
2017-12-23 23:27:45 +00:00
}
}
}
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( sizeof ( $best_contact ) == 0 ) {
foreach ( $contacts as $contact ) {
if ( $contact [ " network " ] == " stat " ) {
2018-01-15 13:05:12 +00:00
$best_contact = [ $contact ];
2017-12-23 23:27:45 +00:00
}
}
}
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( sizeof ( $best_contact ) == 0 ) {
foreach ( $contacts as $contact ) {
if ( $contact [ " network " ] == " pump " ) {
2018-01-15 13:05:12 +00:00
$best_contact = [ $contact ];
2017-12-23 23:27:45 +00:00
}
}
}
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( sizeof ( $best_contact ) == 0 ) {
foreach ( $contacts as $contact ) {
if ( $contact [ " network " ] == " twit " ) {
2018-01-15 13:05:12 +00:00
$best_contact = [ $contact ];
2017-12-23 23:27:45 +00:00
}
}
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( sizeof ( $best_contact ) == 1 ) {
$contacts = $best_contact ;
} else {
2018-01-15 13:05:12 +00:00
$contacts = [ $contacts [ 0 ]];
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-24 02:20:50 +00:00
/**
* Return all or a specified group of the user with the containing contacts .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_friendica_group_show ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$gid = defaults ( $_REQUEST , 'gid' , 0 );
2017-11-10 05:00:50 +00:00
$uid = $user_info [ 'uid' ];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// get data of the specified group id or all groups if not specified
if ( $gid != 0 ) {
$r = q (
" SELECT * FROM `group` WHERE `deleted` = 0 AND `uid` = %d AND `id` = %d " ,
2016-09-25 16:50:08 +00:00
intval ( $uid ),
2017-11-10 05:00:50 +00:00
intval ( $gid )
);
2016-09-25 16:50:08 +00:00
// error message if specified gid is not in database
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " gid not available " );
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
} else {
$r = q (
" SELECT * FROM `group` WHERE `deleted` = 0 AND `uid` = %d " ,
intval ( $uid )
);
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// loop through all groups and retrieve all members for adding data in the user array
2018-04-09 19:34:53 +00:00
$grps = [];
2017-11-10 05:00:50 +00:00
foreach ( $r as $rr ) {
2017-12-09 18:45:17 +00:00
$members = Contact :: getByGroupId ( $rr [ 'id' ]);
2018-01-15 13:05:12 +00:00
$users = [];
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
$user_element = " users " ;
$k = 0 ;
foreach ( $members as $member ) {
$user = api_get_user ( $a , $member [ 'nurl' ]);
$users [ $k ++. " :user " ] = $user ;
}
} else {
$user_element = " user " ;
foreach ( $members as $member ) {
$user = api_get_user ( $a , $member [ 'nurl' ]);
$users [] = $user ;
2016-09-25 16:50:08 +00:00
}
}
2018-01-15 13:05:12 +00:00
$grps [] = [ 'name' => $rr [ 'name' ], 'gid' => $rr [ 'id' ], $user_element => $users ];
2017-11-10 05:00:50 +00:00
}
2018-01-15 13:05:12 +00:00
return api_format_data ( " groups " , $type , [ 'group' => $grps ]);
2017-11-10 05:00:50 +00:00
}
api_register_func ( 'api/friendica/group_show' , 'api_friendica_group_show' , true );
2017-12-24 02:20:50 +00:00
/**
* Delete the specified group of the user .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_friendica_group_delete ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
2017-12-09 18:42:02 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$gid = defaults ( $_REQUEST , 'gid' , 0 );
$name = defaults ( $_REQUEST , 'name' , " " );
2017-11-10 05:00:50 +00:00
$uid = $user_info [ 'uid' ];
// error if no gid specified
2017-12-09 18:42:02 +00:00
if ( $gid == 0 || $name == " " ) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'gid or name not specified' );
2017-12-09 18:42:02 +00:00
}
2017-11-10 05:00:50 +00:00
// get data of the specified group id
$r = q (
" SELECT * FROM `group` WHERE `uid` = %d AND `id` = %d " ,
intval ( $uid ),
intval ( $gid )
);
// error message if specified gid is not in database
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'gid not available' );
2017-12-09 18:42:02 +00:00
}
2017-11-10 05:00:50 +00:00
// get data of the specified group id and group name
$rname = q (
" SELECT * FROM `group` WHERE `uid` = %d AND `id` = %d AND `name` = '%s' " ,
intval ( $uid ),
intval ( $gid ),
2018-07-21 13:10:13 +00:00
DBA :: escape ( $name )
2017-11-10 05:00:50 +00:00
);
// error message if specified gid is not in database
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $rname )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'wrong group name' );
2017-12-09 18:42:02 +00:00
}
2017-11-10 05:00:50 +00:00
// delete group
2017-12-09 18:45:17 +00:00
$ret = Group :: removeByName ( $uid , $name );
2017-11-10 05:00:50 +00:00
if ( $ret ) {
// return success
2018-01-15 13:05:12 +00:00
$success = [ 'success' => $ret , 'gid' => $gid , 'name' => $name , 'status' => 'deleted' , 'wrong users' => []];
return api_format_data ( " group_delete " , $type , [ 'result' => $success ]);
2017-11-10 05:00:50 +00:00
} else {
throw new BadRequestException ( 'other API error' );
}
}
api_register_func ( 'api/friendica/group_delete' , 'api_friendica_group_delete' , true , API_METHOD_DELETE );
2017-12-24 02:20:50 +00:00
/**
2018-04-07 13:54:26 +00:00
* Delete a group .
2017-12-24 02:20:50 +00:00
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
2018-04-07 13:54:26 +00:00
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / create - manage - lists / api - reference / post - lists - destroy
2017-12-24 02:20:50 +00:00
*/
2018-04-07 13:54:26 +00:00
function api_lists_destroy ( $type )
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
2017-12-23 23:27:45 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$gid = defaults ( $_REQUEST , 'list_id' , 0 );
2017-11-10 05:00:50 +00:00
$uid = $user_info [ 'uid' ];
2018-04-07 13:54:26 +00:00
// error if no gid specified
if ( $gid == 0 ) {
throw new BadRequestException ( 'gid not specified' );
}
// get data of the specified group id
2018-07-20 12:19:26 +00:00
$group = DBA :: selectFirst ( 'group' , [], [ 'uid' => $uid , 'id' => $gid ]);
2018-04-07 13:54:26 +00:00
// error message if specified gid is not in database
2018-04-07 17:55:41 +00:00
if ( ! $group ) {
2018-04-07 13:54:26 +00:00
throw new BadRequestException ( 'gid not available' );
}
if ( Group :: remove ( $gid )) {
2018-04-07 17:55:41 +00:00
$list = [
'name' => $group [ 'name' ],
2018-04-07 13:54:26 +00:00
'id' => intval ( $gid ),
'id_str' => ( string ) $gid ,
'user' => $user_info
];
2018-04-07 17:55:41 +00:00
return api_format_data ( " lists " , $type , [ 'lists' => $list ]);
2018-04-07 13:54:26 +00:00
}
}
api_register_func ( 'api/lists/destroy' , 'api_lists_destroy' , true , API_METHOD_DELETE );
/**
* Add a new group to the database .
*
* @ param string $name Group name
* @ param int $uid User ID
* @ param array $users List of users to add to the group
*
* @ return array
*/
function group_create ( $name , $uid , $users = [])
{
2017-11-10 05:00:50 +00:00
// error if no name specified
2017-12-23 23:27:45 +00:00
if ( $name == " " ) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'group name not specified' );
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
// get data of the specified group name
$rname = q (
" SELECT * FROM `group` WHERE `uid` = %d AND `name` = '%s' AND `deleted` = 0 " ,
intval ( $uid ),
2018-07-21 13:10:13 +00:00
DBA :: escape ( $name )
2017-11-10 05:00:50 +00:00
);
// error message if specified group name already exists
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $rname )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'group name already exists' );
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
// check if specified group name is a deleted group
$rname = q (
" SELECT * FROM `group` WHERE `uid` = %d AND `name` = '%s' AND `deleted` = 1 " ,
intval ( $uid ),
2018-07-21 13:10:13 +00:00
DBA :: escape ( $name )
2017-11-10 05:00:50 +00:00
);
// error message if specified group name already exists
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $rname )) {
2017-11-10 05:00:50 +00:00
$reactivate_group = true ;
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
// create group
2017-12-09 18:45:17 +00:00
$ret = Group :: create ( $uid , $name );
2017-11-10 05:00:50 +00:00
if ( $ret ) {
2017-12-09 18:45:17 +00:00
$gid = Group :: getIdByName ( $uid , $name );
2017-11-10 05:00:50 +00:00
} else {
throw new BadRequestException ( 'other API error' );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// add members
$erroraddinguser = false ;
2018-01-15 13:05:12 +00:00
$errorusers = [];
2017-11-10 05:00:50 +00:00
foreach ( $users as $user ) {
$cid = $user [ 'cid' ];
// check if user really exists as contact
$contact = q (
" SELECT * FROM `contact` WHERE `id` = %d AND `uid` = %d " ,
intval ( $cid ),
intval ( $uid )
);
2017-12-22 23:10:32 +00:00
if ( count ( $contact )) {
2018-01-04 01:54:35 +00:00
Group :: addMember ( $gid , $cid );
2017-12-22 23:10:32 +00:00
} else {
2017-11-10 05:00:50 +00:00
$erroraddinguser = true ;
$errorusers [] = $cid ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// return success message incl. missing users in array
2018-04-09 19:34:53 +00:00
$status = ( $erroraddinguser ? " missing user " : (( isset ( $reactivate_group ) && $reactivate_group ) ? " reactivated " : " ok " ));
2018-04-07 13:54:26 +00:00
return [ 'success' => true , 'gid' => $gid , 'name' => $name , 'status' => $status , 'wrong users' => $errorusers ];
}
/**
* Create the specified group with the posted array of contacts .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
function api_friendica_group_create ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-07 13:54:26 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$name = defaults ( $_REQUEST , 'name' , " " );
2018-04-07 13:54:26 +00:00
$uid = $user_info [ 'uid' ];
$json = json_decode ( $_POST [ 'json' ], true );
$users = $json [ 'user' ];
$success = group_create ( $name , $uid , $users );
2018-01-15 13:05:12 +00:00
return api_format_data ( " group_create " , $type , [ 'result' => $success ]);
2017-11-10 05:00:50 +00:00
}
api_register_func ( 'api/friendica/group_create' , 'api_friendica_group_create' , true , API_METHOD_POST );
2018-04-07 13:54:26 +00:00
/**
* Create a new group .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / create - manage - lists / api - reference / post - lists - create
*/
2018-04-07 20:15:35 +00:00
function api_lists_create ( $type )
2018-04-07 13:54:26 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-07 13:54:26 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$name = defaults ( $_REQUEST , 'name' , " " );
2018-04-07 13:54:26 +00:00
$uid = $user_info [ 'uid' ];
$success = group_create ( $name , $uid );
if ( $success [ 'success' ]) {
$grp = [
'name' => $success [ 'name' ],
'id' => intval ( $success [ 'gid' ]),
'id_str' => ( string ) $success [ 'gid' ],
'user' => $user_info
];
return api_format_data ( " lists " , $type , [ 'lists' => $grp ]);
}
}
2018-04-07 20:15:35 +00:00
api_register_func ( 'api/lists/create' , 'api_lists_create' , true , API_METHOD_POST );
2017-11-10 05:00:50 +00:00
2017-12-24 02:20:50 +00:00
/**
* Update the specified group with the posted array of contacts .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_friendica_group_update ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
2017-12-23 23:27:45 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
$uid = $user_info [ 'uid' ];
2018-11-30 14:06:22 +00:00
$gid = defaults ( $_REQUEST , 'gid' , 0 );
$name = defaults ( $_REQUEST , 'name' , " " );
2017-11-10 05:00:50 +00:00
$json = json_decode ( $_POST [ 'json' ], true );
$users = $json [ 'user' ];
// error if no name specified
2017-12-23 23:27:45 +00:00
if ( $name == " " ) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'group name not specified' );
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
// error if no gid specified
2017-12-23 23:27:45 +00:00
if ( $gid == " " ) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( 'gid not specified' );
2017-12-23 23:27:45 +00:00
}
2017-11-10 05:00:50 +00:00
// remove members
2017-12-09 18:45:17 +00:00
$members = Contact :: getByGroupId ( $gid );
2017-11-10 05:00:50 +00:00
foreach ( $members as $member ) {
$cid = $member [ 'id' ];
2016-09-25 16:50:08 +00:00
foreach ( $users as $user ) {
2017-11-10 05:00:50 +00:00
$found = ( $user [ 'cid' ] == $cid ? true : false );
}
2018-04-09 19:34:53 +00:00
if ( ! isset ( $found ) || ! $found ) {
2018-01-04 01:54:35 +00:00
Group :: removeMemberByName ( $uid , $name , $cid );
2016-09-25 16:50:08 +00:00
}
}
2017-11-10 05:00:50 +00:00
// add members
$erroraddinguser = false ;
2018-01-15 13:05:12 +00:00
$errorusers = [];
2017-11-10 05:00:50 +00:00
foreach ( $users as $user ) {
$cid = $user [ 'cid' ];
// check if user really exists as contact
$contact = q (
" SELECT * FROM `contact` WHERE `id` = %d AND `uid` = %d " ,
intval ( $cid ),
intval ( $uid )
);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( count ( $contact )) {
2018-01-04 01:54:35 +00:00
Group :: addMember ( $gid , $cid );
2016-09-25 16:50:08 +00:00
} else {
2017-11-10 05:00:50 +00:00
$erroraddinguser = true ;
$errorusers [] = $cid ;
2016-09-25 16:50:08 +00:00
}
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// return success message incl. missing users in array
$status = ( $erroraddinguser ? " missing user " : " ok " );
2018-01-15 13:05:12 +00:00
$success = [ 'success' => true , 'gid' => $gid , 'name' => $name , 'status' => $status , 'wrong users' => $errorusers ];
return api_format_data ( " group_update " , $type , [ 'result' => $success ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/friendica/group_update' , 'api_friendica_group_update' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2018-04-07 13:54:26 +00:00
/**
* Update information about a group .
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / create - manage - lists / api - reference / post - lists - update
*/
function api_lists_update ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-07 13:54:26 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$gid = defaults ( $_REQUEST , 'list_id' , 0 );
$name = defaults ( $_REQUEST , 'name' , " " );
2018-04-07 13:54:26 +00:00
$uid = $user_info [ 'uid' ];
// error if no gid specified
if ( $gid == 0 ) {
throw new BadRequestException ( 'gid not specified' );
}
// get data of the specified group id
2018-07-20 12:19:26 +00:00
$group = DBA :: selectFirst ( 'group' , [], [ 'uid' => $uid , 'id' => $gid ]);
2018-04-07 13:54:26 +00:00
// error message if specified gid is not in database
2018-04-07 17:55:41 +00:00
if ( ! $group ) {
2018-04-07 13:54:26 +00:00
throw new BadRequestException ( 'gid not available' );
}
if ( Group :: update ( $gid , $name )) {
2018-04-07 17:55:41 +00:00
$list = [
2018-04-07 13:54:26 +00:00
'name' => $name ,
'id' => intval ( $gid ),
'id_str' => ( string ) $gid ,
'user' => $user_info
];
2018-04-07 17:55:41 +00:00
return api_format_data ( " lists " , $type , [ 'lists' => $list ]);
2018-04-07 13:54:26 +00:00
}
}
api_register_func ( 'api/lists/update' , 'api_lists_update' , true , API_METHOD_POST );
2017-12-24 02:20:50 +00:00
/**
*
* @ param string $type Return type ( atom , rss , xml , json )
*
* @ return array | string
*/
2017-11-10 05:00:50 +00:00
function api_friendica_activity ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2017-11-10 05:00:50 +00:00
$verb = strtolower ( $a -> argv [ 3 ]);
$verb = preg_replace ( " | \ ..* $ | " , " " , $verb );
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
$id = defaults ( $_REQUEST , 'id' , 0 );
2016-09-25 16:50:08 +00:00
2018-02-01 19:17:08 +00:00
$res = Item :: performLike ( $id , $verb );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $res ) {
if ( $type == " xml " ) {
$ok = " true " ;
} else {
$ok = " ok " ;
}
2018-01-15 13:05:12 +00:00
return api_format_data ( 'ok' , $type , [ 'ok' => $ok ]);
2017-11-10 05:00:50 +00:00
} else {
throw new BadRequestException ( 'Error adding activity' );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/friendica/activity/like' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/dislike' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/attendyes' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/attendno' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/attendmaybe' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/unlike' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/undislike' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/unattendyes' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/unattendno' , 'api_friendica_activity' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/activity/unattendmaybe' , 'api_friendica_activity' , true , API_METHOD_POST );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief Returns notifications
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_friendica_notification ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-12-23 23:27:45 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
if ( $a -> argc !== 3 ) {
throw new BadRequestException ( " Invalid argument count " );
}
2017-11-10 05:00:50 +00:00
$nm = new NotificationsManager ();
2016-09-25 16:50:08 +00:00
2018-01-15 13:05:12 +00:00
$notes = $nm -> getAll ([], " +seen -date " , 50 );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( $type == " xml " ) {
2018-01-15 13:05:12 +00:00
$xmlnotes = [];
2018-07-01 18:42:38 +00:00
if ( ! empty ( $notes )) {
foreach ( $notes as $note ) {
$xmlnotes [] = [ " @attributes " => $note ];
}
2017-12-23 23:27:45 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$notes = $xmlnotes ;
2016-09-25 16:50:08 +00:00
}
2018-01-15 13:05:12 +00:00
return api_format_data ( " notes " , $type , [ 'note' => $notes ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* POST request with 'id' param as notification id
*
2017-12-24 02:20:50 +00:00
* @ brief Set notification as seen and returns associated item ( if possible )
*
2017-11-10 05:00:50 +00:00
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_friendica_notification_seen ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2018-04-09 15:36:33 +00:00
$user_info = api_get_user ( $a );
2016-09-25 16:50:08 +00:00
2018-04-09 15:36:33 +00:00
if ( api_user () === false || $user_info === false ) {
2017-12-23 23:27:45 +00:00
throw new ForbiddenException ();
}
if ( $a -> argc !== 4 ) {
throw new BadRequestException ( " Invalid argument count " );
}
2016-09-25 16:50:08 +00:00
2018-11-30 14:06:22 +00:00
$id = ( ! empty ( $_REQUEST [ 'id' ]) ? intval ( $_REQUEST [ 'id' ]) : 0 );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
$nm = new NotificationsManager ();
$note = $nm -> getByID ( $id );
2017-12-23 23:27:45 +00:00
if ( is_null ( $note )) {
throw new BadRequestException ( " Invalid argument " );
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
$nm -> setSeen ( $note );
if ( $note [ 'otype' ] == 'item' ) {
// would be really better with an ItemsManager and $im->getByID() :-P
2018-06-17 17:05:17 +00:00
$item = Item :: selectFirstForUser ( api_user (), [], [ 'id' => $note [ 'iid' ], 'uid' => api_user ()]);
2018-08-16 09:46:53 +00:00
if ( DBA :: isResult ( $item )) {
2017-11-10 05:00:50 +00:00
// we found the item, return it to the user
2018-06-09 16:56:37 +00:00
$ret = api_format_items ([ $item ], $user_info , false , $type );
2018-01-15 13:05:12 +00:00
$data = [ 'status' => $ret ];
2017-11-10 05:00:50 +00:00
return api_format_data ( " status " , $type , $data );
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// the item can't be found, but we set the note as seen, so we count this as a success
}
2018-01-15 13:05:12 +00:00
return api_format_data ( 'result' , $type , [ 'result' => " success " ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/friendica/notification/seen' , 'api_friendica_notification_seen' , true , API_METHOD_POST );
api_register_func ( 'api/friendica/notification' , 'api_friendica_notification' , true , API_METHOD_GET );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief update a direct_message to seen state
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array ( success result = ok , error result = error with error message )
2017-11-10 05:00:50 +00:00
*/
function api_friendica_direct_messages_setseen ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
$uid = $user_info [ 'uid' ];
2018-11-30 14:06:22 +00:00
$id = defaults ( $_REQUEST , 'id' , 0 );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// return error if id is zero
if ( $id == " " ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'message id not specified' ];
return api_format_data ( " direct_messages_setseen " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// error message if specified id is not in database
2018-07-20 12:19:26 +00:00
if ( ! DBA :: exists ( 'mail' , [ 'id' => $id , 'uid' => $uid ])) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'message id not in database' ];
return api_format_data ( " direct_messages_setseen " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// update seen indicator
2018-07-20 12:19:26 +00:00
$result = DBA :: update ( 'mail' , [ 'seen' => true ], [ 'id' => $id ]);
2017-11-10 05:00:50 +00:00
if ( $result ) {
// return success
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'ok' , 'message' => 'message set to seen' ];
return api_format_data ( " direct_message_setseen " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
} else {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'unknown error' ];
return api_format_data ( " direct_messages_setseen " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
}
}
2016-11-04 22:45:20 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/friendica/direct_messages_setseen' , 'api_friendica_direct_messages_setseen' , true );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief search for direct_messages containing a searchstring through api
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-02-11 21:13:29 +00:00
* @ param string $box
2018-04-09 17:34:02 +00:00
* @ return string | array ( success : success = true if found and search_result contains found messages ,
2017-11-10 05:00:50 +00:00
* success = false if nothing was found , search_result = 'nothing found' ,
* error : result = error with error message )
*/
2018-02-11 21:13:29 +00:00
function api_friendica_direct_messages_search ( $type , $box = " " )
2017-11-10 05:00:50 +00:00
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// params
$user_info = api_get_user ( $a );
2018-11-30 14:06:22 +00:00
$searchstring = defaults ( $_REQUEST , 'searchstring' , " " );
2017-11-10 05:00:50 +00:00
$uid = $user_info [ 'uid' ];
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// error if no searchstring specified
if ( $searchstring == " " ) {
2018-01-15 13:05:12 +00:00
$answer = [ 'result' => 'error' , 'message' => 'searchstring not specified' ];
return api_format_data ( " direct_messages_search " , $type , [ '$result' => $answer ]);
2017-11-10 05:00:50 +00:00
}
// get data for the specified searchstring
$r = q (
" SELECT `mail`.*, `contact`.`nurl` AS `contact-url` FROM `mail`,`contact` WHERE `mail`.`contact-id` = `contact`.`id` AND `mail`.`uid`=%d AND `body` LIKE '%s' ORDER BY `mail`.`id` DESC " ,
intval ( $uid ),
2018-07-21 13:10:13 +00:00
DBA :: escape ( '%' . $searchstring . '%' )
2017-11-10 05:00:50 +00:00
);
$profile_url = $user_info [ " url " ];
// message if nothing was found
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2018-01-15 13:05:12 +00:00
$success = [ 'success' => false , 'search_results' => 'problem with query' ];
2017-11-10 05:00:50 +00:00
} elseif ( count ( $r ) == 0 ) {
2018-01-15 13:05:12 +00:00
$success = [ 'success' => false , 'search_results' => 'nothing found' ];
2017-11-10 05:00:50 +00:00
} else {
2018-01-15 13:05:12 +00:00
$ret = [];
2017-11-10 05:00:50 +00:00
foreach ( $r as $item ) {
if ( $box == " inbox " || $item [ 'from-url' ] != $profile_url ) {
$recipient = $user_info ;
2018-11-08 16:28:29 +00:00
$sender = api_get_user ( $a , Strings :: normaliseLink ( $item [ 'contact-url' ]));
2017-11-10 05:00:50 +00:00
} elseif ( $box == " sentbox " || $item [ 'from-url' ] == $profile_url ) {
2018-11-08 16:28:29 +00:00
$recipient = api_get_user ( $a , Strings :: normaliseLink ( $item [ 'contact-url' ]));
2017-11-10 05:00:50 +00:00
$sender = $user_info ;
2016-09-25 16:50:08 +00:00
}
2018-04-09 19:34:53 +00:00
if ( isset ( $recipient ) && isset ( $sender )) {
$ret [] = api_format_messages ( $item , $recipient , $sender );
}
2017-11-10 05:00:50 +00:00
}
2018-01-15 13:05:12 +00:00
$success = [ 'success' => true , 'search_results' => $ret ];
2016-09-25 16:50:08 +00:00
}
2017-04-05 20:07:55 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( " direct_message_search " , $type , [ '$result' => $success ]);
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-11-10 05:00:50 +00:00
api_register_func ( 'api/friendica/direct_messages_search' , 'api_friendica_direct_messages_search' , true );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
/**
* @ brief return data of all the profiles a user has to the client
*
* @ param string $type Known types are 'atom' , 'rss' , 'xml' and 'json'
2018-04-09 17:34:02 +00:00
* @ return string | array
2017-11-10 05:00:50 +00:00
*/
function api_friendica_profile_show ( $type )
{
2018-12-28 00:22:35 +00:00
$a = \get_app ();
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
if ( api_user () === false ) {
throw new ForbiddenException ();
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// input params
2018-11-30 14:06:22 +00:00
$profile_id = defaults ( $_REQUEST , 'profile_id' , 0 );
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// retrieve general information about profiles for user
2017-12-04 14:01:27 +00:00
$multi_profiles = Feature :: isEnabled ( api_user (), 'multi_profiles' );
2017-11-10 05:00:50 +00:00
$directory = Config :: get ( 'system' , 'directory' );
2017-04-05 20:07:55 +00:00
2017-11-10 05:00:50 +00:00
// get data of the specified profile id or all profiles of the user if not specified
2018-05-04 21:43:29 +00:00
if ( $profile_id != 0 ) {
2017-11-10 05:00:50 +00:00
$r = q (
" SELECT * FROM `profile` WHERE `uid` = %d AND `id` = %d " ,
intval ( api_user ()),
2018-05-04 21:43:29 +00:00
intval ( $profile_id )
2017-11-10 05:00:50 +00:00
);
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// error message if specified gid is not in database
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $r )) {
2017-11-10 05:00:50 +00:00
throw new BadRequestException ( " profile_id not available " );
}
} else {
$r = q (
" SELECT * FROM `profile` WHERE `uid` = %d " ,
intval ( api_user ())
);
}
// loop through all returned profiles and retrieve data and users
$k = 0 ;
2018-04-09 19:34:53 +00:00
$profiles = [];
2017-11-10 05:00:50 +00:00
foreach ( $r as $rr ) {
2018-01-04 01:54:35 +00:00
$profile = api_format_items_profiles ( $rr );
2017-11-10 05:00:50 +00:00
// select all users from contact table, loop and prepare standard return for user data
2018-01-15 13:05:12 +00:00
$users = [];
2018-04-09 19:34:53 +00:00
$nurls = q (
2017-11-10 05:00:50 +00:00
" SELECT `id`, `nurl` FROM `contact` WHERE `uid`= %d AND `profile-id` = %d " ,
intval ( api_user ()),
2018-08-17 03:19:42 +00:00
intval ( $rr [ 'id' ])
2017-11-10 05:00:50 +00:00
);
2016-09-25 16:50:08 +00:00
2018-04-09 19:34:53 +00:00
foreach ( $nurls as $nurl ) {
$user = api_get_user ( $a , $nurl [ 'nurl' ]);
2017-11-10 05:00:50 +00:00
( $type == " xml " ) ? $users [ $k ++ . " :user " ] = $user : $users [] = $user ;
}
$profile [ 'users' ] = $users ;
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// add prepared profile data to array for final return
if ( $type == " xml " ) {
$profiles [ $k ++ . " :profile " ] = $profile ;
} else {
$profiles [] = $profile ;
2016-09-25 16:50:08 +00:00
}
2017-11-10 05:00:50 +00:00
}
2016-09-25 16:50:08 +00:00
2017-11-10 05:00:50 +00:00
// return settings, authenticated user and profiles data
2018-08-19 12:46:11 +00:00
$self = DBA :: selectFirst ( 'contact' , [ 'nurl' ], [ 'uid' => api_user (), 'self' => true ]);
2017-03-25 14:16:21 +00:00
2018-01-15 13:05:12 +00:00
$result = [ 'multi_profiles' => $multi_profiles ? true : false ,
2017-11-10 05:00:50 +00:00
'global_dir' => $directory ,
2018-08-19 12:46:11 +00:00
'friendica_owner' => api_get_user ( $a , $self [ 'nurl' ]),
2018-01-15 13:05:12 +00:00
'profiles' => $profiles ];
return api_format_data ( " friendica_profiles " , $type , [ '$result' => $result ]);
2017-11-10 05:00:50 +00:00
}
api_register_func ( 'api/friendica/profile/show' , 'api_friendica_profile_show' , true , API_METHOD_GET );
2016-09-25 16:50:08 +00:00
2017-12-19 13:20:32 +00:00
/**
* Returns a list of saved searches .
*
* @ see https :// developer . twitter . com / en / docs / accounts - and - users / manage - account - settings / api - reference / get - saved_searches - list
*
* @ param string $type Return format : json or xml
*
* @ return string | array
*/
function api_saved_searches_list ( $type )
{
2018-07-20 12:19:26 +00:00
$terms = DBA :: select ( 'search' , [ 'id' , 'term' ], [ 'uid' => local_user ()]);
2017-12-19 13:20:32 +00:00
2018-01-15 13:05:12 +00:00
$result = [];
2017-12-19 13:20:32 +00:00
while ( $term = $terms -> fetch ()) {
2018-01-15 13:05:12 +00:00
$result [] = [
2018-03-04 19:25:02 +00:00
'created_at' => api_date ( time ()),
'id' => intval ( $term [ 'id' ]),
2017-12-19 13:20:32 +00:00
'id_str' => $term [ 'id' ],
2018-03-04 19:25:02 +00:00
'name' => $term [ 'term' ],
'position' => null ,
'query' => $term [ 'term' ]
2018-01-15 13:05:12 +00:00
];
2017-12-19 13:20:32 +00:00
}
2018-07-20 12:19:26 +00:00
DBA :: close ( $terms );
2017-12-19 13:42:13 +00:00
2018-01-15 13:05:12 +00:00
return api_format_data ( " terms " , $type , [ 'terms' => $result ]);
2017-12-19 13:20:32 +00:00
}
2017-12-25 20:12:08 +00:00
/// @TODO move to top of file or somewhere better
2017-12-19 13:20:32 +00:00
api_register_func ( 'api/saved_searches/list' , 'api_saved_searches_list' , true );
2016-09-25 16:50:08 +00:00
/*
2017-04-05 20:07:55 +00:00
@ TODO Maybe open to implement ?
2016-09-25 16:50:08 +00:00
To . Do :
2017-12-23 23:27:45 +00:00
[ pagename ] => api / 1.1 / statuses / lookup . json
[ id ] => 605138389168451584
[ include_cards ] => true
[ cards_platform ] => Android - 12
[ include_entities ] => true
[ include_my_retweet ] => 1
[ include_rts ] => 1
[ include_reply_count ] => true
[ include_descendent_reply_count ] => true
2016-09-25 16:50:08 +00:00
( ? )
Not implemented by now :
statuses / retweets_of_me
friendships / create
friendships / destroy
friendships / exists
friendships / show
account / update_location
account / update_profile_background_image
blocks / create
blocks / destroy
friendica / profile / update
friendica / profile / create
friendica / profile / delete
Not implemented in status . net :
statuses / retweeted_to_me
statuses / retweeted_by_me
direct_messages / destroy
account / end_session
account / update_delivery_device
notifications / follow
notifications / leave
blocks / exists
blocks / blocking
lists
*/