You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

875 lines
28 KiB

<?php
use Monolog\Formatter\LineFormatter;
use Monolog\Logger;
use Monolog\Handler\SyslogHandler;
use Monolog\Handler\StreamHandler;
use Movim\Image;
use Movim\ImageSize;
/**
* Log an error
*/
function logError(string|Stringable $logs)
{
$log = new Logger('movim');
$log->pushHandler(new SyslogHandler('movim'));
$stream = new StreamHandler(config('paths.log') . '/errors.log');
$stream->setFormatter(new LineFormatter(null, null, true, true));
$log->pushHandler($stream);
$log->error($logs);
}
/**
* Log an info
*/
function logInfo(string|Stringable $logs)
{
if (config('daemon.verbose')) {
$log = new Logger('movim');
$log->pushHandler(new SyslogHandler('movim'));
$stream = new StreamHandler(config('paths.log') . '/info.log');
$stream->setFormatter(new LineFormatter(null, null, true));
$log->pushHandler($stream);
$log->info($logs);
}
}
/**
* Log a string, only used for debug purposes
*/
function logDebug($logs)
{
$log = new Logger('movim');
$log->pushHandler(new StreamHandler(config('paths.log') . '/debug.log'));
if (is_array($logs)) {
$log->debug('', $logs);
} else {
$log->debug($logs);
}
}
/**
* Return a configuration variable
*/
function config(string $key, $default = null)
{
$path = explode('.', $key);
$config = require(CONFIG_PATH . $path[0] . '.php');
if (!isset($path[1])) return $config;
if (array_key_exists($path[1], $config) && !empty($config[$path[1]])) {
$casted = null;
switch ($config[$path[1]]) {
case 'true':
$casted = true;
break;
case 'false':
$casted = false;
break;
default:
$casted = $config[$path[1]];
break;
}
return $casted;
}
return $default;
}
/**
* Check if Opcache is enabled
*/
function isOpcacheEnabled(): bool
{
return is_array(opcache_get_status());
}
/**
* List compilable Opcache files
*/
function listOpcacheCompilableFiles(): array
{
$files = [];
foreach (['vendor', 'app', 'src'] as $dir) {
$directory = new \RecursiveDirectoryIterator(DOCUMENT_ROOT . '/' . $dir);
$iterator = new \RecursiveIteratorIterator($directory);
$regex = new \RegexIterator($iterator, '/^.+\.php$/i', \RecursiveRegexIterator::GET_MATCH);
foreach ($regex as $key => $file) {
array_push($files, $file[0]);
}
}
return $files;
}
/**
* Compile main files in Opcache
*/
function compileOpcache()
{
error_reporting(0);
foreach (listOpcacheCompilableFiles() as $file) {
if (opcache_is_script_cached($file)) {
yield @opcache_invalidate($file, true);
} else {
yield @opcache_compile_file($file);
}
}
error_reporting(1);
}
/**
* Check if the session exists
*/
function isLogged()
{
return (bool)(\Movim\Session::instance())->get('jid');
}
/**
* Return the list of client types
*/
function getClientTypes()
{
return [
'bot' => __('client.bot'),
'console' => __('client.console'),
'pc' => __('client.desktop'),
'phone' => __('client.phone'),
'gateway' => __('client.gateway'),
'handheld' => __('client.phone'),
'web' => __('client.web'),
'registered' => __('client.registered')
];
}
/**
* Resolve infos from a Posts collection
*/
function resolveInfos($postCollection)
{
$serverNodes = $postCollection->map(function ($item) {
return ['server' => $item->server, 'node' => $item->node];
})->unique(fn ($item) => $item['server'] . $item['node']);
if ($serverNodes->isNotEmpty()) {
$first = $serverNodes->first();
$infos = \App\Info::where([
'server' => $first['server'],
'node' => $first['node'],
]);
$serverNodes->skip(1)->each(function ($serverNode) use ($infos) {
$infos->orWhere([
'server' => $serverNode['server'],
'node' => $serverNode['node'],
]);
});
$infos = $infos->get()->keyBy(fn ($item) => $item['server'] . $item['node']);
$postCollection->map(function ($item) use ($infos) {
$item->info = $infos->get($item->server . $item->node);
return $item;
});
return $postCollection;
}
}
/**
* Form to array
*/
function formToArray(stdClass $form): array
{
$values = [];
foreach ($form as $key => $value) {
$values[$key] = $value->value;
}
return $values;
}
/**
* Return the picture or fallback to the placeholder
*/
function getPicture(?string $key, string $placeholder, ImageSize $size = ImageSize::M): string
{
[$width, $height] = match ($size) {
ImageSize::XXL => [1280, 300],
ImageSize::XL => [512, false],
ImageSize::L => [210, false],
ImageSize::M => [120, false],
ImageSize::S => [50, false],
ImageSize::O => [false, false],
};
return (!empty($key) && $url = Image::getOrCreate($key, $width, $height))
? $url
: avatarPlaceholder($placeholder);
}
/**
* Return a XEP to namespace association
*/
function getXepNamespace()
{
return [
'0004' => ['name' => 'Data Forms', 'category' => 'client', 'ns' => 'jabber:x:data'],
'0012' => ['name' => 'Last Activity', 'category' => 'chat', 'ns' => 'jabber:iq:last'],
'0030' => ['name' => 'Service Discovery', 'category' => 'client', 'ns' => 'http://jabber.org/protocol/disco#info'],
'0045' => ['name' => 'Multi-User Chat', 'category' => 'chat', 'ns' => 'http://jabber.org/protocol/muc'],
'0050' => ['name' => 'Ad-Hoc Commands', 'category' => 'client', 'ns' => 'http://jabber.org/protocol/commands'],
'0054' => ['name' => 'vcard-temp', 'category' => 'client', 'ns' => 'vcard-temp'],
'0071' => ['name' => 'XHTML-IM', 'category' => 'chat', 'ns' => 'http://jabber.org/protocol/xhtml-im'],
'0080' => ['name' => 'User Location', 'category' => 'profile', 'ns' => 'http://jabber.org/protocol/geoloc'],
'0084' => ['name' => 'User Avatar', 'category' => 'profile', 'ns' => 'urn:xmpp:avatar:data'],
'0085' => ['name' => 'Chat State Notifications', 'category' => 'chat', 'ns' => 'http://jabber.org/protocol/chatstates'],
'0092' => ['name' => 'Software Version', 'category' => 'client', 'ns' => 'jabber:iq:version'],
'0107' => ['name' => 'User Mood', 'category' => 'profile', 'ns' => 'http://jabber.org/protocol/mood'],
'0108' => ['name' => 'User Activity', 'category' => 'profile', 'ns' => 'http://jabber.org/protocol/activity'],
'0115' => ['name' => 'Entity Capabilities', 'category' => 'client', 'ns' => 'http://jabber.org/protocol/caps'],
'0118' => ['name' => 'User Tune', 'category' => 'profile', 'ns' => 'http://jabber.org/protocol/tune'],
'0124' => ['name' => 'Bidirectional-streams Over Synchronous HTTP (BOSH]', 'category' => 'client', 'ns' => 'http://jabber.org/protocol/httpbind'],
'0152' => ['name' => 'Reachability Addresses', 'category' => 'client', 'ns' => 'urn:xmpp:reach:0'],
'0166' => ['name' => 'Jingle', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:1'],
'0167' => ['name' => 'Jingle RTP Sessions', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:apps:rtp:1'],
'0172' => ['name' => 'User Nickname', 'category' => 'profile', 'ns' => 'http://jabber.org/protocol/nick'],
'0176' => ['name' => 'Jingle ICE-UDP Transport Method', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:transports:ice-udp:1'],
'0177' => ['name' => 'Jingle Raw UDP Transport Method', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:transports:raw-udp:1'],
'0184' => ['name' => 'Message Delivery Receipts', 'category' => 'chat', 'ns' => 'urn:xmpp:receipts'],
'0186' => ['name' => 'Invisible Command', 'category' => 'chat', 'ns' => 'urn:xmpp:invisible:0'],
'0199' => ['name' => 'XMPP Ping', 'category' => 'client', 'ns' => 'urn:xmpp:ping'],
'0202' => ['name' => 'Entity Time', 'category' => 'client', 'ns' => 'urn:xmpp:time'],
'0224' => ['name' => 'Attention', 'category' => 'chat', 'ns' => 'urn:xmpp:attention:0'],
'0231' => ['name' => 'Bits of Binary', 'category' => 'chat', 'ns' => 'urn:xmpp:bob'],
'0234' => ['name' => 'Jingle File Transfer', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:apps:file-transfer:4'],
'0249' => ['name' => 'Direct MUC Invitations', 'category' => 'chat', 'ns' => 'jabber:x:conference'],
'0277' => ['name' => 'Microblogging over XMPP', 'category' => 'social', 'ns' => 'urn:xmpp:microblog:0'],
'0280' => ['name' => 'Message Carbons', 'category' => 'chat', 'ns' => 'urn:xmpp:carbons:2'],
'0292' => ['name' => 'vCard4 Over XMPP', 'category' => 'profile', 'ns' => 'urn:xmpp:vcard4'],
'0301' => ['name' => 'In-Band Real Time Text', 'category' => 'chat', 'ns' => 'urn:xmpp:rtt:0'],
'0308' => ['name' => 'Last Message Correction', 'category' => 'chat', 'ns' => 'urn:xmpp:message-correct:0'],
'0320' => ['name' => 'Use of DTLS-SRTP in Jingle Sessions', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:apps:dtls:0'],
'0327' => ['name' => 'Rayo', 'category' => 'rayo', 'ns' => 'urn:xmpp:rayo:0'],
'0330' => ['name' => 'Pubsub Subscription', 'category' => 'social', 'ns' => 'urn:xmpp:pubsub:subscription'],
'0332' => ['name' => 'HTTP over XMPP transport', 'category' => 'client', 'ns' => 'urn:xmpp:http'],
'0333' => ['name' => 'Chat Markers', 'category' => 'chat', 'ns' => 'urn:xmpp:chat-markers:0'],
'0337' => ['name' => 'Event Logging over XMPP', 'category' => 'client', 'ns' => 'urn:xmpp:eventlog'],
'0338' => ['name' => 'Jingle Grouping Framework', 'category' => 'jingle', 'ns' => 'urn:ietf:rfc:5888'],
'0339' => ['name' => 'Source-Specific Media Attributes in Jingle', 'category' => 'jingle', 'ns' => 'urn:ietf:rfc:5576'],
'0340' => ['name' => 'COnferences with LIghtweight BRIdging (COLIBRI]', 'category' => 'jingle', 'ns' => 'http://jitsi.org/protocol/colibri'],
'0341' => ['name' => 'Rayo CPA', 'category' => 'rayo', 'ns' => 'urn:xmpp:rayo:cpa:0'],
'0342' => ['name' => 'Rayo Fax', 'category' => 'rayo', 'ns' => 'urn:xmpp:rayo:fax:1'],
'0348' => ['name' => 'Signing Forms', 'category' => 'client', 'ns' => 'urn:xmpp:xdata:signature:oauth1'],
'0390' => ['name' => 'Entity Capabilities 2.0', 'category' => 'client', 'ns' => 'urn:xmpp:caps:optimize'],
'0391' => ['name' => 'Jingle Encrypted Transports', 'category' => 'jingle', 'ns' => 'urn:xmpp:jingle:jet:0'],
];
}
/**
* Return a list of all the country
*/
function getCountries()
{
return [
'AF' => 'Afghanistan',
'AX' => 'Aland Islands',
'AL' => 'Albania',
'DZ' => 'Algeria',
'AS' => 'American Samoa',
'AD' => 'Andorra',
'AO' => 'Angola',
'AI' => 'Anguilla',
'AQ' => 'Antarctica',
'AG' => 'Antigua and Barbuda',
'AR' => 'Argentina',
'AM' => 'Armenia',
'AW' => 'Aruba',
'AU' => 'Australia',
'AT' => 'Austria',
'AZ' => 'Azerbaijan',
'BS' => 'Bahamas The',
'BH' => 'Bahrain',
'BD' => 'Bangladesh',
'BB' => 'Barbados',
'BY' => 'Belarus',
'BE' => 'Belgium',
'BZ' => 'Belize',
'BJ' => 'Benin',
'BM' => 'Bermuda',
'BT' => 'Bhutan',
'BO' => 'Bolivia',
'BA' => 'Bosnia and Herzegovina',
'BW' => 'Botswana',
'BV' => 'Bouvet Island (Bouvetoya)',
'BR' => 'Brazil',
'IO' => 'British Indian Ocean Territory (Chagos Archipelago)',
'VG' => 'British Virgin Islands',
'BN' => 'Brunei Darussalam',
'BG' => 'Bulgaria',
'BF' => 'Burkina Faso',
'BI' => 'Burundi',
'KH' => 'Cambodia',
'CM' => 'Cameroon',
'CA' => 'Canada',
'CV' => 'Cape Verde',
'KY' => 'Cayman Islands',
'CF' => 'Central African Republic',
'TD' => 'Chad',
'CL' => 'Chile',
'CN' => 'China',
'CX' => 'Christmas Island',
'CC' => 'Cocos (Keeling) Islands',
'CO' => 'Colombia',
'KM' => 'Comoros The',
'CD' => 'Congo',
'CG' => 'Congo The',
'CK' => 'Cook Islands',
'CR' => 'Costa Rica',
'CI' => 'Cote d\'Ivoire',
'HR' => 'Croatia',
'CU' => 'Cuba',
'CY' => 'Cyprus',
'CZ' => 'Czech Republic',
'DK' => 'Denmark',
'DJ' => 'Djibouti',
'DM' => 'Dominica',
'DO' => 'Dominican Republic',
'EC' => 'Ecuador',
'EG' => 'Egypt',
'SV' => 'El Salvador',
'GQ' => 'Equatorial Guinea',
'ER' => 'Eritrea',
'EE' => 'Estonia',
'ET' => 'Ethiopia',
'FO' => 'Faroe Islands',
'FK' => 'Falkland Islands (Malvinas)',
'FJ' => 'Fiji The Fiji Islands',
'FI' => 'Finland',
'FR' => 'France, French Republic',
'GF' => 'French Guiana',
'PF' => 'French Polynesia',
'TF' => 'French Southern Territories',
'GA' => 'Gabon',
'GM' => 'Gambia The',
'GE' => 'Georgia',
'DE' => 'Germany',
'GH' => 'Ghana',
'GI' => 'Gibraltar',
'GR' => 'Greece',
'GL' => 'Greenland',
'GD' => 'Grenada',
'GP' => 'Guadeloupe',
'GU' => 'Guam',
'GT' => 'Guatemala',
'GG' => 'Guernsey',
'GN' => 'Guinea',
'GW' => 'Guinea-Bissau',
'GY' => 'Guyana',
'HT' => 'Haiti',
'HM' => 'Heard Island and McDonald Islands',
'VA' => 'Holy See (Vatican City State)',
'HN' => 'Honduras',
'HK' => 'Hong Kong',
'HU' => 'Hungary',
'IS' => 'Iceland',
'IN' => 'India',
'ID' => 'Indonesia',
'IR' => 'Iran',
'IQ' => 'Iraq',
'IE' => 'Ireland',
'IM' => 'Isle of Man',
'IL' => 'Israel',
'IT' => 'Italy',
'JM' => 'Jamaica',
'JP' => 'Japan',
'JE' => 'Jersey',
'JO' => 'Jordan',
'KZ' => 'Kazakhstan',
'KE' => 'Kenya',
'KI' => 'Kiribati',
'KP' => 'Korea',
'KR' => 'Korea',
'KW' => 'Kuwait',
'KG' => 'Kyrgyz Republic',
'LA' => 'Lao',
'LV' => 'Latvia',
'LB' => 'Lebanon',
'LS' => 'Lesotho',
'LR' => 'Liberia',
'LY' => 'Libyan Arab Jamahiriya',
'LI' => 'Liechtenstein',
'LT' => 'Lithuania',
'LU' => 'Luxembourg',
'MO' => 'Macao',
'MK' => 'Macedonia',
'MG' => 'Madagascar',
'MW' => 'Malawi',
'MY' => 'Malaysia',
'MV' => 'Maldives',
'ML' => 'Mali',
'MT' => 'Malta',
'MH' => 'Marshall Islands',
'MQ' => 'Martinique',
'MR' => 'Mauritania',
'MU' => 'Mauritius',
'YT' => 'Mayotte',
'MX' => 'Mexico',
'FM' => 'Micronesia',
'MD' => 'Moldova',
'MC' => 'Monaco',
'MN' => 'Mongolia',
'ME' => 'Montenegro',
'MS' => 'Montserrat',
'MA' => 'Morocco',
'MZ' => 'Mozambique',
'MM' => 'Myanmar',
'NA' => 'Namibia',
'NR' => 'Nauru',
'NP' => 'Nepal',
'AN' => 'Netherlands Antilles',
'NL' => 'Netherlands The',
'NC' => 'New Caledonia',
'NZ' => 'New Zealand',
'NI' => 'Nicaragua',
'NE' => 'Niger',
'NG' => 'Nigeria',
'NU' => 'Niue',
'NF' => 'Norfolk Island',
'MP' => 'Northern Mariana Islands',
'NO' => 'Norway',
'OM' => 'Oman',
'PK' => 'Pakistan',
'PW' => 'Palau',
'PS' => 'Palestinian Territory',
'PA' => 'Panama',
'PG' => 'Papua New Guinea',
'PY' => 'Paraguay',
'PE' => 'Peru',
'PH' => 'Philippines',
'PN' => 'Pitcairn Islands',
'PL' => 'Poland',
'PT' => 'Portugal, Portuguese Republic',
'PR' => 'Puerto Rico',
'QA' => 'Qatar',
'RE' => 'Reunion',
'RO' => 'Romania',
'RU' => 'Russian Federation',
'RW' => 'Rwanda',
'BL' => 'Saint Barthelemy',
'SH' => 'Saint Helena',
'KN' => 'Saint Kitts and Nevis',
'LC' => 'Saint Lucia',
'MF' => 'Saint Martin',
'PM' => 'Saint Pierre and Miquelon',
'VC' => 'Saint Vincent and The Grenadines',
'WS' => 'Samoa',
'SM' => 'San Marino',
'ST' => 'Sao Tome and Principe',
'SA' => 'Saudi Arabia',
'SN' => 'Senegal',
'RS' => 'Serbia',
'SC' => 'Seychelles',
'SL' => 'Sierra Leone',
'SG' => 'Singapore',
'SK' => 'Slovakia (Slovak Republic)',
'SI' => 'Slovenia',
'SB' => 'Solomon Islands',
'SO' => 'Somalia, Somali Republic',
'ZA' => 'South Africa',
'GS' => 'South Georgia and The South Sandwich Islands',
'ES' => 'Spain',
'LK' => 'Sri Lanka',
'SD' => 'Sudan',
'SR' => 'Suriname',
'SJ' => 'Svalbard & Jan Mayen Islands',
'SZ' => 'Swaziland',
'SE' => 'Sweden',
'CH' => 'Switzerland, Swiss Confederation',
'SY' => 'Syrian Arab Republic',
'TW' => 'Taiwan',
'TJ' => 'Tajikistan',
'TZ' => 'Tanzania',
'TH' => 'Thailand',
'TL' => 'Timor-Leste',
'TG' => 'Togo',
'TK' => 'Tokelau',
'TO' => 'Tonga',
'TT' => 'Trinidad and Tobago',
'TN' => 'Tunisia',
'TR' => 'Turkey',
'TM' => 'Turkmenistan',
'TC' => 'Turks and Caicos Islands',
'TV' => 'Tuvalu',
'UG' => 'Uganda',
'UA' => 'Ukraine',
'AE' => 'United Arab Emirates',
'GB' => 'United Kingdom',
'US' => 'United States of America',
'UM' => 'United States Minor Outlying Islands',
'VI' => 'United States Virgin Islands',
'UY' => 'Uruguay, Eastern Republic of',
'UZ' => 'Uzbekistan',
'VU' => 'Vanuatu',
'VE' => 'Venezuela',
'VN' => 'Vietnam',
'WF' => 'Wallis and Futuna',
'EH' => 'Western Sahara',
'YE' => 'Yemen',
'ZM' => 'Zambia',
'ZW' => 'Zimbabwe'
];
}
function getImgurThumbnail(string $uri)
{
$matches = [];
preg_match('/https?:\/\/i.imgur.com\/([a-zA-Z0-9]{7})(.*)/', $uri, $matches);
if (!empty($matches)) {
return 'https://i.imgur.com/' . $matches[1] . 'g' . $matches[2];
}
}
function getPresences()
{
return [
1 => __('presence.online'),
2 => __('presence.away'),
3 => __('presence.dnd'),
4 => __('presence.xa'),
5 => __('presence.offline'),
6 => __('presence.error')
];
}
function getPresencesTxt()
{
return [
1 => 'online',
2 => 'away',
3 => 'dnd',
4 => 'xa',
5 => 'offline',
6 => 'server_error'
];
}
/**
* Map the XMPP form vars to Material Symbols
*/
function varToIcons(string $var)
{
$icons = [
// Pubsub
'pubsub#deliver_payloads' => 'add_box',
'pubsub#deliver_notifications' => 'notifications',
'pubsub#notify_config' => 'notifications',
'pubsub#notify_delete' => 'delete',
'pubsub#notify_retract' => 'delete_sweep',
'pubsub#persist_items' => 'save',
'pubsub#presence_based_delivery' => 'notifications_active',
'pubsub#purge_offline' => 'delete_forever',
'pubsub#subscribe' => 'how_to_reg',
'pubsub#type' => 'space_dashboard',
// Muc
'muc#roomconfig_persistentroom' => 'save',
'muc#roomconfig_publicroom' => 'wifi_tethering',
'muc#roomconfig_passwordprotectedroom' => 'lock',
'muc#roomconfig_membersonly' => 'playlist_add_check',
'muc#roomconfig_moderatedroom' => 'stars',
'muc#roomconfig_changesubject' => 'title',
'muc#roomconfig_allowinvites' => 'mail',
'allow_private_messages' => 'message',
'allow_query_users' => 'portrait',
'allow_visitor_nickchange' => '3p',
'allow_visitor_status' => 'description',
'allow_voice_requests' => 'voice_selection',
'allow_subscription' => 'how_to_reg',
'enable_hats' => 'badge',
'mam' => 'archive',
'members_by_default' => 'remember_me',
'public_list' => 'public',
];
if (array_key_exists($var, $icons)) {
return $icons[$var];
}
return 'tune';
}
/**
* Generate a standard UUID
*/
function generateUUID($string = false)
{
$data = ($string != false) ? $string : openssl_random_pseudo_bytes(16);
$data[6] = chr(ord($data[6]) & 0x0f | 0x40); // set version to 0010
$data[8] = chr(ord($data[8]) & 0x3f | 0x80); // set bits 6-7 to 10
return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}
/**
* @desc Generate a simple random key
* @params The size of the key
*/
function generateKey(?int $size = 16): string
{
$hashChars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
$hash = '';
for ($i = 0; $i < $size; $i++) {
$hash .= $hashChars[random_int(0, strlen($hashChars) - 1)];
}
return $hash;
}
define('DEFAULT_HTTP_USER_AGENT', 'Mozilla/5.0 (Android 4.4; Mobile; rv:41.0) Gecko/41.0 Firefox/41.0');
/**
* @desc Request a url async
*/
function requestAsyncURL(string $url, int $timeout = 10, array $headers = [])
{
$browser = new React\Http\Browser;
return $browser->withTimeout($timeout)->get($url, $headers);
}
/*
* @desc Request a simple url
*/
function requestURL(string $url, int $timeout = 10, $post = false, bool $json = false, array $headers = [])
{
if ($json) {
array_push($headers, 'Accept: application/json');
}
$ch = curl_init($url);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
//curl_setopt($ch, CURLOPT_ENCODING, 'gzip,deflate');
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
curl_setopt($ch, CURLOPT_TIMEOUT, $timeout);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $timeout);
curl_setopt($ch, CURLOPT_USERAGENT, DEFAULT_HTTP_USER_AGENT);
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
if ($post) {
curl_setopt($ch, CURLOPT_POST, 1);
curl_setopt($ch, CURLOPT_POSTFIELDS, $post);
}
// Disable SSL if the host requested is the local one
if (parse_url(config('daemon.url'), PHP_URL_HOST) == parse_url($url, PHP_URL_HOST)) {
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0);
}
$content = curl_exec($ch);
return curl_errno($ch) == 0 ? $content : false;
}
/*
* Request the headers of a URL
*/
function requestHeaders(string $url, $timeout = 2)
{
$ch = curl_init($url);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
//curl_setopt($ch, CURLOPT_ENCODING, 'gzip,deflate');
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
curl_setopt($ch, CURLOPT_TIMEOUT, $timeout);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $timeout);
curl_setopt($ch, CURLOPT_HEADER, 1);
curl_setopt($ch, CURLOPT_NOBODY, 1);
curl_setopt($ch, CURLOPT_USERAGENT, DEFAULT_HTTP_USER_AGENT);
curl_exec($ch);
return curl_getinfo($ch);
}
/**
* Request the internal API
*/
function requestAPI(string $action, int $timeout = 2, $post = false)
{
$ch = curl_init('http:/' . $action);
curl_setopt($ch, CURLOPT_TIMEOUT, $timeout);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $timeout);
curl_setopt($ch, CURLOPT_UNIX_SOCKET_PATH, API_SOCKET);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
if (is_array($post)) {
curl_setopt($ch, CURLOPT_POST, 1);
curl_setopt($ch, CURLOPT_POSTFIELDS, $post);
}
$content = curl_exec($ch);
return curl_errno($ch) == 0 ? $content : false;
}
/**
* @desc Get distance between two coordinates
*
* @param float $latitudeFrom
* @param float $longitudeFrom
* @param float $latitudeTo
* @param float $longitudeTo
*
* @return float [km]
*/
function getDistance(float $latitudeFrom, float $longitudeFrom, float $latitudeTo, float $longitudeTo): float
{
$rad = M_PI / 180;
$theta = $longitudeFrom - $longitudeTo;
$dist = sin($latitudeFrom * $rad)
* sin($latitudeTo * $rad) + cos($latitudeFrom * $rad)
* cos($latitudeTo * $rad) * cos($theta * $rad);
return acos($dist) / $rad * 60 * 1.853;
}
/*
* @desc Get the URI of a smiley
*/
function getSmileyPath($id)
{
return BASE_URI . 'theme/img/emojis/svg/' . $id . '.svg';
}
/**
* @desc Return the url of an avatar placeholder
*/
function avatarPlaceholder(string $id): string
{
return \Movim\Route::urlize('picture', false, ['type' => 'avatar', 'id' => urlencode($id)]);
}
/*
* @desc Protect a picture URL by using the internal Proxy
*/
function protectPicture($url)
{
$emptyPicture = \Movim\Route::urlize('picture', '');
$emptyPicture = preg_replace("(^//)", 'https://', $emptyPicture);
// The picture is already protected
if (substr($url, 0, strlen($emptyPicture)) === $emptyPicture) {
return $url;
}
return \Movim\Route::urlize('picture', false, ['type' => 'picture', 'url' => urlencode($url)]);
}
/*
* @desc Translate something
*/
function __()
{
$args = func_get_args();
$l = Movim\i18n\Locale::start();
$string = array_shift($args);
return $l->translate($string, $args);
}
/*
* @desc Get the browser name from a user agent
*/
function getBrowser(string $userAgent): ?string
{
$t = strtolower($userAgent);
$t = ' ' . $t;
if (strpos($t, 'opera')) return 'Opera';
elseif (strpos($t, 'edge')) return 'Edge';
elseif (strpos($t, 'chrome')) return 'Chrome';
elseif (strpos($t, 'safari')) return 'Safari';
elseif (strpos($t, 'firefox')) return 'Firefox';
}
/*
* @desc Get the platform from the user agent
*/
function getPlatform(string $userAgent): ?string
{
$oses = [
'/windows nt 10/i' => 'Windows 10',
'/windows nt 6.3/i' => 'Windows 8.1',
'/windows nt 6.2/i' => 'Windows 8',
'/windows nt 6.1/i' => 'Windows 7',
'/windows nt 6.0/i' => 'Windows Vista',
'/macintosh|mac os x/i' => 'Mac OS X',
'/mac_powerpc/i' => 'Mac OS 9',
'/linux/i' => 'Linux',
'/ubuntu/i' => 'Ubuntu',
'/iphone/i' => 'iPhone',
'/ipod/i' => 'iPod',
'/ipad/i' => 'iPad',
'/android/i' => 'Android',
'/blackberry/i' => 'BlackBerry',
'/webos/i' => 'Mobile'
];
foreach ($oses as $regex => $value) {
if (preg_match($regex, $userAgent)) {
return $value;
}
}
}
/**
* @desc Get PHP hash to IANA hashes conversion
* https://www.iana.org/assignments/hash-function-text-names/hash-function-text-names.xhtml
* https://www.php.net/manual/en/function.hash-algos.php
*/
function phpToIANAHash(): array
{
return [
'md2' => 'md2',
'md5' => 'md5',
'sha1' => 'sha1',
'sha224' => 'sha-224',
'sha256' => 'sha-256',
'sha384' => 'sha-384',
'sha512' => 'sha-512',
];
}
function IANAHashToPhp(): array
{
return [
'md2' => 'md2',
'md5' => 'md5',
'sha1' => 'sha1', // https://xmpp.org/extensions/xep-0231.html#algo
'sha-224' => 'sha224',
'sha256' => 'sha256', // retro-compatibility
'sha-256' => 'sha256',
'sha-384' => 'sha384',
'sha-512' => 'sha512',
];
}