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

668 lines
18 KiB

<?php
//Disable error report for undefined superglobals
error_reporting( error_reporting() & ~E_NOTICE );
//Security options
$allow_direct_link = true; // Set to false to only allow downloads and not direct link
$allow_show_folders = true; // Set to false to hide all subdirectories
$disallowed_patterns = ['*.php']; // must be an array. Matching files not allowed to be uploaded
$hidden_patterns = ['*.php','.*']; // Matching files hidden in directory index
$PASSWORD = ''; // Set the password, to access the file manager... (optional)
if($PASSWORD) {
session_start();
if(!$_SESSION['_sfm_allowed']) {
// sha1, and random bytes to thwart timing attacks. Not meant as secure hashing.
$t = bin2hex(openssl_random_pseudo_bytes(10));
if($_POST['p'] && sha1($t.$_POST['p']) === sha1($t.$PASSWORD)) {
$_SESSION['_sfm_allowed'] = true;
header('Location: ?');
}
echo '<html><body><form action=? method=post>PASSWORD:<input type=password name=p autofocus/></form></body></html>';
exit;
}
}
// must be in UTF-8 or `basename` doesn't work
setlocale(LC_ALL,'en_US.UTF-8');
$tmp_dir = dirname($_SERVER['SCRIPT_FILENAME']);
if(DIRECTORY_SEPARATOR==='\\') $tmp_dir = str_replace('/',DIRECTORY_SEPARATOR,$tmp_dir);
$tmp = get_absolute_path($tmp_dir . '/' .$_REQUEST['file']);
if($tmp === false)
err(404,'File or Directory Not Found');
if(substr($tmp, 0,strlen($tmp_dir)) !== $tmp_dir)
err(403,"Forbidden");
if(strpos($_REQUEST['file'], DIRECTORY_SEPARATOR) === 0)
err(403,"Forbidden");
if(preg_match('@^.+://@',$_REQUEST['file'])) {
err(403,"Forbidden");
}
if(!$_COOKIE['_sfm_xsrf'])
setcookie('_sfm_xsrf',bin2hex(openssl_random_pseudo_bytes(16)));
if($_POST) {
if($_COOKIE['_sfm_xsrf'] !== $_POST['xsrf'] || !$_POST['xsrf'])
err(403,"XSRF Failure");
}
$file = $_REQUEST['file'] ?: '.';
if($_GET['do'] == 'list') {
if (is_dir($file)) {
$directory = $file;
$result = [];
$files = array_diff(scandir($directory), ['.','..']);
foreach ($files as $entry) if (!is_entry_ignored($entry, $allow_show_folders, $hidden_patterns)) {
$i = $directory . '/' . $entry;
$stat = stat($i);
$result[] = [
'mtime' => $stat['mtime'],
'size' => $stat['size'],
'name' => basename($i),
'path' => preg_replace('@^\./@', '', $i),
'is_dir' => is_dir($i),
'is_readable' => is_readable($i),
'is_writable' => is_writable($i),
'is_executable' => is_executable($i),
];
}
usort($result,function($f1,$f2){
$f1_key = ($f1['is_dir']?:2) . $f1['name'];
$f2_key = ($f2['is_dir']?:2) . $f2['name'];
return $f1_key > $f2_key;
});
} else {
err(412,"Not a Directory");
}
echo json_encode(['success' => true, 'is_writable' => is_writable($file), 'results' =>$result]);
exit;
} elseif ($_GET['do'] == 'download') {
foreach($disallowed_patterns as $pattern)
if(fnmatch($pattern, $file))
err(403,"Files of this type are not allowed.");
$filename = basename($file);
$finfo = finfo_open(FILEINFO_MIME_TYPE);
header('Content-Type: ' . finfo_file($finfo, $file));
header('Content-Length: '. filesize($file));
header(sprintf('Content-Disposition: attachment; filename=%s',
strpos('MSIE',$_SERVER['HTTP_REFERER']) ? rawurlencode($filename) : "\"$filename\"" ));
ob_flush();
readfile($file);
exit;
}
function is_entry_ignored($entry, $allow_show_folders, $hidden_patterns) {
if ($entry === basename(__FILE__)) {
return true;
}
if (is_dir($entry) && !$allow_show_folders) {
return true;
}
foreach($hidden_patterns as $pattern) {
if(fnmatch($pattern,$entry)) {
return true;
}
}
return false;
}
function get_absolute_path($path) {
$path = str_replace(['/', '\\'], DIRECTORY_SEPARATOR, $path);
$parts = explode(DIRECTORY_SEPARATOR, $path);
$absolutes = [];
foreach ($parts as $part) {
if ('.' == $part) continue;
if ('..' == $part) {
array_pop($absolutes);
} else {
$absolutes[] = $part;
}
}
return implode(DIRECTORY_SEPARATOR, $absolutes);
}
function err($code,$msg) {
http_response_code($code);
header("Content-Type: application/json");
echo json_encode(['error' => ['code'=>intval($code), 'msg' => $msg]]);
exit;
}
function asBytes($ini_v) {
$ini_v = trim($ini_v);
$s = ['g'=> 1<<30, 'm' => 1<<20, 'k' => 1<<10];
return intval($ini_v) * ($s[strtolower(substr($ini_v,-1))] ?: 1);
}
$MAX_UPLOAD_SIZE = min(asBytes(ini_get('post_max_size')), asBytes(ini_get('upload_max_filesize')));
?>
<!DOCTYPE html>
<html><head>
<meta http-equiv="content-type" content="text/html; charset=utf-8">
<style>
.aquila {
position: absolute;
top: 8px;
right: 16px;
padding-top: 4px;
width: 100px;
}
body {
font-family: "lucida grande",
"Segoe UI",
Arial,
sans-serif;
font-weight: bold;
font-size:14px;
color: GoldenRod;
width:1024;
padding:1em;
margin:0;
background-color: #222222;
}
th {
font-weight: bold;
color: GoldenRod;
background-color: #025202;
padding: .5em 1em .5em .2em;
text-align: left;
cursor:pointer;
user-select: none;
}
th .indicator {
margin-left: 6px;
color: GoldenRod;
}
thead {
border-top: 5px solid #018a30;
border-bottom: 5px solid #018a30;
border-left: 5px solid #018a30;
border-right: 5px solid #018a30;
}
#top {
height:52px;
color: GoldenRod;
}
#mkdir {
display:inline-block;
float:right;
padding-top:16px;
}
label {
display:block;
font-size:11px;
color:#555;
}
#file_drop_target {
width:500px;
padding:12px 0;
border: 4px dashed #008700;
font-size:12px;
color:#ccc;
text-align: center;
float:right;
margin-right:20px;
}
#file_drop_target.drag_over {
border: 4px dashed #96C4EA;
color: #96C4EA;
}
#upload_progress {
padding: 4px 0;
}
#upload_progress .error {
color:#a00;
}
#upload_progress > div {
padding:3px 0;
}
.no_write #mkdir, .no_write #file_drop_target {
display: none
}
.progress_track {
display:inline-block;
width:200px;
height:10px;
border:1px solid #333;
margin: 0 4px 0 10px;
}
.progress {
background-color: #82CFFA;
height:10px;
}
footer {
font-size:11px;
color:#bbbbc5;
padding:4em 0 0;
text-align: left;
}
footer a, footer a:visited {
color:#bbbbc5;
}
#breadcrumb {
font-weight: bold;
padding-top:34px;
font-size:15px;
color:red;
display:inline-block;
float:left;
}
#folder_actions {
width: 50%;
float:right;
}
a {
text-decoration;
}
a, a:visited {
text-decoration: none;
color: GoldenRod;
}
a:hover {
text-decoration: none;
color: GoldenRod;
}
.sort_hide{
display:none;
}
table {
border-collapse: collapse;
width:100%;
}
thead {
max-width: 1024px
}
td {
padding:.2em 1em .2em .2em;
border-bottom:1px solid grey;
height:30px;
font-size:12px;
white-space: nowrap;
}
td.first {
font-size:14px;
white-space: normal;
}
td.empty {
color:#777;
font-style: italic;
text-align: center;
padding:3em 0;
}
.is_dir .size {
color:transparent;
font-size:0;
}
.is_dir .size:before {
content: "--";
font-size:14px;
color:#333;
}
.is_dir .download {
visibility: hidden;
color: GoldenRod;
}
a.delete {
display:inline-block;
background: url() no-repeat scroll 0 2px;
color:#d00;
margin-left: 15px;
font-size:11px;
padding:0 0 0 13px;
color: GoldenRod;
}
.name {
background: url() no-repeat scroll 0px 12px;
padding:15px 0 10px 40px;
color: GoldenRod;
font-weight: bold;
}
.is_dir .name {
background: url(../Resources/Files/folder.png) no-repeat;
background-position: 5px 10px;
background-size: 27px 27px;
color: GoldenRod;
font-weight: bold;
}
.pdf {
background: url(../Resources/Files/pdf.ico) no-repeat;
background-position: 5px 8px;
background-size: 28px 31px;
color: GoldenRod;
font-weight: bold;
}
.undefined {
background: url(../Resources/Files/undefined.ico) no-repeat;
background-position: 5px 10px;
background-size: 27px 27px;
color: GoldenRod;
font-weight: bold;
}
.audio {
background: url(../Resources/Files/audio.ico) no-repeat;
background-position: 5px 10px;
background-size: 27px 27px;
color: GoldenRod;
font-weight: bold;
}
.binary {
background: url(../Resources/Files/binary.ico) no-repeat;
background-position: 5px 10px;
background-size: 27px 27px;
color: GoldenRod;
font-weight: bold;
}
.tex {
background: url(../Resources/Files/tex.ico) no-repeat;
background-position: 5px 10px;
background-size: 27px 27px;
color: GoldenRod;
font-weight: bold;
}
.texte {
background: url(../Resources/Files/texte.ico) no-repeat;
background-position: 6px 11px;
background-size: 24px 27px;
color: GoldenRod;
font-weight: bold;
}
.image {
background: url(../Resources/Files/image.ico) no-repeat;
background-position: 6px 10px;
background-size: 25px 27px;
color: GoldenRod;
font-weight: bold;
}
.video {
background: url(../Resources/Files/video.ico) no-repeat;
background-position: 8px 13px;
background-size: 22px 25px;
color: GoldenRod;
font-weight: bold;
}
.code {
background: url(../Resources/Files/code.ico) no-repeat;
background-position: 5px 10px;
background-size: 27px 27px;
color: GoldenRod;
font-weight: bold;
}
.download {
background: url() no-repeat scroll 0px 5px;
padding:4px 0 4px 20px;
color: GoldenRod;
font-weight: bold;
}
</style>
<script src="../Resources/Files/jquery-3.5.1.min.js"></script>
<script>
(function($){
$.fn.tablesorter = function() {
var $table = this;
this.find('th').click(function() {
var idx = $(this).index();
var direction = $(this).hasClass('sort_asc');
$table.tablesortby(idx,direction);
});
return this;
};
$.fn.tablesortby = function(idx,direction) {
var $rows = this.find('tbody tr');
function elementToVal(a) {
var $a_elem = $(a).find('td:nth-child('+(idx+1)+')');
var a_val = $a_elem.attr('data-sort') || $a_elem.text();
return (a_val == parseInt(a_val) ? parseInt(a_val) : a_val);
}
$rows.sort(function(a,b){
var a_val = elementToVal(a), b_val = elementToVal(b);
return (a_val > b_val ? 1 : (a_val == b_val ? 0 : -1)) * (direction ? 1 : -1);
})
this.find('th').removeClass('sort_asc sort_desc');
$(this).find('thead th:nth-child('+(idx+1)+')').addClass(direction ? 'sort_desc' : 'sort_asc');
for(var i =0;i<$rows.length;i++)
this.append($rows[i]);
this.settablesortmarkers();
return this;
}
$.fn.retablesort = function() {
var $e = this.find('thead th.sort_asc, thead th.sort_desc');
if($e.length)
this.tablesortby($e.index(), $e.hasClass('sort_desc') );
return this;
}
$.fn.settablesortmarkers = function() {
this.find('thead th span.indicator').remove();
this.find('thead th.sort_asc').append('<span class="indicator">&darr;<span>');
this.find('thead th.sort_desc').append('<span class="indicator">&uarr;<span>');
return this;
}
})(jQuery);
$(function(){
var XSRF = (document.cookie.match('(^|; )_sfm_xsrf=([^;]*)')||0)[2];
var MAX_UPLOAD_SIZE = <?php echo $MAX_UPLOAD_SIZE ?>;
var $tbody = $('#list');
$(window).on('hashchange',list).trigger('hashchange');
$('#table').tablesorter();
$('#table').on('click','.delete',function(data) {
$.post("",{'do':'delete',file:$(this).attr('data-file'),xsrf:XSRF},function(response){
list();
},'json');
return false;
});
$('#mkdir').submit(function(e) {
var hashval = decodeURIComponent(window.location.hash.substr(1)),
$dir = $(this).find('[name=name]');
e.preventDefault();
$dir.val().length && $.post('?',{'do':'mkdir',name:$dir.val(),xsrf:XSRF,file:hashval},function(data){
list();
},'json');
$dir.val('');
return false;
});
function list() {
var hashval = window.location.hash.substr(1);
$.get('?do=list&file='+ hashval,function(data) {
$tbody.empty();
$('#breadcrumb').empty().html(renderBreadcrumbs(hashval));
if(data.success) {
$.each(data.results,function(k,v){
$tbody.append(renderFileRow(v));
});
!data.results.length && $tbody.append('<tr><td class="empty" colspan=5>This folder is empty</td></tr>')
data.is_writable ? $('body').removeClass('no_write') : $('body').addClass('no_write');
} else {
console.warn(data.error.msg);
}
$('#table').retablesort();
},'json');
}
function renderFileRow(data) {
var $link = $('<a class="name" />')
.attr('href', data.is_dir ? '#' + encodeURIComponent(data.path) : './' + data.path)
.text(data.name);
var allow_direct_link = <?php echo $allow_direct_link?'true':'false'; ?>;
if (!data.is_dir) $link.attr('data-type', data.path.split('.').pop());
var texte = ["txt", "rtf", "xls"];
var audio = ["wav", "mp3", "MP3", "WAV", "wave", "flac", "m3u", "ogg", "midi"];
var image = ["bmp", "jpeg", "jpg", "gif", "tif", "webp", "png"];
var video = ["mp4", "wmv", "MP4", "mov", "flv", "webm", "mpeg", "mpg", "flv"];
var binary = ["exe", "bat", "sh"];
var pdf = ["pdf"];
var code = ["c", "c++", "cpp", "h", "hpp", "py", "html", "js", "rkt"];
if (!data.is_dir) {
if (texte.indexOf($link.attr('data-type')) != -1) {
$link.addClass("texte");
} else if (image.indexOf($link.attr('data-type')) != -1) {
$link.addClass("image");
} else if (audio.indexOf($link.attr('data-type')) != -1) {
$link.addClass("audio");
} else if (video.indexOf($link.attr('data-type')) != -1) {
$link.addClass("video");
} else if (pdf.indexOf($link.attr('data-type')) != -1) {
$link.addClass("pdf");
} else if (code.indexOf($link.attr('data-type')) != -1) {
$link.addClass("code");
} else if (binary.indexOf($link.attr('data-type')) != -1) {
$link.addClass("binary");
} else {
$link.addClass("undefined");
}
}
if (!data.is_dir && !allow_direct_link) $link.css('pointer-events','none');
var $dl_link = $('<a/>').attr('href','?do=download&file='+ encodeURIComponent(data.path))
.addClass('download').text('Download');
var $delete_link = $('<a href="#" />').attr('data-file',data.path).addClass('delete').text('delete');
var perms = [];
if(data.is_readable) perms.push('read');
if(data.is_writable) perms.push('write');
if(data.is_executable) perms.push('exec');
var $html = $('<tr />')
.addClass(data.is_dir ? 'is_dir' : '')
.append( $('<td class="first" />').append($link) )
.append( $('<td/>').attr('data-sort',data.is_dir ? -1 : data.size)
.html($('<span class="size" />').text(formatFileSize(data.size))) )
.append( $('<td/>').attr('data-sort',data.mtime).text(formatTimestamp(data.mtime)) )
.append( $('<td/>').text(perms.join('+')) )
.append( $('<td/>').append($dl_link).append( data.is_deleteable ? $delete_link : '') )
return $html;
}
function renderBreadcrumbs(path) {
var base = "",
$html = $('<div/>').append( $('<a href=#>Home</a></div>') );
$.each(path.split('%2F'),function(k,v){
if(v) {
var v_as_text = decodeURIComponent(v);
$html.append( $('<span/>').text(' ▸ ') )
.append( $('<a/>').attr('href','#'+base+v).text(v_as_text) );
base += v + '%2F';
}
});
return $html;
}
function formatTimestamp(unix_timestamp) {
var m = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];
var d = new Date(unix_timestamp*1000);
return [m[d.getMonth()],' ',d.getDate(),', ',d.getFullYear()," ",
(d.getHours() % 12 || 12),":",(d.getMinutes() < 10 ? '0' : '')+d.getMinutes(),
" ",d.getHours() >= 12 ? 'PM' : 'AM'].join('');
}
function formatFileSize(bytes) {
var s = ['bytes', 'KB','MB','GB','TB','PB','EB'];
for(var pos = 0;bytes >= 1000; pos++,bytes /= 1024);
var d = Math.round(bytes*10);
return pos ? [parseInt(d/10),".",d%10," ",s[pos]].join('') : bytes + ' bytes';
}
})
</script>
<head>
</head>
<div>
<img class="aquila" src="../Resources/Files/aquila_large.png">
</div>
<body>
<div id="top">
<div id="breadcrumb">&nbsp;</div>
</div>
<div id="upload_progress"></div>
<table id="table"><thead><tr>
<th> Name</th>
<th>Size</th>
<th>Modified</th>
<th>Permissions</th>
<th>Actions</th>
</tr></thead><tbody id="list">
</tbody></table>
<footer></a></footer>
</body></html>