Warning: php.ini: $ini_name ($ini_val) set lower than $var_name ($var_val)\n"); }; $warn_config_value('upload_max_filesize', 'MAX_FILESIZE', CONFIG::MAX_FILESIZE); $warn_config_value('post_max_size', 'MAX_FILESIZE', CONFIG::MAX_FILESIZE); $warn_config_value('max_input_time', 'UPLOAD_TIMEOUT', CONFIG::UPLOAD_TIMEOUT); $warn_config_value('max_execution_time', 'UPLOAD_TIMEOUT', CONFIG::UPLOAD_TIMEOUT); } //extract extension from a path (does not include the dot) function ext_by_path(string $path) : string { $ext = pathinfo($path, PATHINFO_EXTENSION); //special handling of .tar.* archives $ext2 = pathinfo(substr($path,0,-(strlen($ext)+1)), PATHINFO_EXTENSION); if ($ext2 === 'tar') { $ext = $ext2.'.'.$ext; } return $ext; } function ext_by_finfo(string $path) : string { $finfo = finfo_open(FILEINFO_EXTENSION); $finfo_ext = finfo_file($finfo, $path); finfo_close($finfo); if ($finfo_ext != '???') { return explode('/', $finfo_ext, 2)[0]; } else { $finfo = finfo_open(); $finfo_info = finfo_file($finfo, $path); finfo_close($finfo); if (strstr($finfo_info, 'text') !== false) { return 'txt'; } } return ''; } // store an uploaded file, given its name and temporary path (e.g. values straight out of $_FILES) // files are stored wit a randomised name, but with their original extension // // $name: original filename // $tmpfile: temporary path of uploaded file // $formatted: set to true to display formatted message instead of bare link function store_file(string $name, string $tmpfile, bool $formatted = false) : void { //create folder, if it doesn't exist if (!file_exists(CONFIG::STORE_PATH)) { mkdir(CONFIG::STORE_PATH, 0750, true); //TODO: error handling } //check file size $size = filesize($tmpfile); if ($size > CONFIG::MAX_FILESIZE * 1024 * 1024) { header('HTTP/1.0 413 Payload Too Large'); print("Error 413: Max File Size ({CONFIG::MAX_FILESIZE} MiB) Exceeded\n"); return; } if ($size == 0) { header('HTTP/1.0 400 Bad Request'); print('Error 400: Uploaded file is empty\n'); return; } $ext = ext_by_path($name); if (empty($ext) && CONFIG::AUTO_FILE_EXT) { $ext = ext_by_finfo($tmpfile); } $ext = substr($ext, 0, CONFIG::MAX_EXT_LEN); $tries_per_len=3; //try random names a few times before upping the length $id_length=CONFIG::ID_LENGTH; if(isset($_POST['secret'])) { $id_length=CONFIG::ID_LENGTH_SECRET; } for ($len = $id_length; ; ++$len) { for ($n=0; $n<=$tries_per_len; ++$n) { $id = rnd_str($len); $basename = $id . (empty($ext) ? '' : '.' . $ext); $target_file = CONFIG::STORE_PATH . $basename; if (!file_exists($target_file)) break 2; } } $res = move_uploaded_file($tmpfile, $target_file); if (!$res) { //TODO: proper error handling? header('HTTP/1.0 520 Unknown Error'); return; } if (CONFIG::EXTERNAL_HOOK !== null) { putenv('REMOTE_ADDR='.$_SERVER['REMOTE_ADDR']); putenv('ORIGINAL_NAME='.$name); putenv('STORED_FILE='.$target_file); $ret = -1; $out = null; $last_line = exec(CONFIG::EXTERNAL_HOOK, $out, $ret); if ($last_line !== false && $ret !== 0) { unlink($target_file); header('HTTP/1.0 400 Bad Request'); print("Error: $last_line\n"); return; } } //print the download link of the file $url = sprintf(CONFIG::SITE_URL().CONFIG::DOWNLOAD_PATH, $basename); if ($formatted) { print("
Access your file here: $url
"); } else { print("$url\n"); } // log uploader's IP, original filename, etc. if (CONFIG::LOG_PATH) { file_put_contents( CONFIG::LOG_PATH, implode("\t", array( date('c'), $_SERVER['REMOTE_ADDR'], filesize($tmpfile), escapeshellarg($name), $basename )) . "\n", FILE_APPEND ); } } // purge all files older than their retention period allows. function purge_files() : void { $num_del = 0; //number of deleted files $total_size = 0; //total size of deleted files //for each stored file foreach (scandir(CONFIG::STORE_PATH) as $file) { //skip virtual . and .. files if ($file === '.' || $file === '..') { continue; } $file = CONFIG::STORE_PATH . $file; $file_size = filesize($file) / (1024*1024); //size in MiB $file_age = (time()-filemtime($file)) / (60*60*24); //age in days //keep all files below the min age if ($file_age < CONFIG::MIN_FILEAGE) { continue; } //calculate the maximum age in days for this file $file_max_age = CONFIG::MIN_FILEAGE + (CONFIG::MAX_FILEAGE - CONFIG::MIN_FILEAGE) * pow(1 - ($file_size / CONFIG::MAX_FILESIZE), CONFIG::DECAY_EXP); //delete if older if ($file_age > $file_max_age) { unlink($file); print("deleted $file, $file_size MiB, $file_age days old\n"); $num_del += 1; $total_size += $file_size; } } print("Deleted $num_del files totalling $total_size MiB\n"); } function send_text_file(string $filename, string $content) : void { header('Content-type: application/octet-stream'); header("Content-Disposition: attachment; filename=\"$filename\""); header('Content-Length: '.strlen($content)); print($content); } // send a ShareX custom uploader config as .json function send_sharex_config() : void { $name = $_SERVER['SERVER_NAME']; $site_url = str_replace("?sharex", "", CONFIG::SITE_URL()); send_text_file($name.'.sxcu', << Filehost
 === How To Upload ===
