2017-08-26 02:04:21 -04:00
< ? php
2017-11-19 14:15:25 -05:00
/**
2021-03-29 02:40:20 -04:00
* @ copyright Copyright ( C ) 2010 - 2021 , the Friendica project
2020-02-09 09:45:36 -05:00
*
* @ license GNU AGPL version 3 or any later version
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation , either version 3 of the
* License , or ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU Affero General Public License for more details .
*
* You should have received a copy of the GNU Affero General Public License
* along with this program . If not , see < https :// www . gnu . org / licenses />.
*
2017-11-19 14:15:25 -05:00
*/
2020-02-09 09:45:36 -05:00
2017-08-26 02:04:21 -04:00
namespace Friendica\Core ;
2021-11-04 16:29:59 -04:00
use Exception ;
2019-12-15 16:34:11 -05:00
use Friendica\DI ;
2021-10-29 19:21:07 -04:00
use Friendica\Network\HTTPException\FoundException ;
use Friendica\Network\HTTPException\MovedPermanentlyException ;
use Friendica\Network\HTTPException\TemporaryRedirectException ;
2021-11-04 16:29:59 -04:00
use Friendica\Util\BasePath ;
2018-01-27 11:59:10 -05:00
use Friendica\Util\XML ;
2017-08-26 02:04:21 -04:00
/**
2020-01-19 01:05:23 -05:00
* Contains the class with system relevant stuff
2017-08-26 02:04:21 -04:00
*/
2019-12-15 17:28:01 -05:00
class System
2017-11-19 14:15:25 -05:00
{
2017-08-26 06:01:50 -04:00
/**
2020-01-19 01:05:23 -05:00
* Returns a string with a callstack . Can be used for logging .
2020-01-19 04:51:37 -05:00
*
2020-07-27 00:20:30 -04:00
* @ param integer $depth How many calls to include in the stacks after filtering
* @ param int $offset How many calls to shave off the top of the stack , for example if
* this is called from a centralized method that isn ' t relevant to the callstack
2017-08-26 06:01:50 -04:00
* @ return string
*/
2020-07-27 00:20:30 -04:00
public static function callstack ( int $depth = 4 , int $offset = 0 )
2017-11-19 14:15:25 -05:00
{
2017-10-17 16:51:46 -04:00
$trace = debug_backtrace ( DEBUG_BACKTRACE_IGNORE_ARGS );
2017-08-26 06:01:50 -04:00
2020-07-27 00:20:30 -04:00
// We remove at least the first two items from the list since they contain data that we don't need.
$trace = array_slice ( $trace , 2 + $offset );
2017-08-26 06:01:50 -04:00
2018-01-15 08:05:12 -05:00
$callstack = [];
2020-07-10 03:01:28 -04:00
$previous = [ 'class' => '' , 'function' => '' , 'database' => false ];
2017-10-17 16:51:46 -04:00
// The ignore list contains all functions that are only wrapper functions
2020-07-27 07:50:36 -04:00
$ignore = [ 'call_user_func_array' ];
2017-10-17 16:51:46 -04:00
while ( $func = array_pop ( $trace )) {
2017-08-26 06:01:50 -04:00
if ( ! empty ( $func [ 'class' ])) {
2021-02-14 09:24:48 -05:00
if ( in_array ( $previous [ 'function' ], [ 'insert' , 'fetch' , 'toArray' , 'exists' , 'count' , 'selectFirst' , 'selectToArray' ,
'select' , 'update' , 'delete' , 'selectFirstForUser' , 'selectForUser' ])
&& ( substr ( $previous [ 'class' ], 0 , 15 ) === 'Friendica\Model' )) {
continue ;
}
2020-07-10 03:01:28 -04:00
// Don't show multiple calls from the Database classes to show the essential parts of the callstack
$func [ 'database' ] = in_array ( $func [ 'class' ], [ 'Friendica\Database\DBA' , 'Friendica\Database\Database' ]);
if ( ! $previous [ 'database' ] || ! $func [ 'database' ]) {
2017-10-17 16:51:46 -04:00
$classparts = explode ( " \\ " , $func [ 'class' ]);
$callstack [] = array_pop ( $classparts ) . '::' . $func [ 'function' ];
$previous = $func ;
}
} elseif ( ! in_array ( $func [ 'function' ], $ignore )) {
2020-07-10 03:01:28 -04:00
$func [ 'database' ] = ( $func [ 'function' ] == 'q' );
2017-08-26 06:01:50 -04:00
$callstack [] = $func [ 'function' ];
2018-07-01 00:15:11 -04:00
$func [ 'class' ] = '' ;
2017-10-17 16:51:46 -04:00
$previous = $func ;
2017-08-26 06:01:50 -04:00
}
}
2018-01-15 08:05:12 -05:00
$callstack2 = [];
2017-10-17 16:51:46 -04:00
while (( count ( $callstack2 ) < $depth ) && ( count ( $callstack ) > 0 )) {
$callstack2 [] = array_pop ( $callstack );
}
return implode ( ', ' , $callstack2 );
2017-08-26 06:01:50 -04:00
}
2018-01-27 11:59:10 -05:00
/**
* Generic XML return
* Outputs a basic dfrn XML status structure to STDOUT , with a < status > variable
* of $st and an optional text < message > of $message and terminates the current process .
2019-01-06 16:06:53 -05:00
*
* @ param $st
* @ param string $message
* @ throws \Exception
2018-01-27 11:59:10 -05:00
*/
public static function xmlExit ( $st , $message = '' )
{
$result = [ 'status' => $st ];
if ( $message != '' ) {
$result [ 'message' ] = $message ;
}
if ( $st ) {
2021-11-03 19:19:24 -04:00
Logger :: notice ( 'xml_status returning non_zero: ' . $st . " message= " . $message );
2018-01-27 11:59:10 -05:00
}
header ( " Content-type: text/xml " );
$xmldata = [ " result " => $result ];
echo XML :: fromArray ( $xmldata , $xml );
2018-12-26 00:40:12 -05:00
exit ();
2018-01-27 11:59:10 -05:00
}
/**
2020-01-19 01:05:23 -05:00
* Send HTTP status header and exit .
2018-01-27 11:59:10 -05:00
*
2019-05-01 21:33:33 -04:00
* @ param integer $val HTTP status result value
* @ param string $message Error message . Optional .
* @ param string $content Response body . Optional .
* @ throws \Exception
2018-01-27 11:59:10 -05:00
*/
2019-05-01 21:33:33 -04:00
public static function httpExit ( $val , $message = '' , $content = '' )
2018-01-27 11:59:10 -05:00
{
2021-10-29 19:21:07 -04:00
if ( $val >= 400 ) {
Logger :: debug ( 'Exit with error' , [ 'code' => $val , 'message' => $message , 'callstack' => System :: callstack ( 20 ), 'method' => $_SERVER [ 'REQUEST_METHOD' ], 'agent' => $_SERVER [ 'HTTP_USER_AGENT' ] ? ? '' ]);
}
2019-05-01 21:33:33 -04:00
header ( $_SERVER [ " SERVER_PROTOCOL " ] . ' ' . $val . ' ' . $message );
2018-01-27 11:59:10 -05:00
2019-05-01 21:33:33 -04:00
echo $content ;
2018-01-27 11:59:10 -05:00
2018-10-21 22:24:47 -04:00
exit ();
2018-01-27 11:59:10 -05:00
}
2019-02-23 15:25:41 -05:00
public static function jsonError ( $httpCode , $data , $content_type = 'application/json' )
{
2021-10-29 19:21:07 -04:00
if ( $httpCode >= 400 ) {
Logger :: debug ( 'Exit with error' , [ 'code' => $httpCode , 'content_type' => $content_type , 'callstack' => System :: callstack ( 20 ), 'method' => $_SERVER [ 'REQUEST_METHOD' ], 'agent' => $_SERVER [ 'HTTP_USER_AGENT' ] ? ? '' ]);
}
2019-02-23 15:25:41 -05:00
header ( $_SERVER [ " SERVER_PROTOCOL " ] . ' ' . $httpCode );
self :: jsonExit ( $data , $content_type );
}
2018-01-27 11:59:10 -05:00
/**
2020-01-19 01:05:23 -05:00
* Encodes content to json .
2018-01-27 11:59:10 -05:00
*
* This function encodes an array to json format
* and adds an application / json HTTP header to the output .
* After finishing the process is getting killed .
*
2020-07-13 09:26:09 -04:00
* @ param mixed $x The input content .
* @ param string $content_type Type of the input ( Default : 'application/json' ) .
* @ param integer $options JSON options
2018-01-27 11:59:10 -05:00
*/
2020-07-13 12:24:44 -04:00
public static function jsonExit ( $x , $content_type = 'application/json' , int $options = 0 ) {
2018-06-18 17:05:44 -04:00
header ( " Content-type: $content_type " );
2020-07-13 09:26:09 -04:00
echo json_encode ( $x , $options );
2018-12-26 00:40:12 -05:00
exit ();
2018-01-27 11:59:10 -05:00
}
2018-09-26 16:03:46 -04:00
/**
* Generates a random string in the UUID format
*
2019-01-06 16:06:53 -05:00
* @ param bool | string $prefix A given prefix ( default is empty )
2018-09-26 16:03:46 -04:00
* @ return string a generated UUID
2019-01-06 16:06:53 -05:00
* @ throws \Exception
2018-09-26 16:03:46 -04:00
*/
2018-09-27 07:52:15 -04:00
public static function createUUID ( $prefix = '' )
2018-09-26 16:03:46 -04:00
{
$guid = System :: createGUID ( 32 , $prefix );
2018-11-22 11:25:43 -05:00
return substr ( $guid , 0 , 8 ) . '-' . substr ( $guid , 8 , 4 ) . '-' . substr ( $guid , 12 , 4 ) . '-' . substr ( $guid , 16 , 4 ) . '-' . substr ( $guid , 20 , 12 );
2018-09-26 16:03:46 -04:00
}
2018-07-09 15:38:16 -04:00
/**
* Generates a GUID with the given parameters
*
2019-01-06 16:06:53 -05:00
* @ param int $size The size of the GUID ( default is 16 )
* @ param bool | string $prefix A given prefix ( default is empty )
2018-07-09 15:38:16 -04:00
* @ return string a generated GUID
2019-01-06 16:06:53 -05:00
* @ throws \Exception
2018-07-09 15:38:16 -04:00
*/
public static function createGUID ( $size = 16 , $prefix = '' )
{
if ( is_bool ( $prefix ) && ! $prefix ) {
$prefix = '' ;
2018-07-09 16:10:35 -04:00
} elseif ( empty ( $prefix )) {
2019-12-15 18:47:24 -05:00
$prefix = hash ( 'crc32' , DI :: baseUrl () -> getHostname ());
2018-07-09 15:38:16 -04:00
}
while ( strlen ( $prefix ) < ( $size - 13 )) {
$prefix .= mt_rand ();
}
if ( $size >= 24 ) {
$prefix = substr ( $prefix , 0 , $size - 22 );
return str_replace ( '.' , '' , uniqid ( $prefix , true ));
} else {
$prefix = substr ( $prefix , 0 , max ( $size - 13 , 0 ));
return uniqid ( $prefix );
}
}
2018-10-13 12:57:31 -04:00
/**
* Returns the current Load of the System
*
* @ return integer
*/
public static function currentLoad ()
{
if ( ! function_exists ( 'sys_getloadavg' )) {
return false ;
}
$load_arr = sys_getloadavg ();
if ( ! is_array ( $load_arr )) {
return false ;
}
return max ( $load_arr [ 0 ], $load_arr [ 1 ]);
}
2018-10-19 14:11:27 -04:00
/**
* Redirects to an external URL ( fully qualified URL )
* If you want to route relative to the current Friendica base , use App -> internalRedirect ()
*
2019-05-04 03:16:37 -04:00
* @ param string $url The new Location to redirect
* @ param int $code The redirection code , which is used ( Default is 302 )
2018-10-19 14:11:27 -04:00
*/
2019-05-04 03:16:37 -04:00
public static function externalRedirect ( $url , $code = 302 )
2018-10-19 14:11:27 -04:00
{
2018-12-03 10:59:53 -05:00
if ( empty ( parse_url ( $url , PHP_URL_SCHEME ))) {
2021-11-01 17:21:03 -04:00
Logger :: warning ( 'No fully qualified URL provided' , [ 'url' => $url , 'callstack' => self :: callstack ( 20 )]);
DI :: baseUrl () -> redirect ( $url );
2018-10-19 14:11:27 -04:00
}
2021-10-29 19:21:07 -04:00
header ( " Location: $url " );
2019-05-04 03:16:37 -04:00
switch ( $code ) {
case 302 :
2021-10-29 19:21:07 -04:00
throw new FoundException ();
2019-05-04 03:16:37 -04:00
case 301 :
2021-10-29 19:21:07 -04:00
throw new MovedPermanentlyException ();
2019-05-04 07:42:26 -04:00
case 307 :
2021-10-29 19:21:07 -04:00
throw new TemporaryRedirectException ();
2019-05-04 03:16:37 -04:00
}
2018-10-19 14:11:27 -04:00
exit ();
}
2019-02-03 16:22:04 -05:00
/**
2020-01-19 01:05:23 -05:00
* Returns the system user that is executing the script
2019-02-03 16:22:04 -05:00
*
* This mostly returns something like " www-data " .
*
* @ return string system username
*/
public static function getUser ()
{
if ( ! function_exists ( 'posix_getpwuid' ) || ! function_exists ( 'posix_geteuid' )) {
return '' ;
}
$processUser = posix_getpwuid ( posix_geteuid ());
return $processUser [ 'name' ];
}
2019-02-05 16:30:18 -05:00
/**
2020-01-19 01:05:23 -05:00
* Checks if a given directory is usable for the system
2019-02-05 16:30:18 -05:00
*
* @ param $directory
* @ param bool $check_writable
*
* @ return boolean the directory is usable
*/
public static function isDirectoryUsable ( $directory , $check_writable = true )
{
if ( $directory == '' ) {
2021-11-03 19:19:24 -04:00
Logger :: info ( 'Directory is empty. This shouldn\'t happen.' );
2019-02-05 16:30:18 -05:00
return false ;
}
if ( ! file_exists ( $directory )) {
2021-11-03 19:19:24 -04:00
Logger :: info ( 'Path "' . $directory . '" does not exist for user ' . static :: getUser ());
2019-02-05 16:30:18 -05:00
return false ;
}
if ( is_file ( $directory )) {
2021-11-03 19:19:24 -04:00
Logger :: info ( 'Path "' . $directory . '" is a file for user ' . static :: getUser ());
2019-02-05 16:30:18 -05:00
return false ;
}
if ( ! is_dir ( $directory )) {
2021-11-03 19:19:24 -04:00
Logger :: info ( 'Path "' . $directory . '" is not a directory for user ' . static :: getUser ());
2019-02-05 16:30:18 -05:00
return false ;
}
if ( $check_writable && ! is_writable ( $directory )) {
2021-11-03 19:19:24 -04:00
Logger :: info ( 'Path "' . $directory . '" is not writable for user ' . static :: getUser ());
2019-02-05 16:30:18 -05:00
return false ;
}
return true ;
}
2020-02-16 04:32:56 -05:00
/**
* Exit method used by asynchronous update modules
*
* @ param string $o
*/
public static function htmlUpdateExit ( $o )
{
header ( " Content-type: text/html " );
echo " <!DOCTYPE html><html><body> \r \n " ;
// We can remove this hack once Internet Explorer recognises HTML5 natively
echo " <section> " ;
// reportedly some versions of MSIE don't handle tabs in XMLHttpRequest documents very well
echo str_replace ( " \t " , " " , $o );
echo " </section> " ;
echo " </body></html> \r \n " ;
exit ();
}
2021-11-04 16:29:59 -04:00
/**
* Fetch the temp path of the system
*
* @ return string Path for temp files
*/
public static function getTempPath ()
{
$temppath = DI :: config () -> get ( " system " , " temppath " );
if (( $temppath != " " ) && System :: isDirectoryUsable ( $temppath )) {
// We have a temp path and it is usable
return BasePath :: getRealPath ( $temppath );
}
// We don't have a working preconfigured temp path, so we take the system path.
$temppath = sys_get_temp_dir ();
// Check if it is usable
if (( $temppath != " " ) && System :: isDirectoryUsable ( $temppath )) {
// Always store the real path, not the path through symlinks
$temppath = BasePath :: getRealPath ( $temppath );
// To avoid any interferences with other systems we create our own directory
$new_temppath = $temppath . " / " . DI :: baseUrl () -> getHostname ();
if ( ! is_dir ( $new_temppath )) {
/// @TODO There is a mkdir()+chmod() upwards, maybe generalize this (+ configurable) into a function/method?
mkdir ( $new_temppath );
}
if ( System :: isDirectoryUsable ( $new_temppath )) {
// The new path is usable, we are happy
DI :: config () -> set ( " system " , " temppath " , $new_temppath );
return $new_temppath ;
} else {
// We can't create a subdirectory, strange.
// But the directory seems to work, so we use it but don't store it.
return $temppath ;
}
}
// Reaching this point means that the operating system is configured badly.
return '' ;
}
/**
* Returns the path where spool files are stored
*
* @ return string Spool path
*/
public static function getSpoolPath ()
{
$spoolpath = DI :: config () -> get ( 'system' , 'spoolpath' );
if (( $spoolpath != " " ) && System :: isDirectoryUsable ( $spoolpath )) {
// We have a spool path and it is usable
return $spoolpath ;
}
// We don't have a working preconfigured spool path, so we take the temp path.
$temppath = self :: getTempPath ();
if ( $temppath != " " ) {
// To avoid any interferences with other systems we create our own directory
$spoolpath = $temppath . " /spool " ;
if ( ! is_dir ( $spoolpath )) {
mkdir ( $spoolpath );
}
if ( System :: isDirectoryUsable ( $spoolpath )) {
// The new path is usable, we are happy
DI :: config () -> set ( " system " , " spoolpath " , $spoolpath );
return $spoolpath ;
} else {
// We can't create a subdirectory, strange.
// But the directory seems to work, so we use it but don't store it.
return $temppath ;
}
}
// Reaching this point means that the operating system is configured badly.
return " " ;
}
2017-08-26 02:04:21 -04:00
}