1
0
mirror of https://github.com/Rouji/single_php_filehost.git synced 2025-04-04 16:39:34 +00:00

clean up some things

also added logging and (clunky) scan using clamd
This commit is contained in:
Rouji 2019-03-23 18:12:25 +01:00
parent 3a2d84d7de
commit 8bbc9138da
2 changed files with 180 additions and 141 deletions

View File

@ -1,12 +1,12 @@
# Single .php Filehost # Single .php Filehost
Simple PHP script, mainly for sharing random files with people using curl. (and thus in an easily scriptable way) Simple PHP script, mainly for sharing random files with people using curl. (and thus in an easily scriptable way)
It receives files uploaded via HTTP POST, and saves them to a configured directory, with a randomised filename (but preserving the original file extension). Puts a file sent via POST into a configured directory with a randomised filename but preserving the original file extension, and returns a link to it.
On successful upload, it returns a link to the uploaded file. Serving the file to people you've shared the link with can then simply be left to apache to figure out. Actually serving the file to people is left to apache to figure out.
There's also a mechanism for removing files over a certain age, which can be invoked by calling the script with a commandline argument. There's also a mechanism for removing files over a certain age, which can be invoked by calling the script with a commandline argument.
## Config # Config
All configuration is done using the global variables at the top of **index.php**. Hopefully, they're explained well enough in the short comments besides them. All configuration is done using the global variables at the top of **index.php**. Hopefully, they're explained well enough in the short comments besides them.
To accommodate for larger uploads, you'll also need to set the following values in your php.ini: To accommodate for larger uploads, you'll also need to set the following values in your php.ini:
@ -18,10 +18,32 @@ max_execution_time
The code responsible for the default info text can be found at the very bottom of index.php, in case you want to reword anything. The code responsible for the default info text can be found at the very bottom of index.php, in case you want to reword anything.
## Apache
Pretty straight forward, I use something like this:
## Purging Old Files
To check for any files, that exceed their max age, and delete them, you need to call index.php with the argument "purge"
``` ```
<Directory /path/to/webroot/>
Options +FollowSymLinks -MultiViews -Indexes
AddDefaultCharset UTF-8
AllowOverride None
RewriteEngine On
RewriteCond "%{ENV:REDIRECT_STATUS}" "^$"
RewriteRule "^/?$" "index.php" [L,END]
RewriteRule "^(.+)$" "files/$1" [L,END]
</Directory>
<Directory /path/to/webroot/files>
Options -ExecCGI
php_flag engine off
SetHandler None
AddType text/plain .php .php5 .html .htm .cpp .c .h .sh
</Directory>
```
# Purging Old Files
To check for any files, that exceed their max age, and delete them, you need to call index.php with the argument "purge"
```bash
php index.php purge php index.php purge
``` ```
@ -32,12 +54,12 @@ To automate this, simply create a cron job:
If you specify **$STORE_PATH** using an absolute path, you can omit the **cd** If you specify **$STORE_PATH** using an absolute path, you can omit the **cd**
### Max. File Age ## Max. File Age
The max age of a file is computed using the following formula: The max age of a file is computed using the following formula:
``` ```
$fileMaxAge = $MIN_FILEAGE + $file_max_age = $MIN_FILEAGE +
($MAX_FILEAGE - $MIN_FILEAGE) * ($MAX_FILEAGE - $MIN_FILEAGE) *
pow(1-($fileSize/$MAX_FILESIZE),$DECAY_EXP); pow(1-($fileSize/$MAX_FILESIZE),$DECAY_EXP);
``` ```
...which is a basic exponential decay curve that favours smaller files, meaning small files are kept longer and really big ones are deleted relatively quickly. ...which is a basic exponential decay curve that favours smaller files, meaning small files are kept longer and really big ones are deleted relatively quickly.
**$DECAY_EXP** is one of the configurable globals and basically makes the curve more or less exponential-looking. Set to 1 for a completely linear relationship. **$DECAY_EXP** is one of the configurable globals and basically makes the curve more or less exponential-looking. Set to 1 for a completely linear relationship.