You can upload files to this site via a simple HTTP POST, e.g. using curl:
curl -F "file=@/path/to/your/file.jpg" $site_url

Or if you want to pipe to curl *and* have a file extension, add a "filename":
echo "hello" | curl -F "file=@-;filename=.txt" $site_url

To use a longer ID length of $id_length_secret, add -F secret= to the
curl options. This makes the key less easy to guess. Please don't
consider it secure, the file is stored on the server in plain format.

On Windows, you can use ShareX and import this custom uploader.
On Android, you can use an app called Hupl with this uploader.


Or simply choose a file and click "Upload" below:
(Hint: If you're lucky, your browser may support drag-and-drop onto the file 
selection input.)


 === File Sizes etc. ===
The maximum allowed file size is $max_size MiB.

Files are kept for a minimum of $min_age, and a maximum of $max_age Days.

How long a file is kept depends on its size. Larger files are deleted earlier 
than small ones. This relation is non-linear and skewed in favour of small 
files.

The exact formula for determining the maximum age for a file is:

MIN_AGE + (MAX_AGE - MIN_AGE) * (1-(FILE_SIZE/MAX_SIZE))^$decay


 === Source ===
The PHP script used to provide this service is open source and available on 
GitHub


 === Contact ===
If you want to report abuse of this service, or have any other inquiries, 
please write an email to $mail
EOT; } // decide what to do, based on POST parameters etc. if (isset($_FILES['file']['name']) && isset($_FILES['file']['tmp_name']) && is_uploaded_file($_FILES['file']['tmp_name'])) { //file was uploaded, store it $formatted = isset($_REQUEST['formatted']); store_file($_FILES['file']['name'], $_FILES['file']['tmp_name'], $formatted); } else if (isset($_GET['sharex'])) { send_sharex_config(); } else if (isset($_GET['hupl'])) { send_hupl_config(); } else if ($argv[1] ?? null === 'purge') { purge_files(); } else { check_config(); print_index(); }