2016-08-18 20:39:31 +00:00
|
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
G\ library
|
|
|
|
|
http://gbackslash.com
|
|
|
|
|
|
|
|
|
|
@author Rodolfo Berrios A. <http://rodolfoberrios.com/>
|
|
|
|
|
|
|
|
|
|
Copyright (c) Rodolfo Berrios <inbox@rodolfoberrios.com> All rights reserved.
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
Licensed under the MIT license
|
|
|
|
|
http://opensource.org/licenses/MIT
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
--------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
namespace G {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-10-12 16:32:18 +00:00
|
|
|
|
use Exception;
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* ROUTE HELPERS
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns true if the $route is current /route or mapped-route -> /route
|
|
|
|
|
function is_route($route){
|
|
|
|
|
return Handler::$base_request == $route;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns true if the $route is has route file
|
|
|
|
|
function is_route_available($route) {
|
|
|
|
|
$route_file = 'route.'.$route.'.php';
|
|
|
|
|
return file_exists(G_APP_PATH_ROUTES . $route_file) or file_exists(G_APP_PATH_ROUTES_OVERRIDES . $route_file);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns true if the route is prevented
|
|
|
|
|
function is_prevented_route() {
|
|
|
|
|
return Handler::$prevented_route == true;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get the route path
|
|
|
|
|
// $full=true returns route/and/sub/routes, $full=false returns just the /route base
|
2017-11-09 19:02:18 +00:00
|
|
|
|
function get_route_path($full=false) {
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return Handler::getRoutePath($full);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get route name from route.name.php
|
|
|
|
|
function get_route_name() {
|
|
|
|
|
return Handler::getRouteName();
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get the route template basename file used. Useful when you do route mapping.
|
2017-11-09 19:02:18 +00:00
|
|
|
|
function get_template_used() {
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return Handler::getTemplateUsed();
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* GLOBAL HELPERS
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Outputs a well formatted HTML output of anything (strings, arrays, comma separated "things", anything)
|
|
|
|
|
function debug($arguments) {
|
|
|
|
|
if(empty($arguments)) return;
|
|
|
|
|
echo '<pre>';
|
|
|
|
|
foreach(func_get_args() as $value) {
|
|
|
|
|
print_r($value);
|
|
|
|
|
}
|
|
|
|
|
echo '</pre>';
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Universal check for setted values for strings and arrays
|
|
|
|
|
function check_value($anything) {
|
|
|
|
|
if((@count($anything)>0 and !@empty($anything) and @isset($anything)) || $anything=='0') {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_global($var) {
|
|
|
|
|
global ${$var};
|
|
|
|
|
return ${$var};
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function is_apache() {
|
|
|
|
|
return (isset($_SERVER['SERVER_SOFTWARE']) and preg_match('/Apache/i', $_SERVER['SERVER_SOFTWARE']));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Fixed from the original (not working) at: http://stackoverflow.com/a/7859707
|
|
|
|
|
function random_values($min, $max, $limit) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Numbers?
|
|
|
|
|
if(!is_numeric($min) or !is_numeric($max)) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get the accurate min and max
|
|
|
|
|
$min = min($min, $max);
|
|
|
|
|
$max = max($min, $max);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Go home
|
|
|
|
|
if($min == $max) return array($min);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// is the limit ok?
|
|
|
|
|
$minmax_limit = abs($max - $min);
|
|
|
|
|
if($limit > $minmax_limit) {
|
|
|
|
|
$limit = $minmax_limit;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$array = Array();
|
|
|
|
|
for($i = 0; $i < $limit; $i++) {
|
|
|
|
|
$rand = rand($min,$max);
|
|
|
|
|
while(in_array($rand,$array)) {
|
|
|
|
|
$rand = mt_rand($min,$max);
|
|
|
|
|
}
|
|
|
|
|
$array[$i] = $rand;
|
2017-11-09 19:02:18 +00:00
|
|
|
|
}
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return $array;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Generates a random string
|
|
|
|
|
* @autor Baba
|
|
|
|
|
* @url http://stackoverflow.com/a/17267718
|
|
|
|
|
*/
|
|
|
|
|
function random_string($length) {
|
|
|
|
|
switch(true) {
|
2017-01-22 18:50:02 +00:00
|
|
|
|
case function_exists('random_bytes') :
|
2017-01-15 20:53:11 +00:00
|
|
|
|
$r = random_bytes($length);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
break;
|
|
|
|
|
case function_exists('openssl_random_pseudo_bytes') :
|
|
|
|
|
$r = openssl_random_pseudo_bytes($length);
|
|
|
|
|
break;
|
|
|
|
|
case is_readable('/dev/urandom') : // deceze
|
|
|
|
|
$r = file_get_contents('/dev/urandom', false, null, 0, $length);
|
|
|
|
|
break;
|
|
|
|
|
default :
|
|
|
|
|
$i = 0;
|
|
|
|
|
$r = '';
|
|
|
|
|
while($i ++ < $length) {
|
|
|
|
|
$r .= chr(mt_rand(0, 255));
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return substr(bin2hex($r), 0, $length);
|
|
|
|
|
}
|
2017-05-21 20:19:25 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* A timing safe equals comparison
|
|
|
|
|
*
|
|
|
|
|
* To prevent leaking length information, it is important
|
|
|
|
|
* that user input is always used as the second parameter.
|
|
|
|
|
*
|
|
|
|
|
* @param string $safe The internal (safe) value to be checked
|
|
|
|
|
* @param string $user The user submitted (unsafe) value
|
|
|
|
|
*
|
|
|
|
|
* @return boolean True if the two strings are identical.
|
|
|
|
|
*/
|
|
|
|
|
function timing_safe_compare($safe, $user) {
|
|
|
|
|
// Prevent issues if string length is 0
|
|
|
|
|
$safe .= chr(0);
|
|
|
|
|
$user .= chr(0);
|
|
|
|
|
|
|
|
|
|
$safeLen = strlen($safe);
|
|
|
|
|
$userLen = strlen($user);
|
|
|
|
|
|
|
|
|
|
// Set the result to the difference between the lengths
|
|
|
|
|
$result = $safeLen - $userLen;
|
|
|
|
|
|
|
|
|
|
// Note that we ALWAYS iterate over the user-supplied length
|
|
|
|
|
// This is to prevent leaking length information
|
|
|
|
|
for ($i = 0; $i < $userLen; $i++) {
|
|
|
|
|
// Using % here is a trick to prevent notices
|
|
|
|
|
// It's safe, since if the lengths are different
|
|
|
|
|
// $result is already non-0
|
|
|
|
|
$result |= (ord($safe[$i % $safeLen]) ^ ord($user[$i]));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// They are only identical strings if $result is exactly 0...
|
|
|
|
|
return $result === 0;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function str_replace_first($search, $replace, $subject) {
|
|
|
|
|
$pos = strpos($subject, $search);
|
|
|
|
|
if ($pos !== false) {
|
|
|
|
|
$subject = substr_replace($subject, $replace, $pos, strlen($search));
|
|
|
|
|
}
|
|
|
|
|
return $subject;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function str_replace_last($search, $replace, $subject) {
|
|
|
|
|
$pos = strrpos($subject, $search);
|
|
|
|
|
if($pos !== false) {
|
|
|
|
|
$subject = substr_replace($subject, $replace, $pos, strlen($search));
|
|
|
|
|
}
|
|
|
|
|
return $subject;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Taken from http://stackoverflow.com/a/834355
|
|
|
|
|
function starts_with($needle, $haystack) {
|
|
|
|
|
$length = strlen($needle);
|
|
|
|
|
return (substr($haystack, 0, $length) === $needle);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Taken from http://stackoverflow.com/a/834355
|
|
|
|
|
function ends_with($needle, $haystack) {
|
|
|
|
|
$length = strlen($needle);
|
|
|
|
|
if($length == 0) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return (substr($haystack, -$length) === $needle);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* DATA HANDLING
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Filter array with another array
|
|
|
|
|
function array_filter_array($array, $filter_keys, $get='exclusion') {
|
|
|
|
|
$arr = $array;
|
|
|
|
|
$return = [];
|
|
|
|
|
$get = strtolower($get);
|
|
|
|
|
$default_get = 'exclusion';
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
foreach($filter_keys as $k => $v) {
|
|
|
|
|
switch($get) {
|
|
|
|
|
default: case $default_get:
|
|
|
|
|
$get = $default_get;
|
|
|
|
|
if(!array_key_exists($v, $array)) continue;
|
|
|
|
|
$return[$v] = $array[$v];
|
|
|
|
|
break;
|
|
|
|
|
case 'rest':
|
|
|
|
|
unset($array[$v]);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return $get == $default_get ? $return : $array;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function key_asort(&$array, $key) {
|
|
|
|
|
$sorter = [];
|
|
|
|
|
$ret = [];
|
|
|
|
|
reset($array);
|
|
|
|
|
foreach ($array as $ii => $va) {
|
|
|
|
|
$sorter[$ii] = $va[$key];
|
|
|
|
|
}
|
|
|
|
|
asort($sorter);
|
|
|
|
|
foreach ($sorter as $ii => $va) {
|
|
|
|
|
$ret[$ii] = $array[$ii];
|
|
|
|
|
}
|
|
|
|
|
$array = $ret;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Recursive UTF-8 encode array
|
|
|
|
|
*/
|
|
|
|
|
function array_utf8encode(&$arr){
|
|
|
|
|
array_walk_recursive($arr, function(&$val, $key){
|
|
|
|
|
$val = utf8_encode($val);
|
|
|
|
|
});
|
|
|
|
|
return $arr;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Recursive remove empty properties
|
|
|
|
|
*/
|
|
|
|
|
function array_remove_empty($haystack) {
|
|
|
|
|
foreach($haystack as $key => $value) {
|
|
|
|
|
if(is_array($value)) {
|
|
|
|
|
$haystack[$key] = array_remove_empty($haystack[$key]);
|
|
|
|
|
}
|
|
|
|
|
if(empty($haystack[$key])) {
|
|
|
|
|
unset($haystack[$key]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return $haystack;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Abbreviate a number with suffix output
|
|
|
|
|
*/
|
|
|
|
|
function abbreviate_number($number) {
|
|
|
|
|
|
|
|
|
|
// strip any formatting
|
|
|
|
|
$number = (0+str_replace(',', '', $number));
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Not a number, keep it "as is"
|
|
|
|
|
if(!is_numeric($number) or $number == 0) return $number;
|
|
|
|
|
|
|
|
|
|
$abbreviations = [
|
|
|
|
|
24 => 'Y',
|
|
|
|
|
21 => 'Z',
|
|
|
|
|
18 => 'E',
|
|
|
|
|
15 => 'P',
|
|
|
|
|
12 => 'T',
|
2017-05-21 20:19:25 +00:00
|
|
|
|
9 => 'G',
|
2016-08-18 20:39:31 +00:00
|
|
|
|
6 => 'M',
|
|
|
|
|
3 => 'K',
|
|
|
|
|
0 => NULL
|
|
|
|
|
];
|
|
|
|
|
foreach($abbreviations as $exponent => $abbreviation) {
|
|
|
|
|
if($number >= pow(10, $exponent)) {
|
|
|
|
|
return round(floatval($number / pow(10, $exponent))) . $abbreviation;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function nullify_string(&$string) {
|
|
|
|
|
if(is_string($string) and $string == '') {
|
|
|
|
|
$string = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// http://bavotasan.com/2011/convert-hex-color-to-rgb-using-php/
|
|
|
|
|
function hex_to_rgb($hex) {
|
|
|
|
|
$hex = str_replace('#', '', $hex);
|
|
|
|
|
if(strlen($hex) == 3) {
|
|
|
|
|
$r = hexdec(substr($hex,0,1).substr($hex,0,1));
|
|
|
|
|
$g = hexdec(substr($hex,1,1).substr($hex,1,1));
|
|
|
|
|
$b = hexdec(substr($hex,2,1).substr($hex,2,1));
|
|
|
|
|
} else {
|
|
|
|
|
$r = hexdec(substr($hex,0,2));
|
|
|
|
|
$g = hexdec(substr($hex,2,2));
|
|
|
|
|
$b = hexdec(substr($hex,4,2));
|
|
|
|
|
}
|
|
|
|
|
$rgb = array($r, $g, $b);
|
|
|
|
|
return $rgb; // returns an array with the rgb values
|
|
|
|
|
}
|
|
|
|
|
function rgb_to_hex($rgb) {
|
|
|
|
|
$hex = '#';
|
|
|
|
|
$hex .= str_pad(dechex($rgb[0]), 2, '0', STR_PAD_LEFT);
|
|
|
|
|
$hex .= str_pad(dechex($rgb[1]), 2, '0', STR_PAD_LEFT);
|
|
|
|
|
$hex .= str_pad(dechex($rgb[2]), 2, '0', STR_PAD_LEFT);
|
|
|
|
|
return $hex; // returns the hex value including the number sign (#)
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Convert HTML code to BBCode
|
|
|
|
|
* http://kuikie.com/snippets/snippet.php/90-17/php-function-to-convert-bbcode-to-html
|
|
|
|
|
*/
|
|
|
|
|
function html_to_bbcode($text) {
|
|
|
|
|
$htmltags = array(
|
|
|
|
|
'/\<b\>(.*?)\<\/b\>/is',
|
|
|
|
|
'/\<i\>(.*?)\<\/i\>/is',
|
|
|
|
|
'/\<u\>(.*?)\<\/u\>/is',
|
|
|
|
|
'/\<ul.*?\>(.*?)\<\/ul\>/is',
|
|
|
|
|
'/\<li\>(.*?)\<\/li\>/is',
|
|
|
|
|
'/\<img(.*?) src=\"(.*?)\" alt=\"(.*?)\" title=\"Smile(y?)\" \/\>/is',
|
|
|
|
|
'/\<img(.*?) src=\"(.*?)\" (.*?)\>/is',
|
|
|
|
|
'/\<img(.*?) src=\"(.*?)\" alt=\":(.*?)\" .*? \/\>/is',
|
|
|
|
|
'/\<div class=\"quotecontent\"\>(.*?)\<\/div\>/is',
|
2017-11-09 19:02:18 +00:00
|
|
|
|
'/\<div class=\"codecontent\"\>(.*?)\<\/div\>/is',
|
|
|
|
|
'/\<div class=\"quotetitle\"\>(.*?)\<\/div\>/is',
|
2016-08-18 20:39:31 +00:00
|
|
|
|
'/\<div class=\"codetitle\"\>(.*?)\<\/div\>/is',
|
|
|
|
|
'/\<cite.*?\>(.*?)\<\/cite\>/is',
|
|
|
|
|
'/\<blockquote.*?\>(.*?)\<\/blockquote\>/is',
|
|
|
|
|
'/\<div\>(.*?)\<\/div\>/is',
|
|
|
|
|
'/\<code\>(.*?)\<\/code\>/is',
|
|
|
|
|
'/\<br(.*?)\>/is',
|
|
|
|
|
'/\<strong\>(.*?)\<\/strong\>/is',
|
|
|
|
|
'/\<em\>(.*?)\<\/em\>/is',
|
|
|
|
|
'/\<a href=\"mailto:(.*?)\"(.*?)\>(.*?)\<\/a\>/is',
|
|
|
|
|
'/\<a .*?href=\"(.*?)\"(.*?)\>http:\/\/(.*?)\<\/a\>/is',
|
|
|
|
|
'/\<a .*?href=\"(.*?)\"(.*?)\>(.*?)\<\/a\>/is'
|
|
|
|
|
);
|
|
|
|
|
$bbtags = array(
|
|
|
|
|
'[b]$1[/b]',
|
|
|
|
|
'[i]$1[/i]',
|
|
|
|
|
'[u]$1[/u]',
|
|
|
|
|
'[list]$1[/list]',
|
|
|
|
|
'[*]$1',
|
|
|
|
|
'$3',
|
|
|
|
|
'[img]$2[/img]',
|
|
|
|
|
':$3',
|
|
|
|
|
'\[quote\]$1\[/quote\]',
|
|
|
|
|
'\[code\]$1\[/code\]',
|
|
|
|
|
'',
|
|
|
|
|
'',
|
|
|
|
|
'',
|
|
|
|
|
'\[quote\]$1\[/quote\]',
|
|
|
|
|
'$1',
|
|
|
|
|
'\[code\]$1\[/code\]',
|
|
|
|
|
"\n",
|
|
|
|
|
'[b]$1[/b]',
|
|
|
|
|
'[i]$1[/i]',
|
|
|
|
|
'[email=$1]$3[/email]',
|
|
|
|
|
'[url]$1[/url]',
|
|
|
|
|
'[url=$1]$3[/url]'
|
|
|
|
|
);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$text = str_replace ("\n", ' ', $text);
|
|
|
|
|
$ntext = preg_replace ($htmltags, $bbtags, $text);
|
|
|
|
|
$ntext = preg_replace ($htmltags, $bbtags, $ntext);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// for too large text and cannot handle by str_replace
|
|
|
|
|
if(!$ntext) {
|
|
|
|
|
$ntext = str_replace(array('<br>', '<br />'), "\n", $text);
|
|
|
|
|
$ntext = str_replace(array('<strong>', '</strong>'), array('[b]', '[/b]'), $ntext);
|
|
|
|
|
$ntext = str_replace(array('<em>', '</em>'), array('[i]', '[/i]'), $ntext);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$ntext = strip_tags($ntext);
|
|
|
|
|
$ntext = trim(html_entity_decode($ntext, ENT_QUOTES, 'UTF-8'));
|
|
|
|
|
return $ntext;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Linkify functions borrowed from https://github.com/misd-service-development/php-linkify
|
|
|
|
|
function linkify($text, array $options = array()) {
|
2017-01-15 20:53:11 +00:00
|
|
|
|
$attr = '';
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if (true === array_key_exists('attr', $options)) {
|
|
|
|
|
foreach ($options['attr'] as $key => $value) {
|
|
|
|
|
if (true === is_array($value)) {
|
|
|
|
|
$value = array_pop($value);
|
|
|
|
|
}
|
|
|
|
|
$attr .= sprintf(' %s="%s"', $key, $value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
$options['attr'] = $attr;
|
|
|
|
|
$ignoreTags = array('head', 'link', 'a', 'script', 'style', 'code', 'pre', 'select', 'textarea', 'button');
|
|
|
|
|
$chunks = preg_split('/(<.+?>)/is', $text, 0, PREG_SPLIT_DELIM_CAPTURE);
|
|
|
|
|
$openTag = null;
|
|
|
|
|
for ($i = 0; $i < count($chunks); $i++) {
|
|
|
|
|
if ($i % 2 === 0) { // even numbers are text
|
|
|
|
|
// Only process this chunk if there are no unclosed $ignoreTags
|
|
|
|
|
if (null === $openTag) {
|
|
|
|
|
$chunks[$i] = linkify_urls($chunks[$i], $options);
|
|
|
|
|
$chunks[$i] = linkify_emails($chunks[$i], $options);
|
|
|
|
|
}
|
|
|
|
|
} else { // odd numbers are tags
|
|
|
|
|
// Only process this tag if there are no unclosed $ignoreTags
|
|
|
|
|
if (null === $openTag) {
|
|
|
|
|
// Check whether this tag is contained in $ignoreTags and is not self-closing
|
|
|
|
|
if (preg_match("`<(" . implode('|', $ignoreTags) . ").*(?<!/)>$`is", $chunks[$i], $matches)) {
|
|
|
|
|
$openTag = $matches[1];
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Otherwise, check whether this is the closing tag for $openTag.
|
|
|
|
|
if (preg_match('`</\s*' . $openTag . '>`i', $chunks[$i], $matches)) {
|
|
|
|
|
$openTag = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
$text = implode($chunks);
|
|
|
|
|
return $text;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function linkify_emails($text, $options = array('attr' => '')) {
|
|
|
|
|
$pattern = '~(?xi)
|
|
|
|
|
\b
|
|
|
|
|
(?<!=) # Not part of a query string
|
|
|
|
|
[A-Z0-9._\'%+-]+ # Username
|
|
|
|
|
@ # At
|
|
|
|
|
[A-Z0-9.-]+ # Domain
|
|
|
|
|
\. # Dot
|
|
|
|
|
[A-Z]{2,4} # Something
|
|
|
|
|
~';
|
|
|
|
|
$callback = function ($match) use ($options) {
|
|
|
|
|
if (isset($options['callback'])) {
|
|
|
|
|
$cb = $options['callback']($match[0], $match[0], true);
|
|
|
|
|
if (!is_null($cb)) {
|
|
|
|
|
return $cb;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return '<a href="mailto:' . $match[0] . '"' . $options['attr'] . '>' . $match[0] . '</a>';
|
|
|
|
|
};
|
|
|
|
|
return preg_replace_callback($pattern, $callback, $text);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function linkify_urls($text, $options = array('attr' => '')) {
|
|
|
|
|
$pattern = '~(?xi)
|
|
|
|
|
(?:
|
|
|
|
|
((ht|f)tps?://) # scheme://
|
|
|
|
|
| # or
|
|
|
|
|
www\d{0,3}\. # "www.", "www1.", "www2." ... "www999."
|
|
|
|
|
| # or
|
|
|
|
|
www\- # "www-"
|
|
|
|
|
| # or
|
|
|
|
|
[a-z0-9.\-]+\.[a-z]{2,4}(?=/) # looks like domain name followed by a slash
|
|
|
|
|
)
|
|
|
|
|
(?: # Zero or more:
|
|
|
|
|
[^\s()<>]+ # Run of non-space, non-()<>
|
|
|
|
|
| # or
|
|
|
|
|
\(([^\s()<>]+|(\([^\s()<>]+\)))*\) # balanced parens, up to 2 levels
|
|
|
|
|
)*
|
|
|
|
|
(?: # End with:
|
|
|
|
|
\(([^\s()<>]+|(\([^\s()<>]+\)))*\) # balanced parens, up to 2 levels
|
|
|
|
|
| # or
|
|
|
|
|
[^\s`!\-()\[\]{};:\'".,<>?«»“”‘’] # not a space or one of these punct chars
|
|
|
|
|
)
|
|
|
|
|
~';
|
|
|
|
|
$callback = function ($match) use ($options) {
|
|
|
|
|
$caption = $match[0];
|
|
|
|
|
$pattern = "~^(ht|f)tps?://~";
|
|
|
|
|
if (0 === preg_match($pattern, $match[0])) {
|
|
|
|
|
$match[0] = 'http://' . $match[0];
|
|
|
|
|
}
|
|
|
|
|
if (isset($options['callback'])) {
|
2017-01-15 20:53:11 +00:00
|
|
|
|
$cb = $options['callback']($match[0], $caption, $options);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if (!is_null($cb)) {
|
|
|
|
|
return $cb;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return '<a href="' . $match[0] . '"' . $options['attr'] . '>' . $caption . '</a>';
|
|
|
|
|
};
|
|
|
|
|
return preg_replace_callback($pattern, $callback, $text);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2017-01-15 20:53:11 +00:00
|
|
|
|
function linkify_safe($text, $options=[]) {
|
|
|
|
|
$options = array_merge(['attr' => ['rel' => 'nofollow', 'target' => '_blank']], $options);
|
|
|
|
|
return linkify(htmlspecialchars($text), $options);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* ERROR HANDLING
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Converts an Exception to a formatted PHP like error
|
|
|
|
|
function exception_to_error($e, $die=TRUE) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$internal_code = 500;
|
|
|
|
|
$internal_error = '<b>'.G_APP_NAME.' error:</b> ' . get_set_status_header_desc($internal_code);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
set_status_header($internal_code);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Debug levels
|
|
|
|
|
// 0:NONE 1:ERROR_LOG 2:PRINT(NO ERROR_LOG) 3:PRINT+ERROR_LOG
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$debug_level = get_app_setting('debug_level');
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!in_array($debug_level, [0,1,2,3])) {
|
|
|
|
|
$debug_level = 1;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(in_array($debug_level, [1,3])) {
|
|
|
|
|
error_log($e);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!in_array($debug_level, [2,3])) { // No print here
|
|
|
|
|
die($internal_error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$message = [];
|
|
|
|
|
$message[] = '<b>Fatal error ['.$e->getCode().']:</b> ' . safe_html($e->getMessage());
|
|
|
|
|
$message[] = 'Triggered in ' . absolute_to_relative($e->getFile()) . ' at line ' . $e->getLine() . "\n";
|
|
|
|
|
$message[] = '<b>Stack trace:</b>';
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$rtn = '';
|
|
|
|
|
$count = 0;
|
|
|
|
|
foreach ($e->getTrace() as $frame) {
|
|
|
|
|
$args = '';
|
|
|
|
|
if (isset($frame['args'])) {
|
|
|
|
|
$args = array();
|
|
|
|
|
foreach ($frame['args'] as $arg) {
|
|
|
|
|
switch(true) {
|
|
|
|
|
case is_string($arg):
|
|
|
|
|
if(file_exists($arg)) {
|
|
|
|
|
$arg = absolute_to_relative($arg);
|
|
|
|
|
}
|
|
|
|
|
$args[] = "'" . $arg . "'";
|
|
|
|
|
break;
|
|
|
|
|
case is_array($arg):
|
|
|
|
|
$args[] = 'Array';
|
|
|
|
|
break;
|
|
|
|
|
case is_null($arg):
|
|
|
|
|
$args[] = 'NULL';
|
|
|
|
|
break;
|
|
|
|
|
case is_bool($arg):
|
|
|
|
|
$args[] = ($arg) ? 'true' : 'false';
|
|
|
|
|
break;
|
|
|
|
|
case is_object($arg):
|
|
|
|
|
$args[] = get_class($arg);
|
|
|
|
|
break;
|
|
|
|
|
case is_resource($arg):
|
|
|
|
|
$args[] = get_resource_type($arg);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
$args[] = $arg;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-09 19:02:18 +00:00
|
|
|
|
}
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$args = join(', ', $args);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$rtn .= sprintf("#%s %s(%s): %s(%s)\n",
|
|
|
|
|
$count,
|
|
|
|
|
isset($frame['file']) ? absolute_to_relative($frame['file']) : 'unknown file',
|
|
|
|
|
isset($frame['line']) ? $frame['line'] : 'unknown line',
|
|
|
|
|
(isset($frame['class'])) ? $frame['class'].$frame['type'].$frame['function'] : $frame['function'],
|
|
|
|
|
$args);
|
|
|
|
|
$count++;
|
|
|
|
|
}
|
|
|
|
|
$message[] = $rtn;
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$message = implode("\n", $message);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($die) {
|
|
|
|
|
echo nl2br($message);
|
|
|
|
|
die();
|
|
|
|
|
} else {
|
|
|
|
|
return $message;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* MATH
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function fraction_to_decimal($fraction) {
|
|
|
|
|
list($top, $bottom) = explode('/', $fraction);
|
|
|
|
|
return $bottom == 0 ? $fraction : ($top / $bottom);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* TIME
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns current GMT datetime
|
|
|
|
|
function datetimegmt($format=NULL) {
|
|
|
|
|
return gmdate(!is_null($format) ? $format : 'Y-m-d H:i:s');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns current datetime (for the system timezone)
|
|
|
|
|
function datetime($format=NULL) {
|
|
|
|
|
return date(!is_null($format) ? $format : 'Y-m-d H:i:s');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns current datetime in the specified timezone
|
|
|
|
|
function datetime_tz($tz, $format=NULL) {
|
|
|
|
|
$date = date_create(NULL, timezone_open($tz));
|
|
|
|
|
return date_format($date, !is_null($format) ? $format : 'Y-m-d H:i:s');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// http://stackoverflow.com/a/5878722
|
|
|
|
|
function is_valid_timezone($tzid){
|
|
|
|
|
$valid = [];
|
|
|
|
|
$tza = timezone_abbreviations_list();
|
|
|
|
|
foreach ($tza as $zone)
|
|
|
|
|
foreach ($zone as $item)
|
|
|
|
|
$valid[$item['timezone_id']] = true;
|
|
|
|
|
unset($valid['']);
|
|
|
|
|
return !!$valid[$tzid];
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// http://stackoverflow.com/a/18602474
|
|
|
|
|
function time_elapsed_string($datetime, $full=false) {
|
|
|
|
|
$now = new \DateTime(datetimegmt());
|
|
|
|
|
$ago = new \DateTime($datetime);
|
|
|
|
|
$diff = $now->diff($ago);
|
|
|
|
|
|
|
|
|
|
$diff->w = floor($diff->d / 7);
|
|
|
|
|
$diff->d -= $diff->w * 7;
|
|
|
|
|
|
|
|
|
|
$string = array(
|
|
|
|
|
'y' => 'year',
|
|
|
|
|
'm' => 'month',
|
|
|
|
|
'w' => 'week',
|
|
|
|
|
'd' => 'day',
|
|
|
|
|
'h' => 'hour',
|
|
|
|
|
'i' => 'minute',
|
|
|
|
|
's' => 'second',
|
|
|
|
|
);
|
|
|
|
|
foreach ($string as $k => &$v) {
|
|
|
|
|
if ($diff->$k) {
|
|
|
|
|
$v = $diff->$k . ' ' . $v . ($diff->$k > 1 ? 's' : '');
|
|
|
|
|
} else {
|
|
|
|
|
unset($string[$k]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!$full) $string = array_slice($string, 0, 1);
|
|
|
|
|
return $string ? implode(', ', $string) . ' ago' : 'just now';
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function datetimegmt_convert_tz($datetimegmt, $tz) {
|
|
|
|
|
if(!is_valid_timezone($tz)) return $datetimegmt;
|
|
|
|
|
$date = new \DateTime($datetimegmt . "+00");
|
|
|
|
|
$date->setTimezone(new \DateTimeZone($tz));
|
|
|
|
|
return $date->format('Y-m-d H:i:s');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns the difference between two dates in the given format (default seconds)
|
|
|
|
|
function datetime_diff($older, $newer=NULL, $format='s') {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!in_array($format, ['s', 'm', 'h', 'd'])) {
|
|
|
|
|
$format = 's';
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!$newer or $newer == NULL) {
|
|
|
|
|
$newer = datetimegmt();
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$datetime1 = new \DateTime($older);
|
|
|
|
|
$datetime2 = new \DateTime($newer);
|
|
|
|
|
$diff = $datetime2->getTimestamp() - $datetime1->getTimestamp(); // In seconds
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$timeconstant = [
|
|
|
|
|
's' => 1,
|
|
|
|
|
'm' => 60,
|
|
|
|
|
'h' => 3600,
|
2017-11-09 19:02:18 +00:00
|
|
|
|
'd' => 86400
|
2016-08-18 20:39:31 +00:00
|
|
|
|
];
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return $diff/$timeconstant[$format];
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function datetime_add($datetime, $add) {
|
|
|
|
|
return datetime_alter($datetime, $add, 'add');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function datetime_sub($datetime, $sub) {
|
|
|
|
|
return datetime_alter($datetime, $sub, 'sub');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function datetime_modify($datetime, $var) {
|
|
|
|
|
return datetime_alter($datetime, $var, 'modify');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function datetime_alter($datetime, $var, $action='add') {
|
|
|
|
|
if(!in_array($action, ['add', 'sub', 'modify'])) {
|
2018-04-22 02:03:51 +00:00
|
|
|
|
return $datetime;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
|
|
|
|
try {
|
2017-05-21 20:19:25 +00:00
|
|
|
|
$DateTime = new \DateTime($datetime);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($action=='modify') {
|
2017-05-21 20:19:25 +00:00
|
|
|
|
$DateTime->$action($var);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
} else {
|
2017-05-21 20:19:25 +00:00
|
|
|
|
$interval = dateinterval($var); // validation
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!$interval) {
|
|
|
|
|
return $datetime;
|
|
|
|
|
}
|
2017-05-21 20:19:25 +00:00
|
|
|
|
$DateTime->$action($interval);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-05-21 20:19:25 +00:00
|
|
|
|
return $DateTime->format('Y-m-d H:i:s');
|
2016-10-12 16:32:18 +00:00
|
|
|
|
} catch(Exception $e) {
|
2016-08-18 20:39:31 +00:00
|
|
|
|
throw new Exception($e->getMessage() . ' in ' . __FUNCTION__ . ' (' . $action . ')', $e->getCode());
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function dateinterval($var) {
|
|
|
|
|
try {
|
|
|
|
|
$di = new \DateInterval($var);
|
|
|
|
|
return $di;
|
2016-10-12 16:32:18 +00:00
|
|
|
|
} catch(Exception $e) {}
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* CLIENT
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_client_ip(){
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-10-12 16:32:18 +00:00
|
|
|
|
$client_ip = !empty($_SERVER['REMOTE_ADDR']) ? $_SERVER['REMOTE_ADDR'] : (!empty($_ENV['REMOTE_ADDR']) ? $_ENV['REMOTE_ADDR'] : NULL);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-10-12 16:32:18 +00:00
|
|
|
|
if(array_key_exists('HTTP_CF_CONNECTING_IP', $_SERVER) && $_SERVER['HTTP_CF_CONNECTING_IP'] == $_SERVER['REMOTE_ADDR']) {
|
|
|
|
|
return $_SERVER['HTTP_CF_CONNECTING_IP'];
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
|
|
|
|
|
|
2016-10-12 16:32:18 +00:00
|
|
|
|
$entries = preg_split('/[\s,]/', $_SERVER['HTTP_X_FORWARDED_FOR'], -1, PREG_SPLIT_NO_EMPTY);
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
reset($entries);
|
2016-10-12 16:32:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
while(list(, $entry) = each($entries)) {
|
|
|
|
|
$entry = trim($entry);
|
|
|
|
|
if(preg_match('/^([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+)/', $entry, $ip_list)){
|
|
|
|
|
$private_ip = array(
|
2017-11-09 19:02:18 +00:00
|
|
|
|
'/^0\./',
|
|
|
|
|
'/^127\.0\.0\.1/',
|
|
|
|
|
'/^192\.168\..*/',
|
|
|
|
|
'/^172\.((1[6-9])|(2[0-9])|(3[0-1]))\..*/',
|
2016-08-18 20:39:31 +00:00
|
|
|
|
'/^10\..*/');
|
|
|
|
|
|
|
|
|
|
$found_ip = preg_replace($private_ip, $client_ip, $ip_list[1]);
|
|
|
|
|
|
|
|
|
|
if($client_ip != $found_ip) { // and !isset($_SERVER['HTTP_CF_CONNECTING_IP']
|
|
|
|
|
$client_ip = $found_ip;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-10-12 16:32:18 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return $client_ip;
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_client_languages($getSortedList=true, $acceptedLanguages=false) {
|
|
|
|
|
|
|
|
|
|
if (empty($acceptedLanguages)) {
|
|
|
|
|
$acceptedLanguages = $_SERVER['HTTP_ACCEPT_LANGUAGE'];
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// regex inspired from @GabrielAnderson on http://stackoverflow.com/questions/6038236/http-accept-language
|
|
|
|
|
preg_match_all('/([a-z]{1,8}(-[a-z]{1,8})*)\s*(;\s*q\s*=\s*(1|0\.[0-9]+))?/i', $acceptedLanguages, $lang_parse);
|
|
|
|
|
$langs = $lang_parse[1];
|
|
|
|
|
$ranks = $lang_parse[4];
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// (create an associative array 'language' => 'preference')
|
|
|
|
|
$lang2pref = array();
|
|
|
|
|
for($i=0; $i<count($langs); $i++) {
|
|
|
|
|
$lang2pref[$langs[$i]] = (float) (!empty($ranks[$i]) ? $ranks[$i] : 1);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$cmpLangs = function($a, $b) use ($lang2pref) {
|
|
|
|
|
if ($lang2pref[$a] > $lang2pref[$b]) {
|
|
|
|
|
return -1;
|
|
|
|
|
} else if($lang2pref[$a] < $lang2pref[$b]) {
|
|
|
|
|
return 1;
|
|
|
|
|
} else if(strlen($a) > strlen($b)) {
|
|
|
|
|
return -1;
|
|
|
|
|
} else if(strlen($a) < strlen($b)) {
|
|
|
|
|
return 1;
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
};
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Just in case the server is running eAccelerator
|
|
|
|
|
if(is_callable($cmpLangs)) {
|
|
|
|
|
uksort($lang2pref, $cmpLangs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($getSortedList) {
|
|
|
|
|
return $lang2pref;
|
|
|
|
|
}
|
|
|
|
|
// return the first value's key
|
|
|
|
|
reset($lang2pref);
|
|
|
|
|
return key($lang2pref);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Parses a user agent string into its important parts
|
|
|
|
|
*
|
|
|
|
|
* @author Jesse G. Donat <donatj@gmail.com>
|
|
|
|
|
* @link https://github.com/donatj/PhpUserAgent
|
|
|
|
|
* @link http://donatstudios.com/PHP-Parser-HTTP_USER_AGENT
|
|
|
|
|
* @param string|null $u_agent
|
|
|
|
|
* @return array an array with browser, version and platform keys
|
|
|
|
|
*/
|
|
|
|
|
function parse_user_agent( $u_agent = null ) {
|
|
|
|
|
if( is_null($u_agent) && isset($_SERVER['HTTP_USER_AGENT']) ) $u_agent = $_SERVER['HTTP_USER_AGENT'];
|
|
|
|
|
|
|
|
|
|
$platform = null;
|
|
|
|
|
$browser = null;
|
|
|
|
|
$version = null;
|
|
|
|
|
|
|
|
|
|
$empty = array( 'platform' => $platform, 'browser' => $browser, 'version' => $version );
|
|
|
|
|
|
|
|
|
|
if( !$u_agent ) return $empty;
|
|
|
|
|
|
|
|
|
|
if( preg_match('/\((.*?)\)/im', $u_agent, $parent_matches) ) {
|
|
|
|
|
|
|
|
|
|
preg_match_all('/(?P<platform>Android|CrOS|iPhone|iPad|Linux|Macintosh|Windows(\ Phone\ OS)?|Silk|linux-gnu|BlackBerry|PlayBook|Nintendo\ (WiiU?|3DS)|Xbox)
|
|
|
|
|
(?:\ [^;]*)?
|
|
|
|
|
(?:;|$)/imx', $parent_matches[1], $result, PREG_PATTERN_ORDER);
|
|
|
|
|
|
|
|
|
|
$priority = array( 'Android', 'Xbox' );
|
|
|
|
|
$result['platform'] = array_unique($result['platform']);
|
|
|
|
|
if( count($result['platform']) > 1 ) {
|
|
|
|
|
if( $keys = array_intersect($priority, $result['platform']) ) {
|
|
|
|
|
$platform = reset($keys);
|
|
|
|
|
} else {
|
|
|
|
|
$platform = $result['platform'][0];
|
|
|
|
|
}
|
|
|
|
|
} elseif( isset($result['platform'][0]) ) {
|
|
|
|
|
$platform = $result['platform'][0];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if( $platform == 'linux-gnu' ) {
|
|
|
|
|
$platform = 'Linux';
|
|
|
|
|
} elseif( $platform == 'CrOS' ) {
|
|
|
|
|
$platform = 'Chrome OS';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
preg_match_all('%(?P<browser>Camino|Kindle(\ Fire\ Build)?|Firefox|Iceweasel|Safari|MSIE|Trident/.*rv|AppleWebKit|Chrome|IEMobile|Opera|OPR|Silk|Lynx|Midori|Version|Wget|curl|NintendoBrowser|PLAYSTATION\ (\d|Vita)+)
|
|
|
|
|
(?:\)?;?)
|
|
|
|
|
(?:(?:[:/ ])(?P<version>[0-9A-Z.]+)|/(?:[A-Z]*))%ix',
|
|
|
|
|
$u_agent, $result, PREG_PATTERN_ORDER);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// If nothing matched, return null (to avoid undefined index errors)
|
|
|
|
|
if( !isset($result['browser'][0]) || !isset($result['version'][0]) ) {
|
|
|
|
|
return $empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$browser = $result['browser'][0];
|
|
|
|
|
$version = $result['version'][0];
|
|
|
|
|
|
|
|
|
|
$find = function ( $search, &$key ) use ( $result ) {
|
|
|
|
|
$xkey = array_search(strtolower($search),array_map('strtolower',$result['browser']));
|
|
|
|
|
if( $xkey !== false ) {
|
|
|
|
|
$key = $xkey;
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
$key = 0;
|
|
|
|
|
if( $browser == 'Iceweasel' ) {
|
|
|
|
|
$browser = 'Firefox';
|
|
|
|
|
}elseif( $find('Playstation Vita', $key) ) {
|
|
|
|
|
$platform = 'PlayStation Vita';
|
|
|
|
|
$browser = 'Browser';
|
|
|
|
|
} elseif( $find('Kindle Fire Build', $key) || $find('Silk', $key) ) {
|
|
|
|
|
$browser = $result['browser'][$key] == 'Silk' ? 'Silk' : 'Kindle';
|
|
|
|
|
$platform = 'Kindle Fire';
|
|
|
|
|
if( !($version = $result['version'][$key]) || !is_numeric($version[0]) ) {
|
|
|
|
|
$version = $result['version'][array_search('Version', $result['browser'])];
|
|
|
|
|
}
|
|
|
|
|
} elseif( $find('NintendoBrowser', $key) || $platform == 'Nintendo 3DS' ) {
|
|
|
|
|
$browser = 'NintendoBrowser';
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
} elseif( $find('Kindle', $key) ) {
|
|
|
|
|
$browser = $result['browser'][$key];
|
|
|
|
|
$platform = 'Kindle';
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
} elseif( $find('OPR', $key) ) {
|
|
|
|
|
$browser = 'Opera Next';
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
} elseif( $find('Opera', $key) ) {
|
|
|
|
|
$browser = 'Opera';
|
|
|
|
|
$find('Version', $key);
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
}elseif ( $find('Chrome', $key) ) {
|
|
|
|
|
$browser = 'Chrome';
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
} elseif( $find('Midori', $key) ) {
|
|
|
|
|
$browser = 'Midori';
|
2017-11-09 19:02:18 +00:00
|
|
|
|
$version = $result['version'][$key];
|
2016-08-18 20:39:31 +00:00
|
|
|
|
} elseif( $browser == 'AppleWebKit' ) {
|
|
|
|
|
if( ($platform == 'Android' && !($key = 0)) ) {
|
|
|
|
|
$browser = 'Android Browser';
|
|
|
|
|
} elseif( $platform == 'BlackBerry' || $platform == 'PlayBook' ) {
|
|
|
|
|
$browser = 'BlackBerry Browser';
|
|
|
|
|
} elseif( $find('Safari', $key) ) {
|
|
|
|
|
$browser = 'Safari';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$find('Version', $key);
|
|
|
|
|
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
} elseif( $browser == 'MSIE' || strpos($browser, 'Trident') !== false ) {
|
|
|
|
|
if( $find('IEMobile', $key) ) {
|
|
|
|
|
$browser = 'IEMobile';
|
|
|
|
|
} else {
|
|
|
|
|
$browser = 'MSIE';
|
|
|
|
|
$key = 0;
|
|
|
|
|
}
|
|
|
|
|
$version = $result['version'][$key];
|
|
|
|
|
} elseif( $key = preg_grep("/playstation \d/i", array_map('strtolower', $result['browser']))) {
|
|
|
|
|
$key = reset($key);
|
|
|
|
|
|
|
|
|
|
$platform = 'PlayStation ' . preg_replace('/[^\d]/i', '', $key);
|
|
|
|
|
$browser = 'NetFront';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return array( 'platform' => $platform, 'browser' => $browser, 'version' => $version );
|
|
|
|
|
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* MISC. VALIDATIONS
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// This checks for real email address.
|
|
|
|
|
function is_real_email_address($email) {
|
|
|
|
|
$valid = true;
|
|
|
|
|
$atIndex = strrpos($email, '@');
|
|
|
|
|
if(is_bool($atIndex) && !$atIndex) {
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else {
|
|
|
|
|
$domain = substr($email, $atIndex+1);
|
|
|
|
|
$local = substr($email, 0, $atIndex);
|
|
|
|
|
$localLen = strlen($local);
|
|
|
|
|
$domainLen = strlen($domain);
|
|
|
|
|
if($localLen < 1 || $localLen > 64) {
|
|
|
|
|
// local part length exceeded
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else if($domainLen < 1 || $domainLen > 255) {
|
|
|
|
|
// domain part length exceeded
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else if($local[0] == '.' || $local[$localLen-1] == '.') {
|
|
|
|
|
// local part starts or ends with '.'
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else if(preg_match('/\\.\\./', $local)) {
|
|
|
|
|
// local part has two consecutive dots
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else if(!preg_match('/^[A-Za-z0-9\\-\\.]+$/', $domain)) {
|
|
|
|
|
// character not valid in domain part
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else if(preg_match('/\\.\\./', $domain)) {
|
|
|
|
|
// domain part has two consecutive dots
|
|
|
|
|
$valid = false;
|
|
|
|
|
} else if(!preg_match('/^(\\\\.|[A-Za-z0-9!#%&`_=\\/$\'*+?^{}|~.-])+$/',
|
|
|
|
|
str_replace('\\\\','',$local))) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
// character not valid in local part unless
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// local part is quoted
|
|
|
|
|
if(!preg_match('/^"(\\\\"|[^"])+"$/', str_replace("\\\\","",$local))) {
|
|
|
|
|
$valid = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if($valid && !(checkdnsrr($domain,'MX') || checkdnsrr($domain,'A'))) {
|
|
|
|
|
// domain not found in DNS
|
|
|
|
|
$valid = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return $valid;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function is_valid_hex_color($string, $prefix=true) {
|
|
|
|
|
return preg_match('/#' . ($prefix ? '?' : NULL) . '([\da-fA-F]{2})([\da-fA-F]{2})([\da-fA-F]{2})/', $string);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function is_valid_ip($ip) {
|
2018-04-17 21:25:26 +00:00
|
|
|
|
return filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4) || filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_IPV6);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* SANITIZATION
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// Removes all the spaces on a given string
|
|
|
|
|
function remove_spaces($string) {
|
|
|
|
|
return str_replace(' ', '', $string);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Sanitizes double or more slahes in a path
|
|
|
|
|
function sanitize_path_slashes($path) {
|
|
|
|
|
return preg_replace('#/+#','/', $path);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function sanitize_directory_separator($path) {
|
|
|
|
|
return preg_replace('#' . DIRECTORY_SEPARATOR . '+#', DIRECTORY_SEPARATOR , $path);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function sanitize_relative_path($path) {
|
|
|
|
|
$clean = forward_slash($path);
|
|
|
|
|
$clean = sanitize_path_slashes($path);
|
|
|
|
|
$clean = preg_replace('#(\.+/)+#', '', $clean);
|
|
|
|
|
$clean = sanitize_path_slashes($clean);
|
|
|
|
|
return $clean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a sanitized string, typically for URLs
|
|
|
|
|
* This function was borrowed from chyrp.net (MIT License)
|
|
|
|
|
*/
|
|
|
|
|
function sanitize_string($string, $force_lowercase = true, $only_alphanumerics = false, $truncate = 100) {
|
|
|
|
|
$strip = array('~', '`', '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '=', '+', '{',
|
|
|
|
|
'}', '\\', '|', ';', ':', '\'', "'", '‘', '’', '“', '”', '–', '—',
|
|
|
|
|
'—', '–', ',', '<', '.', '>', '/', '?');
|
|
|
|
|
$clean = trim(str_replace($strip, "", strip_tags($string)));
|
|
|
|
|
$clean = preg_replace('/\s+/', '-', $clean);
|
|
|
|
|
$clean = ($only_alphanumerics ? preg_replace('/[^a-zA-Z0-9]/', '', $clean) : $clean);
|
|
|
|
|
$clean = ($truncate ? substr($clean, 0, $truncate) : $clean);
|
|
|
|
|
return ($force_lowercase) ? (function_exists('mb_strtolower')) ? mb_strtolower($clean, 'UTF-8') : strtolower($clean) : $clean;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Original PHP code by Chirp Internet: www.chirp.com.au
|
|
|
|
|
function truncate($string, $limit, $break=NULL, $pad='...') {
|
|
|
|
|
$encoding = 'UTF-8';
|
|
|
|
|
if(mb_strlen($string, $encoding) <= $limit) return $string;
|
|
|
|
|
if(is_null($break) or $break == '') {
|
|
|
|
|
$string = trim(mb_substr($string, 0, $limit - strlen($pad), $encoding)) . $pad;
|
|
|
|
|
} else {
|
|
|
|
|
if(false !== ($breakpoint = strpos($string, $break, $limit))) {
|
|
|
|
|
if($breakpoint < mb_strlen($string, $encoding) - 1) {
|
|
|
|
|
$string = mb_substr($string, 0, $breakpoint, $encoding) . $pad;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return $string;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Thanks to http://www.evaisse.net/2008/php-translit-remove-accent-unaccent-21001
|
|
|
|
|
function unaccent_string($string) {
|
|
|
|
|
$string = (string)$string;
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(function_exists('mb_detect_encoding')) {
|
|
|
|
|
$utf8 = strtolower(mb_detect_encoding($string)) == 'utf-8';
|
|
|
|
|
} else {
|
|
|
|
|
$length = strlen($string);
|
|
|
|
|
$utf8 = true;
|
|
|
|
|
for ($i=0; $i < $length; $i++) {
|
|
|
|
|
$c = ord($string[$i]);
|
|
|
|
|
if ($c < 0x80) $n = 0; # 0bbbbbbb
|
|
|
|
|
elseif (($c & 0xE0) == 0xC0) $n=1; # 110bbbbb
|
|
|
|
|
elseif (($c & 0xF0) == 0xE0) $n=2; # 1110bbbb
|
|
|
|
|
elseif (($c & 0xF8) == 0xF0) $n=3; # 11110bbb
|
|
|
|
|
elseif (($c & 0xFC) == 0xF8) $n=4; # 111110bb
|
|
|
|
|
elseif (($c & 0xFE) == 0xFC) $n=5; # 1111110b
|
|
|
|
|
else return false; # Does not match any model
|
|
|
|
|
for ($j=0; $j<$n; $j++) { # n bytes matching 10bbbbbb follow ?
|
|
|
|
|
if ((++$i == $length)
|
|
|
|
|
|| ((ord($string[$i]) & 0xC0) != 0x80)) {
|
|
|
|
|
$utf8 = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!$utf8) $string = utf8_encode($string);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$transliteration = array(
|
|
|
|
|
'IJ' => 'I', 'Ö' => 'O', 'Œ' => 'O', 'Ü' => 'U', 'ä' => 'a', 'æ' => 'a',
|
|
|
|
|
'ij' => 'i', 'ö' => 'o', 'œ' => 'o', 'ü' => 'u', 'ß' => 's', 'ſ' => 's',
|
|
|
|
|
'À' => 'A', 'Á' => 'A', 'Â' => 'A', 'Ã' => 'A', 'Ä' => 'A', 'Å' => 'A',
|
|
|
|
|
'Æ' => 'A', 'Ā' => 'A', 'Ą' => 'A', 'Ă' => 'A', 'Ç' => 'C', 'Ć' => 'C',
|
|
|
|
|
'Č' => 'C', 'Ĉ' => 'C', 'Ċ' => 'C', 'Ď' => 'D', 'Đ' => 'D', 'È' => 'E',
|
|
|
|
|
'É' => 'E', 'Ê' => 'E', 'Ë' => 'E', 'Ē' => 'E', 'Ę' => 'E', 'Ě' => 'E',
|
|
|
|
|
'Ĕ' => 'E', 'Ė' => 'E', 'Ĝ' => 'G', 'Ğ' => 'G', 'Ġ' => 'G', 'Ģ' => 'G',
|
|
|
|
|
'Ĥ' => 'H', 'Ħ' => 'H', 'Ì' => 'I', 'Í' => 'I', 'Î' => 'I', 'Ï' => 'I',
|
|
|
|
|
'Ī' => 'I', 'Ĩ' => 'I', 'Ĭ' => 'I', 'Į' => 'I', 'İ' => 'I', 'Ĵ' => 'J',
|
|
|
|
|
'Ķ' => 'K', 'Ľ' => 'K', 'Ĺ' => 'K', 'Ļ' => 'K', 'Ŀ' => 'K', 'Ł' => 'L',
|
|
|
|
|
'Ñ' => 'N', 'Ń' => 'N', 'Ň' => 'N', 'Ņ' => 'N', 'Ŋ' => 'N', 'Ò' => 'O',
|
|
|
|
|
'Ó' => 'O', 'Ô' => 'O', 'Õ' => 'O', 'Ø' => 'O', 'Ō' => 'O', 'Ő' => 'O',
|
|
|
|
|
'Ŏ' => 'O', 'Ŕ' => 'R', 'Ř' => 'R', 'Ŗ' => 'R', 'Ś' => 'S', 'Ş' => 'S',
|
|
|
|
|
'Ŝ' => 'S', 'Ș' => 'S', 'Š' => 'S', 'Ť' => 'T', 'Ţ' => 'T', 'Ŧ' => 'T',
|
|
|
|
|
'Ț' => 'T', 'Ù' => 'U', 'Ú' => 'U', 'Û' => 'U', 'Ū' => 'U', 'Ů' => 'U',
|
|
|
|
|
'Ű' => 'U', 'Ŭ' => 'U', 'Ũ' => 'U', 'Ų' => 'U', 'Ŵ' => 'W', 'Ŷ' => 'Y',
|
|
|
|
|
'Ÿ' => 'Y', 'Ý' => 'Y', 'Ź' => 'Z', 'Ż' => 'Z', 'Ž' => 'Z', 'à' => 'a',
|
|
|
|
|
'á' => 'a', 'â' => 'a', 'ã' => 'a', 'ā' => 'a', 'ą' => 'a', 'ă' => 'a',
|
|
|
|
|
'å' => 'a', 'ç' => 'c', 'ć' => 'c', 'č' => 'c', 'ĉ' => 'c', 'ċ' => 'c',
|
|
|
|
|
'ď' => 'd', 'đ' => 'd', 'è' => 'e', 'é' => 'e', 'ê' => 'e', 'ë' => 'e',
|
|
|
|
|
'ē' => 'e', 'ę' => 'e', 'ě' => 'e', 'ĕ' => 'e', 'ė' => 'e', 'ƒ' => 'f',
|
|
|
|
|
'ĝ' => 'g', 'ğ' => 'g', 'ġ' => 'g', 'ģ' => 'g', 'ĥ' => 'h', 'ħ' => 'h',
|
|
|
|
|
'ì' => 'i', 'í' => 'i', 'î' => 'i', 'ï' => 'i', 'ī' => 'i', 'ĩ' => 'i',
|
|
|
|
|
'ĭ' => 'i', 'į' => 'i', 'ı' => 'i', 'ĵ' => 'j', 'ķ' => 'k', 'ĸ' => 'k',
|
|
|
|
|
'ł' => 'l', 'ľ' => 'l', 'ĺ' => 'l', 'ļ' => 'l', 'ŀ' => 'l', 'ñ' => 'n',
|
|
|
|
|
'ń' => 'n', 'ň' => 'n', 'ņ' => 'n', 'ʼn' => 'n', 'ŋ' => 'n', 'ò' => 'o',
|
|
|
|
|
'ó' => 'o', 'ô' => 'o', 'õ' => 'o', 'ø' => 'o', 'ō' => 'o', 'ő' => 'o',
|
|
|
|
|
'ŏ' => 'o', 'ŕ' => 'r', 'ř' => 'r', 'ŗ' => 'r', 'ś' => 's', 'š' => 's',
|
|
|
|
|
'ť' => 't', 'ù' => 'u', 'ú' => 'u', 'û' => 'u', 'ū' => 'u', 'ů' => 'u',
|
|
|
|
|
'ű' => 'u', 'ŭ' => 'u', 'ũ' => 'u', 'ų' => 'u', 'ŵ' => 'w', 'ÿ' => 'y',
|
|
|
|
|
'ý' => 'y', 'ŷ' => 'y', 'ż' => 'z', 'ź' => 'z', 'ž' => 'z', 'Α' => 'A',
|
|
|
|
|
'Ά' => 'A', 'Ἀ' => 'A', 'Ἁ' => 'A', 'Ἂ' => 'A', 'Ἃ' => 'A', 'Ἄ' => 'A',
|
|
|
|
|
'Ἅ' => 'A', 'Ἆ' => 'A', 'Ἇ' => 'A', 'ᾈ' => 'A', 'ᾉ' => 'A', 'ᾊ' => 'A',
|
|
|
|
|
'ᾋ' => 'A', 'ᾌ' => 'A', 'ᾍ' => 'A', 'ᾎ' => 'A', 'ᾏ' => 'A', 'Ᾰ' => 'A',
|
|
|
|
|
'Ᾱ' => 'A', 'Ὰ' => 'A', 'ᾼ' => 'A', 'Β' => 'B', 'Γ' => 'G', 'Δ' => 'D',
|
|
|
|
|
'Ε' => 'E', 'Έ' => 'E', 'Ἐ' => 'E', 'Ἑ' => 'E', 'Ἒ' => 'E', 'Ἓ' => 'E',
|
|
|
|
|
'Ἔ' => 'E', 'Ἕ' => 'E', 'Ὲ' => 'E', 'Ζ' => 'Z', 'Η' => 'I', 'Ή' => 'I',
|
|
|
|
|
'Ἠ' => 'I', 'Ἡ' => 'I', 'Ἢ' => 'I', 'Ἣ' => 'I', 'Ἤ' => 'I', 'Ἥ' => 'I',
|
|
|
|
|
'Ἦ' => 'I', 'Ἧ' => 'I', 'ᾘ' => 'I', 'ᾙ' => 'I', 'ᾚ' => 'I', 'ᾛ' => 'I',
|
|
|
|
|
'ᾜ' => 'I', 'ᾝ' => 'I', 'ᾞ' => 'I', 'ᾟ' => 'I', 'Ὴ' => 'I', 'ῌ' => 'I',
|
|
|
|
|
'Θ' => 'T', 'Ι' => 'I', 'Ί' => 'I', 'Ϊ' => 'I', 'Ἰ' => 'I', 'Ἱ' => 'I',
|
|
|
|
|
'Ἲ' => 'I', 'Ἳ' => 'I', 'Ἴ' => 'I', 'Ἵ' => 'I', 'Ἶ' => 'I', 'Ἷ' => 'I',
|
|
|
|
|
'Ῐ' => 'I', 'Ῑ' => 'I', 'Ὶ' => 'I', 'Κ' => 'K', 'Λ' => 'L', 'Μ' => 'M',
|
|
|
|
|
'Ν' => 'N', 'Ξ' => 'K', 'Ο' => 'O', 'Ό' => 'O', 'Ὀ' => 'O', 'Ὁ' => 'O',
|
|
|
|
|
'Ὂ' => 'O', 'Ὃ' => 'O', 'Ὄ' => 'O', 'Ὅ' => 'O', 'Ὸ' => 'O', 'Π' => 'P',
|
|
|
|
|
'Ρ' => 'R', 'Ῥ' => 'R', 'Σ' => 'S', 'Τ' => 'T', 'Υ' => 'Y', 'Ύ' => 'Y',
|
|
|
|
|
'Ϋ' => 'Y', 'Ὑ' => 'Y', 'Ὓ' => 'Y', 'Ὕ' => 'Y', 'Ὗ' => 'Y', 'Ῠ' => 'Y',
|
|
|
|
|
'Ῡ' => 'Y', 'Ὺ' => 'Y', 'Φ' => 'F', 'Χ' => 'X', 'Ψ' => 'P', 'Ω' => 'O',
|
|
|
|
|
'Ώ' => 'O', 'Ὠ' => 'O', 'Ὡ' => 'O', 'Ὢ' => 'O', 'Ὣ' => 'O', 'Ὤ' => 'O',
|
|
|
|
|
'Ὥ' => 'O', 'Ὦ' => 'O', 'Ὧ' => 'O', 'ᾨ' => 'O', 'ᾩ' => 'O', 'ᾪ' => 'O',
|
|
|
|
|
'ᾫ' => 'O', 'ᾬ' => 'O', 'ᾭ' => 'O', 'ᾮ' => 'O', 'ᾯ' => 'O', 'Ὼ' => 'O',
|
|
|
|
|
'ῼ' => 'O', 'α' => 'a', 'ά' => 'a', 'ἀ' => 'a', 'ἁ' => 'a', 'ἂ' => 'a',
|
|
|
|
|
'ἃ' => 'a', 'ἄ' => 'a', 'ἅ' => 'a', 'ἆ' => 'a', 'ἇ' => 'a', 'ᾀ' => 'a',
|
|
|
|
|
'ᾁ' => 'a', 'ᾂ' => 'a', 'ᾃ' => 'a', 'ᾄ' => 'a', 'ᾅ' => 'a', 'ᾆ' => 'a',
|
|
|
|
|
'ᾇ' => 'a', 'ὰ' => 'a', 'ᾰ' => 'a', 'ᾱ' => 'a', 'ᾲ' => 'a', 'ᾳ' => 'a',
|
|
|
|
|
'ᾴ' => 'a', 'ᾶ' => 'a', 'ᾷ' => 'a', 'β' => 'b', 'γ' => 'g', 'δ' => 'd',
|
|
|
|
|
'ε' => 'e', 'έ' => 'e', 'ἐ' => 'e', 'ἑ' => 'e', 'ἒ' => 'e', 'ἓ' => 'e',
|
|
|
|
|
'ἔ' => 'e', 'ἕ' => 'e', 'ὲ' => 'e', 'ζ' => 'z', 'η' => 'i', 'ή' => 'i',
|
|
|
|
|
'ἠ' => 'i', 'ἡ' => 'i', 'ἢ' => 'i', 'ἣ' => 'i', 'ἤ' => 'i', 'ἥ' => 'i',
|
|
|
|
|
'ἦ' => 'i', 'ἧ' => 'i', 'ᾐ' => 'i', 'ᾑ' => 'i', 'ᾒ' => 'i', 'ᾓ' => 'i',
|
|
|
|
|
'ᾔ' => 'i', 'ᾕ' => 'i', 'ᾖ' => 'i', 'ᾗ' => 'i', 'ὴ' => 'i', 'ῂ' => 'i',
|
|
|
|
|
'ῃ' => 'i', 'ῄ' => 'i', 'ῆ' => 'i', 'ῇ' => 'i', 'θ' => 't', 'ι' => 'i',
|
|
|
|
|
'ί' => 'i', 'ϊ' => 'i', 'ΐ' => 'i', 'ἰ' => 'i', 'ἱ' => 'i', 'ἲ' => 'i',
|
|
|
|
|
'ἳ' => 'i', 'ἴ' => 'i', 'ἵ' => 'i', 'ἶ' => 'i', 'ἷ' => 'i', 'ὶ' => 'i',
|
|
|
|
|
'ῐ' => 'i', 'ῑ' => 'i', 'ῒ' => 'i', 'ῖ' => 'i', 'ῗ' => 'i', 'κ' => 'k',
|
|
|
|
|
'λ' => 'l', 'μ' => 'm', 'ν' => 'n', 'ξ' => 'k', 'ο' => 'o', 'ό' => 'o',
|
|
|
|
|
'ὀ' => 'o', 'ὁ' => 'o', 'ὂ' => 'o', 'ὃ' => 'o', 'ὄ' => 'o', 'ὅ' => 'o',
|
|
|
|
|
'ὸ' => 'o', 'π' => 'p', 'ρ' => 'r', 'ῤ' => 'r', 'ῥ' => 'r', 'σ' => 's',
|
|
|
|
|
'ς' => 's', 'τ' => 't', 'υ' => 'y', 'ύ' => 'y', 'ϋ' => 'y', 'ΰ' => 'y',
|
|
|
|
|
'ὐ' => 'y', 'ὑ' => 'y', 'ὒ' => 'y', 'ὓ' => 'y', 'ὔ' => 'y', 'ὕ' => 'y',
|
|
|
|
|
'ὖ' => 'y', 'ὗ' => 'y', 'ὺ' => 'y', 'ῠ' => 'y', 'ῡ' => 'y', 'ῢ' => 'y',
|
|
|
|
|
'ῦ' => 'y', 'ῧ' => 'y', 'φ' => 'f', 'χ' => 'x', 'ψ' => 'p', 'ω' => 'o',
|
|
|
|
|
'ώ' => 'o', 'ὠ' => 'o', 'ὡ' => 'o', 'ὢ' => 'o', 'ὣ' => 'o', 'ὤ' => 'o',
|
|
|
|
|
'ὥ' => 'o', 'ὦ' => 'o', 'ὧ' => 'o', 'ᾠ' => 'o', 'ᾡ' => 'o', 'ᾢ' => 'o',
|
|
|
|
|
'ᾣ' => 'o', 'ᾤ' => 'o', 'ᾥ' => 'o', 'ᾦ' => 'o', 'ᾧ' => 'o', 'ὼ' => 'o',
|
|
|
|
|
'ῲ' => 'o', 'ῳ' => 'o', 'ῴ' => 'o', 'ῶ' => 'o', 'ῷ' => 'o', 'А' => 'A',
|
|
|
|
|
'Б' => 'B', 'В' => 'V', 'Г' => 'G', 'Д' => 'D', 'Е' => 'E', 'Ё' => 'E',
|
|
|
|
|
'Ж' => 'Z', 'З' => 'Z', 'И' => 'I', 'Й' => 'I', 'К' => 'K', 'Л' => 'L',
|
|
|
|
|
'М' => 'M', 'Н' => 'N', 'О' => 'O', 'П' => 'P', 'Р' => 'R', 'С' => 'S',
|
|
|
|
|
'Т' => 'T', 'У' => 'U', 'Ф' => 'F', 'Х' => 'K', 'Ц' => 'T', 'Ч' => 'C',
|
|
|
|
|
'Ш' => 'S', 'Щ' => 'S', 'Ы' => 'Y', 'Э' => 'E', 'Ю' => 'Y', 'Я' => 'Y',
|
|
|
|
|
'а' => 'A', 'б' => 'B', 'в' => 'V', 'г' => 'G', 'д' => 'D', 'е' => 'E',
|
|
|
|
|
'ё' => 'E', 'ж' => 'Z', 'з' => 'Z', 'и' => 'I', 'й' => 'I', 'к' => 'K',
|
|
|
|
|
'л' => 'L', 'м' => 'M', 'н' => 'N', 'о' => 'O', 'п' => 'P', 'р' => 'R',
|
|
|
|
|
'с' => 'S', 'т' => 'T', 'у' => 'U', 'ф' => 'F', 'х' => 'K', 'ц' => 'T',
|
|
|
|
|
'ч' => 'C', 'ш' => 'S', 'щ' => 'S', 'ы' => 'Y', 'э' => 'E', 'ю' => 'Y',
|
|
|
|
|
'я' => 'Y', 'ð' => 'd', 'Ð' => 'D', 'þ' => 't', 'Þ' => 'T', 'ა' => 'a',
|
|
|
|
|
'ბ' => 'b', 'გ' => 'g', 'დ' => 'd', 'ე' => 'e', 'ვ' => 'v', 'ზ' => 'z',
|
|
|
|
|
'თ' => 't', 'ი' => 'i', 'კ' => 'k', 'ლ' => 'l', 'მ' => 'm', 'ნ' => 'n',
|
|
|
|
|
'ო' => 'o', 'პ' => 'p', 'ჟ' => 'z', 'რ' => 'r', 'ს' => 's', 'ტ' => 't',
|
|
|
|
|
'უ' => 'u', 'ფ' => 'p', 'ქ' => 'k', 'ღ' => 'g', 'ყ' => 'q', 'შ' => 's',
|
|
|
|
|
'ჩ' => 'c', 'ც' => 't', 'ძ' => 'd', 'წ' => 't', 'ჭ' => 'c', 'ხ' => 'k',
|
|
|
|
|
'ჯ' => 'j', 'ჰ' => 'h', 'ḩ' => 'h', 'ừ' => 'u', 'ế' => 'e', 'ả' => 'a',
|
|
|
|
|
'ị' => 'i', 'ậ' => 'a', 'ệ' => 'e', 'ỉ' => 'i', 'ộ' => 'o', 'ồ' => 'o',
|
|
|
|
|
'ề' => 'e', 'ơ' => 'o', 'ạ' => 'a', 'ẵ' => 'a', 'ư' => 'u', 'ắ' => 'a',
|
|
|
|
|
'ằ' => 'a', 'ầ' => 'a', 'ḑ' => 'd', 'Ḩ' => 'H', 'Ḑ' => 'D', 'ḑ' => 'd',
|
|
|
|
|
'ş' => 's', 'ā' => 'a', 'ţ' => 't', 'ễ' => 'e'
|
|
|
|
|
);
|
|
|
|
|
$string = str_replace(array_keys($transliteration), array_values($transliteration), $string);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Missing something?
|
|
|
|
|
if(strpos($string = htmlentities($string, ENT_QUOTES, 'UTF-8'), '&') !== false) {
|
|
|
|
|
$string = html_entity_decode(preg_replace('~&([a-z]{1,2})(?:acute|cedil|circ|grave|lig|orn|ring|slash|tilde|uml);~i', '$1', $string), ENT_QUOTES, 'UTF-8');
|
|
|
|
|
}
|
|
|
|
|
return $string;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Safe for HTML output
|
|
|
|
|
function safe_html($var, $flag=ENT_QUOTES) {
|
|
|
|
|
if(!is_array($var)) {
|
|
|
|
|
return $var === NULL ? NULL : htmlspecialchars($var, $flag, 'UTF-8'); // htmlspecialchars keeps ñ, á and all the UTF-8 valid chars
|
|
|
|
|
}
|
|
|
|
|
$safe_array = array();
|
|
|
|
|
foreach($var as $k => $v) {
|
|
|
|
|
$safe_array[$k] = is_array($v) ? safe_html($v) : ($v === NULL ? NULL : htmlspecialchars($v, $flag, 'UTF-8'));
|
|
|
|
|
}
|
|
|
|
|
return $safe_array;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* BYTE AND NUMBER HANDLING
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// Converts bytes to whatever
|
|
|
|
|
function format_bytes($bytes, $round=1) {
|
|
|
|
|
if(!is_numeric($bytes)) return false;
|
|
|
|
|
if ($bytes < 1000) return "$bytes B";
|
|
|
|
|
$units = ['KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];
|
|
|
|
|
foreach($units as $k => $v) {
|
|
|
|
|
$multiplier = pow(1000, $k + 1);
|
|
|
|
|
$threshold = $multiplier * 1000;
|
|
|
|
|
if($bytes < $threshold) {
|
|
|
|
|
$size = round($bytes / $multiplier, $round);
|
|
|
|
|
return "$size $v";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2017-01-15 20:53:11 +00:00
|
|
|
|
// Returns bytes for SIZE + Suffix format (dec + bin)
|
|
|
|
|
function get_bytes($size, $cut=NULL) {
|
|
|
|
|
if($cut == NULL) {
|
|
|
|
|
$suffix = substr($size, -3);
|
|
|
|
|
$suffix = preg_match('/([A-Za-z]){3}/', $suffix) ? $suffix : substr($size, -2);
|
|
|
|
|
} else {
|
|
|
|
|
$suffix = substr($size, $cut);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2017-01-15 20:53:11 +00:00
|
|
|
|
$suffix = strtoupper($suffix);
|
|
|
|
|
|
|
|
|
|
$units = ['KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']; // Default dec units
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2017-01-15 20:53:11 +00:00
|
|
|
|
if(strlen($suffix) == 3) { // Convert units to bin
|
|
|
|
|
foreach($units as &$unit) {
|
|
|
|
|
$split = str_split($unit);
|
|
|
|
|
$unit = $split[0] . 'I' . $split[1];
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(strlen($suffix) == 1) {
|
2017-01-22 18:50:02 +00:00
|
|
|
|
$suffix .= 'B'; // Adds missing "B" for shorthand ini notation (Turns 1G into 1GB)
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
|
|
|
|
if(!in_array($suffix, $units)) {
|
|
|
|
|
return $size;
|
|
|
|
|
}
|
|
|
|
|
$pow_factor = array_search($suffix, $units) + 1;
|
2017-01-15 20:53:11 +00:00
|
|
|
|
return $size * pow(strlen($suffix) == 2 ? 1000 : 1024, $pow_factor);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Converts bytes to MB
|
|
|
|
|
function bytes_to_mb($bytes) {
|
|
|
|
|
return round($bytes / pow(10, 6));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Returns bytes (used for the ini_get functions)
|
|
|
|
|
function get_ini_bytes($size) {
|
|
|
|
|
return get_bytes($size, -1);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* PATHS AND URL HANDLING
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// Add trailing slashes
|
|
|
|
|
function add_trailing_slashes($string) {
|
|
|
|
|
return add_ending_slash(add_starting_slash($string));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function add_starting_slash($string) {
|
|
|
|
|
return '/' . ltrim($string, '/');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function add_ending_slash($string) {
|
|
|
|
|
return rtrim($string, '/') . '/';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts backslash into forward slash
|
|
|
|
|
function forward_slash($string){
|
|
|
|
|
return str_replace('\\','/', $string);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts relative path to absolute path
|
|
|
|
|
function relative_to_absolute($filepath) {
|
|
|
|
|
return str_replace(G_ROOT_PATH_RELATIVE, G_ROOT_PATH, forward_slash($filepath));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts relative path to url
|
|
|
|
|
function relative_to_url($filepath, $root_url=NULL) {
|
|
|
|
|
if(!check_value($root_url)) {
|
|
|
|
|
$root_url = G_ROOT_URL;
|
|
|
|
|
}
|
|
|
|
|
return str_replace(G_ROOT_PATH_RELATIVE, $root_url, forward_slash($filepath));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts app URL to relative path
|
|
|
|
|
function url_to_relative($url, $root_url=NULL) {
|
|
|
|
|
if(!check_value($root_url)) {
|
|
|
|
|
$root_url = G_ROOT_URL;
|
|
|
|
|
}
|
|
|
|
|
return str_replace($root_url, G_ROOT_PATH_RELATIVE, $url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts absolute path to relative path
|
|
|
|
|
function absolute_to_relative($filepath) {
|
|
|
|
|
return str_replace(G_ROOT_PATH, G_ROOT_PATH_RELATIVE, forward_slash($filepath));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts absolute path to URL
|
|
|
|
|
function absolute_to_url($filepath, $root_url=NULL) {
|
|
|
|
|
if(!check_value($root_url)) {
|
|
|
|
|
$root_url = G_ROOT_URL;
|
|
|
|
|
}
|
|
|
|
|
if(G_ROOT_PATH === G_ROOT_PATH_RELATIVE) {
|
|
|
|
|
return $root_url . ltrim($filepath, '/');
|
|
|
|
|
}
|
|
|
|
|
return str_replace(G_ROOT_PATH, $root_url, forward_slash($filepath));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts app URL to absolute path
|
|
|
|
|
function url_to_absolute($url, $root_url=NULL) {
|
|
|
|
|
if(!check_value($root_url)) {
|
|
|
|
|
$root_url = G_ROOT_URL;
|
|
|
|
|
}
|
|
|
|
|
return str_replace($root_url, G_ROOT_PATH, $url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* GET AND FETCH SOME DATA
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get current G\ version
|
|
|
|
|
function get_version() {
|
|
|
|
|
return G_VERSION;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get the current app version
|
|
|
|
|
function get_app_version($full=true) {
|
|
|
|
|
if($full) {
|
|
|
|
|
return G_APP_VERSION;
|
|
|
|
|
} else {
|
|
|
|
|
preg_match('/\d\.\d/', G_APP_VERSION, $return);
|
|
|
|
|
return $return[0];
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get the $settings value for the app
|
|
|
|
|
function get_app_setting($key) {
|
|
|
|
|
return get_global('settings')[$key];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function get_domain() {
|
|
|
|
|
return HTTP_HOST;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function get_base_url($path='') {
|
|
|
|
|
$return = G_ROOT_URL . sanitize_relative_path($path);
|
|
|
|
|
return rtrim($return, '/');
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_current_url() {
|
|
|
|
|
return get_base_url(preg_replace('#'.G_ROOT_PATH_RELATIVE.'#', '', $_SERVER['REQUEST_URI'], 1));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function settings_has_db_info() {
|
|
|
|
|
$settings = get_global('settings');
|
|
|
|
|
if(!is_array($settings)) return FALSE;
|
|
|
|
|
$has = TRUE;
|
|
|
|
|
foreach(['db_driver', 'db_host', 'db_name', 'db_user'] as $k) {
|
|
|
|
|
if(!array_key_exists($k, $settings)) {
|
|
|
|
|
$has = FALSE;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return $has;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_regex_match($regex, $delimiter='/', $subject, $key=NULL) {
|
|
|
|
|
preg_match($delimiter . $regex . $delimiter, $subject, $matches);
|
|
|
|
|
if(array_key_exists($key, $matches)) {
|
|
|
|
|
return $matches[$key];
|
|
|
|
|
} else {
|
|
|
|
|
return $matches;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Fetch the contents from an URL
|
|
|
|
|
* if $file is set the downloaed file will be saved there
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
function fetch_url($url, $file=NULL, $options=[]) {
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!$url) {
|
2016-10-12 16:32:18 +00:00
|
|
|
|
throw new Exception('missing $url in ' . __FUNCTION__);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
return FALSE;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-10-12 16:32:18 +00:00
|
|
|
|
if(ini_get('allow_url_fopen') !== 1 && !function_exists('curl_init')) {
|
|
|
|
|
throw new Exception("Fatal error in " .__FUNCTION__. ": cURL isn't installed and allow_url_fopen is disabled. Can't perform HTTP requests.");
|
|
|
|
|
return FALSE;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2016-10-12 16:32:18 +00:00
|
|
|
|
|
|
|
|
|
// File get contents is the failover fn
|
2017-11-09 19:02:18 +00:00
|
|
|
|
$fn = (!function_exists('curl_init') ? 'fgc' : 'curl');
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($fn == 'curl') {
|
|
|
|
|
$ch = curl_init();
|
|
|
|
|
curl_setopt($ch, CURLOPT_URL, $url);
|
|
|
|
|
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0);
|
|
|
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_AUTOREFERER, 1);
|
2016-10-12 16:32:18 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_TIMEOUT, 60);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_HEADER, 0);
|
|
|
|
|
curl_setopt($ch, CURLOPT_FAILONERROR, 0);
|
2018-04-17 21:25:26 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_ENCODING, 'gzip');
|
2016-11-06 02:19:47 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_VERBOSE, 0);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
|
|
|
|
if(is_array($options) && count($options) > 0) {
|
|
|
|
|
foreach($options as $k => $v) {
|
|
|
|
|
curl_setopt($ch, $k, $v);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($file) {
|
|
|
|
|
// Save the file to $file destination
|
|
|
|
|
$out = @fopen($file, 'wb');
|
|
|
|
|
if(!$out) {
|
2018-04-17 21:25:26 +00:00
|
|
|
|
throw new Exception("Can't open " . __FUNCTION__ . "() file for read and write");
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
curl_setopt($ch, CURLOPT_FILE, $out);
|
|
|
|
|
@curl_exec($ch);
|
|
|
|
|
fclose($out);
|
|
|
|
|
} else {
|
|
|
|
|
// Return the file string
|
|
|
|
|
$file_get_contents = @curl_exec($ch);
|
|
|
|
|
}
|
|
|
|
|
if(curl_errno($ch)) {
|
|
|
|
|
$curl_error = curl_error($ch);
|
|
|
|
|
curl_close($ch);
|
2018-04-17 21:25:26 +00:00
|
|
|
|
throw new Exception('Curl error ' . $curl_error);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if($file == NULL) {
|
|
|
|
|
curl_close($ch);
|
|
|
|
|
return $file_get_contents;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
$context = stream_context_create([
|
2018-04-17 21:25:26 +00:00
|
|
|
|
'http' => ['ignore_errors' => TRUE],
|
2017-11-09 19:02:18 +00:00
|
|
|
|
]);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$result = @file_get_contents($url, FALSE, $context);
|
|
|
|
|
if(!$result) {
|
2018-04-17 21:25:26 +00:00
|
|
|
|
throw new Exception("Can't fetch target URL (file_get_contents)");
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if($file) {
|
|
|
|
|
if(file_put_contents($file, $result) === false) {
|
2018-04-17 21:25:26 +00:00
|
|
|
|
throw new Exception("Can't fetch target URL (file_put_contents)");
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return $result;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/* get complete raw, add success + error properties */
|
|
|
|
|
function getUrlHeaders($url, $options=[]) {
|
|
|
|
|
$ch = curl_init();
|
2018-04-17 21:25:26 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_URL, $url);
|
|
|
|
|
curl_setopt($ch, CURLOPT_HEADER, 1);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_NOBODY, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_AUTOREFERER, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0);
|
2018-04-17 21:25:26 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_USERAGENT, 'Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36');
|
|
|
|
|
// Inject custom options
|
|
|
|
|
if(is_array($options)) {
|
|
|
|
|
foreach($options as $k => $v) {
|
|
|
|
|
curl_setopt($ch, $k, $v);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
$raw = curl_exec($ch);
|
|
|
|
|
if(curl_errno($ch)) {
|
|
|
|
|
$return['error'] = curl_error($ch);
|
|
|
|
|
$return['http_code'] = 500;
|
|
|
|
|
} else {
|
|
|
|
|
$return = curl_getinfo($ch);
|
|
|
|
|
$return['raw'] = $raw;
|
|
|
|
|
}
|
2018-04-17 21:25:26 +00:00
|
|
|
|
curl_close($ch);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return $return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Returns float Execution time at this point of call
|
|
|
|
|
function get_execution_time() {
|
|
|
|
|
return microtime(true) - G_APP_TIME_EXECUTION_START;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get bcrypt optimal cost
|
|
|
|
|
function bcrypt_cost($time=0.2, $cost=9) {
|
|
|
|
|
do {
|
|
|
|
|
$cost++;
|
|
|
|
|
$inicio = microtime(true);
|
|
|
|
|
password_hash('test', PASSWORD_BCRYPT, ['cost' => $cost]);
|
|
|
|
|
$fin = microtime(true);
|
|
|
|
|
} while (($fin - $inicio) < $time);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return $cost;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* CONDITIONALS
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2017-05-21 20:19:25 +00:00
|
|
|
|
// Checks if $var is a positive integer using filter_var
|
|
|
|
|
function is_integer($var, $range=[]) {
|
|
|
|
|
$options = [];
|
|
|
|
|
if(!empty($range) && is_array($range)) {
|
|
|
|
|
foreach(['min', 'max'] as $k) {
|
|
|
|
|
if(is_int($range[$k])) {
|
|
|
|
|
$options['options'][$k . '_range'] = $range[$k];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return filter_var($var, FILTER_VALIDATE_INT, $options) !== FALSE;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This will tell if the string is an URL
|
|
|
|
|
function is_url($string) {
|
|
|
|
|
// Only for strings
|
|
|
|
|
if(!is_string($string)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// Internal PHP system
|
|
|
|
|
if(filter_var($string, FILTER_VALIDATE_URL)) { // Note: This doesn't validate any non-alphanumeric URL
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
// Test if it has foreign chars
|
|
|
|
|
if(strlen($string) !== strlen(utf8_encode($string))) {
|
|
|
|
|
$parsed_url = parse_url($string);
|
|
|
|
|
if(count($parsed_url) < 2) { // At least scheme and host
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
$schemes = ['http', 'https', 'ftp', 'ftps', 'mailto', 'news', 'irc', 'gopher', 'nntp', 'feed', 'telnet', 'mms', 'rtsp', 'svn', 'tel', 'fax', 'xmpp'];
|
|
|
|
|
if(!in_array(strtolower($parsed_url['scheme']), $schemes)) { // Must be a valid scheme
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if(!array_key_exists('host', $parsed_url)) { // Host must be there
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// At this point this thing looks like an URL
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Tells if the given url is https or not
|
|
|
|
|
function is_https($string) {
|
|
|
|
|
return strpos($string, 'https://') !== false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Tell if the string is an URL and if is valid
|
|
|
|
|
function is_valid_url($string) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!is_url($string)){
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$url = preg_replace('/^https/', 'http', $string, 1);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(function_exists('curl_init')) {
|
|
|
|
|
$ch = curl_init();
|
|
|
|
|
curl_setopt($ch, CURLOPT_URL, $url);
|
|
|
|
|
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER , 0);
|
|
|
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_NOBODY, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_FAILONERROR, 0);
|
|
|
|
|
curl_setopt($ch, CURLOPT_AUTOREFERER, 1);
|
|
|
|
|
curl_setopt($ch, CURLOPT_TIMEOUT, 120);
|
|
|
|
|
$result = @curl_exec($ch);
|
|
|
|
|
curl_close($ch);
|
|
|
|
|
return $result !== false ? true : false;
|
|
|
|
|
} else {
|
|
|
|
|
if(ini_get('allow_url_fopen')) {
|
|
|
|
|
$result = file_get_contents($url);
|
|
|
|
|
return $result === false ? false : true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Tell if the string is an image URL
|
|
|
|
|
function is_image_url($string) {
|
|
|
|
|
if(!is_string($string)) return false;
|
|
|
|
|
return preg_match('/(?:ftp|https?):\/\/(\w+:\w+@)?([0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}(:[0-9]{1,4}){0,1}|(?:[\w\-]+\.)+[a-z]{2,6})(?:\/[^\/#\?]+)+\.(?:jpe?g|gif|png|bmp)/i', $string);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Returns true if the system is in development mode
|
|
|
|
|
// nota: buscar los is_localhost()
|
|
|
|
|
function is_development_env() {
|
|
|
|
|
return G_APP_ENV == 'develompent';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function is_windows_os() {
|
|
|
|
|
return (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' ? true : false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Taken from http://php.net/manual/en/function.imagecreatefromgif.php#104473
|
|
|
|
|
function is_animated_image($filename){
|
|
|
|
|
if(!($fh = @fopen($filename, 'rb'))) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
$count = 0;
|
|
|
|
|
while(!feof($fh) && $count < 2) {
|
|
|
|
|
$chunk = fread($fh, 1024 * 100); //read 100kb at a time
|
|
|
|
|
$count += preg_match_all('#\x00\x21\xF9\x04.{4}\x00(\x2C|\x21)#s', $chunk, $matches);
|
|
|
|
|
}
|
|
|
|
|
fclose($fh);
|
|
|
|
|
return $count > 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* FILE RELATED
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Get mimetype of $file according to your php version
|
|
|
|
|
function get_mimetype($file) {
|
|
|
|
|
if(function_exists('finfo_open')) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
return finfo_file(finfo_open(FILEINFO_MIME_TYPE), $file);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
} else {
|
|
|
|
|
if(function_exists('mime_content_type')) {
|
|
|
|
|
return mime_content_type($file);
|
|
|
|
|
} else {
|
|
|
|
|
return extension_to_mime(get_file_extension($file));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// For now this only works for images
|
|
|
|
|
function mime_to_extension($mime, $reverse=false) {
|
|
|
|
|
$mime_to_extension = array(
|
|
|
|
|
'image/x-windows-bmp' => 'bmp',
|
|
|
|
|
'image/x-ms-bmp' => 'bmp',
|
|
|
|
|
'image/bmp' => 'bmp',
|
|
|
|
|
'image/gif' => 'gif',
|
|
|
|
|
'image/pjpeg' => 'jpg',
|
|
|
|
|
'image/jpeg' => 'jpg',
|
|
|
|
|
'image/x-png' => 'png',
|
|
|
|
|
'image/png' => 'png',
|
|
|
|
|
'image/x-tiff' => 'tiff',
|
|
|
|
|
'image/tiff' => 'tiff',
|
|
|
|
|
'image/x-icon' => 'ico',
|
|
|
|
|
'image/x-rgb' => 'rgb'
|
|
|
|
|
);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Used to get ext -> mime
|
|
|
|
|
if($reverse) $mime_to_extension = array_flip($mime_to_extension);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return $mime_to_extension[$mime];
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function extension_to_mime($ext) {
|
|
|
|
|
return mime_to_extension($ext, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Retrieves info about the current image file
|
|
|
|
|
function get_image_fileinfo($file) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
|
|
|
|
clearstatcache($file); // http://php.net/manual/es/function.clearstatcache.php (add G\ magic for those)
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$info = getimagesize($file);
|
|
|
|
|
$filesize = @filesize($file);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
|
|
|
|
if(!$info || $filesize === FALSE) return FALSE;
|
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$mime = strtolower($info['mime']);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
return array(
|
|
|
|
|
'filename' => basename($file), // image.jpg
|
|
|
|
|
'name' => basename($file, '.' . get_file_extension($file)), // image
|
|
|
|
|
'width' => intval($info[0]),
|
|
|
|
|
'height' => intval($info[1]),
|
|
|
|
|
'ratio' => $info[0] / $info[1],
|
|
|
|
|
'size' => intval($filesize),
|
|
|
|
|
'size_formatted' => format_bytes($filesize),
|
|
|
|
|
'mime' => $mime,
|
|
|
|
|
'extension' => mime_to_extension($mime),
|
|
|
|
|
'bits' => $info['bits'],
|
|
|
|
|
'channels' => $info['channels'],
|
|
|
|
|
'url' => absolute_to_url($file),
|
|
|
|
|
'md5' => md5_file($file)
|
|
|
|
|
);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_file_extension($file) {
|
|
|
|
|
return strtolower(pathinfo($file, PATHINFO_EXTENSION));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_filename($file) {
|
|
|
|
|
return basename($file);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function get_filename_without_extension($file) {
|
|
|
|
|
return preg_replace('/\\.[^.\\s]{2,4}$/', '', basename($file));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This creates .htaccess file on the target dir using the param rules
|
|
|
|
|
* This can be also used to add rules to a existing .htaccess file
|
|
|
|
|
*/
|
|
|
|
|
function generate_htaccess($rules, $directory, $before=NULL, $output=false) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$htaccess = $directory.'.htaccess';
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$rules_stock = [
|
|
|
|
|
'static' => '<Files .*>'."\n".
|
|
|
|
|
'order allow,deny'."\n".
|
|
|
|
|
'deny from all'."\n".
|
2017-11-09 19:02:18 +00:00
|
|
|
|
'</Files>'."\n\n".
|
2016-08-18 20:39:31 +00:00
|
|
|
|
'AddHandler cgi-script .php .php3 .phtml .pl .py .jsp .asp .htm .shtml .sh .cgi .fcgi'."\n".
|
|
|
|
|
'Options -ExecCGI',
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
'deny_php' => '<FilesMatch "\.php$">'."\n".
|
|
|
|
|
'Order Deny,Allow'."\n".
|
|
|
|
|
'Deny from all'."\n".
|
|
|
|
|
'</FilesMatch>',
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
'deny' => 'deny from all'
|
|
|
|
|
];
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(array_key_exists($rules, $rules_stock)) {
|
|
|
|
|
$rules = $rules_stock[$rules];
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(file_exists($htaccess)) {
|
|
|
|
|
$fgc = file_get_contents($htaccess);
|
|
|
|
|
if(strpos($fgc, $rules)) {
|
|
|
|
|
$done = true;
|
|
|
|
|
} else {
|
|
|
|
|
if(!is_null($before)) {
|
|
|
|
|
$rules = str_replace($before, $rules."\n".$before, $fgc);
|
|
|
|
|
$f = 'w';
|
|
|
|
|
} else {
|
|
|
|
|
$rules = "\n\n".$rules;
|
|
|
|
|
$f = 'a';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
$f = 'w';
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if(!$done) {
|
|
|
|
|
$fh = @fopen($htaccess, $f);
|
|
|
|
|
if(!$fh) return false;
|
|
|
|
|
if(fwrite($fh, $rules)) {
|
|
|
|
|
@fclose($fh); return $output ? $rules : true;
|
|
|
|
|
} else {
|
|
|
|
|
@fclose($fh); return $output ? $rules : false;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return $output ? $rules : true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Get a safe filename
|
|
|
|
|
* $method: original | random | mixed | id
|
|
|
|
|
* $filename: name of the original file
|
|
|
|
|
*/
|
|
|
|
|
function get_filename_by_method($method, $filename) {
|
|
|
|
|
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$max_length = 200; // Safe limit, ideally this should be 255 - 4
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$extension = get_file_extension($filename);
|
|
|
|
|
$clean_filename = substr($filename, 0, -(strlen($extension) + 1));
|
|
|
|
|
$clean_filename = unaccent_string($clean_filename); // change áéíóú to aeiou
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$clean_filename = preg_replace('/\s+/','-', $clean_filename); // change all spaces with dash
|
|
|
|
|
$clean_filename = trim($clean_filename, '-'); // get rid of those ugly dashes
|
|
|
|
|
$clean_filename = preg_replace('/[^\.\w\d-]/i', '', $clean_filename); // remove any non alphanumeric, non underscore, non hyphen and non dot
|
2016-08-18 20:39:31 +00:00
|
|
|
|
|
|
|
|
|
// Non alphanumeric name uh..
|
|
|
|
|
if(strlen($clean_filename) == 0) {
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$clean_filename = random_string(32);
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$unlimited_filename = $clean_filename; // No max_length limit
|
|
|
|
|
$capped_filename = substr($clean_filename, 0, $max_length); // 1->200
|
2016-08-18 20:39:31 +00:00
|
|
|
|
|
|
|
|
|
switch($method){
|
|
|
|
|
default:
|
|
|
|
|
case 'original':
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$name = $capped_filename;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'random':
|
|
|
|
|
$name = random_string(32);
|
|
|
|
|
break;
|
|
|
|
|
case 'mixed':
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$mixed_chars_length = 16;
|
|
|
|
|
$mixed_chars = random_string($mixed_chars_length);
|
|
|
|
|
if(strlen($capped_filename) + $mixed_chars_length > $max_length) {
|
|
|
|
|
// Bring the scissors Morty
|
|
|
|
|
$capped_filename = substr($capped_filename, 0, $max_length - $mixed_chars_length - strlen($capped_filename));
|
|
|
|
|
// Well done Morty you little piece of shit
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2018-04-17 21:25:26 +00:00
|
|
|
|
$name = $capped_filename . $mixed_chars;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'id':
|
|
|
|
|
$name = $unlimited_filename;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2018-04-17 21:25:26 +00:00
|
|
|
|
return $name . '.' . $extension; // 200 + 4
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function name_unique_file($path, $method='original', $filename) {
|
|
|
|
|
$file = $path . get_filename_by_method($method, $filename);
|
|
|
|
|
if($method == 'id') {
|
|
|
|
|
return $file;
|
|
|
|
|
}
|
|
|
|
|
while(file_exists($file)) {
|
|
|
|
|
if($method == 'original') $method = 'mixed';
|
|
|
|
|
$file = $path . get_filename_by_method($method, $filename);
|
|
|
|
|
}
|
|
|
|
|
return $file;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* IMAGE RELATED
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* PNG ALPHA CHANNEL SUPPORT for imagecopymerge();
|
|
|
|
|
* by Sina Salek
|
|
|
|
|
*
|
|
|
|
|
* Bugfix by Ralph Voigt (bug which causes it
|
|
|
|
|
* to work only for $src_x = $src_y = 0.
|
|
|
|
|
* Also, inverting opacity is not necessary.)
|
|
|
|
|
* 08-JAN-2011
|
|
|
|
|
*
|
|
|
|
|
**/
|
|
|
|
|
function imagecopymerge_alpha($dst_im, $src_im, $dst_x, $dst_y, $src_x, $src_y, $src_w, $src_h, $pct, $dst_im_ext) {
|
|
|
|
|
if($dst_im_ext=='jpg' && $pct==100) {
|
|
|
|
|
imagealphablending($dst_im, true);
|
|
|
|
|
imagealphablending($src_im, true);
|
|
|
|
|
imagecopy($dst_im, $src_im, $dst_x, $dst_y, 0, 0, $src_w, $src_h);
|
|
|
|
|
} else {
|
|
|
|
|
$transparent_index = imagecolortransparent($dst_im);
|
|
|
|
|
$colors_total = imagecolorstotal($dst_im);
|
|
|
|
|
$cut = imagecreatetruecolor($src_w, $src_h);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($transparent_index >= 0) {
|
|
|
|
|
$transparent_color = imagecolorsforindex($dst_im, $transparent_index);
|
|
|
|
|
$transparent_index = imagecolorallocatealpha($cut, $transparent_color['red'], $transparent_color['green'], $transparent_color['blue'], 127);
|
|
|
|
|
imagefill($cut, 0, 0, $transparent_index);
|
|
|
|
|
imagecolortransparent($cut, $transparent_index);
|
|
|
|
|
} else {
|
|
|
|
|
$color = imagecolorallocatealpha($cut, 0, 0, 0, 127);
|
|
|
|
|
imagefill($cut, 0, 0, $color);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($dst_im_ext=='png') {
|
|
|
|
|
imagealphablending($dst_im, false);
|
|
|
|
|
imagesavealpha($dst_im, true);
|
|
|
|
|
} else {
|
|
|
|
|
if($dst_im_ext!=='jpg') {
|
|
|
|
|
imagetruecolortopalette($dst_im, true, 255);
|
|
|
|
|
imagesavealpha($dst_im, false);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
}
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
if($dst_im_ext=='png' && $colors_total==0) {
|
|
|
|
|
if($pct<100) imagefilteropacity($src_im, $pct);
|
|
|
|
|
imagealphablending($dst_im, true);
|
|
|
|
|
imagesavealpha($dst_im, true);
|
|
|
|
|
imagecopy($dst_im, $src_im, $dst_x, $dst_y, 0, 0, $src_w, $src_h);
|
|
|
|
|
imagealphablending($dst_im, false);
|
|
|
|
|
} else {
|
|
|
|
|
imagecopy($cut, $dst_im, 0, 0, $dst_x, $dst_y, $src_w, $src_h);
|
|
|
|
|
imagecopy($cut, $src_im, 0, 0, $src_x, $src_y, $src_w, $src_h);
|
|
|
|
|
imagecopymerge($dst_im, $cut, $dst_x, $dst_y, 0, 0, $src_w, $src_h, $pct);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
imagedestroy($cut);
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
imagedestroy($src_im);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Taken from http://www.php.net/manual/en/function.imagefilter.php#82162
|
|
|
|
|
function imagefilteropacity(&$img, $opacity) {
|
|
|
|
|
if(!isset($opacity)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
$opacity /= 100;
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$w = imagesx($img);
|
|
|
|
|
$h = imagesy($img);
|
|
|
|
|
|
|
|
|
|
imagealphablending($img, false);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
//find the most opaque pixel in the image (the one with the smallest alpha value)
|
|
|
|
|
$minalpha = 127;
|
|
|
|
|
for($x = 0; $x < $w; $x++)
|
|
|
|
|
for($y = 0; $y < $h; $y++) {
|
|
|
|
|
$alpha = (imagecolorat($img, $x, $y) >> 24 ) & 0xFF;
|
|
|
|
|
if($alpha < $minalpha) {
|
|
|
|
|
$minalpha = $alpha;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
//loop through image pixels and modify alpha for each
|
|
|
|
|
for($x = 0; $x < $w; $x++) {
|
|
|
|
|
for($y = 0; $y < $h; $y++) {
|
|
|
|
|
//get current alpha value (represents the TANSPARENCY!)
|
|
|
|
|
$colorxy = imagecolorat($img, $x, $y);
|
|
|
|
|
$alpha = ($colorxy >> 24) & 0xFF;
|
|
|
|
|
//calculate new alpha
|
|
|
|
|
if($minalpha !== 127) {
|
|
|
|
|
$alpha = 127 + 127 * $opacity * ( $alpha - 127 ) / ( 127 - $minalpha );
|
|
|
|
|
} else { $alpha += 127 * $opacity; }
|
|
|
|
|
//get the color index with new alpha
|
|
|
|
|
$alphacolorxy = imagecolorallocatealpha($img, ($colorxy >> 16) & 0xFF, ($colorxy >> 8) & 0xFF, $colorxy & 0xFF, $alpha);
|
|
|
|
|
//set pixel with the new color + opacity
|
|
|
|
|
if(!imagesetpixel($img, $x, $y, $alphacolorxy)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function image_allocate_transparency($image, $extension) {
|
|
|
|
|
if($extension == 'png') {
|
|
|
|
|
imagealphablending($image, false);
|
|
|
|
|
imagesavealpha($image, true);
|
|
|
|
|
} else {
|
|
|
|
|
imagetruecolortopalette($image, true, 255);
|
|
|
|
|
imagesavealpha($image, false);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function image_copy_transparency($image_source, $image_target) {
|
|
|
|
|
$transparent_index = imagecolortransparent($image_source);
|
|
|
|
|
$palletsize = imagecolorstotal($image_source);
|
|
|
|
|
if($transparent_index >= 0 and $transparent_index < $palletsize) {
|
|
|
|
|
$transparent_color = imagecolorsforindex($image_source, $transparent_index);
|
|
|
|
|
$transparent_index = imagecolorallocatealpha($image_target, $transparent_color['red'], $transparent_color['green'], $transparent_color['blue'], 127);
|
|
|
|
|
imagefill($image_target, 0, 0, $transparent_index);
|
|
|
|
|
imagecolortransparent($image_target, $transparent_index);
|
|
|
|
|
} else {
|
|
|
|
|
$color = imagecolorallocatealpha($image_target, 0, 0, 0, 127);
|
|
|
|
|
imagefill($image_target, 0, 0, $color);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// http://www.programmierer-forum.de/function-imagecreatefrombmp-welche-variante-laeuft-t143137.htm
|
|
|
|
|
function imagecreatefrombmp($file) {
|
|
|
|
|
if(function_exists('imagecreatefrombmp')) return imagecreatefrombmp($file);
|
|
|
|
|
// version 1.00
|
|
|
|
|
if (!($fh = fopen($file, 'rb'))) {
|
|
|
|
|
trigger_error('imagecreatefrombmp: Can not open ' . $file, E_USER_WARNING);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// read file header
|
|
|
|
|
$meta = unpack('vtype/Vfilesize/Vreserved/Voffset', fread($fh, 14));
|
|
|
|
|
// check for bitmap
|
|
|
|
|
if ($meta['type'] != 19778) {
|
|
|
|
|
trigger_error('imagecreatefrombmp: ' . $file . ' is not a bitmap!', E_USER_WARNING);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// read image header
|
|
|
|
|
$meta += unpack('Vheadersize/Vwidth/Vheight/vplanes/vbits/Vcompression/Vimagesize/Vxres/Vyres/Vcolors/Vimportant', fread($fh, 40));
|
|
|
|
|
// read additional 16bit header
|
|
|
|
|
if ($meta['bits'] == 16) {
|
|
|
|
|
$meta += unpack('VrMask/VgMask/VbMask', fread($fh, 12));
|
|
|
|
|
}
|
|
|
|
|
// set bytes and padding
|
|
|
|
|
$meta['bytes'] = $meta['bits'] / 8;
|
|
|
|
|
$meta['decal'] = 4 - (4 * (($meta['width'] * $meta['bytes'] / 4)- floor($meta['width'] * $meta['bytes'] / 4)));
|
|
|
|
|
if ($meta['decal'] == 4) {
|
|
|
|
|
$meta['decal'] = 0;
|
|
|
|
|
}
|
|
|
|
|
// obtain imagesize
|
|
|
|
|
if ($meta['imagesize'] < 1) {
|
|
|
|
|
$meta['imagesize'] = $meta['filesize'] - $meta['offset'];
|
|
|
|
|
// in rare cases filesize is equal to offset so we need to read physical size
|
|
|
|
|
if ($meta['imagesize'] < 1) {
|
|
|
|
|
$meta['imagesize'] = @filesize($file) - $meta['offset'];
|
|
|
|
|
if ($meta['imagesize'] < 1) {
|
|
|
|
|
trigger_error('imagecreatefrombmp: Can not obtain filesize of ' . $file . '!', E_USER_WARNING);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// calculate colors
|
|
|
|
|
$meta['colors'] = !$meta['colors'] ? pow(2, $meta['bits']) : $meta['colors'];
|
|
|
|
|
// read color palette
|
|
|
|
|
$palette = array();
|
|
|
|
|
if ($meta['bits'] < 16) {
|
|
|
|
|
$palette = unpack('l' . $meta['colors'], fread($fh, $meta['colors'] * 4));
|
|
|
|
|
// in rare cases the color value is signed
|
|
|
|
|
if ($palette[1] < 0) {
|
|
|
|
|
foreach ($palette as $i => $color) {
|
|
|
|
|
$palette[$i] = $color + 16777216;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// create gd image
|
|
|
|
|
$im = imagecreatetruecolor($meta['width'], $meta['height']);
|
|
|
|
|
$data = fread($fh, $meta['imagesize']);
|
|
|
|
|
$p = 0;
|
|
|
|
|
$vide = chr(0);
|
|
|
|
|
$y = $meta['height'] - 1;
|
|
|
|
|
$error = 'imagecreatefrombmp: ' . $file . ' has not enough data!';
|
|
|
|
|
// loop through the image data beginning with the lower left corner
|
|
|
|
|
while ($y >= 0) {
|
|
|
|
|
$x = 0;
|
|
|
|
|
while ($x < $meta['width']) {
|
|
|
|
|
switch ($meta['bits']) {
|
|
|
|
|
case 32:
|
|
|
|
|
case 24:
|
|
|
|
|
if (!($part = substr($data, $p, 3))) {
|
|
|
|
|
trigger_error($error, E_USER_WARNING);
|
|
|
|
|
return $im;
|
|
|
|
|
}
|
|
|
|
|
$color = unpack('V', $part . $vide);
|
|
|
|
|
break;
|
|
|
|
|
case 16:
|
|
|
|
|
if (!($part = substr($data, $p, 2))) {
|
|
|
|
|
trigger_error($error, E_USER_WARNING);
|
|
|
|
|
return $im;
|
|
|
|
|
}
|
|
|
|
|
$color = unpack('v', $part);
|
|
|
|
|
$color[1] = (($color[1] & 0xf800) >> 8) * 65536 + (($color[1] & 0x07e0) >> 3) * 256 + (($color[1] & 0x001f) << 3);
|
|
|
|
|
break;
|
|
|
|
|
case 8:
|
|
|
|
|
$color = unpack('n', $vide . substr($data, $p, 1));
|
|
|
|
|
$color[1] = $palette[ $color[1] + 1 ];
|
|
|
|
|
break;
|
|
|
|
|
case 4:
|
|
|
|
|
$color = unpack('n', $vide . substr($data, floor($p), 1));
|
|
|
|
|
$color[1] = ($p * 2) % 2 == 0 ? $color[1] >> 4 : $color[1] & 0x0F;
|
|
|
|
|
$color[1] = $palette[ $color[1] + 1 ];
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
$color = unpack('n', $vide . substr($data, floor($p), 1));
|
|
|
|
|
switch (($p * 8) % 8) {
|
|
|
|
|
case 0:
|
|
|
|
|
$color[1] = $color[1] >> 7;
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
$color[1] = ($color[1] & 0x40) >> 6;
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
$color[1] = ($color[1] & 0x20) >> 5;
|
|
|
|
|
break;
|
|
|
|
|
case 3:
|
|
|
|
|
$color[1] = ($color[1] & 0x10) >> 4;
|
|
|
|
|
break;
|
|
|
|
|
case 4:
|
|
|
|
|
$color[1] = ($color[1] & 0x8) >> 3;
|
|
|
|
|
break;
|
|
|
|
|
case 5:
|
|
|
|
|
$color[1] = ($color[1] & 0x4) >> 2;
|
|
|
|
|
break;
|
|
|
|
|
case 6:
|
|
|
|
|
$color[1] = ($color[1] & 0x2) >> 1;
|
|
|
|
|
break;
|
|
|
|
|
case 7:
|
|
|
|
|
$color[1] = ($color[1] & 0x1);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
$color[1] = $palette[ $color[1] + 1 ];
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
trigger_error('imagecreatefrombmp: ' . $file . ' has ' . $meta['bits'] . ' bits and this is not supported!', E_USER_WARNING);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
imagesetpixel($im, $x, $y, $color[1]);
|
|
|
|
|
$x++;
|
|
|
|
|
$p += $meta['bytes'];
|
|
|
|
|
}
|
|
|
|
|
$y--;
|
|
|
|
|
$p += $meta['decal'];
|
|
|
|
|
}
|
|
|
|
|
fclose($fh);
|
|
|
|
|
return $im;
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* JSON
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
// Prepare json to only serve XMLHttpRequest
|
|
|
|
|
function json_prepare() {
|
|
|
|
|
if(is_development_env()) return;
|
|
|
|
|
if($_SERVER['HTTP_X_REQUESTED_WITH'] !== 'XMLHttpRequest') {
|
|
|
|
|
Render\json_output(['status_code' => 400]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function json_error($args) {
|
|
|
|
|
if(func_num_args($args) == 1 and is_object($args)) {
|
|
|
|
|
if(method_exists($args, 'getMessage') and method_exists($args, 'getCode')) {
|
|
|
|
|
$message = $args->getMessage();
|
|
|
|
|
$code = $args->getCode();
|
|
|
|
|
$context = get_class($args);
|
|
|
|
|
error_log($message); // log class errors
|
|
|
|
|
} else {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if(func_num_args($args) == 1) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
$message = $args;
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$code = NULL;
|
|
|
|
|
$context = NULL;
|
|
|
|
|
} else {
|
|
|
|
|
$message = func_get_arg(0);
|
|
|
|
|
$code = func_get_arg(1);
|
|
|
|
|
$context = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return array(
|
|
|
|
|
'status_code' => 400,
|
|
|
|
|
'error' => array(
|
|
|
|
|
'message' => $message,
|
|
|
|
|
'code' => $code,
|
|
|
|
|
'context' => $context
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* HTTP
|
|
|
|
|
* ---------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Redirects to another URL
|
2017-11-09 19:02:18 +00:00
|
|
|
|
*/
|
2016-08-18 20:39:31 +00:00
|
|
|
|
function redirect($to='', $status=301) {
|
|
|
|
|
if(!filter_var($to, FILTER_VALIDATE_URL)){
|
|
|
|
|
$to = get_base_url($to);
|
|
|
|
|
}
|
|
|
|
|
$to = preg_replace('|[^a-z0-9-~+_.?#=&;,/:%!]|i', '', $to);
|
|
|
|
|
if(php_sapi_name() != 'cgi-fcgi') set_status_header($status);
|
|
|
|
|
header("Location: $to");
|
|
|
|
|
die();
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* Set HTTP status header from status code
|
|
|
|
|
* @Inspired from WordPress
|
|
|
|
|
*/
|
|
|
|
|
function set_status_header($code) {
|
|
|
|
|
$desc = get_set_status_header_desc($code);
|
|
|
|
|
if(empty($desc)) return false;
|
|
|
|
|
$protocol = $_SERVER['SERVER_PROTOCOL'];
|
|
|
|
|
if('HTTP/1.1' != $protocol && 'HTTP/1.0' != $protocol) $protocol = 'HTTP/1.0';
|
|
|
|
|
$set_status_header = "$protocol $code $desc";
|
|
|
|
|
return @header($set_status_header, true, $code);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Gets header description according to its code
|
|
|
|
|
* @Inspired from WordPress
|
|
|
|
|
*/
|
|
|
|
|
function get_set_status_header_desc($code) {
|
|
|
|
|
$codes_to_desc = array(
|
|
|
|
|
100 => 'Continue',
|
|
|
|
|
101 => 'Switching Protocols',
|
|
|
|
|
102 => 'Processing',
|
|
|
|
|
200 => 'OK',
|
|
|
|
|
201 => 'Created',
|
|
|
|
|
202 => 'Accepted',
|
|
|
|
|
203 => 'Non-Authoritative Information',
|
|
|
|
|
204 => 'No Content',
|
|
|
|
|
205 => 'Reset Content',
|
|
|
|
|
206 => 'Partial Content',
|
|
|
|
|
207 => 'Multi-Status',
|
|
|
|
|
226 => 'IM Used',
|
|
|
|
|
300 => 'Multiple Choices',
|
|
|
|
|
301 => 'Moved Permanently',
|
|
|
|
|
302 => 'Found',
|
|
|
|
|
303 => 'See Other',
|
|
|
|
|
304 => 'Not Modified',
|
|
|
|
|
305 => 'Use Proxy',
|
|
|
|
|
306 => 'Reserved',
|
|
|
|
|
307 => 'Temporary Redirect',
|
|
|
|
|
400 => 'Bad Request',
|
|
|
|
|
401 => 'Unauthorized',
|
|
|
|
|
402 => 'Payment Required',
|
|
|
|
|
403 => 'Forbidden',
|
|
|
|
|
404 => 'Not Found',
|
|
|
|
|
405 => 'Method Not Allowed',
|
|
|
|
|
406 => 'Not Acceptable',
|
|
|
|
|
407 => 'Proxy Authentication Required',
|
|
|
|
|
408 => 'Request Timeout',
|
|
|
|
|
409 => 'Conflict',
|
|
|
|
|
410 => 'Gone',
|
|
|
|
|
411 => 'Length Required',
|
|
|
|
|
412 => 'Precondition Failed',
|
|
|
|
|
413 => 'Request Entity Too Large',
|
|
|
|
|
414 => 'Request-URI Too Long',
|
|
|
|
|
415 => 'Unsupported Media Type',
|
|
|
|
|
416 => 'Requested Range Not Satisfiable',
|
|
|
|
|
417 => 'Expectation Failed',
|
|
|
|
|
422 => 'Unprocessable Entity',
|
|
|
|
|
423 => 'Locked',
|
|
|
|
|
424 => 'Failed Dependency',
|
|
|
|
|
426 => 'Upgrade Required',
|
|
|
|
|
500 => 'Internal Server Error',
|
|
|
|
|
501 => 'Not Implemented',
|
|
|
|
|
502 => 'Bad Gateway',
|
|
|
|
|
503 => 'Service Unavailable',
|
|
|
|
|
504 => 'Gateway Timeout',
|
|
|
|
|
505 => 'HTTP Version Not Supported',
|
|
|
|
|
506 => 'Variant Also Negotiates',
|
|
|
|
|
507 => 'Insufficient Storage',
|
|
|
|
|
510 => 'Not Extended'
|
|
|
|
|
);
|
|
|
|
|
if(check_value($codes_to_desc[$code])) {
|
2017-11-09 19:02:18 +00:00
|
|
|
|
return $codes_to_desc[$code];
|
2016-08-18 20:39:31 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// @Inspired from WordPress
|
|
|
|
|
function clean_header_comment($string) {
|
|
|
|
|
return trim(preg_replace('/\s*(?:\*\/|\?>).*/', '', $string));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
} // G Namespace
|
|
|
|
|
|
|
|
|
|
// Global namespace
|
|
|
|
|
namespace {
|
|
|
|
|
|
2017-11-09 19:02:18 +00:00
|
|
|
|
function class_autoloader($class) {
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$array = ['class' => $class, 'exists' => class_exists($class)];
|
|
|
|
|
$explode = explode('\\', $class);
|
|
|
|
|
$last_key = key(array_slice($explode, -1, 1, TRUE));
|
|
|
|
|
$file = ($explode[0] == 'G' ? G_PATH_CLASSES : G_APP_PATH_CLASSES) . 'class.' . strtolower($explode[$last_key]) . '.php';
|
|
|
|
|
if(file_exists($file)) {
|
|
|
|
|
require_once($file);
|
|
|
|
|
} else {
|
|
|
|
|
//trigger_error("Can't autoload ".$class.' class. Class should be at '. $file, E_USER_ERROR);
|
|
|
|
|
// Don't trigger an error so it allows multiple autoloaders to be called
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
spl_autoload_register('class_autoloader');
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
/**
|
|
|
|
|
* A Compatibility library with PHP 5.5's simplified password hashing API.
|
|
|
|
|
*
|
|
|
|
|
* @author Anthony Ferrara <ircmaxell@php.net>
|
|
|
|
|
* @license http://www.opensource.org/licenses/mit-license.html MIT License
|
|
|
|
|
* @copyright 2012 The Authors
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
if(!defined('PASSWORD_DEFAULT')) {
|
|
|
|
|
|
|
|
|
|
define('PASSWORD_BCRYPT', 1);
|
|
|
|
|
define('PASSWORD_DEFAULT', PASSWORD_BCRYPT);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Hash the password using the specified algorithm
|
|
|
|
|
*
|
|
|
|
|
* @param string $password The password to hash
|
|
|
|
|
* @param int $algo The algorithm to use (Defined by PASSWORD_* constants)
|
|
|
|
|
* @param array $options The options for the algorithm to use
|
|
|
|
|
*
|
|
|
|
|
* @return string|false The hashed password, or false on error.
|
|
|
|
|
*/
|
|
|
|
|
function password_hash($password, $algo, array $options = array()) {
|
|
|
|
|
if (!function_exists('crypt')) {
|
|
|
|
|
trigger_error("Crypt must be loaded for password_hash to function", E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if (!is_string($password)) {
|
|
|
|
|
trigger_error("password_hash(): Password must be a string", E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if (!is_int($algo)) {
|
|
|
|
|
trigger_error("password_hash() expects parameter 2 to be long, " . gettype($algo) . " given", E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
switch ($algo) {
|
|
|
|
|
case PASSWORD_BCRYPT:
|
|
|
|
|
// Note that this is a C constant, but not exposed to PHP, so we don't define it here.
|
|
|
|
|
$cost = 10;
|
|
|
|
|
if (isset($options['cost'])) {
|
|
|
|
|
$cost = $options['cost'];
|
|
|
|
|
if ($cost < 4 || $cost > 31) {
|
|
|
|
|
trigger_error(sprintf('password_hash(): Invalid bcrypt cost parameter specified: %d', $cost), E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// The length of salt to generate
|
|
|
|
|
$raw_salt_len = 16;
|
|
|
|
|
// The length required in the final serialization
|
|
|
|
|
$required_salt_len = 22;
|
|
|
|
|
$hash_format = sprintf("$2y$%02d$", $cost);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
trigger_error(sprintf('password_hash(): Unknown password hashing algorithm: %s', $algo), E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if (isset($options['salt'])) {
|
|
|
|
|
switch (gettype($options['salt'])) {
|
|
|
|
|
case 'NULL':
|
|
|
|
|
case 'boolean':
|
|
|
|
|
case 'integer':
|
|
|
|
|
case 'double':
|
|
|
|
|
case 'string':
|
|
|
|
|
$salt = (string) $options['salt'];
|
|
|
|
|
break;
|
|
|
|
|
case 'object':
|
|
|
|
|
if (method_exists($options['salt'], '__tostring')) {
|
|
|
|
|
$salt = (string) $options['salt'];
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 'array':
|
|
|
|
|
case 'resource':
|
|
|
|
|
default:
|
|
|
|
|
trigger_error('password_hash(): Non-string salt parameter supplied', E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if (strlen($salt) < $required_salt_len) {
|
|
|
|
|
trigger_error(sprintf('password_hash(): Provided salt is too short: %d expecting %d', strlen($salt), $required_salt_len), E_USER_WARNING);
|
|
|
|
|
return null;
|
|
|
|
|
} elseif (0 == preg_match('#^[a-zA-Z0-9./]+$#D', $salt)) {
|
|
|
|
|
$salt = str_replace('+', '.', base64_encode($salt));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
$buffer = '';
|
|
|
|
|
$buffer_valid = false;
|
|
|
|
|
if (function_exists('mcrypt_create_iv') && !defined('PHALANGER')) {
|
|
|
|
|
$buffer = mcrypt_create_iv($raw_salt_len, MCRYPT_DEV_URANDOM);
|
|
|
|
|
if ($buffer) {
|
|
|
|
|
$buffer_valid = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!$buffer_valid && function_exists('openssl_random_pseudo_bytes')) {
|
|
|
|
|
$buffer = openssl_random_pseudo_bytes($raw_salt_len);
|
|
|
|
|
if ($buffer) {
|
|
|
|
|
$buffer_valid = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!$buffer_valid && is_readable('/dev/urandom')) {
|
|
|
|
|
$f = fopen('/dev/urandom', 'r');
|
|
|
|
|
$read = strlen($buffer);
|
|
|
|
|
while ($read < $raw_salt_len) {
|
|
|
|
|
$buffer .= fread($f, $raw_salt_len - $read);
|
|
|
|
|
$read = strlen($buffer);
|
|
|
|
|
}
|
|
|
|
|
fclose($f);
|
|
|
|
|
if ($read >= $raw_salt_len) {
|
|
|
|
|
$buffer_valid = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!$buffer_valid || strlen($buffer) < $raw_salt_len) {
|
|
|
|
|
$bl = strlen($buffer);
|
|
|
|
|
for ($i = 0; $i < $raw_salt_len; $i++) {
|
|
|
|
|
if ($i < $bl) {
|
|
|
|
|
$buffer[$i] = $buffer[$i] ^ chr(mt_rand(0, 255));
|
|
|
|
|
} else {
|
|
|
|
|
$buffer .= chr(mt_rand(0, 255));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
$salt = str_replace('+', '.', base64_encode($buffer));
|
|
|
|
|
}
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$salt = substr($salt, 0, $required_salt_len);
|
2017-11-09 19:02:18 +00:00
|
|
|
|
|
2016-08-18 20:39:31 +00:00
|
|
|
|
$hash = $hash_format . $salt;
|
|
|
|
|
$ret = crypt($password, $hash);
|
|
|
|
|
|
|
|
|
|
if (!is_string($ret) || strlen($ret) <= 13) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Get information about the password hash. Returns an array of the information
|
|
|
|
|
* that was used to generate the password hash.
|
|
|
|
|
*
|
|
|
|
|
* array(
|
|
|
|
|
* 'algo' => 1,
|
|
|
|
|
* 'algoName' => 'bcrypt',
|
|
|
|
|
* 'options' => array(
|
|
|
|
|
* 'cost' => 10,
|
|
|
|
|
* ),
|
|
|
|
|
* )
|
|
|
|
|
*
|
|
|
|
|
* @param string $hash The password hash to extract info from
|
|
|
|
|
*
|
|
|
|
|
* @return array The array of information about the hash.
|
|
|
|
|
*/
|
|
|
|
|
function password_get_info($hash) {
|
|
|
|
|
$return = array(
|
|
|
|
|
'algo' => 0,
|
|
|
|
|
'algoName' => 'unknown',
|
|
|
|
|
'options' => array(),
|
|
|
|
|
);
|
|
|
|
|
if (substr($hash, 0, 4) == '$2y$' && strlen($hash) == 60) {
|
|
|
|
|
$return['algo'] = PASSWORD_BCRYPT;
|
|
|
|
|
$return['algoName'] = 'bcrypt';
|
|
|
|
|
list($cost) = sscanf($hash, "$2y$%d$");
|
|
|
|
|
$return['options']['cost'] = $cost;
|
|
|
|
|
}
|
|
|
|
|
return $return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Determine if the password hash needs to be rehashed according to the options provided
|
|
|
|
|
*
|
|
|
|
|
* If the answer is true, after validating the password using password_verify, rehash it.
|
|
|
|
|
*
|
|
|
|
|
* @param string $hash The hash to test
|
|
|
|
|
* @param int $algo The algorithm used for new password hashes
|
|
|
|
|
* @param array $options The options array passed to password_hash
|
|
|
|
|
*
|
|
|
|
|
* @return boolean True if the password needs to be rehashed.
|
|
|
|
|
*/
|
|
|
|
|
function password_needs_rehash($hash, $algo, array $options = array()) {
|
|
|
|
|
$info = password_get_info($hash);
|
|
|
|
|
if ($info['algo'] != $algo) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
switch ($algo) {
|
|
|
|
|
case PASSWORD_BCRYPT:
|
|
|
|
|
$cost = isset($options['cost']) ? $options['cost'] : 10;
|
|
|
|
|
if ($cost != $info['options']['cost']) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Verify a password against a hash using a timing attack resistant approach
|
|
|
|
|
*
|
|
|
|
|
* @param string $password The password to verify
|
|
|
|
|
* @param string $hash The hash to verify against
|
|
|
|
|
*
|
|
|
|
|
* @return boolean If the password matches the hash
|
|
|
|
|
*/
|
|
|
|
|
function password_verify($password, $hash) {
|
|
|
|
|
if (!function_exists('crypt')) {
|
|
|
|
|
trigger_error('Crypt must be loaded for password_verify to function', E_USER_WARNING);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
$ret = crypt($password, $hash);
|
|
|
|
|
if (!is_string($ret) || strlen($ret) != strlen($hash) || strlen($ret) <= 13) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$status = 0;
|
|
|
|
|
for ($i = 0; $i < strlen($ret); $i++) {
|
|
|
|
|
$status |= (ord($ret[$i]) ^ ord($hash[$i]));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $status === 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-17 21:25:26 +00:00
|
|
|
|
}
|