281
index.php
View File

@ -1,110 +1,99 @@
<?php <?php
//////////////////////////////////////////////////////////////////////////////// $MAX_FILESIZE=512; //max. filesize in MiB
// global config
////////////////////////////////////////////////////////////////////////////////
$MAX_FILESIZE=512; //max. filesize in MiB
$MAX_FILEAGE=180; //max. age of files in days $MAX_FILEAGE=180; //max. age of files in days
$MIN_FILEAGE=31; //min. age of files in days $MIN_FILEAGE=31; //min. age of files in days
$DECAY_EXP=2; //high values penalise larger files $DECAY_EXP=2; //high values penalise larger files more
$UPLOAD_TIMEOUT=5*60; //max. time an upload can take before it times out $UPLOAD_TIMEOUT=5*60; //max. time an upload can take before it times out
$ID_LENGTH=3; //length of the random file ID $ID_LENGTH=3; //length of the random file ID
$STORE_PATH="files/"; //directory to store uploaded files in $STORE_PATH="files/"; //directory to store uploaded files in
$DOWNLOAD_URL="%s"; //%s = placeholder for filename $LOG_PATH=null; //path to log uploads + resulting links to
$DOWNLOAD_PATH="%s"; //the path part of the download url. %s = placeholder for filename
$HTTP_PROTO="https"; //protocol to use in links $HTTP_PROTO="https"; //protocol to use in links
$CLAM_SCAN=false; //scan files using clamd
$MAX_EXT_LEN=7; //max. length for file extensions
$ADMIN_EMAIL="complaintsgo@here.com"; //address for inquiries $ADMIN_EMAIL="admin@example.com"; //address for inquiries
////////////////////////////////////////////////////////////////////////////////
// decide what to do, based on POST parameters etc.
////////////////////////////////////////////////////////////////////////////////
if (isset($_FILES["file"]["name"]) && //file was uploaded, store it
isset($_FILES["file"]["tmp_name"]) &&
is_uploaded_file($_FILES["file"]["tmp_name"]))
{
$formatted = isset($_GET["formatted"]) || isset($_POST["formatted"]);
storeFile($_FILES["file"]["name"],
$_FILES["file"]["tmp_name"],
$formatted);
}
else if (isset($_GET['sharex'])) //send sharex config
{
sendShareXConfig();
}
else if (isset($_GET['hupl'])) //send hupl config
{
sendHuplConfig();
}
else if (isset($argv[1]) && //file was called from cmd, to purge old files
$argv[1] === 'purge')
{
purgeFiles();
}
else //nothing special going on, print info text
{
checkConfig(); //check for any php.ini config problems
printInfo(); //print info page
}
////////////////////////////////////////////////////////////////////////////////
// generate a random string of characters with given length // generate a random string of characters with given length
//////////////////////////////////////////////////////////////////////////////// function rnd_str($len)
function rndStr($len)
{ {
$chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_'; $chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_';
$maxIdx = strlen($chars) - 1; $max_idx = strlen($chars) - 1;
$out = ''; $out = '';
while ($len--) while ($len--)
{ {
$out .= $chars[mt_rand(0,$maxIdx)]; $out .= $chars[mt_rand(0,$max_idx)];
} }
return $out; return $out;
} }
// check php.ini settings and print warnings if anything's not configured properly
//////////////////////////////////////////////////////////////////////////////// function check_config()
// check php.ini settings and print warnings if anything's not configured
// properly
////////////////////////////////////////////////////////////////////////////////
function checkConfig()
{ {
global $MAX_FILESIZE; global $MAX_FILESIZE;
global $UPLOAD_TIMEOUT; global $UPLOAD_TIMEOUT;
warnConfig('upload_max_filesize', "MAX_FILESIZE", $MAX_FILESIZE); warn_config_value('upload_max_filesize', "MAX_FILESIZE", $MAX_FILESIZE);
warnConfig('post_max_size', "MAX_FILESIZE", $MAX_FILESIZE); warn_config_value('post_max_size', "MAX_FILESIZE", $MAX_FILESIZE);
warnConfig('max_input_time', "UPLOAD_TIMEOUT", $UPLOAD_TIMEOUT); warn_config_value('max_input_time', "UPLOAD_TIMEOUT", $UPLOAD_TIMEOUT);
warnConfig('max_execution_time', "UPLOAD_TIMEOUT", $UPLOAD_TIMEOUT); warn_config_value('max_execution_time', "UPLOAD_TIMEOUT", $UPLOAD_TIMEOUT);
} }
function warnConfig($iniName, $varName, $varValue)
function warn_config_value($ini_name, $var_name, $var_val)
{ {
$iniValue = intval(ini_get($iniName)); $ini_val = intval(ini_get($ini_name));
if ($iniValue < $varValue) if ($ini_val < $var_val)
printf("<pre>Warning: php.ini: %s (%s) set lower than %s (%s)\n</pre>", printf("<pre>Warning: php.ini: %s (%s) set lower than %s (%s)\n</pre>",
$iniName, $ini_name,
$iniValue, $ini_val,
$varName, $var_name,
$varValue); $var_val);
} }
// runs a file through clamdscan
// fails silently(!) if clamd is not running or something
function clam($path)
{
$cmd = 'clamdscan --quiet ' . escapeshellarg($path);
$out = '';
$ret = -1;
exec($cmd, $out, $ret);
return $ret != 1;
}
//////////////////////////////////////////////////////////////////////////////// //extract extension from a path (does not include the dot)
// store an uploaded file, given its name and temporary path, (e.g. values function get_ext($path)
// straight out of $_FILES) {
global $MAX_EXT_LEN;
$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;
}
//trim extension to max. 7 chars
$ext = substr($ext, 0, $MAX_EXT_LEN);
return $ext;
}
// 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 // files are stored wit a randomised name, but with their original extension
// //
// $name: original filename // $name: original filename
// $tmpFile: temporary path of uploaded file // $tmpFile: temporary path of uploaded file
// $formatted: set to true to display formatted message instead of bare link // $formatted: set to true to display formatted message instead of bare link
//////////////////////////////////////////////////////////////////////////////// function store_file($name, $tmpFile, $formatted = false)
function storeFile($name, $tmpFile, $formatted = false)
{ {
global $STORE_PATH; global $STORE_PATH;
global $ID_LENGTH; global $ID_LENGTH;
global $HTTP_PROTO; global $HTTP_PROTO;
global $DOWNLOAD_URL; global $DOWNLOAD_PATH;
global $MAX_FILESIZE; global $MAX_FILESIZE;
global $CLAM_SCAN;
global $LOG_PATH;
//create folder, if it doesn't exist //create folder, if it doesn't exist
if (!file_exists($STORE_PATH)) if (!file_exists($STORE_PATH))
@ -119,13 +108,13 @@ function storeFile($name, $tmpFile, $formatted = false)
return; return;
} }
$ext = getExtension($name); $ext = get_ext($name);
$tries_per_len=3; //try random names a few times before upping the length $tries_per_len=3; //try random names a few times before upping the length
for ($len = $ID_LENGTH; ; ++$len) for ($len = $ID_LENGTH; ; ++$len)
{ {
for ($n=0; $n<=$tries_per_len; ++$n) for ($n=0; $n<=$tries_per_len; ++$n)
{ {
$id = rndStr($len); $id = rnd_str($len);
$basename = $id . (empty($ext) ? '' : '.' . $ext); $basename = $id . (empty($ext) ? '' : '.' . $ext);
$target_file = $STORE_PATH . $basename; $target_file = $STORE_PATH . $basename;
@ -137,48 +126,54 @@ function storeFile($name, $tmpFile, $formatted = false)
$res = move_uploaded_file($tmpFile, $target_file); $res = move_uploaded_file($tmpFile, $target_file);
if ($res) if ($res)
{ {
//scan file using clam
if ($CLAM_SCAN && !clam($target_file))
{
unlink($target_file);
header("HTTP/1.0 400 Bad Request");
print("Error 400: File rejected by virus scan");
return;
}
//print the download link of the file //print the download link of the file
$url = sprintf("%s://%s/".$DOWNLOAD_URL, $url = sprintf('%s://%s/'.$DOWNLOAD_PATH,
$HTTP_PROTO, $HTTP_PROTO,
$_SERVER["SERVER_NAME"], $_SERVER["SERVER_NAME"],
$basename); $basename);
if ($formatted) if ($formatted)
{ {
printf("<pre>Access your file here:\n<a href=\"%s\">%s</a></pre>", printf('<pre>Access your file here:\n<a href="%s">%s</a></pre>', $url, $url);
$url,$url);
} }
else else
{ {
printf($url); printf($url);
} }
// log uploader's IP, original filename, etc.
if ($LOG_PATH)
{
file_put_contents(
$LOG_PATH,
implode("\t", array(
date('c'),
$_SERVER['REMOTE_ADDR'],
filesize($tmpFile),
escapeshellarg($name),
$basename
)) . "\n",
FILE_APPEND
);
}
} }
else else
{ {
//TODO: proper error handling //TODO: proper error handling?
header("HTTP/1.0 520 Unknown Error"); header("HTTP/1.0 520 Unknown Error");
} }
} }
//extract extension from a path (does not include the dot)
function getExtension($path)
{
$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;
}
//trim extension to max. 7 chars
$ext = substr($ext,0,7);
return $ext;
}
////////////////////////////////////////////////////////////////////////////////
// purge all files older than their retention period allows. // purge all files older than their retention period allows.
//////////////////////////////////////////////////////////////////////////////// function purge_files()
function purgeFiles()
{ {
global $STORE_PATH; global $STORE_PATH;
global $MAX_FILEAGE; global $MAX_FILEAGE;
@ -186,61 +181,50 @@ function purgeFiles()
global $MIN_FILEAGE; global $MIN_FILEAGE;
global $DECAY_EXP; global $DECAY_EXP;
$numDel = 0; //number of deleted files $num_del = 0; //number of deleted files
$totalSize = 0; //total size of deleted files $total_size = 0; //total size of deleted files
//for each stored file //for each stored file
foreach (scandir($STORE_PATH) as $file) foreach (scandir($STORE_PATH) as $file)
{ {
//skip virtual . and .. files //skip virtual . and .. files
if ($file == '.' || if ($file === '.' ||
$file == '..' || $file === '..')
$file == '.htaccess' ||
$file == '.htpasswd')
{ {
continue; continue;
} }
$file = $STORE_PATH . $file; $file = $STORE_PATH . $file;
$fileSize = filesize($file) / (1024*1024); //size in MiB $file_size = filesize($file) / (1024*1024); //size in MiB
$fileAge = (time()-filemtime($file)) / (60*60*24); //age in days $file_age = (time()-filemtime($file)) / (60*60*24); //age in days
//keep all files below the min age //keep all files below the min age
if ($fileAge < $MIN_FILEAGE) if ($file_age < $MIN_FILEAGE)
{ {
continue; continue;
} }
//calculate the maximum age, in days, for this file //calculate the maximum age in days for this file
//minage + (maxage-minage) * (1-(size/maxsize))^exp; $file_max_age = $MIN_FILEAGE +
$fileMaxAge = $MIN_FILEAGE +
($MAX_FILEAGE - $MIN_FILEAGE) * ($MAX_FILEAGE - $MIN_FILEAGE) *
pow(1-($fileSize/$MAX_FILESIZE),$DECAY_EXP); pow(1-($file_size/$MAX_FILESIZE),$DECAY_EXP);
//delete if older //delete if older
if ($fileAge > $fileMaxAge) if ($file_age > $file_max_age)
{ {
unlink($file); unlink($file);
printf("deleted \"%s\", %d MiB, %d days old\n", printf("deleted \"%s\", %d MiB, %d days old\n", $file, $file_size, $file_age);
$file, $num_del += 1;
$fileSize, $total_size += $file_size;
$fileAge);
$numDel += 1;
$totalSize += $fileSize;
} }
} }
printf("Purge finished. Deleted %d files totalling %d MiB\n", printf("Deleted %d files totalling %d MiB\n", $num_del, $total_size);
$numDel,
$totalSize);
} }
////////////////////////////////////////////////////////////////////////////////
// send a ShareX custom uploader config as .json // send a ShareX custom uploader config as .json
//////////////////////////////////////////////////////////////////////////////// function send_sharex_config()
function sendShareXConfig()
{ {
global $HTTP_PROTO; global $HTTP_PROTO;
$host = $_SERVER["HTTP_HOST"]; $host = $_SERVER["HTTP_HOST"];
@ -261,10 +245,8 @@ EOT;
print($content); print($content);
} }
////////////////////////////////////////////////////////////////////////////////
// send a Hupl uploader config as .hupl (which is just JSON) // send a Hupl uploader config as .hupl (which is just JSON)
//////////////////////////////////////////////////////////////////////////////// function send_hupl_config()
function sendHuplConfig()
{ {
global $HTTP_PROTO; global $HTTP_PROTO;
$host = $_SERVER["HTTP_HOST"]; $host = $_SERVER["HTTP_HOST"];
@ -283,31 +265,35 @@ EOT;
print($content); print($content);
} }
////////////////////////////////////////////////////////////////////////////////
// print a plaintext info page, explaining what this script does and how to // print a plaintext info page, explaining what this script does and how to
// use it, how to upload, etc. // use it, how to upload, etc.
// essentially the homepage function print_index()
////////////////////////////////////////////////////////////////////////////////
function printInfo()
{ {
global $ADMIN_EMAIL; global $ADMIN_EMAIL;
global $HTTP_PROTO; global $MAX_FILEAGE; global $HTTP_PROTO;
global $MAX_FILEAGE;
global $MAX_FILESIZE; global $MAX_FILESIZE;
global $MIN_FILEAGE; global $MIN_FILEAGE;
global $DECAY_EXP; global $DECAY_EXP;
$url = $HTTP_PROTO."://".$_SERVER["HTTP_HOST"].$_SERVER['REQUEST_URI']; $url = $HTTP_PROTO."://".$_SERVER["HTTP_HOST"].$_SERVER['REQUEST_URI'];
$sharexUrl = $url."?sharex"; $sharex_url = $url."?sharex";
$huplUrl = $url."?hupl"; $hupl_url = $url."?hupl";
echo <<<EOT echo <<<EOT
<html>
<head>
<title>Filehost</title>
<meta name="description" content="Minimalistic service for sharing temporary files." />
</title>
<body>
<pre> <pre>
=== How To Upload === === How To Upload ===
You can upload files to this site via a simple HTTP POST, e.g. using curl: You can upload files to this site via a simple HTTP POST, e.g. using curl:
curl -F "file=@/path/to/your/file.jpg" $url curl -F "file=@/path/to/your/file.jpg" $url
On Windows, you can use <a href="https://getsharex.com/">ShareX</a> and import <a href="$sharexUrl">this</a> custom uploader. On Windows, you can use <a href="https://getsharex.com/">ShareX</a> and import <a href="$sharex_url">this</a> custom uploader.
On Android, you can use an app called <a href="https://github.com/Rouji/Hupl">Hupl</a> with <a href="$huplUrl">this</a> uploader. On Android, you can use an app called <a href="https://github.com/Rouji/Hupl">Hupl</a> with <a href="$hupl_url">this</a> uploader.
Or simply choose a file and click "Upload" below: Or simply choose a file and click "Upload" below:
@ -345,6 +331,37 @@ The PHP script used to provide this service is open source and available on
If you want to report abuse of this service, or have any other inquiries, If you want to report abuse of this service, or have any other inquiries,
please write an email to $ADMIN_EMAIL please write an email to $ADMIN_EMAIL
</pre> </pre>
</body>
</html>
EOT; 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($_GET["formatted"]) || isset($_POST["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 (isset($argv[1]) && $argv[1] === 'purge')
{
purge_files();
}
else
{
check_config();
print_index();
}