<?php /******************************** Simple PHP File Manager Copyright John Campbell (jcampbell1) Liscense: MIT ********************************/ // Thanks for providing such a great PHP application! //Disable error report for undefined superglobals error_reporting( error_reporting() & ~E_NOTICE ); //Security options $allow_delete = true; // Set to false to disable delete button and delete POST request. $allow_upload = true; // Set to true to allow upload files $allow_create_folder = true; // Set to false to disable folder creation $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 = 'admin'; // Set the password, to access the file manager... (optional) if($PASSWORD) { session_start(); // Disabled to prevent using cookies to determine if a computer is trusted. 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_deleteable' => $allow_delete && ((!is_dir($i) && is_writable($directory)) || (is_dir($i) && is_writable($directory) && is_recursively_deleteable($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 ($_POST['do'] == 'delete') { if($allow_delete) { rmrf($file); } exit; } elseif ($_POST['do'] == 'mkdir' && $allow_create_folder) { // don't allow actions outside root. we also filter out slashes to catch args like './../outside' $dir = $_POST['name']; $dir = str_replace('/', '', $dir); if(substr($dir, 0, 2) === '..') exit; chdir($file); @mkdir($_POST['name']); exit; } elseif ($_POST['do'] == 'upload' && $allow_upload) { foreach($disallowed_patterns as $pattern) if(fnmatch($pattern, $_FILES['file_data']['name'])) err(403,"Files of this type are not allowed."); $res = move_uploaded_file($_FILES['file_data']['tmp_name'], $file.'/'.$_FILES['file_data']['name']); 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 rmrf($dir) { if(is_dir($dir)) { $files = array_diff(scandir($dir), ['.','..']); foreach ($files as $file) rmrf("$dir/$file"); rmdir($dir); } else { unlink($dir); } } function is_recursively_deleteable($d) { $stack = [$d]; while($dir = array_pop($stack)) { if(!is_readable($dir) || !is_writable($dir)) return false; $files = array_diff(scandir($dir), ['.','..']); foreach($files as $file) if(is_dir($file)) { $stack[] = "$dir/$file"; } } return true; } // from: http://php.net/manual/en/function.realpath.php#84012 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> body {font-family: "lucida grande","Segoe UI",Arial, sans-serif; font-size: 14px;width:1024;padding:1em;margin:0;} th {font-weight: normal; color: #1F75CC; background-color: #F0F9FF; padding:.5em 1em .5em .2em; text-align: left;cursor:pointer;user-select: none;} th .indicator {margin-left: 6px } thead {border-top: 1px solid #82CFFA; border-bottom: 1px solid #96C4EA;border-left: 1px solid #E7F2FB; border-right: 1px solid #E7F2FB; } #top {height:52px;} #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 #ccc;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 { padding-top:34px; font-size:15px; color:#aaa;display:inline-block;float:left;} #folder_actions {width: 50%;float:right;} a, a:visited { color:#00c; text-decoration: none} a:hover {text-decoration: underline} .sort_hide{ display:none;} table {border-collapse: collapse;width:100%;} thead {max-width: 1024px} td { padding:.2em 1em .2em .2em; border-bottom:1px solid #def;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} 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; } .name { background: url() no-repeat scroll 0px 12px; padding:15px 0 10px 40px; } .is_dir .name { background: url() no-repeat scroll 0px 10px; padding:15px 0 10px 40px; } .download { background: url() no-repeat scroll 0px 5px; padding:4px 0 4px 20px; } </style> <script src="//ajax.googleapis.com/ajax/libs/jquery/1.8.2/jquery.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">↓<span>'); this.find('thead th.sort_desc').append('<span class="indicator">↑<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; }); <?php if($allow_upload): ?> // file upload stuff $('#file_drop_target').on('dragover',function(){ $(this).addClass('drag_over'); return false; }).on('dragend',function(){ $(this).removeClass('drag_over'); return false; }).on('drop',function(e){ e.preventDefault(); var files = e.originalEvent.dataTransfer.files; $.each(files,function(k,file) { uploadFile(file); }); $(this).removeClass('drag_over'); }); $('input[type=file]').change(function(e) { e.preventDefault(); $.each(this.files,function(k,file) { uploadFile(file); }); }); function uploadFile(file) { var folder = decodeURIComponent(window.location.hash.substr(1)); if(file.size > MAX_UPLOAD_SIZE) { var $error_row = renderFileSizeErrorRow(file,folder); $('#upload_progress').append($error_row); window.setTimeout(function(){$error_row.fadeOut();},5000); return false; } var $row = renderFileUploadRow(file,folder); $('#upload_progress').append($row); var fd = new FormData(); fd.append('file_data',file); fd.append('file',folder); fd.append('xsrf',XSRF); fd.append('do','upload'); var xhr = new XMLHttpRequest(); xhr.open('POST', '?'); xhr.onload = function() { $row.remove(); list(); }; xhr.upload.onprogress = function(e){ if(e.lengthComputable) { $row.find('.progress').css('width',(e.loaded/e.total*100 | 0)+'%' ); } }; xhr.send(fd); } function renderFileUploadRow(file,folder) { return $row = $('<div/>') .append( $('<span class="fileuploadname" />').text( (folder ? folder+'/':'')+file.name)) .append( $('<div class="progress_track"><div class="progress"></div></div>') ) .append( $('<span class="size" />').text(formatFileSize(file.size)) ) }; function renderFileSizeErrorRow(file,folder) { return $row = $('<div class="error" />') .append( $('<span class="fileuploadname" />').text( 'Error: ' + (folder ? folder+'/':'')+file.name)) .append( $('<span/>').html(' file size - <b>' + formatFileSize(file.size) + '</b>' +' exceeds max upload size of <b>' + formatFileSize(MAX_UPLOAD_SIZE) + '</b>') ); } <?php endif; ?> 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 && !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><body> <script> document.addEventListener('contextmenu', event => event.preventDefault()); </script> <div id="top"> <?php if($allow_create_folder): ?> <form action="?" method="post" id="mkdir" /> <label for=dirname>Create New Folder</label><input id=dirname type=text name=name value="" /> <input type="submit" value="create" /> </form> <?php endif; ?> <?php if($allow_upload): ?> <div id="file_drop_target"> Drag Files Here To Upload <b>or</b> <input type="file" multiple /> </div> <?php endif; ?> <div id="breadcrumb"> </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 onclick="lockRoot()">Index of <?php echo preg_replace('/^' . preg_quote($_SERVER['DOCUMENT_ROOT'], '/') . '/', '', __DIR__); ?> | Press to lock root on this browser</a></footer> <script> function lockRoot() { document.cookie = "PHPSESSID=; expires = Thu, 01 Jan 1970 00:00:00 GMT" document.cookie = "_sfm_xsrf=; expires = Thu, 01 Jan 1970 00:00:00 GMT" window.location.href = '/home/user/index.php'; } // If you are like me and use the same browsing session for ages at a time, this can easily prevent usage of the root file browser. </script> </body></html>