2021-12-14 18:32:47 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2023-01-10 22:44:09 +00:00
|
|
|
|
|
|
|
namespace Shimmie2;
|
|
|
|
|
2018-11-05 22:30:18 +00:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *\
|
|
|
|
* Misc *
|
|
|
|
\* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
2019-07-05 03:28:39 +00:00
|
|
|
const DATA_DIR = "data";
|
|
|
|
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
function get_theme(): string
|
|
|
|
{
|
|
|
|
global $config;
|
2019-08-02 19:40:03 +00:00
|
|
|
$theme = $config->get_string(SetupConfig::THEME, "default");
|
2019-05-28 16:59:38 +00:00
|
|
|
if (!file_exists("themes/$theme")) {
|
|
|
|
$theme = "default";
|
|
|
|
}
|
|
|
|
return $theme;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
function contact_link(): ?string
|
|
|
|
{
|
|
|
|
global $config;
|
|
|
|
$text = $config->get_string('contact_link');
|
|
|
|
if (is_null($text)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
2020-10-25 19:15:13 +00:00
|
|
|
str_starts_with($text, "http:") ||
|
|
|
|
str_starts_with($text, "https:") ||
|
|
|
|
str_starts_with($text, "mailto:")
|
2019-05-28 16:59:38 +00:00
|
|
|
) {
|
|
|
|
return $text;
|
|
|
|
}
|
|
|
|
|
2020-10-25 19:31:58 +00:00
|
|
|
if (str_contains($text, "@")) {
|
2019-05-28 16:59:38 +00:00
|
|
|
return "mailto:$text";
|
|
|
|
}
|
|
|
|
|
2020-10-25 19:31:58 +00:00
|
|
|
if (str_contains($text, "/")) {
|
2019-05-28 16:59:38 +00:00
|
|
|
return "http://$text";
|
|
|
|
}
|
|
|
|
|
|
|
|
return $text;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if HTTPS is enabled for the server.
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function is_https_enabled(): bool
|
|
|
|
{
|
2021-12-24 03:36:21 +00:00
|
|
|
// check forwarded protocol
|
2024-01-04 14:50:36 +00:00
|
|
|
if (is_trusted_proxy() && !empty($_SERVER['HTTP_X_FORWARDED_PROTO']) && $_SERVER['HTTP_X_FORWARDED_PROTO'] == 'https') {
|
2023-11-11 21:49:12 +00:00
|
|
|
$_SERVER['HTTPS'] = 'on';
|
2021-12-24 03:36:21 +00:00
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
return (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] !== 'off');
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compare two Block objects, used to sort them before being displayed
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function blockcmp(Block $a, Block $b): int
|
|
|
|
{
|
|
|
|
if ($a->position == $b->position) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
2020-01-26 13:19:35 +00:00
|
|
|
return ($a->position > $b->position) ? 1 : -1;
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Figure out PHP's internal memory limit
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function get_memory_limit(): int
|
|
|
|
{
|
|
|
|
global $config;
|
|
|
|
|
|
|
|
// thumbnail generation requires lots of memory
|
2023-11-11 21:49:12 +00:00
|
|
|
$default_limit = 8 * 1024 * 1024; // 8 MB of memory is PHP's default.
|
2020-01-26 13:19:35 +00:00
|
|
|
$shimmie_limit = $config->get_int(MediaConfig::MEM_LIMIT);
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($shimmie_limit < 3 * 1024 * 1024) {
|
2019-05-28 16:59:38 +00:00
|
|
|
// we aren't going to fit, override
|
|
|
|
$shimmie_limit = $default_limit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Get PHP's configured memory limit.
|
|
|
|
Note that this is set to -1 for NO memory limit.
|
|
|
|
|
2020-03-25 11:47:00 +00:00
|
|
|
https://ca2.php.net/manual/en/ini.core.php#ini.memory-limit
|
2019-05-28 16:59:38 +00:00
|
|
|
*/
|
|
|
|
$memory = parse_shorthand_int(ini_get("memory_limit"));
|
|
|
|
|
|
|
|
if ($memory == -1) {
|
|
|
|
// No memory limit.
|
|
|
|
// Return the larger of the set limits.
|
|
|
|
return max($shimmie_limit, $default_limit);
|
|
|
|
} else {
|
|
|
|
// PHP has a memory limit set.
|
|
|
|
if ($shimmie_limit > $memory) {
|
|
|
|
// Shimmie wants more memory than what PHP is currently set for.
|
|
|
|
|
|
|
|
// Attempt to set PHP's memory limit.
|
2020-01-26 13:19:35 +00:00
|
|
|
if (ini_set("memory_limit", "$shimmie_limit") === false) {
|
2019-05-28 16:59:38 +00:00
|
|
|
/* We can't change PHP's limit, oh well, return whatever its currently set to */
|
|
|
|
return $memory;
|
|
|
|
}
|
|
|
|
$memory = parse_shorthand_int(ini_get("memory_limit"));
|
|
|
|
}
|
|
|
|
|
|
|
|
// PHP's memory limit is more than Shimmie needs.
|
|
|
|
return $memory; // return the current setting
|
|
|
|
}
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 17:47:28 +00:00
|
|
|
/**
|
|
|
|
* Check if PHP has the GD library installed
|
|
|
|
*/
|
|
|
|
function check_gd_version(): int
|
|
|
|
{
|
|
|
|
$gdversion = 0;
|
|
|
|
|
|
|
|
if (function_exists('gd_info')) {
|
|
|
|
$gd_info = gd_info();
|
|
|
|
if (substr_count($gd_info['GD Version'], '2.')) {
|
|
|
|
$gdversion = 2;
|
|
|
|
} elseif (substr_count($gd_info['GD Version'], '1.')) {
|
|
|
|
$gdversion = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $gdversion;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether ImageMagick's `convert` command
|
|
|
|
* is installed and working
|
|
|
|
*/
|
|
|
|
function check_im_version(): int
|
|
|
|
{
|
2024-01-09 15:26:45 +00:00
|
|
|
$convert_check = exec("convert --version");
|
2020-01-27 17:47:28 +00:00
|
|
|
|
|
|
|
return (empty($convert_check) ? 0 : 1);
|
|
|
|
}
|
|
|
|
|
2024-01-04 14:50:36 +00:00
|
|
|
function is_trusted_proxy(): bool
|
2022-07-09 22:37:43 +00:00
|
|
|
{
|
2024-01-04 14:50:36 +00:00
|
|
|
$ra = $_SERVER['REMOTE_ADDR'] ?? "0.0.0.0";
|
2024-01-15 13:53:54 +00:00
|
|
|
// @phpstan-ignore-next-line - TRUSTED_PROXIES is defined in config
|
2024-01-04 14:50:36 +00:00
|
|
|
foreach(TRUSTED_PROXIES as $proxy) {
|
|
|
|
if(ip_in_range($ra, $proxy)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2022-01-17 17:06:20 +00:00
|
|
|
}
|
2024-01-04 14:50:36 +00:00
|
|
|
|
2022-01-17 17:06:20 +00:00
|
|
|
/**
|
|
|
|
* Get real IP if behind a reverse proxy
|
|
|
|
*/
|
2024-01-15 14:31:51 +00:00
|
|
|
function get_real_ip(): string
|
2022-07-09 22:37:43 +00:00
|
|
|
{
|
2024-01-04 14:50:36 +00:00
|
|
|
$ip = $_SERVER['REMOTE_ADDR'];
|
|
|
|
|
|
|
|
if(is_trusted_proxy()) {
|
|
|
|
if (isset($_SERVER['HTTP_X_REAL_IP'])) {
|
2024-01-20 20:48:47 +00:00
|
|
|
if(filter_var_ex($ip, FILTER_VALIDATE_IP)) {
|
2024-01-04 14:50:36 +00:00
|
|
|
$ip = $_SERVER['HTTP_X_REAL_IP'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
|
|
|
|
$ips = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);
|
|
|
|
$last_ip = $ips[count($ips) - 1];
|
2024-01-20 20:48:47 +00:00
|
|
|
if(filter_var_ex($last_ip, FILTER_VALIDATE_IP)) {
|
2024-01-04 14:50:36 +00:00
|
|
|
$ip = $last_ip;
|
|
|
|
}
|
2022-07-09 22:37:43 +00:00
|
|
|
}
|
2022-01-17 17:06:20 +00:00
|
|
|
}
|
2022-07-09 22:37:43 +00:00
|
|
|
|
|
|
|
return $ip;
|
2022-01-17 17:06:20 +00:00
|
|
|
}
|
|
|
|
|
2018-11-05 22:30:18 +00:00
|
|
|
/**
|
|
|
|
* Get the currently active IP, masked to make it not change when the last
|
|
|
|
* octet or two change, for use in session cookies and such
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function get_session_ip(Config $config): string
|
|
|
|
{
|
|
|
|
$mask = $config->get_string("session_hash_mask", "255.255.0.0");
|
2022-07-09 22:37:43 +00:00
|
|
|
$addr = get_real_ip();
|
2024-01-20 20:48:47 +00:00
|
|
|
$addr = inet_ntop_ex(inet_pton_ex($addr) & inet_pton_ex($mask));
|
2019-05-28 16:59:38 +00:00
|
|
|
return $addr;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A shorthand way to send a TextFormattingEvent and get the results.
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function format_text(string $string): string
|
|
|
|
{
|
2023-02-04 20:50:26 +00:00
|
|
|
$event = send_event(new TextFormattingEvent($string));
|
2023-01-11 14:04:35 +00:00
|
|
|
return $event->formatted;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-07-05 03:28:39 +00:00
|
|
|
/**
|
|
|
|
* Generates the path to a file under the data folder based on the file's hash.
|
|
|
|
* This process creates subfolders based on octet pairs from the file's hash.
|
|
|
|
* The calculated folder follows this pattern data/$base/octet_pairs/$hash
|
|
|
|
* @param string $base
|
|
|
|
* @param string $hash
|
|
|
|
* @param bool $create
|
|
|
|
* @param int $splits The number of octet pairs to split the hash into. Caps out at strlen($hash)/2.
|
|
|
|
* @return string
|
|
|
|
*/
|
2023-11-11 21:49:12 +00:00
|
|
|
function warehouse_path(string $base, string $hash, bool $create = true, int $splits = WH_SPLITS): string
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2023-11-11 21:49:12 +00:00
|
|
|
$dirs = [DATA_DIR, $base];
|
2019-07-05 03:28:39 +00:00
|
|
|
$splits = min($splits, strlen($hash) / 2);
|
2019-09-29 13:30:55 +00:00
|
|
|
for ($i = 0; $i < $splits; $i++) {
|
2019-07-05 03:28:39 +00:00
|
|
|
$dirs[] = substr($hash, $i * 2, 2);
|
|
|
|
}
|
|
|
|
$dirs[] = $hash;
|
2019-06-15 16:18:52 +00:00
|
|
|
|
2019-07-05 03:28:39 +00:00
|
|
|
$pa = join_path(...$dirs);
|
2019-06-15 16:18:52 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
if ($create && !file_exists(dirname($pa))) {
|
|
|
|
mkdir(dirname($pa), 0755, true);
|
|
|
|
}
|
|
|
|
return $pa;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-07-05 03:28:39 +00:00
|
|
|
/**
|
|
|
|
* Determines the path to the specified file in the data folder.
|
|
|
|
*/
|
|
|
|
function data_path(string $filename, bool $create = true): string
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2019-07-05 03:28:39 +00:00
|
|
|
$filename = join_path("data", $filename);
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($create && !file_exists(dirname($filename))) {
|
2019-05-28 16:59:38 +00:00
|
|
|
mkdir(dirname($filename), 0755, true);
|
|
|
|
}
|
|
|
|
return $filename;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2023-11-11 21:49:12 +00:00
|
|
|
function load_balance_url(string $tmpl, string $hash, int $n = 0): string
|
2020-02-09 16:02:16 +00:00
|
|
|
{
|
|
|
|
static $flexihashes = [];
|
|
|
|
$matches = [];
|
|
|
|
if (preg_match("/(.*){(.*)}(.*)/", $tmpl, $matches)) {
|
|
|
|
$pre = $matches[1];
|
|
|
|
$opts = $matches[2];
|
|
|
|
$post = $matches[3];
|
|
|
|
|
|
|
|
if (isset($flexihashes[$opts])) {
|
|
|
|
$flexihash = $flexihashes[$opts];
|
|
|
|
} else {
|
2023-01-10 22:44:09 +00:00
|
|
|
$flexihash = new \Flexihash\Flexihash();
|
2020-02-09 16:02:16 +00:00
|
|
|
foreach (explode(",", $opts) as $opt) {
|
|
|
|
$parts = explode("=", $opt);
|
|
|
|
$parts_count = count($parts);
|
|
|
|
$opt_val = "";
|
|
|
|
$opt_weight = 0;
|
|
|
|
if ($parts_count === 2) {
|
|
|
|
$opt_val = $parts[0];
|
2023-06-27 14:54:24 +00:00
|
|
|
$opt_weight = (int)$parts[1];
|
2020-02-09 16:02:16 +00:00
|
|
|
} elseif ($parts_count === 1) {
|
|
|
|
$opt_val = $parts[0];
|
|
|
|
$opt_weight = 1;
|
|
|
|
}
|
|
|
|
$flexihash->addTarget($opt_val, $opt_weight);
|
|
|
|
}
|
|
|
|
$flexihashes[$opts] = $flexihash;
|
|
|
|
}
|
|
|
|
|
|
|
|
// $choice = $flexihash->lookup($pre.$post);
|
|
|
|
$choices = $flexihash->lookupList($hash, $n + 1); // hash doesn't change
|
|
|
|
$choice = $choices[$n];
|
|
|
|
$tmpl = $pre . $choice . $post;
|
|
|
|
}
|
|
|
|
return $tmpl;
|
|
|
|
}
|
|
|
|
|
2024-01-20 19:47:26 +00:00
|
|
|
class FetchException extends \Exception
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
2024-01-20 19:47:26 +00:00
|
|
|
* @return array<string, string|string[]>
|
2024-01-20 14:10:59 +00:00
|
|
|
*/
|
2024-01-20 19:47:26 +00:00
|
|
|
function fetch_url(string $url, string $mfile): array
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $config;
|
|
|
|
|
2020-06-16 23:29:59 +00:00
|
|
|
if ($config->get_string(UploadConfig::TRANSLOAD_ENGINE) === "curl" && function_exists("curl_init")) {
|
2019-05-28 16:59:38 +00:00
|
|
|
$ch = curl_init($url);
|
2024-01-20 14:10:59 +00:00
|
|
|
assert($ch !== false);
|
2024-01-20 19:47:26 +00:00
|
|
|
$fp = false_throws(fopen($mfile, "w"));
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
|
|
|
curl_setopt($ch, CURLOPT_VERBOSE, 1);
|
|
|
|
curl_setopt($ch, CURLOPT_HEADER, 1);
|
|
|
|
curl_setopt($ch, CURLOPT_REFERER, $url);
|
|
|
|
curl_setopt($ch, CURLOPT_USERAGENT, "Shimmie-".VERSION);
|
|
|
|
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0);
|
|
|
|
|
|
|
|
$response = curl_exec($ch);
|
2020-01-26 23:23:01 +00:00
|
|
|
if ($response === false) {
|
2024-01-20 19:47:26 +00:00
|
|
|
throw new FetchException("cURL failed: ".curl_error($ch));
|
|
|
|
}
|
|
|
|
if ($response === true) { // we use CURLOPT_RETURNTRANSFER, so this should never happen
|
|
|
|
throw new FetchException("cURL failed successfully??");
|
2020-01-26 23:23:01 +00:00
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
$header_size = curl_getinfo($ch, CURLINFO_HEADER_SIZE);
|
2024-01-20 19:47:26 +00:00
|
|
|
$header_text = trim(substr($response, 0, $header_size));
|
|
|
|
$headers = http_parse_headers(implode("\n", false_throws(preg_split('/\R/', $header_text))));
|
2019-05-28 16:59:38 +00:00
|
|
|
$body = substr($response, $header_size);
|
|
|
|
|
|
|
|
curl_close($ch);
|
|
|
|
fwrite($fp, $body);
|
|
|
|
fclose($fp);
|
2024-01-20 19:47:26 +00:00
|
|
|
} elseif ($config->get_string(UploadConfig::TRANSLOAD_ENGINE) === "wget") {
|
2019-05-28 16:59:38 +00:00
|
|
|
$s_url = escapeshellarg($url);
|
|
|
|
$s_mfile = escapeshellarg($mfile);
|
|
|
|
system("wget --no-check-certificate $s_url --output-document=$s_mfile");
|
2024-01-20 19:47:26 +00:00
|
|
|
if(!file_exists($mfile)) {
|
|
|
|
throw new FetchException("wget failed");
|
|
|
|
}
|
|
|
|
$headers = [];
|
|
|
|
} elseif ($config->get_string(UploadConfig::TRANSLOAD_ENGINE) === "fopen") {
|
2019-05-28 16:59:38 +00:00
|
|
|
$fp_in = @fopen($url, "r");
|
|
|
|
$fp_out = fopen($mfile, "w");
|
|
|
|
if (!$fp_in || !$fp_out) {
|
2024-01-20 19:47:26 +00:00
|
|
|
throw new FetchException("fopen failed");
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
$length = 0;
|
2020-06-16 23:29:59 +00:00
|
|
|
while (!feof($fp_in) && $length <= $config->get_int(UploadConfig::SIZE)) {
|
2024-01-20 19:47:26 +00:00
|
|
|
$data = false_throws(fread($fp_in, 8192));
|
2019-05-28 16:59:38 +00:00
|
|
|
$length += strlen($data);
|
|
|
|
fwrite($fp_out, $data);
|
|
|
|
}
|
|
|
|
fclose($fp_in);
|
|
|
|
fclose($fp_out);
|
|
|
|
|
|
|
|
$headers = http_parse_headers(implode("\n", $http_response_header));
|
2024-01-20 19:47:26 +00:00
|
|
|
} else {
|
|
|
|
throw new FetchException("No transload engine configured");
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2024-01-20 19:47:26 +00:00
|
|
|
if (filesize($mfile) == 0) {
|
|
|
|
@unlink($mfile);
|
|
|
|
throw new FetchException("No data found in $url -- perhaps the site has hotlink protection?");
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
|
2024-01-20 19:47:26 +00:00
|
|
|
return $headers;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2024-01-04 15:54:44 +00:00
|
|
|
/**
|
|
|
|
* @return string[]
|
|
|
|
*/
|
|
|
|
function path_to_tags(string $path): array
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
$matches = [];
|
2019-06-21 08:12:44 +00:00
|
|
|
$tags = [];
|
|
|
|
if (preg_match("/\d+ - (.+)\.([a-zA-Z0-9]+)/", basename($path), $matches)) {
|
|
|
|
$tags = explode(" ", $matches[1]);
|
|
|
|
}
|
2019-06-20 04:26:30 +00:00
|
|
|
|
2022-10-27 15:56:12 +00:00
|
|
|
$path = str_replace("\\", "/", $path);
|
2019-06-14 14:45:40 +00:00
|
|
|
$path = str_replace(";", ":", $path);
|
2019-06-21 08:12:44 +00:00
|
|
|
$path = str_replace("__", " ", $path);
|
2022-10-27 15:56:12 +00:00
|
|
|
$path = dirname($path);
|
|
|
|
if ($path == "\\" || $path == "/" || $path == ".") {
|
|
|
|
$path = "";
|
|
|
|
}
|
2019-06-14 14:45:40 +00:00
|
|
|
|
|
|
|
$category = "";
|
2019-06-21 08:12:44 +00:00
|
|
|
foreach (explode("/", $path) as $dir) {
|
2019-06-14 14:45:40 +00:00
|
|
|
$category_to_inherit = "";
|
2019-06-21 08:12:44 +00:00
|
|
|
foreach (explode(" ", $dir) as $tag) {
|
2019-06-14 14:45:40 +00:00
|
|
|
$tag = trim($tag);
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($tag == "") {
|
2019-06-14 14:45:40 +00:00
|
|
|
continue;
|
|
|
|
}
|
2019-06-21 08:12:44 +00:00
|
|
|
if (substr_compare($tag, ":", -1) === 0) {
|
2019-06-14 14:45:40 +00:00
|
|
|
// This indicates a tag that ends in a colon,
|
|
|
|
// which is for inheriting to tags on the subfolder
|
|
|
|
$category_to_inherit = $tag;
|
|
|
|
} else {
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($category != "" && !str_contains($tag, ":")) {
|
2019-06-14 14:45:40 +00:00
|
|
|
// This indicates that category inheritance is active,
|
|
|
|
// and we've encountered a tag that does not specify a category.
|
|
|
|
// So we attach the inherited category to the tag.
|
|
|
|
$tag = $category.$tag;
|
|
|
|
}
|
2019-06-20 04:26:30 +00:00
|
|
|
$tags[] = $tag;
|
2019-06-14 14:45:40 +00:00
|
|
|
}
|
|
|
|
}
|
2019-06-21 08:12:44 +00:00
|
|
|
// Category inheritance only works on the immediate subfolder,
|
2019-06-14 14:45:40 +00:00
|
|
|
// so we hold a category until the next iteration, and then set
|
|
|
|
// it back to an empty string after that iteration
|
|
|
|
$category = $category_to_inherit;
|
2019-06-14 12:16:58 +00:00
|
|
|
}
|
2019-06-20 04:26:30 +00:00
|
|
|
|
2024-01-04 15:54:44 +00:00
|
|
|
return $tags;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @return string[]
|
|
|
|
*/
|
2019-10-10 15:25:37 +00:00
|
|
|
function get_dir_contents(string $dir): array
|
|
|
|
{
|
2020-01-26 13:19:35 +00:00
|
|
|
assert(!empty($dir));
|
|
|
|
|
2019-11-02 19:57:34 +00:00
|
|
|
if (!is_dir($dir)) {
|
2019-10-10 15:25:37 +00:00
|
|
|
return [];
|
|
|
|
}
|
2020-01-26 13:19:35 +00:00
|
|
|
return array_diff(
|
2024-01-20 19:47:26 +00:00
|
|
|
false_throws(scandir($dir)),
|
2019-11-02 19:57:34 +00:00
|
|
|
['..', '.']
|
|
|
|
);
|
2019-10-10 15:25:37 +00:00
|
|
|
}
|
|
|
|
|
2020-05-28 18:34:55 +00:00
|
|
|
function remove_empty_dirs(string $dir): bool
|
|
|
|
{
|
|
|
|
$result = true;
|
|
|
|
|
2024-01-20 19:47:26 +00:00
|
|
|
$items = get_dir_contents($dir);
|
|
|
|
;
|
2020-06-02 22:47:53 +00:00
|
|
|
foreach ($items as $item) {
|
2020-05-28 18:34:55 +00:00
|
|
|
$path = join_path($dir, $item);
|
2020-06-02 22:47:53 +00:00
|
|
|
if (is_dir($path)) {
|
2020-05-28 18:34:55 +00:00
|
|
|
$result = $result && remove_empty_dirs($path);
|
|
|
|
} else {
|
|
|
|
$result = false;
|
|
|
|
}
|
|
|
|
}
|
2023-11-11 21:49:12 +00:00
|
|
|
if ($result === true) {
|
2021-03-14 23:43:50 +00:00
|
|
|
$result = rmdir($dir);
|
2020-05-28 18:34:55 +00:00
|
|
|
}
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @return string[]
|
|
|
|
*/
|
2020-05-28 18:34:55 +00:00
|
|
|
function get_files_recursively(string $dir): array
|
|
|
|
{
|
2024-01-20 19:47:26 +00:00
|
|
|
$things = get_dir_contents($dir);
|
2020-05-28 18:34:55 +00:00
|
|
|
|
|
|
|
$output = [];
|
|
|
|
|
2020-06-02 22:47:53 +00:00
|
|
|
foreach ($things as $thing) {
|
|
|
|
$path = join_path($dir, $thing);
|
|
|
|
if (is_file($path)) {
|
2020-05-28 18:34:55 +00:00
|
|
|
$output[] = $path;
|
|
|
|
} else {
|
|
|
|
$output = array_merge($output, get_files_recursively($path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
2019-10-10 15:25:37 +00:00
|
|
|
/**
|
|
|
|
* Returns amount of files & total size of dir.
|
2024-01-20 14:10:59 +00:00
|
|
|
*
|
|
|
|
* @return array{"path": string, "total_files": int, "total_mb": string}
|
2019-10-10 15:25:37 +00:00
|
|
|
*/
|
|
|
|
function scan_dir(string $path): array
|
|
|
|
{
|
|
|
|
$bytestotal = 0;
|
|
|
|
$nbfiles = 0;
|
|
|
|
|
2023-01-10 22:44:09 +00:00
|
|
|
$ite = new \RecursiveDirectoryIterator(
|
2019-10-10 15:25:37 +00:00
|
|
|
$path,
|
2023-01-10 22:44:09 +00:00
|
|
|
\FilesystemIterator::KEY_AS_PATHNAME |
|
|
|
|
\FilesystemIterator::CURRENT_AS_FILEINFO |
|
|
|
|
\FilesystemIterator::SKIP_DOTS
|
2019-11-02 19:57:34 +00:00
|
|
|
);
|
2023-01-10 22:44:09 +00:00
|
|
|
foreach (new \RecursiveIteratorIterator($ite) as $filename => $cur) {
|
2019-10-10 15:25:37 +00:00
|
|
|
try {
|
|
|
|
$filesize = $cur->getSize();
|
|
|
|
$bytestotal += $filesize;
|
|
|
|
$nbfiles++;
|
2023-01-10 22:44:09 +00:00
|
|
|
} catch (\RuntimeException $e) {
|
2019-10-10 15:25:37 +00:00
|
|
|
// This usually just means that the file got eaten by the import
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$size_mb = $bytestotal / 1048576; // to mb
|
|
|
|
$size_mb = number_format($size_mb, 2, '.', '');
|
|
|
|
return ['path' => $path, 'total_files' => $nbfiles, 'total_mb' => $size_mb];
|
|
|
|
}
|
|
|
|
|
2018-11-05 22:30:18 +00:00
|
|
|
|
2023-01-11 13:27:57 +00:00
|
|
|
/**
|
|
|
|
* because microtime() returns string|float, and we only ever want float
|
|
|
|
*/
|
|
|
|
function ftime(): float
|
|
|
|
{
|
|
|
|
return (float)microtime(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-05 22:30:18 +00:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *\
|
|
|
|
* Debugging functions *
|
|
|
|
\* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
2023-01-11 13:27:57 +00:00
|
|
|
$_shm_load_start = ftime();
|
2018-11-05 22:30:18 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Collects some debug information (execution time, memory usage, queries, etc)
|
|
|
|
* and formats it to stick in the footer of the page.
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function get_debug_info(): string
|
|
|
|
{
|
2023-01-28 19:03:15 +00:00
|
|
|
$d = get_debug_info_arr();
|
|
|
|
|
|
|
|
$debug = "<br>Took {$d['time']} seconds (db:{$d['dbtime']}) and {$d['mem_mb']}MB of RAM";
|
|
|
|
$debug .= "; Used {$d['files']} files and {$d['query_count']} queries";
|
|
|
|
$debug .= "; Sent {$d['event_count']} events";
|
|
|
|
$debug .= "; {$d['cache_hits']} cache hits and {$d['cache_misses']} misses";
|
|
|
|
$debug .= "; Shimmie version {$d['version']}";
|
|
|
|
|
|
|
|
return $debug;
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2024-01-15 15:08:22 +00:00
|
|
|
/**
|
|
|
|
* Collects some debug information (execution time, memory usage, queries, etc)
|
|
|
|
*
|
|
|
|
* @return array<string, mixed>
|
|
|
|
*/
|
2023-01-28 19:03:15 +00:00
|
|
|
function get_debug_info_arr(): array
|
|
|
|
{
|
|
|
|
global $cache, $config, $_shm_event_count, $database, $_shm_load_start;
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
if ($config->get_string("commit_hash", "unknown") == "unknown") {
|
|
|
|
$commit = "";
|
|
|
|
} else {
|
|
|
|
$commit = " (".$config->get_string("commit_hash").")";
|
|
|
|
}
|
|
|
|
|
2023-01-28 19:03:15 +00:00
|
|
|
return [
|
|
|
|
"time" => round(ftime() - $_shm_load_start, 2),
|
|
|
|
"dbtime" => round($database->dbtime, 2),
|
2023-11-11 21:49:12 +00:00
|
|
|
"mem_mb" => round(((memory_get_peak_usage(true) + 512) / 1024) / 1024, 2),
|
2023-01-28 19:03:15 +00:00
|
|
|
"files" => count(get_included_files()),
|
|
|
|
"query_count" => $database->query_count,
|
|
|
|
// "query_log" => $database->queries,
|
|
|
|
"event_count" => $_shm_event_count,
|
2023-02-02 16:46:25 +00:00
|
|
|
"cache_hits" => $cache->get("__etc_cache_hits"),
|
|
|
|
"cache_misses" => $cache->get("__etc_cache_misses"),
|
2023-01-28 19:03:15 +00:00
|
|
|
"version" => VERSION . $commit,
|
|
|
|
];
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *\
|
|
|
|
* Request initialisation stuff *
|
|
|
|
\* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
2024-01-15 15:08:22 +00:00
|
|
|
/**
|
|
|
|
* @param string[] $files
|
|
|
|
*/
|
2020-01-29 00:49:21 +00:00
|
|
|
function require_all(array $files): void
|
|
|
|
{
|
2020-01-27 18:24:11 +00:00
|
|
|
foreach ($files as $filename) {
|
2020-02-06 03:10:30 +00:00
|
|
|
require_once $filename;
|
2020-01-27 18:24:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 14:31:51 +00:00
|
|
|
function _load_core_files(): void
|
2020-01-29 00:49:21 +00:00
|
|
|
{
|
2020-01-27 19:05:43 +00:00
|
|
|
require_all(array_merge(
|
|
|
|
zglob("core/*.php"),
|
2020-01-27 20:00:23 +00:00
|
|
|
zglob("core/imageboard/*.php"),
|
2020-01-27 19:05:43 +00:00
|
|
|
zglob("ext/*/info.php")
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2024-01-15 14:31:51 +00:00
|
|
|
function _load_extension_files(): void
|
2023-12-16 00:37:40 +00:00
|
|
|
{
|
|
|
|
ExtensionInfo::load_all_extension_info();
|
|
|
|
Extension::determine_enabled_extensions();
|
|
|
|
require_all(zglob("ext/{".Extension::get_enabled_extensions_as_string()."}/main.php"));
|
|
|
|
}
|
|
|
|
|
2024-01-15 14:31:51 +00:00
|
|
|
function _load_theme_files(): void
|
2020-01-29 00:49:21 +00:00
|
|
|
{
|
2023-01-10 22:44:09 +00:00
|
|
|
$theme = get_theme();
|
2023-12-16 00:37:40 +00:00
|
|
|
require_once('themes/'.$theme.'/page.class.php');
|
|
|
|
require_once('themes/'.$theme.'/themelet.class.php');
|
|
|
|
require_all(zglob("ext/{".Extension::get_enabled_extensions_as_string()."}/theme.php"));
|
|
|
|
require_all(zglob('themes/'.$theme.'/{'.Extension::get_enabled_extensions_as_string().'}.theme.php'));
|
2020-01-27 19:05:43 +00:00
|
|
|
}
|
|
|
|
|
2020-06-24 13:53:36 +00:00
|
|
|
function _set_up_shimmie_environment(): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
2020-03-13 09:23:54 +00:00
|
|
|
global $tracer_enabled;
|
2020-01-27 19:05:43 +00:00
|
|
|
|
|
|
|
if (file_exists("images") && !file_exists("data/images")) {
|
2020-06-24 13:53:36 +00:00
|
|
|
die_nicely("Upgrade error", "As of Shimmie 2.7 images and thumbs should be moved to data/images and data/thumbs");
|
2020-01-27 19:05:43 +00:00
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
|
|
|
|
if (TIMEZONE) {
|
|
|
|
date_default_timezone_set(TIMEZONE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (DEBUG) {
|
|
|
|
error_reporting(E_ALL);
|
|
|
|
}
|
|
|
|
|
2020-01-27 19:05:43 +00:00
|
|
|
// The trace system has a certain amount of memory consumption every time it is used,
|
|
|
|
// so to prevent running out of memory during complex operations code that uses it should
|
|
|
|
// check if tracer output is enabled before making use of it.
|
2024-01-15 13:53:54 +00:00
|
|
|
// @phpstan-ignore-next-line - TRACE_FILE is defined in config
|
2024-01-15 13:40:18 +00:00
|
|
|
$tracer_enabled = !is_null('TRACE_FILE');
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used to display fatal errors to the web user.
|
|
|
|
*/
|
2023-01-10 22:44:09 +00:00
|
|
|
function _fatal_error(\Exception $e): void
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
$version = VERSION;
|
|
|
|
$message = $e->getMessage();
|
2019-12-09 14:18:25 +00:00
|
|
|
$phpver = phpversion();
|
2018-11-05 22:30:18 +00:00
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
//$hash = exec("git rev-parse HEAD");
|
|
|
|
//$h_hash = $hash ? "<p><b>Hash:</b> $hash" : "";
|
|
|
|
//'.$h_hash.'
|
2018-11-05 22:30:18 +00:00
|
|
|
|
2019-12-07 22:53:59 +00:00
|
|
|
if (PHP_SAPI === 'cli' || PHP_SAPI == 'phpdbg') {
|
2019-12-15 15:31:44 +00:00
|
|
|
print("Trace: ");
|
|
|
|
$t = array_reverse($e->getTrace());
|
|
|
|
foreach ($t as $n => $f) {
|
|
|
|
$c = $f['class'] ?? '';
|
|
|
|
$t = $f['type'] ?? '';
|
2024-01-20 19:47:26 +00:00
|
|
|
$i = $f['file'] ?? 'unknown file';
|
|
|
|
$l = $f['line'] ?? -1;
|
2023-04-03 22:14:41 +00:00
|
|
|
$a = implode(", ", array_map("Shimmie2\stringer", $f['args'] ?? []));
|
2024-01-20 19:47:26 +00:00
|
|
|
print("$n: {$i}({$l}): {$c}{$t}{$f['function']}({$a})\n");
|
2019-12-15 15:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
print("Message: $message\n");
|
|
|
|
|
2024-01-04 22:48:56 +00:00
|
|
|
if (is_a($e, DatabaseException::class)) {
|
|
|
|
print("Query: {$e->query}\n");
|
|
|
|
print("Args: ".var_export($e->args, true)."\n");
|
2019-12-15 15:31:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
print("Version: $version (on $phpver)\n");
|
|
|
|
} else {
|
2024-01-04 22:48:56 +00:00
|
|
|
$query = is_a($e, DatabaseException::class) ? $e->query : null;
|
|
|
|
$code = is_a($e, SCoreException::class) ? $e->http_code : 500;
|
|
|
|
|
|
|
|
$q = "";
|
|
|
|
if(is_a($e, DatabaseException::class)) {
|
|
|
|
$q .= "<p><b>Query:</b> " . html_escape($query);
|
|
|
|
$q .= "<p><b>Args:</b> " . html_escape(var_export($e->args, true));
|
|
|
|
}
|
2021-11-16 14:52:26 +00:00
|
|
|
if ($code >= 500) {
|
|
|
|
error_log("Shimmie Error: $message (Query: $query)\n{$e->getTraceAsString()}");
|
|
|
|
}
|
|
|
|
header("HTTP/1.0 $code Error");
|
2019-12-15 15:31:44 +00:00
|
|
|
echo '
|
2020-02-01 18:11:00 +00:00
|
|
|
<!doctype html>
|
|
|
|
<html lang="en">
|
2018-11-05 22:30:18 +00:00
|
|
|
<head>
|
|
|
|
<title>Internal error - SCore-'.$version.'</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<h1>Internal Error</h1>
|
2019-11-11 16:43:04 +00:00
|
|
|
<p><b>Message:</b> '.html_escape($message).'
|
|
|
|
'.$q.'
|
2019-12-09 14:18:25 +00:00
|
|
|
<p><b>Version:</b> '.$version.' (on '.$phpver.')
|
2020-05-28 18:34:55 +00:00
|
|
|
<p><b>Stack Trace:</b></p><pre>'.$e->getTraceAsString().'</pre>
|
2018-11-05 22:30:18 +00:00
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
';
|
2019-12-15 15:31:44 +00:00
|
|
|
}
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
function _get_user(): User
|
|
|
|
{
|
|
|
|
global $config, $page;
|
2019-11-11 16:24:13 +00:00
|
|
|
$my_user = null;
|
2023-02-07 13:24:56 +00:00
|
|
|
if (isset($_SERVER['HTTP_AUTHORIZATION'])) {
|
|
|
|
$parts = explode(" ", $_SERVER['HTTP_AUTHORIZATION'], 2);
|
|
|
|
if (count($parts) == 2 && $parts[0] == "Bearer") {
|
|
|
|
$parts = explode(":", $parts[1], 2);
|
|
|
|
if (count($parts) == 2) {
|
|
|
|
$my_user = User::by_session($parts[0], $parts[1]);
|
|
|
|
}
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
|
|
|
}
|
2023-12-17 00:08:06 +00:00
|
|
|
if (is_null($my_user) && $page->get_cookie("user") && $page->get_cookie("session")) {
|
2023-02-07 13:24:56 +00:00
|
|
|
$my_user = User::by_session($page->get_cookie("user"), $page->get_cookie("session"));
|
|
|
|
}
|
2019-11-11 16:24:13 +00:00
|
|
|
if (is_null($my_user)) {
|
|
|
|
$my_user = User::by_id($config->get_int("anon_id", 0));
|
2019-05-28 16:59:38 +00:00
|
|
|
}
|
2019-11-11 16:24:13 +00:00
|
|
|
assert(!is_null($my_user));
|
2019-05-28 16:59:38 +00:00
|
|
|
|
2019-11-11 16:24:13 +00:00
|
|
|
return $my_user;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
2019-05-28 16:59:38 +00:00
|
|
|
function _get_query(): string
|
|
|
|
{
|
2024-02-07 22:47:12 +00:00
|
|
|
// if q is set in POST, use that
|
|
|
|
if(isset($_POST["q"])) {
|
|
|
|
return $_POST["q"];
|
|
|
|
}
|
|
|
|
|
|
|
|
// if q is set in GET, use that
|
|
|
|
// (we need to manually parse the query string because PHP's $_GET
|
|
|
|
// does an extra round of URL decoding, which we don't want)
|
|
|
|
$parts = parse_url($_SERVER['REQUEST_URI']);
|
|
|
|
$qs = [];
|
|
|
|
foreach(explode('&', $parts['query'] ?? "") as $z) {
|
|
|
|
$qps = explode('=', $z, 2);
|
|
|
|
if(count($qps) == 2) {
|
|
|
|
$qs[$qps[0]] = $qps[1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(isset($qs["q"])) {
|
|
|
|
return $qs["q"];
|
2023-11-07 20:58:46 +00:00
|
|
|
}
|
2024-02-07 22:47:12 +00:00
|
|
|
|
2023-11-07 20:58:46 +00:00
|
|
|
// if we're just looking at index.php, use the default query
|
2024-02-08 00:50:41 +00:00
|
|
|
if(str_ends_with($parts["path"] ?? "", "index.php")) {
|
2023-11-07 20:58:46 +00:00
|
|
|
return "/";
|
|
|
|
}
|
2024-02-07 22:47:12 +00:00
|
|
|
|
2024-02-07 23:22:24 +00:00
|
|
|
// otherwise, use the request URI minus the base path
|
2024-02-08 00:50:41 +00:00
|
|
|
return substr($parts["path"] ?? "", strlen(get_base_href()));
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *\
|
|
|
|
* HTML Generation *
|
|
|
|
\* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Give a HTML string which shows an IP (if the user is allowed to see IPs),
|
|
|
|
* and a link to ban that IP (if the user is allowed to ban IPs)
|
|
|
|
*
|
|
|
|
* FIXME: also check that IP ban ext is installed
|
|
|
|
*/
|
2019-05-28 16:59:38 +00:00
|
|
|
function show_ip(string $ip, string $ban_reason): string
|
|
|
|
{
|
|
|
|
global $user;
|
|
|
|
$u_reason = url_escape($ban_reason);
|
|
|
|
$u_end = url_escape("+1 week");
|
2020-03-28 14:11:14 +00:00
|
|
|
$ban = $user->can(Permissions::BAN_IP) ? ", <a href='".make_link("ip_ban/list", "c_ip=$ip&c_reason=$u_reason&c_expires=$u_end", "create")."'>Ban</a>" : "";
|
2019-07-09 14:10:21 +00:00
|
|
|
$ip = $user->can(Permissions::VIEW_IP) ? $ip.$ban : "";
|
2019-05-28 16:59:38 +00:00
|
|
|
return $ip;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Make a form tag with relevant auth token and stuff
|
|
|
|
*/
|
2023-11-11 21:49:12 +00:00
|
|
|
function make_form(string $target, string $method = "POST", bool $multipart = false, string $form_id = "", string $onsubmit = ""): string
|
2019-05-28 16:59:38 +00:00
|
|
|
{
|
|
|
|
global $user;
|
|
|
|
if ($method == "GET") {
|
2024-02-09 10:50:07 +00:00
|
|
|
die("make_form: GET method is not supported");
|
2019-05-28 16:59:38 +00:00
|
|
|
} else {
|
|
|
|
$extra_inputs = $user->get_auth_html();
|
|
|
|
}
|
|
|
|
|
|
|
|
$extra = empty($form_id) ? '' : 'id="'. $form_id .'"';
|
|
|
|
if ($multipart) {
|
|
|
|
$extra .= " enctype='multipart/form-data'";
|
|
|
|
}
|
|
|
|
if ($onsubmit) {
|
|
|
|
$extra .= ' onsubmit="'.$onsubmit.'"';
|
|
|
|
}
|
|
|
|
return '<form action="'.$target.'" method="'.$method.'" '.$extra.'>'.$extra_inputs;
|
2018-11-05 22:30:18 +00:00
|
|
|
}
|
2019-12-09 14:18:25 +00:00
|
|
|
|
2020-02-24 14:49:40 +00:00
|
|
|
const BYTE_DENOMINATIONS = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];
|
2024-01-15 14:31:51 +00:00
|
|
|
function human_filesize(int $bytes, int $decimals = 2): string
|
2020-02-24 14:49:40 +00:00
|
|
|
{
|
|
|
|
$factor = floor((strlen(strval($bytes)) - 1) / 3);
|
|
|
|
return sprintf("%.{$decimals}f", $bytes / pow(1024, $factor)) . @BYTE_DENOMINATIONS[$factor];
|
|
|
|
}
|
2020-06-17 00:03:03 +00:00
|
|
|
|
2021-03-14 23:43:50 +00:00
|
|
|
/**
|
2020-06-17 00:03:03 +00:00
|
|
|
* Generates a unique key for the website to prevent unauthorized access.
|
|
|
|
*/
|
2021-03-14 23:43:50 +00:00
|
|
|
function generate_key(int $length = 20): string
|
2020-06-17 00:03:03 +00:00
|
|
|
{
|
|
|
|
$characters = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
|
|
|
|
$randomString = '';
|
|
|
|
|
|
|
|
for ($i = 0; $i < $length; $i++) {
|
|
|
|
$randomString .= $characters [rand(0, strlen($characters) - 1)];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $randomString;
|
|
|
|
}
|
2024-01-20 20:48:47 +00:00
|
|
|
|
|
|
|
function shm_tempnam(string $prefix = ""): string
|
|
|
|
{
|
|
|
|
if(!is_dir("data/temp")) {
|
|
|
|
mkdir("data/temp");
|
|
|
|
}
|
|
|
|
$temp = false_throws(realpath("data/temp"));
|
|
|
|
return false_throws(tempnam($temp, $prefix));
|
|
|
|
}
|