mirror of
https://github.com/misterunknown/ifm.git
synced 2025-08-11 10:34:00 +02:00
misterunknown: complete refactor of code including splitting it up to multiple files and build a compiler
This commit is contained in:
35
compiler.php
Normal file
35
compiler.php
Normal file
@@ -0,0 +1,35 @@
|
||||
<?php
|
||||
|
||||
// This compiles the source files into one file
|
||||
|
||||
$IFM_CONFIG = "src/config.php";
|
||||
$IFM_INCLUDES = "src/includes.php";
|
||||
$IFM_MAIN = "src/main.php";
|
||||
$IFM_STYLE = "src/style.css";
|
||||
$IFM_JS = "src/ifm.js";
|
||||
$IFM_OTHER_PHPFILES = array("src/zip_file.php");
|
||||
|
||||
$filename = "ifm.php";
|
||||
|
||||
// config
|
||||
file_put_contents($filename, file_get_contents($IFM_CONFIG));
|
||||
|
||||
// includes
|
||||
$content_includes = file($IFM_INCLUDES);
|
||||
unset($content_includes[0]);
|
||||
file_put_contents($filename, $content_includes, FILE_APPEND);
|
||||
|
||||
// other php classes
|
||||
foreach ( $IFM_OTHER_PHPFILES as $file) {
|
||||
$content_file = file($file);
|
||||
unset($content_file[0]);
|
||||
file_put_contents($filename, $content_file, FILE_APPEND);
|
||||
}
|
||||
|
||||
// main
|
||||
$content_main = file($IFM_MAIN);
|
||||
unset($content_main[0]);
|
||||
$content_main = implode($content_main);
|
||||
$content_main = str_replace("@@@COMPILE:style.css@@@", file_get_contents($IFM_STYLE), $content_main);
|
||||
$content_main = str_replace("@@@COMPILE:ifm.js@@@", file_get_contents($IFM_JS), $content_main);
|
||||
file_put_contents($filename, $content_main, FILE_APPEND);
|
54
src/config.php
Normal file
54
src/config.php
Normal file
@@ -0,0 +1,54 @@
|
||||
<?php
|
||||
|
||||
/* =======================================================================
|
||||
* Improved File Manager
|
||||
* ---------------------
|
||||
* License: This project is provided under the terms of the MIT LICENSE
|
||||
* http://github.com/misterunknown/ifm/blob/master/LICENSE
|
||||
* =======================================================================
|
||||
*
|
||||
* config
|
||||
*/
|
||||
|
||||
class IFMConfig {
|
||||
|
||||
// 0 = no/not allowed;; 1 = yes/allowed;; default: no/forbidden;
|
||||
|
||||
// action controls
|
||||
const upload = 1; // allow uploads
|
||||
const remoteupload = 1; // allow remote uploads using cURL
|
||||
const delete = 1; // allow deletions
|
||||
const rename = 1; // allow renamings
|
||||
const edit = 1; // allow editing
|
||||
const chmod = 1; // allow to change rights
|
||||
const extract = 1; // allow extracting zip archives
|
||||
const download = 1; // allow to download files and skripts (even php scripts!)
|
||||
const selfdownload = 1; // allow to download this skript itself
|
||||
const createdir = 1; // allow to create directorys
|
||||
const createfile = 1; // allow to create files
|
||||
const zipnload = 1; // allow to zip and download directorys
|
||||
|
||||
// view controls
|
||||
const multiselect = 1; // implement multiselect of files and directories
|
||||
const showlastmodified = 0; // show the last modified date?
|
||||
const showfilesize = 1; // show filesize?
|
||||
const showowner = 1; // show file owner?
|
||||
const showgroup = 1; // show file group?
|
||||
const showpath = 0; // show real path of directory (not only root)?
|
||||
const showrights = 2; // show permissions 0 -> not; 1 -> octal, 2 -> human readable
|
||||
const showhtdocs = 1; // show .htaccess and .htpasswd
|
||||
const showhiddenfiles = 1; // show files beginning with a dot (e.g. ".bashrc")
|
||||
|
||||
// general config
|
||||
const auth = 0;
|
||||
const auth_source = 'inline;admin:$2y$10$0Bnm5L4wKFHRxJgNq.oZv.v7yXhkJZQvinJYR2p6X1zPvzyDRUVRC';
|
||||
const defaulttimezone = "Europe/Berlin"; // set default timezone
|
||||
|
||||
// development tools
|
||||
const ajaxrequest = 1; // formular to perform an ajax request
|
||||
|
||||
static function getConstants() {
|
||||
$oClass = new ReflectionClass(__CLASS__);
|
||||
return $oClass->getConstants();
|
||||
}
|
||||
}
|
764
src/ifm.js
Normal file
764
src/ifm.js
Normal file
@@ -0,0 +1,764 @@
|
||||
// declare ifm object
|
||||
if(!ifm) {
|
||||
var ifm = {
|
||||
IFM_SCFN: "<?=basename($_SERVER['SCRIPT_NAME'])?>", // IFM_SCFM = ifm script file name; used for querying via AJAX
|
||||
config: jQuery.parseJSON('<?php echo json_encode(IFMConfig::getConstants()); ?>'), // serialize the PHP config array, so we can use it in JS too
|
||||
editor: null, // global ace editor
|
||||
fileChanged: false, // was our file edited?
|
||||
icons: { // base64 encoded icons we use dynamically
|
||||
"edit.png": "",
|
||||
"delete.png": "",
|
||||
"rename.png": "",
|
||||
"folder.png": "",
|
||||
"extract.png": "",
|
||||
"download.png": "",
|
||||
"zipnload.png": "",
|
||||
"rar.png": "",
|
||||
"sql.png": "",
|
||||
"aac.png": "",
|
||||
"doc.png": "",
|
||||
"xlsx.png": "",
|
||||
"qt.png": "",
|
||||
"bmp.png": "",
|
||||
"c.png": "",
|
||||
"tiff.png": "",
|
||||
"xml.png": "",
|
||||
"dxf.png": "",
|
||||
"avi.png": "",
|
||||
"zip.png": "",
|
||||
"jpg.png": "",
|
||||
"eps.png": "",
|
||||
"mp4.png": "",
|
||||
"ppt.png": "",
|
||||
"java.png": "",
|
||||
"dat.png": "",
|
||||
"dwg.png": "",
|
||||
"tga.png": "",
|
||||
"ai.png": "",
|
||||
"dmg.png": "",
|
||||
"php.png": "",
|
||||
"html.png": "",
|
||||
"aiff.png": "",
|
||||
"odf.png": "",
|
||||
"h.png": "",
|
||||
"cpp.png": "",
|
||||
"exe.png": "",
|
||||
"_page.png": "",
|
||||
"png.png": "",
|
||||
"iso.png": "",
|
||||
"mid.png": "",
|
||||
"gif.png": "",
|
||||
"mp3.png": "",
|
||||
"mpg.png": "",
|
||||
"ods.png": "",
|
||||
"_blank.png": "",
|
||||
"py.png": "",
|
||||
"otp.png": "",
|
||||
"flv.png": "",
|
||||
"dotx.png": "",
|
||||
"xls.png": "",
|
||||
"odt.png": "",
|
||||
"rb.png": "",
|
||||
"txt.png": "",
|
||||
"hpp.png": "",
|
||||
"pdf.png": "",
|
||||
"wav.png": "",
|
||||
"tgz.png": "",
|
||||
"psd.png": "",
|
||||
"key.png": "",
|
||||
"ott.png": "",
|
||||
"css.png": "",
|
||||
"ics.png": "",
|
||||
"rtf.png": "",
|
||||
"ots.png": "",
|
||||
"yml.png": ""
|
||||
},
|
||||
currentDir: "", // this is the global variable for the current directory; it is used for AJAX requests
|
||||
// --------------
|
||||
// main functions
|
||||
// --------------
|
||||
refreshFileTable: function () {
|
||||
if(document.getElementById("multiseloptions"))$("#multiseloptions").remove();
|
||||
var id=ifm.generateGuid();
|
||||
ifm.task_add("Refresh", id);
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: "api=getFiles&dir="+ifm.currentDir,
|
||||
dataType: "json",
|
||||
success:
|
||||
function(data) {
|
||||
$("#filetable tbody tr").remove();
|
||||
for(i=0;i<data.length;i++) {
|
||||
var newrow = "<tr>";
|
||||
var multisel = '';
|
||||
if(ifm.config.multiselect == 1) {
|
||||
multisel = '<input type="checkbox" ';
|
||||
multisel += (!ifm.inArray(data[i].name, ["..", "."]))?'id="'+data[i].name+'" name="multisel"':'style="visibility:hidden"';
|
||||
multisel += '>';
|
||||
}
|
||||
if(data[i].type=="file")
|
||||
newrow += '<td>'+multisel+'<a href="'+ifm.pathCombine(ifm.currentDir,data[i].name)+'"><img src="'+ifm.icons[data[i].picture]+'" alt=""> '+data[i].name+'</td>';
|
||||
else
|
||||
newrow += '<td>'+multisel+'<a onclick="ifm.changeDirectory(\''+data[i].name+'\')"><img src="'+ifm.icons[data[i].picture]+'" alt=""> '+data[i].name+'</td>'
|
||||
if(ifm.config.download == 1) {
|
||||
if(data[i].type != "dir")
|
||||
newrow += '<td class="download-link">\
|
||||
<form style="display:none;" id="fdownload'+i+'" method="post">\
|
||||
<fieldset>\
|
||||
<input type="hidden" name="dir" value="'+ifm.currentDir+'">\
|
||||
<input type="hidden" name="filename" value="'+data[i].name+'">\
|
||||
<input type="hidden" name="api" value="downloadFile">\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
<a onclick="$(\'#fdownload'+i+'\').submit();"><img src="'+ifm.icons["download.png"]+'" style="width:19px;" alt="download"></a>\
|
||||
</td>';
|
||||
else
|
||||
newrow += '<td></td>'; // empty cell for download link
|
||||
}
|
||||
if(data[i].lastmodified) newrow += '<td>'+data[i].lastmodified+'</td>';
|
||||
if(data[i].filesize) newrow += '<td>'+data[i].filesize+'</td>';
|
||||
if(data[i].fileperms) {
|
||||
newrow += '<td><input type="text" name="newperms" value="'+data[i].fileperms+'"';
|
||||
if(ifm.config.chmod == 1)
|
||||
newrow += ' onkeypress="ifm.changePermissions(event, \''+data[i].name+'\');"';
|
||||
else
|
||||
newrow += " readonly";
|
||||
newrow += '></td>';
|
||||
}
|
||||
if(data[i].owner) newrow += '<td>'+data[i].owner+'</td>';
|
||||
if(data[i].group) newrow += '<td>'+data[i].group+'</td>';
|
||||
if(ifm.inArray(1,[ifm.config.edit, ifm.config.rename, ifm.config.delete, ifm.config.zipnload, ifm.config.extract])) {
|
||||
newrow += '<td>';
|
||||
if(data[i].name != ".." && data[i].name != ".") {
|
||||
if(data[i].type == "dir") {
|
||||
if(ifm.config.zipnload == 1)
|
||||
newrow += '<form method="post" style="display:inline-block;padding:0;margin:0;border:0;">\
|
||||
<fieldset style="display:inline-block;padding:0;margin:0;border:0;">\
|
||||
<input type="hidden" name="dir" value="'+ifm.currentDir+'">\
|
||||
<input type="hidden" name="filename" value="'+data[i].name+'">\
|
||||
<input type="hidden" name="api" value="zipnload">\
|
||||
<button type="submit"><img src="'+ifm.icons["zipnload.png"]+'" alt="zip&load"></button>\
|
||||
</fieldset>\
|
||||
</form>';
|
||||
}
|
||||
else if(data[i].name.toLowerCase().substr(-4) == ".zip") {
|
||||
if(ifm.config.extract == 1) newrow += '<button onclick="ifm.extractFileDialog(\''+ifm.JSEncode(data[i].name)+'\');return false;"><img src="'+ifm.icons["extract.png"]+'" alt="extract" /></button>';
|
||||
}
|
||||
else {
|
||||
if(ifm.config.edit == 1) newrow += '<button onclick="ifm.showLoading();ifm.editFile(\''+ifm.JSEncode(data[i].name)+'\');return false;"><img src="'+ifm.icons["edit.png"]+'" alt="edit" /></button>';
|
||||
}
|
||||
if(ifm.config.rename == 1) newrow += '<button onclick="ifm.renameFileDialog(\''+ifm.JSEncode(data[i].name)+'\');return false;"><img src="'+ifm.icons["rename.png"]+'" alt="rename" /></button>';
|
||||
if(ifm.config.delete == 1) newrow += '<button onclick="ifm.deleteFileDialog(\''+ifm.JSEncode(data[i].name)+'\');return false;"><img src="'+ifm.icons["delete.png"]+'" alt="delete" /></button>';
|
||||
}
|
||||
newrow += '</td></tr>';
|
||||
}
|
||||
$("#filetable tbody").append(newrow);
|
||||
}
|
||||
// bind multiselect handler
|
||||
if(ifm.config.multiselect == 1) {
|
||||
$("input[name=multisel]").on("change", function(){ ifm.handleMultiSelect(); });
|
||||
}
|
||||
// bind Fancybox
|
||||
var piclinks = $('a[href$=".bmp"],a[href$=".gif"],a[href$=".jpg"],a[href$=".jpeg"],a[href$=".png"],a[href$=".BMP"],a[href$=".GIF"],a[href$=".JPG"],a[href$=".JPEG"],a[href$=".PNG"]');
|
||||
piclinks.attr('rel', 'fancybox').fancybox();
|
||||
var xOffset = 10;
|
||||
var yOffset = 30;
|
||||
piclinks.hover(function(e){this.t = this.title;this.title = "";var c = (this.t != "") ? "<br/>" + this.t : "";$("body").append("<p id='mdPicPreview'><img src='"+ this.href +"' alt='Image preview' />"+ c +"</p>");$("#mdPicPreview").css("top",(e.pageY - xOffset) + "px").css("left",(e.pageX + yOffset) + "px").fadeIn("fast");},function(){this.title = this.t;$("#mdPicPreview").remove();});
|
||||
piclinks.mousemove(function(e){$("#mdPicPreview").css("top",(e.pageY - xOffset) + "px").css("left",(e.pageX + yOffset) + "px");});
|
||||
},
|
||||
error: function(response) { ifm.showMessage("General error occured: No or broken response", "e"); },
|
||||
complete: function() { ifm.task_done(id); }
|
||||
});
|
||||
},
|
||||
changeDirectory: function(newdir) {
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "getRealpath",
|
||||
dir: ifm.pathCombine(ifm.currentDir, newdir)
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data){
|
||||
ifm.currentDir = data.realpath;
|
||||
ifm.refreshFileTable();
|
||||
$("#currentDir").val(ifm.currentDir);
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured: No or broken response", "e"); }
|
||||
});
|
||||
},
|
||||
showFileForm: function () {
|
||||
var filename = arguments.length > 0 ? arguments[0] : "newfile.txt";
|
||||
var content = arguments.length > 1 ? arguments[1] : "";
|
||||
var overlay = '<div class="overlay">\
|
||||
<form id="showFile">\
|
||||
<fieldset>\
|
||||
<label>Filename:</label>\
|
||||
<input type="text" name="filename" value="'+filename+'" />\
|
||||
<div id="content" name="content"></div>\
|
||||
<div id="showFileButtons"><button onclick="ifm.saveFile();ifm.closeFileForm();return false;">Save</button>\
|
||||
<button onclick="ifm.saveFile();return false;">Save without closing</button>\
|
||||
<button onclick="ifm.closeFileForm();return false;">Close</button>\
|
||||
<select id="aceTheme">\
|
||||
<option value="chrome">chrome</option>\
|
||||
<option value="eclipse">eclipse</option>\
|
||||
<option value="ambiance">ambiance</option>\
|
||||
<option value="clouds_midnight">clouds_midnight</option>\
|
||||
<option value="mono_industrial">mono_industrial</option>\
|
||||
<option value="monokai">monokai</option>\
|
||||
<option value="terminal">terminal</option>\
|
||||
<option value="twilight">twilight</option>\
|
||||
</select>\
|
||||
<select id="aceMode">\
|
||||
<option value="apache_conf">apache_conf</option>\
|
||||
<option value="c_cpp">c_cpp</option>\
|
||||
<option value="csharp">csharp</option>\
|
||||
<option value="css">css</option>\
|
||||
<option value="html">html</option>\
|
||||
<option value="html_completions">html_completions</option>\
|
||||
<option value="ini">ini</option>\
|
||||
<option value="java">java</option>\
|
||||
<option value="javascript">javascript</option>\
|
||||
<option value="json">json</option>\
|
||||
<option value="mysql">mysql</option>\
|
||||
<option value="perl">perl</option>\
|
||||
<option value="php">php</option>\
|
||||
<option value="python">python</option>\
|
||||
<option value="sql">sql</option>\
|
||||
<option value="text">text</option>\
|
||||
</select>\
|
||||
</div>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>';
|
||||
$(document.body).prepend(overlay);
|
||||
$("#aceTheme").change(function() { ifm.editor.setTheme("ace/theme/"+this.value); });
|
||||
$("#aceMode").change(function() { ifm.editor.getSession().setMode("ace/mode/"+this.value); });
|
||||
if(filename=="")$("#showFile input[name=filename]").focus();
|
||||
else $("#showFile #content").focus();
|
||||
$(".overlay").on("remove", function () { ifm.editor = null; ifm.fileChanged = false; });
|
||||
ifm.bindOverlayClickEvent();
|
||||
// Start ACE
|
||||
ifm.editor = ace.edit("content");
|
||||
ifm.editor.getSession().setValue(content);
|
||||
var modelist = ace.require('ace/ext/modelist');
|
||||
var settingsMenu = ace.require('ace/ext/settings_menu');
|
||||
settingsMenu.init();
|
||||
ifm.editor.getSession().setUseWorker(false);
|
||||
ifm.editor.getSession().setMode(modelist.getModeForPath(filename).mode);
|
||||
ifm.editor.commands.addCommands([{
|
||||
name: "showSettingsMenu",
|
||||
bindKey: {win: "Ctrl-q", mac: "Command-q"},
|
||||
exec: function(editor) {
|
||||
ifm.editor.showSettingsMenu();
|
||||
},
|
||||
readOnly: true
|
||||
}]);
|
||||
var curMode = ifm.editor.getSession().getMode().$id;
|
||||
console.log("curMode: "+curMode);
|
||||
$("#aceMode option[value="+curMode.substring(curMode.lastIndexOf('/')+1)+"]").attr("selected", true);
|
||||
ifm.editor.focus();
|
||||
ifm.editor.on("change", function() { ifm.fileChanged = true; });
|
||||
},
|
||||
closeFileForm: function() {
|
||||
ifm.fileChanged = false;
|
||||
ifm.editor = null;
|
||||
ifm.hideSaveQuestion();
|
||||
ifm.removeOverlay();
|
||||
},
|
||||
removeOverlay: function() {
|
||||
$('.overlay').remove();
|
||||
},
|
||||
createDirForm: function() {
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="createDir">\
|
||||
<fieldset>\
|
||||
<label>Directoy name:</label>\
|
||||
<input type="text" name="dirname" value="" />\
|
||||
<button onclick="ifm.createDir();$(\'.overlay\').remove();return false;">Save</button><button onclick="$(\'.overlay\').remove();return false;">Cancel</button>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
ajaxRequestDialog: function() {
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="ajaxrequest">\
|
||||
<fieldset>\
|
||||
<label>URL</label><br>\
|
||||
<input type="text" id="ajaxurl" required><br>\
|
||||
<label>Data</label><br>\
|
||||
<textarea id="ajaxdata"></textarea><br>\
|
||||
<label>Method</label><br>\
|
||||
<input type="radio" name="arMethod" value="GET">GET</input><input type="radio" name="arMethod" value="POST" checked="checked">POST</input><br>\
|
||||
<button onclick="ifm.ajaxRequest();return false;">Request</button>\
|
||||
<button onclick="$(\'.overlay\').remove();return false;">Close</button><br>\
|
||||
<label>Response</label><br>\
|
||||
<textarea id="ajaxresponse"></textarea>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
ajaxRequest: function() {
|
||||
ifm.showLoading();
|
||||
$.ajax({
|
||||
url : $("#ajaxurl").val(),
|
||||
cache : false,
|
||||
data : $('#ajaxdata').val().replace(/\n/g,"&"),
|
||||
type : $('#ajaxrequest input[name=arMethod]:checked').val(),
|
||||
success : function(response) { $("#ajaxresponse").text(response); },
|
||||
error : function(e) { ifm.showMessage("Error: "+e, "e"); console.log(e); },
|
||||
complete: function() { ifm.hideLoading(); }
|
||||
});
|
||||
},
|
||||
saveFile: function() {
|
||||
var _content = ifm.editor.getValue();
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "saveFile",
|
||||
dir: ifm.currentDir,
|
||||
filename: $("#showFile input[name^=filename]").val(),
|
||||
content: _content
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("File successfully edited/created.", "s");
|
||||
ifm.refreshFileTable();
|
||||
} else ifm.showMessage("File could not be edited/created:"+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); }
|
||||
});
|
||||
ifm.fileChanged = false;
|
||||
},
|
||||
editFile: function(name) {
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
dataType: "json",
|
||||
data: ({
|
||||
api: "getContent",
|
||||
dir: ifm.currentDir,
|
||||
filename: name
|
||||
}),
|
||||
success: function(data) {
|
||||
if(data.status == "OK" && data.data.content != null) {
|
||||
ifm.showFileForm(data.data.filename, data.data.content);
|
||||
}
|
||||
else if(data.status == "OK" && data.data.content == null) {
|
||||
ifm.showMessage("The content of this file cannot be fetched.", "e");
|
||||
}
|
||||
else ifm.showMessage("Error: "+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("This file can not be displayed or edited.", "e"); },
|
||||
complete: function() { ifm.hideLoading(); }
|
||||
});
|
||||
},
|
||||
deleteFileDialog: function(name) {
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="deleteFile">\
|
||||
<fieldset>\
|
||||
<label>Do you really want to delete the file '+name+'?\
|
||||
<button onclick="ifm.deleteFile(\''+ifm.JSEncode(name)+'\');$(\'.overlay\').remove();return false;">Yes</button><button onclick="$(\'.overlay\').remove();return false;">No</button>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
deleteFile: function(name) {
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "deleteFile",
|
||||
dir: ifm.currentDir,
|
||||
filename: name
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("File successfully deleted", "s");
|
||||
ifm.refreshFileTable();
|
||||
} else ifm.showMessage("File could not be deleted", "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); }
|
||||
});
|
||||
},
|
||||
createDir: function() {
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "createDir",
|
||||
dir: ifm.currentDir,
|
||||
dirname: $("#createDir input[name^=dirname]").val()
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data){
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("Directory sucessfully created.", "s");
|
||||
ifm.refreshFileTable();
|
||||
}
|
||||
else {
|
||||
ifm.showMessage("Directory could not be created: "+data.message, "e");
|
||||
}
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured.", "e"); }
|
||||
});
|
||||
},
|
||||
renameFileDialog: function(name) {
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="renameFile">\
|
||||
<fieldset>\
|
||||
<label>Rename '+name+' to:</label>\
|
||||
<input type="text" name="newname" />\
|
||||
<button onclick="ifm.renameFile(\''+ifm.JSEncode(name)+'\');$(\'.overlay\').remove();return false;">Rename</button><button onclick="$(\'.overlay\').remove();return false;">Cancel</button>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
renameFile: function(name) {
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "renameFile",
|
||||
dir: ifm.currentDir,
|
||||
filename: name,
|
||||
newname: $("#renameFile input[name^=newname]").val()
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("File successfully renamed", "s");
|
||||
ifm.refreshFileTable();
|
||||
} else ifm.showMessage("File could not be renamed: "+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); }
|
||||
});
|
||||
},
|
||||
extractFileDialog: function(name) {
|
||||
var fuckWorkarounds="";
|
||||
if(fuckWorkarounds.lastIndexOf(".") > 1)
|
||||
fuckWorkarounds = name.substr(0,name.length-4);
|
||||
else fuckWorkarounds = name;
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="extractFile">\
|
||||
<fieldset>\
|
||||
<label>Extract '+name+' to:</label>\
|
||||
<button onclick="ifm.extractFile(\''+ifm.JSEncode(name)+'\', 0);$(\'.overlay\').remove();return false;">here</button>\
|
||||
<button onclick="ifm.extractFile(\''+ifm.JSEncode(name)+'\', 1);$(\'.overlay\').remove();return false;">'+fuckWorkarounds+'/</button>\
|
||||
<button onclick="$(\'.overlay\').remove();return false;">Cancel</button>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
extractFile: function(name, t) {
|
||||
var td = (t == 1)? name.substr(0,name.length-4) : "";
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "extractFile",
|
||||
dir: ifm.currentDir,
|
||||
filename: name,
|
||||
targetdir: td
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("File successfully extracted", "s");
|
||||
ifm.refreshFileTable();
|
||||
} else ifm.showMessage("File could not be extracted. Error: "+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); }
|
||||
});
|
||||
},
|
||||
uploadFileDialog: function() {
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="uploadFile">\
|
||||
<fieldset>\
|
||||
<label>Upload file</label><br>\
|
||||
<input type="file" name="ufile" id="ufile"><br>\
|
||||
<label>new filename</label>\
|
||||
<input type="text" name="newfilename"><br>\
|
||||
<button onclick="ifm.uploadFile();$(\'.overlay\').remove();return false;">Upload</button>\
|
||||
<button onclick="$(\'.overlay\').remove();return false;">Cancel</button>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
uploadFile: function() {
|
||||
var ufile = document.getElementById('ufile').files[0];
|
||||
var data = new FormData();
|
||||
var newfilename = $("#uploadFile input[name^=newfilename]").val();
|
||||
data.append('api', 'uploadFile');
|
||||
data.append('dir', ifm.currentDir);
|
||||
data.append('file', ufile);
|
||||
data.append('newfilename', newfilename);
|
||||
var id = ifm.generateGuid();
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: data,
|
||||
processData: false,
|
||||
contentType: false,
|
||||
dataType: "json",
|
||||
xhr: function(){
|
||||
var xhr = $.ajaxSettings.xhr() ;
|
||||
xhr.upload.onprogress = function(evt){ ifm.task_update(evt.loaded/evt.total*100,id); } ;
|
||||
xhr.upload.onload = function(){ console.log('Uploading '+newfilename+' done.') } ;
|
||||
return xhr ;
|
||||
},
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("File successfully uploaded", "s");
|
||||
if(data.cd == ifm.currentDir) ifm.refreshFileTable();
|
||||
} else ifm.showMessage("File could not be uploaded: "+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); },
|
||||
complete: function() { ifm.task_done(id); }
|
||||
});
|
||||
ifm.task_add("Upload "+ufile.name, id);
|
||||
},
|
||||
changePermissions: function(e, name) {
|
||||
if(e.keyCode == '13')
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "changePermissions",
|
||||
dir: ifm.currentDir,
|
||||
filename: name,
|
||||
chmod: e.target.value
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data){
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("Permissions successfully changed.", "s");
|
||||
ifm.refreshFileTable();
|
||||
}
|
||||
else {
|
||||
ifm.showMessage("Permissions could not be changed: "+data.message, "e");
|
||||
}
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured.", "e"); }
|
||||
});
|
||||
},
|
||||
remoteUploadDialog: function() {
|
||||
$(document.body).prepend('<div class="overlay">\
|
||||
<form id="uploadFile">\
|
||||
<fieldset>\
|
||||
<label>Remote upload URL</label><br>\
|
||||
<input type="text" id="url" name="url" required><br>\
|
||||
<label>Filename (required)</label>\
|
||||
<input type="text" id="filename" name="filename" required><br>\
|
||||
<label>Method</label>\
|
||||
<input type="radio" name="method" value="curl" checked="checked">cURL<input type="radio" name="method" value="file">file</input><br>\
|
||||
<button onclick="ifm.remoteUpload();$(\'.overlay\').remove();return false;">Upload</button>\
|
||||
<button onclick="$(\'.overlay\').remove();return false;">Cancel</button>\
|
||||
</fieldset>\
|
||||
</form>\
|
||||
</div>');
|
||||
// guess the wanted filename, because it is required
|
||||
// e.g http:/example.com/example.txt => filename: example.txt
|
||||
$("#url").on("change keyup", function(){
|
||||
$("#filename").val($(this).val().substr($(this).val().lastIndexOf("/")+1));
|
||||
});
|
||||
// if the filename input field was edited manually remove the event handler above
|
||||
$("#filename").on("keyup", function() { $("#url").off(); });
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
remoteUpload: function() {
|
||||
var filename = $("#filename").val();
|
||||
var id = ifm.generateGuid();
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "remoteUpload",
|
||||
dir: ifm.currentDir,
|
||||
filename: filename,
|
||||
method: $("input[name=method]:checked").val(),
|
||||
url: encodeURI($("#url").val())
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
ifm.showMessage("File successfully uploaded", "s");
|
||||
ifm.refreshFileTable();
|
||||
} else ifm.showMessage("File could not be uploaded:<br />"+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); },
|
||||
complete: function() { ifm.task_done(id); }
|
||||
});
|
||||
ifm.task_add("Remote upload: "+filename, id);
|
||||
},
|
||||
// --------------------
|
||||
// additional functions
|
||||
// --------------------
|
||||
showMessage: function(m, t) {
|
||||
var message = '<div id="mess"><div class="';
|
||||
if(t == "e") message += "message_error";
|
||||
else if(t == "s") message += "message_successful";
|
||||
else message += "message";
|
||||
message += '">'+m+'</div></div>';
|
||||
$(document.body).prepend(message);
|
||||
$("#mess").delay(2000).fadeOut('slow');
|
||||
setTimeout(function() { // remove the message from the DOM after 3 seconds
|
||||
$('#mess').remove();
|
||||
}, 3000);
|
||||
},
|
||||
showLoading: function() {
|
||||
var loading = '<div id="loading">'+ifm.loadingAnim+'</div>';
|
||||
if(document.getElementById("loading")==null)$(document.body).prepend(loading);
|
||||
},
|
||||
hideLoading: function() {
|
||||
$("#loading").remove();
|
||||
},
|
||||
pathCombine: function(a, b) {
|
||||
if(a == "" && b == "") return "";
|
||||
if(b[0] == "/") b = b.substring(1);
|
||||
if(a == "") return b;
|
||||
if(a[a.length-1] == "/") a = a.substring(0, a.length-1);
|
||||
if(b == "") return a;
|
||||
return a+"/"+b;
|
||||
},
|
||||
bindOverlayClickEvent: function() {
|
||||
$(".overlay").off('click');
|
||||
$(".overlay *").off('click');
|
||||
$(".overlay").click(function() {
|
||||
var isFileForm = !!$(".overlay #showFile")[0];
|
||||
var isSaveQuestion = !!$("#savequestion")[0];
|
||||
if(isFileForm && !isSaveQuestion && ifm.fileChanged) {
|
||||
ifm.showSaveQuestion();
|
||||
}
|
||||
else if(isFileForm && isSaveQuestion) {
|
||||
ifm.hideSaveQuestion();
|
||||
ifm.closeFileForm();
|
||||
}
|
||||
else { ifm.removeOverlay(); }
|
||||
});
|
||||
$(".overlay *").click(function(e) { e.stopPropagation(); });
|
||||
},
|
||||
showSaveQuestion: function() {
|
||||
var a = '<div id="savequestion"><label>Do you want to save this file?</label><br><button onclick="ifm.saveFile();ifm.closeFileForm(); return false;">Save</button><button onclick="ifm.closeFileForm();return false;">Dismiss</button>';
|
||||
$(document.body).prepend(a);
|
||||
ifm.bindOverlayClickEvent();
|
||||
},
|
||||
hideSaveQuestion: function() { $("#savequestion").remove(); },
|
||||
handleMultiSelect: function() {
|
||||
var amount = $("input[name=multisel]:checked").length;
|
||||
if(amount > 0) {
|
||||
if(document.getElementById("multiseloptions")===null) {
|
||||
$(document.body).prepend('<div id="multiseloptions">\
|
||||
<div style="font-size:0.8em;background-color:#00A3A3;padding:2px;">\
|
||||
<a style="color:#FFF" onclick="$(\'input[name=multisel]\').attr(\'checked\', false);$(\'#multiseloptions\').remove();">[close]</a>\
|
||||
</div>\
|
||||
<ul><li><a onclick="ifm.multiDelete();"><img src="'+ifm.icons['delete.png']+'" alt=""> delete (<span class="amount"></span>)</a></li></ul></div>\
|
||||
');
|
||||
$("#multiseloptions").draggable();
|
||||
//$("#multiseloptions").resizable({ghost: true, minHeight: 50, minWidth: 100});
|
||||
}
|
||||
$("#multiseloptions .amount").text(amount);
|
||||
}
|
||||
else {
|
||||
if(document.getElementById("multiseloptions")!==null)
|
||||
$("#multiseloptions").remove();
|
||||
}
|
||||
},
|
||||
multiDelete: function() {
|
||||
var jel = $("input[name=multisel]:checked");
|
||||
if(jel.length == 0) {
|
||||
ifm.showMessage("No files chosen");
|
||||
return;
|
||||
}
|
||||
var ids = [];
|
||||
for(var i = 0; i < jel.length; i++) ids.push(jel[i].id);
|
||||
$.ajax({
|
||||
url: ifm.IFM_SCFN,
|
||||
type: "POST",
|
||||
data: ({
|
||||
api: "deleteMultipleFiles",
|
||||
dir: ifm.currentDir,
|
||||
filenames: ids
|
||||
}),
|
||||
dataType: "json",
|
||||
success: function(data) {
|
||||
if(data.status == "OK") {
|
||||
if(data.errflag == 1)
|
||||
ifm.showMessage("All files successfully deleted.", "s");
|
||||
else if(data.errflag == 0)
|
||||
ifm.showMessage("Some files successfully deleted. "+data.message);
|
||||
else
|
||||
ifm.showMessage("Files could not be deleted. "+data.message, "e");
|
||||
ifm.refreshFileTable();
|
||||
} else ifm.showMessage("Files could not be deleted:<br />"+data.message, "e");
|
||||
},
|
||||
error: function() { ifm.showMessage("General error occured", "e"); }
|
||||
});
|
||||
},
|
||||
inArray: function(needle, haystack) {
|
||||
for(var i = 0; i < haystack.length; i++) { if(haystack[i] == needle) return true; } return false;
|
||||
},
|
||||
task_add: function(name,id) { // uFI stands for uploadFileInformation
|
||||
if(!document.getElementById("waitqueue")) {
|
||||
$(document.body).prepend('<div id="waitqueue"></div>');
|
||||
$("#waitqueue").on("mouseover", function() { $(this).toggleClass("left"); });
|
||||
}
|
||||
$("#waitqueue").append('<div id="'+id+'"><div class="progress"></div> '+name+'</div>');
|
||||
$("#"+id+" .progress").progressbar({value: false});
|
||||
},
|
||||
task_done: function(id) {
|
||||
$("#"+id).remove();
|
||||
if($("#waitqueue>div").length == 0) {
|
||||
$("#waitqueue").remove();
|
||||
}
|
||||
},
|
||||
task_update: function(progress, id) {
|
||||
$("#"+id+" .progress").progressbar({ value: progress});
|
||||
},
|
||||
generateGuid: function() {
|
||||
var result, i, j;
|
||||
result = '';
|
||||
for(j=0; j<20; j++) {
|
||||
i = Math.floor(Math.random()*16).toString(16).toUpperCase();
|
||||
result = result + i;
|
||||
}
|
||||
return result;
|
||||
},
|
||||
JSEncode: function(s) {
|
||||
return s.replace(/'/g, '\\x27').replace(/"/g, '\\x22');
|
||||
},
|
||||
// static button bindings and filetable initial filling
|
||||
init: function() {
|
||||
// fill file table
|
||||
this.refreshFileTable();
|
||||
// bind static buttons
|
||||
$("#refresh").click(function(){
|
||||
ifm.refreshFileTable();
|
||||
});
|
||||
$("#createFile").click(function(){
|
||||
ifm.showFileForm();
|
||||
});
|
||||
$("#createDir").click(function(){
|
||||
ifm.createDirForm();
|
||||
});
|
||||
$("#upload").click(function(){
|
||||
ifm.uploadFileDialog();
|
||||
});
|
||||
},
|
||||
// ---------------
|
||||
// further members
|
||||
// ---------------
|
||||
loadingAnim: '<div id="loadingAnim"><div class="blockG" id="rotateG_01"></div><div class="blockG" id="rotateG_02"></div><div class="blockG" id="rotateG_03"></div><div class="blockG" id="rotateG_04"></div><div class="blockG" id="rotateG_05"></div><div class="blockG" id="rotateG_06"></div><div class="blockG" id="rotateG_07"></div><div class="blockG" id="rotateG_08"></div></div>'
|
||||
}
|
||||
}
|
||||
$(document).ready(function() {ifm.init()}); // init ifm
|
189
src/includes.php
Normal file
189
src/includes.php
Normal file
File diff suppressed because one or more lines are too long
810
src/main.php
Normal file
810
src/main.php
Normal file
@@ -0,0 +1,810 @@
|
||||
<?php
|
||||
|
||||
/* =======================================================================
|
||||
* Improved File Manager
|
||||
* ---------------------
|
||||
* License: This project is provided under the terms of the MIT LICENSE
|
||||
* http://github.com/misterunknown/ifm/blob/master/LICENSE
|
||||
* =======================================================================
|
||||
*
|
||||
* main
|
||||
*/
|
||||
|
||||
class IFM {
|
||||
const VERSION = '2.0';
|
||||
|
||||
public function __construct() {
|
||||
session_start();
|
||||
}
|
||||
|
||||
/*
|
||||
this function contains the client-side application
|
||||
*/
|
||||
|
||||
public function getApplication() {
|
||||
print '<!DOCTYPE HTML>
|
||||
<html>
|
||||
<head>
|
||||
<title>IFM - improved file manager</title>
|
||||
<meta charset="utf-8"><script>'.IFMIncludes::getJquery().'</script>
|
||||
<script>'.IFMIncludes::getJqueryUI().'</script>
|
||||
<script>'.IFMIncludes::getAce().'</script>
|
||||
<script>'.IFMIncludes::getJqueryFancybox().'</script>
|
||||
<style type="text/css">
|
||||
'; ?>
|
||||
@@@COMPILE:style.css@@@
|
||||
<?php
|
||||
print ''.IFMIncludes::getJqueryUICSS()."\n".IFMIncludes::getJqueryFancyboxCSS().'
|
||||
</style>
|
||||
<script type="text/javascript">
|
||||
'; ?>
|
||||
@@@COMPILE:ifm.js@@@
|
||||
<?php print '
|
||||
</script>
|
||||
</head>
|
||||
<body>
|
||||
<table id="tooltab">
|
||||
<tbody>
|
||||
<tr>
|
||||
<td class="cell_content">
|
||||
Content of <span id="docroot">';
|
||||
if(IFMConfig::showpath == 1) print getScriptRoot().'/'; else print '/';
|
||||
print '</span><input id="currentDir" type="text">
|
||||
</td>
|
||||
<td class="cell_buttons">
|
||||
<div>';
|
||||
// refresh button - always shows
|
||||
print '<button id="refresh">';
|
||||
echo ('<img src="" alt="refresh" />');
|
||||
print '</button>';
|
||||
// upload button
|
||||
if(IFMConfig::upload == 1) {
|
||||
print '<button id="upload">';
|
||||
print '<img src="" alt="Upload" />';
|
||||
print '</button>';
|
||||
}
|
||||
// create file button
|
||||
if(IFMConfig::createfile == 1) {
|
||||
print '<button id="createFile">';
|
||||
print '<img src="" alt="create file" />';
|
||||
print '</button>';
|
||||
}
|
||||
// create directory button
|
||||
if(IFMConfig::createdir == 1) {
|
||||
print '<button id="createDir">';
|
||||
print '<img src="" alt="create directory" />';
|
||||
print '</button>';
|
||||
}
|
||||
// create options button
|
||||
print ' <div id="options"><img src="" alt="options" /><ul>';
|
||||
$options = false;
|
||||
if(IFMConfig::remoteupload == 1) {
|
||||
print '<li><a onclick="ifm.remoteUploadDialog()">Remote upload</a></li>';
|
||||
$options = true;
|
||||
}
|
||||
if(IFMConfig::ajaxrequest == 1) {
|
||||
print '<li><a onclick="ifm.ajaxRequestDialog()">AJAX Request</a></li>';
|
||||
$options = true;
|
||||
}
|
||||
if(!$options) print '<li>No options available</li>';
|
||||
print '</ul></div>';
|
||||
print '<span id="version">ver '.IFM::VERSION.'</span>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
<table id="filetable">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Filename</th>';
|
||||
if(IFMConfig::download == 1) print '<th><!-- column for download link --></th>';
|
||||
if(IFMConfig::showlastmodified == 1) print '<th>last modified</th>';
|
||||
if(IFMConfig::showfilesize == 1) print '<th>Filesize</th>';
|
||||
if(IFMConfig::showrights > 0)print '<th>Permissions</th>';
|
||||
if(IFMConfig::showowner == 1) print '<th>Owner</th>';
|
||||
if(IFMConfig::showgroup == 1) print '<th>Group</th>';
|
||||
if(in_array(1,array(IFMConfig::edit,IFMConfig::rename,IFMConfig::delete,IFMConfig::zipnload,IFMConfig::extract))) print '<th><!-- column for buttons --></th>';
|
||||
print '</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
</tbody>
|
||||
</table>
|
||||
<footer>IFM - improved file manager | ifm.php hidden | <a href="http://github.com/misterunknown/ifm">Visit the project on GitHub</a></footer>
|
||||
</body>
|
||||
</html>
|
||||
';
|
||||
}
|
||||
|
||||
/*
|
||||
main functions
|
||||
*/
|
||||
|
||||
private function handleRequest() {
|
||||
if($_REQUEST["api"] == "getRealpath") {
|
||||
if( isset( $_REQUEST["dir"] ) && $_REQUEST["dir"] != "" )
|
||||
$this->getRealpath( $_REQUEST["dir"] );
|
||||
else
|
||||
echo json_encode(array("realpath"=>""));
|
||||
}
|
||||
elseif( $_REQUEST["api"] == "getFiles" ) {
|
||||
if( isset( $_REQUEST["dir"] ) && $this->isPathValid( $_REQUEST["dir"] ) )
|
||||
$this->getFiles( $_REQUEST["dir"] );
|
||||
else
|
||||
$this->getFiles( "" );
|
||||
}
|
||||
else {
|
||||
if( isset( $_REQUEST["dir"] ) && $this->isPathValid( $_REQUEST["dir"] ) ) {
|
||||
switch( $_REQUEST["api"] ) {
|
||||
case "createDir": $this->createDir( $_REQUEST["dir"], $_REQUEST["dirname"] ); break;
|
||||
case "saveFile": $this->saveFile( $_REQUEST ); break;
|
||||
case "getContent": $this->getContent( $_REQUEST ); break;
|
||||
case "deleteFile": $this->deleteFile( $_REQUEST ); break;
|
||||
case "renameFile": $this->renameFile( $_REQUEST ); break;
|
||||
case "downloadFile": $this->downloadFile( $_REQUEST ); break;
|
||||
case "extractFile": $this->extractFile( $_REQUEST ); break;
|
||||
case "uploadFile": $this->uploadFile( $_REQUEST ); break;
|
||||
case "changePermissions": $this->changePermissions( $_REQUEST ); break;
|
||||
case "zipnload": $this->zipnload( $_REQUEST); break;
|
||||
case "remoteUpload": $this->remoteUpload( $_REQUEST ); break;
|
||||
case "deleteMultipleFiles": $this->deleteMultipleFiles( $_REQUEST ); break;
|
||||
default: echo json_encode(array("status"=>"ERROR", "message"=>"No valid api action given")); break;
|
||||
}
|
||||
} else {
|
||||
print json_encode(array("status"=>"ERROR", "message"=>"No valid working directory"));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public function run() {
|
||||
if ( $this->checkAuth() ) {
|
||||
if ( ! isset($_REQUEST['api']) ) {
|
||||
$this->getApplication();
|
||||
} else {
|
||||
$this->handleRequest();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
api functions
|
||||
*/
|
||||
|
||||
private function getFiles($dir) {
|
||||
// SECURITY FUNCTION (check that we don't operate on a higher level that the script itself)
|
||||
$dir=$this->getValidDir($dir);
|
||||
// now we change in our target directory
|
||||
$this->chDirIfNecessary($dir);
|
||||
// unset our file and directory arrays
|
||||
unset($files); unset($dirs); $files = array(); $dirs = array();
|
||||
// so lets loop over our directory
|
||||
if ($handle = opendir(".")) {
|
||||
while (false !== ($result = readdir($handle))) { // this awesome statement is the correct way to loop over a directory :)
|
||||
if( $result == basename( $_SERVER['SCRIPT_NAME'] ) && $this->getScriptRoot() == getcwd() ) { } // we don't want to see the script itself
|
||||
elseif( ( $result == ".htaccess" || $result==".htpasswd" ) && IFMConfig::showhtdocs != 1 ) {} // check if we are granted to see .ht-docs
|
||||
elseif( $result == "." ) {} // the folder itself will also be invisible
|
||||
elseif( $result != ".." && substr( $result, 0, 1 ) == "." && IFMConfig::showhiddenfiles != 1 ) {} // eventually hide hidden files, if we should not see them
|
||||
elseif( ! @is_readable( $result ) ) {}
|
||||
else { // thats are the files we should see
|
||||
$item = array();
|
||||
$i = 0;
|
||||
$item["name"] = $result;
|
||||
$i++;
|
||||
if( is_dir($result) ) {
|
||||
$item["type"] = "dir";
|
||||
} else {
|
||||
$item["type"] = "file";
|
||||
}
|
||||
if( is_dir( $result ) ) {
|
||||
$item["picture"] = "folder.png";
|
||||
} else {
|
||||
$type = substr( strrchr( $result, "." ), 1 );
|
||||
$item["picture"] = $this->getTypePicture( $type );
|
||||
}
|
||||
if( IFMConfig::showlastmodified == 1 ) { $item["lastmodified"] = date( "d.m.Y, G:i e", filemtime( $result ) ); }
|
||||
if( IFMConfig::showfilesize == 1 ) {
|
||||
$item["filesize"] = filesize( $result );
|
||||
if( $item["filesize"] > 1073741824 ) $item["filesize"] = round( ( $item["filesize"]/1073741824 ), 2 ) . " GB";
|
||||
elseif($item["filesize"]>1048576)$item["filesize"] = round( ( $item["filesize"]/1048576 ), 2 ) . " MB";
|
||||
elseif($item["filesize"]>1024)$item["filesize"] = round( ( $item["filesize"]/1024 ), 2 ) . " KB";
|
||||
else $item["filesize"] = $item["filesize"] . " Byte";
|
||||
}
|
||||
if( IFMConfig::showrights > 0 ) {
|
||||
if( IFMConfig::showrights == 1 ) $item["fileperms"] = substr( decoct( fileperms( $result ) ), -3 );
|
||||
elseif( IFMConfig::showrights == 2 ) $item["fileperms"] = $this->filePermsDecode( fileperms( $result ) );
|
||||
if( $item["fileperms"] == "" ) $item["fileperms"] = " ";
|
||||
}
|
||||
if( IFMConfig::showowner == 1 ) {
|
||||
if ( function_exists( "posix_getpwuid" ) && fileowner($result) !== false ) {
|
||||
$ownerarr = posix_getpwuid( fileowner( $result ) );
|
||||
$item["owner"] = $ownerarr['name'];
|
||||
} else $item["owner"] = "";
|
||||
}
|
||||
if( IFMConfig::showgroup == 1 ) {
|
||||
if( function_exists( "posix_getgrgid" ) && filegroup( $result ) !== false ) {
|
||||
$grouparr = posix_getgrgid( filegroup( $result ) );
|
||||
$item["group"] = $grouparr['name'];
|
||||
} else $item["group"] = "";
|
||||
}
|
||||
if( is_dir( $result ) ) $dirs[] = $item;
|
||||
else $files[] = $item;
|
||||
}
|
||||
}
|
||||
closedir( $handle );
|
||||
}
|
||||
usort( $dirs, array( $this, "sortByName" ) );
|
||||
usort( $files, array( $this, "sortByName" ) );
|
||||
echo json_encode( array_merge( $dirs, $files ) );
|
||||
}
|
||||
|
||||
// creates a directory
|
||||
private function createDir($w, $dn) {
|
||||
if( $dn == "" ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No valid directory name") );
|
||||
} elseif( strpos( $dn, '/' ) !== false ) echo json_encode( array( "status" => "ERROR", "message" => "No slashes allowed in directory names" ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $w );
|
||||
if( @mkdir( $dn ) ) {
|
||||
echo json_encode( array( "status" => "OK", "message" => "Directory successful created" ) );
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Could not create directory" ) );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// save a file
|
||||
private function saveFile(array $d) {
|
||||
if( isset( $d['filename'] ) && $d['filename'] != "" ) {
|
||||
// if you are not allowed to see .ht-docs you can't save one
|
||||
if( IFMConfig::showhtdocs != 1 && substr( $d['filename'], 0, 3 ) == ".ht" ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "You are not allowed to edit or create htdocs" ) );
|
||||
}
|
||||
// same with hidden files
|
||||
elseif( IFMConfig::showhiddenfiles != 1 && substr( $d['filename'], 0, 1 ) == "." ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "You are not allowed to edit or create hidden files" ) );
|
||||
}
|
||||
elseif(strpos($d['filename'],'/')!==false) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Filenames cannot contain slashes." ) );
|
||||
} else {
|
||||
if( isset( $d['content'] ) ) {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
// work around magic quotes
|
||||
$content = get_magic_quotes_gpc() == 1 ? stripslashes( $d['content'] ) : $d['content'];
|
||||
if( @file_put_contents( $d['filename'], $content ) !== false ) {
|
||||
echo json_encode( array( "status" => "OK", "message" => "File successfully saved" ) );
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Could not write content" ) );
|
||||
}
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Got no content" ) );
|
||||
}
|
||||
}
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No filename specified" ) );
|
||||
}
|
||||
}
|
||||
|
||||
// gets the content of a file
|
||||
// notice: if the content is not JSON encodable it returns an error
|
||||
private function getContent( array $d ) {
|
||||
if( IFMConfig::edit != 1 ) echo json_encode( array( "status" => "ERROR", "message" => "No permission to edit files" ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
if( file_exists( $d['filename'] ) ) {
|
||||
$content = @file_get_contents( $d['filename'] );
|
||||
$utf8content = mb_convert_encoding( $content, 'UTF-8', mb_detect_encoding( $content, 'UTF-8, ISO-8859-1', true ) );
|
||||
echo json_encode( array( "status" => "OK", "data" => array( "filename" => $d['filename'], "content" => $utf8content ) ) );
|
||||
} else echo json_encode( array( "status" => "ERROR", "message" => "File not found" ) );
|
||||
}
|
||||
}
|
||||
|
||||
// deletes a file or a directory (recursive!)
|
||||
private function deleteFile( array $d ) {
|
||||
if( IFMConfig::delete != 1 ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to delete files" ) );
|
||||
}
|
||||
else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
if( is_dir( $d['filename'] ) ) {
|
||||
$res = rec_rmdir( $d['filename'] );
|
||||
if( $res != 0 ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to delete files" ) );
|
||||
} else {
|
||||
echo json_encode( array( "status" => "OK", "message" => "Directoy successful deleted" ) );
|
||||
}
|
||||
}
|
||||
else{
|
||||
if( @unlink( $d['filename'] ) ) {
|
||||
echo json_encode( array( "status" => "OK", "message" => "File successful deleted" ) );
|
||||
} else {
|
||||
echo json_encode( array( "status"=>"ERROR", "message" => "File could not be deleted" ) );
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// deletes a bunch of files or directories
|
||||
private function deleteMultipleFiles( array $d ) {
|
||||
if( IFMConfig::delete != 1 || IFMConfig::multiselect != 1 ) echo json_encode( array( "status" => "ERROR", "message" => "No permission to delete multiple files" ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
$err = array(); $errFLAG = -1; // -1 -> no files deleted; 0 -> at least some files deleted; 1 -> all files deleted
|
||||
foreach( $d['filenames'] as $file ) {
|
||||
if( is_dir($file) ){
|
||||
$res = $this->rec_rmdir( $file );
|
||||
if( $res != 0 )
|
||||
array_push( $err, $file );
|
||||
else
|
||||
$errFLAG = 0;
|
||||
} else {
|
||||
if( @unlink($file) )
|
||||
$errFLAG = 0;
|
||||
else
|
||||
array_push($err, $file);
|
||||
}
|
||||
}
|
||||
if( empty( $err ) ) {
|
||||
echo json_encode( array( "status" => "OK", "message" => "Files deleted successfully", "errflag" => "1" ) );
|
||||
}
|
||||
else {
|
||||
$errmsg = "The following files could not be deleted:<ul>";
|
||||
foreach($err as $item)
|
||||
$errmsg .= "<li>".$item."</li>";
|
||||
$errmsg .= "</ul>";
|
||||
echo json_encode( array( "status" => "OK", "message" => $errmsg, "flag" => $errFLAG ) );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// renames a file
|
||||
private function renameFile( array $d ) {
|
||||
if( IFMConfig::rename != 1 ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to rename files" ) );
|
||||
} else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
if( strpos( $d['newname'], '/' ) !== false )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No slashes allowed in filenames" ) );
|
||||
elseif( IFMConfig::showhtdocs != 1 && ( substr( $d['newname'], 0, 3) == ".ht" || substr( $d['filename'], 0, 3 ) == ".ht" ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to rename this file" ) );
|
||||
elseif( IFMConfig::showhiddenfiles != 1 && ( substr( $d['newname'], 0, 1) == "." || substr( $d['filename'], 0, 1 ) == "." ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to rename file" ) );
|
||||
else {
|
||||
if( @rename( $d['filename'], $d['newname'] ) )
|
||||
echo json_encode( array( "status" => "OK", "message" => "File successful renamed" ) );
|
||||
else
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "File could not be renamed" ) );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// provides a file for downloading
|
||||
private function downloadFile( array $d ) {
|
||||
if( IFMConfig::download != 1 )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to download files" ) );
|
||||
elseif( IFMConfig::showhtdocs != 1 && ( substr( $d['filename'], 0, 3 ) == ".ht" || substr( $d['filename'],0,3 ) == ".ht" ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message"=>"Not allowed to download htdocs" ) );
|
||||
elseif( IFMConfig::showhiddenfiles != 1 && ( substr( $d['filename'], 0, 1 ) == "." || substr( $d['filename'],0,1 ) == "." ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to download hidden files" ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $d["dir"] );
|
||||
header( "Content-Type: application/octet-stream" );
|
||||
header( "Content-Disposition: attachment; filename=\"" . $d["filename"] . "\"" );
|
||||
@readfile( $d["filename"] );
|
||||
}
|
||||
}
|
||||
|
||||
// extracts a zip-archive
|
||||
private function extractFile( array $d ) {
|
||||
if( IFMConfig::extract != 1 )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to extract files" ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
if( ! file_exists( $d['filename'] ) || substr( $d['filename'],-4 ) != ".zip" )
|
||||
echo json_encode( array( "status" => "ERROR","message" => "No valid zip file found" ) );
|
||||
else {
|
||||
if( ! isset( $d['targetdir'] ) )
|
||||
$d['targetdir'] = "";
|
||||
if( strpos( $d['targetdir'], "/" ) !== false )
|
||||
echo json_encode( array( "status" => "ERROR","message" => "Target directory must not contain slashes" ) );
|
||||
else {
|
||||
switch( $d['targetdir'] ){
|
||||
case "":
|
||||
if( $this->unzip( $_POST["filename"] ) )
|
||||
echo json_encode( array( "status" => "OK","message" => "File successfully extracted." ) );
|
||||
else
|
||||
echo json_encode( array( "status" => "ERROR","message" => "File could not be extracted" ) );
|
||||
break;
|
||||
default:
|
||||
if( ! mkdir( $d['targetdir'] ) )
|
||||
echo json_encode( array( "status" => "ERROR","message" => "Could not create target directory" ) );
|
||||
else {
|
||||
chdir( $d['targetdir'] );
|
||||
if( ! $this->unzip( "../" . $d["filename"] ) ) {
|
||||
chdir( ".." );
|
||||
rmdir( $d['targetdir'] );
|
||||
echo json_encode( array( "status" => "ERROR","message" => "Could not extract file" ) );
|
||||
}
|
||||
else {
|
||||
chdir( ".." );
|
||||
echo json_encode( array( "status" => "OK","message" => "File successfully extracted" ) );
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// uploads a file
|
||||
private function uploadFile( array $d ) {
|
||||
if( IFMConfig::upload != 1 )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to upload files" ) );
|
||||
elseif( !isset( $_FILES['file'] ) )
|
||||
echo json_encode( array( "file" => $_FILE,"files" => $_FILES ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
$newfilename = ( isset( $d["newfilename"] ) && $d["newfilename"]!="" ) ? $d["newfilename"] : $_FILES['file']['name'];
|
||||
if( IFMConfig::showhtdocs != 1 && ( substr( $newfilename, 0, 3 ) == ".ht" || substr( $newfilename,0,3 ) == ".ht" ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to upload htdoc file" ) );
|
||||
elseif( IFMConfig::showhiddenfiles != 1 && ( substr( $newfilename, 0, 1 ) == "." || substr( $newfilename,0,1 ) == "." ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to upload hidden file" ) );
|
||||
else {
|
||||
if( $_FILES['file']['tmp_name'] ) {
|
||||
if( is_writable( getcwd( ) ) ) {
|
||||
if( move_uploaded_file( $_FILES['file']['tmp_name'], $newfilename ) )
|
||||
echo json_encode( array( "status" => "OK", "message" => "The file ".$_FILES['file']['name']." was uploaded successfully", "cd" => $d['dir'] ) );
|
||||
else
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "File could not be uploaded" ) );
|
||||
}
|
||||
else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "File could not be uploaded since it has no permissions to write in this directory" ) );
|
||||
}
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No file found" ) );
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// change permissions of a file
|
||||
private function changePermissions( array $d ) {
|
||||
if( IFMConfig::chmod != 1 ) echo json_encode( array( "status" => "ERROR", "message" => "No rights to change permissions" ) );
|
||||
elseif( ! isset( $d["chmod"] )||$d['chmod']=="" ) echo json_encode( array( "status" => "ERROR", "message" => "Could not identify new permissions" ) );
|
||||
elseif( ! isPathValid( pathCombine( $d['dir'],$d['filename'] ) ) ) { echo json_encode( array( "status" => "ERROR", "message" => "Not allowed to change the permissions" ) ); }
|
||||
else {
|
||||
chDirIfNecessary( $d['dir'] ); $chmod = $d["chmod"]; $cmi = true;
|
||||
if( ! is_numeric( $chmod ) ) {
|
||||
$cmi = false;
|
||||
$chmod = str_replace( " ","",$chmod );
|
||||
if( strlen( $chmod )==9 ) {
|
||||
$cmi = true;
|
||||
$arr = array( substr( $chmod,0,3 ),substr( $chmod,3,3 ),substr( $chmod,6,3 ) );
|
||||
$chtmp = "0";
|
||||
foreach( $arr as $right ) {
|
||||
$rtmp = 0;
|
||||
if( substr( $right,0,1 )=="r" ) $rtmp = $rtmp + 4; elseif( substr( $right,0,1 )<>"-" ) $cmi = false;
|
||||
if( substr( $right,1,1 )=="w" ) $rtmp = $rtmp + 2; elseif( substr( $right,1,1 )<>"-" ) $cmi = false;
|
||||
if( substr( $right,2,1 )=="x" ) $rtmp = $rtmp + 1; elseif( substr( $right,2,1 )<>"-" ) $cmi = false;
|
||||
$chtmp = $chtmp . $rtmp;
|
||||
}
|
||||
$chmod = intval( $chtmp );
|
||||
}
|
||||
}
|
||||
else $chmod = "0" . $chmod;
|
||||
|
||||
if( $cmi ) {
|
||||
try {
|
||||
chmod( $d["filename"], (int)octdec( $chmod ) );
|
||||
echo json_encode( array( "status" => "OK", "message" => "Permissions changed successfully" ) );
|
||||
} catch ( Exception $e ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Error while changing permissions" ) );
|
||||
}
|
||||
}
|
||||
else echo json_encode( array( "status" => "ERROR", "message" => "Could not determine permission format" ) );
|
||||
}
|
||||
}
|
||||
|
||||
// zips a directory and provides it for downloading
|
||||
// it creates a temporary zip file in the current directory, so it has to be as much space free as the file size is
|
||||
private function zipnload( array $d ) {
|
||||
if( IFMConfig::zipnload != 1 )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to download directories" ) );
|
||||
else {
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
if( ! file_exists( $d['filename'] ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Directory not found" ) );
|
||||
else {
|
||||
unset( $zip );
|
||||
$dfile = uniqid( "ifm-tmp-" ) . ".zip"; // temporary filename
|
||||
$zip = new zip_file( $dfile );
|
||||
$zip->add_files( $d['filename'] );
|
||||
$zip->create_archive( );
|
||||
header( "Content-Disposition: attachment; filename=\"".$d['filename'].".zip\"" );
|
||||
echo file_get_contents( $dfile );
|
||||
unlink( $dfile ); // delete temporary file
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// uploads a file from an other server using the curl extention
|
||||
private function remoteUpload( array $d ) {
|
||||
if( IFMConfig::remoteupload != 1 )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "No permission to remote upload files" ) );
|
||||
elseif( !isset( $d['method'] ) || !in_array( $d['method'], array( "curl", "file" ) ) )
|
||||
echo json_encode( array( "status" => "error", "message" => "No valid method given. Valid methods: ['curl', 'file']" ) );
|
||||
elseif( $d['method']=="curl" && $this->checkCurl( ) == false )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "cURL extention not installed. Please install the cURL extention to use remote file upload." ) );
|
||||
elseif( $d['method']=="curl" && $this->checkCurl( ) == true ) {
|
||||
$filename = ( isset( $d['filename'] )&&$d['filename']!="" )?$d['filename']:"curl_".uniqid( );
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
$ch = curl_init( );
|
||||
if( $ch ) {
|
||||
if( $this->allowedFileName( $filename ) == false )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "This filename is not allowed due to the config." ) );
|
||||
elseif( filter_var( $d['url'], FILTER_VALIDATE_URL ) === false )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "The passed URL is not valid" ) );
|
||||
else {
|
||||
$fp = fopen( $filename, "w" );
|
||||
if( $fp ) {
|
||||
if( !curl_setopt( $ch, CURLOPT_URL, $d['url'] ) || !curl_setopt( $ch, CURLOPT_FILE, $fp ) || !curl_setopt( $ch, CURLOPT_HEADER, 0 ) || !curl_exec( $ch ) )
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Failed to set options and execute cURL" ) );
|
||||
else {
|
||||
echo json_encode( array( "status" => "OK", "message" => "File sucessfully uploaded" ) );
|
||||
}
|
||||
curl_close( $ch );
|
||||
fclose( $fp );
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Failed to open file" ) );
|
||||
}
|
||||
}
|
||||
} else {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => "Failed to init cURL." ) );
|
||||
}
|
||||
}
|
||||
elseif( $d['method']=='file' ) {
|
||||
$filename = ( isset( $d['filename'] ) && $d['filename']!="" ) ? $d['filename'] : "curl_".uniqid( );
|
||||
$this->chDirIfNecessary( $d['dir'] );
|
||||
try {
|
||||
file_put_contents( $filename, file_get_contents( $d['url'] ) );
|
||||
echo json_encode( array( "status" => "OK", "message" => "File successfully uploaded" ) );
|
||||
} catch( Exception $e ) {
|
||||
echo json_encode( array( "status" => "ERROR", "message" => $e->getMessage() ) );
|
||||
}
|
||||
}
|
||||
else echo json_encode( array( "status" => "error", "message" => "Corrupt parameter data" ) );
|
||||
}
|
||||
|
||||
//apis
|
||||
|
||||
/*
|
||||
help functions
|
||||
*/
|
||||
|
||||
public function checkAuth() {
|
||||
if(IFMConfig::auth == 1 && $_SESSION['auth'] !== true) {
|
||||
$login_failed = false;
|
||||
if(isset($_POST["user"]) && isset($_POST["pass"])) {
|
||||
if($this->checkCredentials($_POST["user"], $_POST["pass"])) {
|
||||
$_SESSION['auth'] = true;
|
||||
}
|
||||
else {
|
||||
$_SESSION['auth'] = false;
|
||||
$login_failed = true;
|
||||
}
|
||||
}
|
||||
|
||||
if($_SESSION['auth'] !== true) {
|
||||
if(isset($_POST["api"]) && $login_failed === true)
|
||||
echo json_encode(array("status"=>"ERROR", "message"=>"authentication failed"));
|
||||
elseif(isset($_POST["api"]) && $login_failed !== true)
|
||||
echo json_encode(array("status"=>"ERROR", "message"=>"not authenticated"));
|
||||
else
|
||||
$this->loginForm($login_failed);
|
||||
return false;
|
||||
} else {
|
||||
return true;
|
||||
}
|
||||
} else {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
private function checkCredentials($user, $pass) {
|
||||
list($src, $srcopt) = explode(";", IFMConfig::auth_source, 2);
|
||||
switch($src) {
|
||||
case "inline":
|
||||
list($uname, $hash) = explode(":", $srcopt);
|
||||
break;
|
||||
case "file":
|
||||
if(@file_exists($srcopt) && @is_readable($srcopt)) {
|
||||
list($uname, $hash) = explode(":", fgets(fopen($srcopt, 'r')));
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
}
|
||||
return password_verify($pass, trim($hash))?($uname == $user):false;
|
||||
}
|
||||
|
||||
private function loginForm($loginFailed=false) {
|
||||
print '<!DOCTYPE HTML>
|
||||
<html>
|
||||
<head>
|
||||
<title>IFM - improved file manager</title>
|
||||
<meta charset="utf-8">
|
||||
<style type="text/css">
|
||||
* { box-sizing: border-box; font-family: Arial, sans-serif; }
|
||||
html { text-align: center; }
|
||||
body { margin:auto; width: auto; display: inline-block; }
|
||||
form { padding: 1em; border: 1px dashed #CCC; }
|
||||
button { margin-top: 1em; }
|
||||
div.err { color: red; font-weight: bold; margin-bottom: 1em; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<h1>IFM - Login</h1>
|
||||
<form method="post">';
|
||||
if($loginFailed){ print '<div class="err">Login attempt failed. Please try again.</div>'; }
|
||||
print '<label>username:</label> <input type="text" name="user" size="12"><br>
|
||||
<label>password:</label> <input type="password" name="pass" size="12"><br>
|
||||
<button type="submit">login</button>
|
||||
</form>
|
||||
</body>
|
||||
</html>';
|
||||
}
|
||||
|
||||
private function getValidDir($dir) {
|
||||
if( $this->getScriptRoot() != substr( realpath( $dir ), 0, strlen( $this->getScriptRoot() ) ) ) {
|
||||
return "";
|
||||
} else {
|
||||
return ( file_exists( realpath( $dir ) ) ) ? substr( realpath( $dir ), strlen( $this->getScriptRoot() ) + 1 ) : "";
|
||||
}
|
||||
}
|
||||
|
||||
private function filePermsDecode( $perms ) {
|
||||
$oct = str_split( strrev( decoct( $perms ) ), 1 );
|
||||
$masks = array( '---', '--x', '-w-', '-wx', 'r--', 'r-x', 'rw-', 'rwx' );
|
||||
return(
|
||||
sprintf(
|
||||
'%s %s %s',
|
||||
array_key_exists( $oct[ 2 ], $masks ) ? $masks[ $oct[ 2 ] ] : '###',
|
||||
array_key_exists( $oct[ 1 ], $masks ) ? $masks[ $oct[ 1 ] ] : '###',
|
||||
array_key_exists( $oct[ 0 ], $masks ) ? $masks[ $oct[ 0 ] ] : '###')
|
||||
);
|
||||
}
|
||||
|
||||
private function isPathValid($p) {
|
||||
if( $p == "" ) {
|
||||
return true;
|
||||
} elseif( $this->getScriptRoot() == substr( realpath( $p ), 0, strlen( $this->getScriptRoot() ) ) ) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private function getScriptRoot() {
|
||||
return realpath( substr( $_SERVER["SCRIPT_FILENAME"], 0, strrpos( $_SERVER["SCRIPT_FILENAME"], "/" ) ) );
|
||||
}
|
||||
|
||||
private function chDirIfNecessary($d) {
|
||||
if( substr( getcwd(), strlen( $this->getScriptRoot() ) ) != $this->getValidDir($d) ) {
|
||||
chdir( $d );
|
||||
}
|
||||
}
|
||||
|
||||
private function getTypePicture( $type ) {
|
||||
switch( $type ) {
|
||||
case "aac": return "aac.png"; break;case "ai": return "ai.png"; break;case "aiff": return "aiff.png"; break;
|
||||
case "avi": return "avi.png"; break;case "bmp": return "bmp.png"; break;case "c": return "c.png"; break;
|
||||
case "cpp": return "cpp.png"; break;case "css": return "css.png"; break;case "dat": return "dat.png"; break;
|
||||
case "dmg": return "dmg.png"; break;case "doc": return "doc.png"; break;case "dotx": return "dotx.png"; break;
|
||||
case "dwg": return "dwg.png"; break;case "dxf": return "dxf.png"; break;case "eps": return "eps.png"; break;
|
||||
case "exe": return "exe.png"; break;case "flv": return "flv.png"; break;case "gif": return "gif.png"; break;
|
||||
case "h": return "h.png"; break;case "hpp": return "hpp.png"; break;case "html": return "html.png"; break;
|
||||
case "ics": return "ics.png"; break;case "iso": return "iso.png"; break;case "java": return "java.png"; break;
|
||||
case "jpg": return "jpg.png"; break;case "key": return "key.png"; break;case "mid": return "mid.png"; break;
|
||||
case "mp3": return "mp3.png"; break;case "mp4": return "mp4.png"; break;case "mpg": return "mpg.png"; break;
|
||||
case "odf": return "odf.png"; break;case "ods": return "ods.png"; break;case "odt": return "odt.png"; break;
|
||||
case "otp": return "otp.png"; break;case "ots": return "ots.png"; break;case "ott": return "ott.png"; break;
|
||||
case "pdf": return "pdf.png"; break;case "php": return "php.png"; break;case "png": return "png.png"; break;
|
||||
case "ppt": return "ppt.png"; break;case "psd": return "psd.png"; break;case "py": return "py.png"; break;
|
||||
case "qt": return "qt.png"; break;case "rb": return "rb.png"; break;case "rtf": return "rtf.png"; break;
|
||||
case "sql": return "sql.png"; break;case "tga": return "tga.png"; break;case "tgz": return "tgz.png"; break;
|
||||
case "tiff": return "tiff.png"; break;case "txt": return "txt.png"; break;case "wav": return "wav.png"; break;
|
||||
case "xls": return "xls.png"; break;case "xlsx": return "xlsx.png"; break;case "xml": return "xml.png"; break;
|
||||
case "yml": return "yml.png"; break;case "zip": return "zip.png"; break;
|
||||
default: return "_blank.png"; break;
|
||||
}
|
||||
}
|
||||
|
||||
private function getRealpath($dir) {
|
||||
if( $this->getScriptRoot() != substr( realpath( $_POST["dir"] ), 0, strlen( $this->getScriptRoot() ) ) ) {
|
||||
echo json_encode( array( "realpath" => "" ) );
|
||||
} else {
|
||||
$rp = substr( realpath( $_POST["dir"] ), strlen( $this->getScriptRoot() ) + 1 );
|
||||
if( $rp == false ) $rp = "";
|
||||
echo json_encode( array( "realpath" => $rp ) );
|
||||
}
|
||||
}
|
||||
private function rec_rmdir( $path ) {
|
||||
if( !is_dir( $path ) ) {
|
||||
return -1;
|
||||
}
|
||||
$dir = @opendir( $path );
|
||||
if( !$dir ) {
|
||||
return -2;
|
||||
}
|
||||
while( ( $entry = @readdir( $dir ) ) !== false ) {
|
||||
if( $entry == '.' || $entry == '..' ) continue;
|
||||
if( is_dir( $path . '/' . $entry ) ) {
|
||||
$res = rec_rmdir( $path . '/' . $entry );
|
||||
if( $res == -1 ) { @closedir( $dir ); return -2; }
|
||||
else if( $res == -2 ) { @closedir( $dir ); return -2; }
|
||||
else if( $res == -3 ) { @closedir( $dir ); return -3; }
|
||||
else if( $res != 0 ) { @closedir( $dir ); return -2; }
|
||||
} else if( is_file( $path . '/' . $entry ) || is_link( $path . '/' . $entry ) ) {
|
||||
$res = @unlink( $path . '/' . $entry );
|
||||
if( !$res ) { @closedir( $dir ); return -2; }
|
||||
} else { @closedir( $dir ); return -3; }
|
||||
}
|
||||
@closedir( $dir );
|
||||
$res = @rmdir( $path );
|
||||
if( !$res ) { return -2; }
|
||||
return 0;
|
||||
}
|
||||
|
||||
// combines two parts to a valid path
|
||||
private function pathCombine( $a, $b ) {
|
||||
if( $a=="" && $b=="" )
|
||||
return "";
|
||||
else
|
||||
return ltrim( rtrim( $a, '/' ) . '/' . ltrim( $b, '/' ), '/' );
|
||||
}
|
||||
|
||||
// check if filename is allowed
|
||||
private function allowedFileName( $f ) {
|
||||
if( IFMConfig::showhtdocs != 1 && substr( $f, 0, 3 ) == ".ht" )
|
||||
return false;
|
||||
elseif( IFMConfig::showhiddenfiles != 1 && substr( $f, 0, 1 ) == "." )
|
||||
return false;
|
||||
return true;
|
||||
}
|
||||
|
||||
// sorting function for file and dir arrays
|
||||
private function sortByName( $a, $b ) {
|
||||
if( strtolower( $a['name'] ) == strtolower( $b['name'] ) ) return 0;
|
||||
return ( strtolower( $a['name'] ) < strtolower( $b['name'] ) ) ? -1 : 1;
|
||||
}
|
||||
|
||||
// unzip an archive
|
||||
private function unzip( $file ) {
|
||||
$zip = new ZipArchive;
|
||||
$res = $zip->open( $file );
|
||||
if( $res === true ) {
|
||||
$zip->extractTo( './' );
|
||||
$zip->close();
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// is cURL extention avaliable?
|
||||
private function checkCurl() {
|
||||
if( ! function_exists( "curl_init" ) &&
|
||||
!function_exists( "curl_setopt" ) &&
|
||||
!function_exists( "curl_exec" ) &&
|
||||
!function_exists( "curl_close" ) ) return false;
|
||||
else return true;
|
||||
}
|
||||
|
||||
///helper
|
||||
}
|
||||
|
||||
/*
|
||||
start program
|
||||
*/
|
||||
|
||||
$ifm = new IFM();
|
||||
$ifm->run();
|
46
src/style.css
Normal file
46
src/style.css
Normal file
@@ -0,0 +1,46 @@
|
||||
* { -moz-box-sizing: border-box; box-sizing: border-box; } html, body { font-family: Arial, non-serif; font-weight: 100; font-size: 12pt; background: #FFF;}
|
||||
footer { margin-top:20px; margin-bottom:10px; border:1px #CCC dashed; padding:3px; color:#222; font-size:0.6em; }
|
||||
footer a { color:#00E; } a { text-decoration:none; color: #000; cursor: pointer; } a:hover { color: #333; } img { vertical-align:middle; border:0; }
|
||||
button > img { width: 19px; } a > img { width: 25px; } #multiseloptions a > img { width:19px; } button { margin: 1px; margin-top: 10px; padding: 10px; border: 1px solid black; }
|
||||
#tooltab button, #filetable button { margin:1px; padding:1px 4px; background: none; border: 0; } #version { font-size:0.8em; }
|
||||
#mess { position: fixed; top: 2em; left: 50%; z-index: 10; }
|
||||
#mess>div { position:relative; left:-50%;display: inline-block; padding: 25px; background-color: #EEE; border-radius: 10px; box-shadow:#333 3px 3px 10px; }
|
||||
#mess .message_error { background: #ff0000; /* Old browsers */ background: -moz-linear-gradient(top, #ff0000 0%, #ff7777 100%); /* FF3.6+ */
|
||||
background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#ff0000), color-stop(100%,#ff7777)); /* Chrome,Safari4+ */
|
||||
background: -webkit-linear-gradient(top, #ff0000 0%,#ff7777 100%); /* Chrome10+,Safari5.1+ */ background: -o-linear-gradient(top, #ff0000 0%,#ff7777 100%); /* Opera 11.10+ */
|
||||
background: -ms-linear-gradient(top, #ff0000 0%,#ff7777 100%); /* IE10+ */ background: linear-gradient(to bottom, #ff0000 0%,#ff7777 100%); /* W3C */
|
||||
filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#ff0000', endColorstr='#ff7777',GradientType=0 ); /* IE6-9 */ }
|
||||
#mess .message_successful { background: #ff0000; /* Old browsers */ background: -moz-linear-gradient(top, #00dd00 0%, #77ee77 100%); /* FF3.6+ */
|
||||
background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,#00dd00), color-stop(100%,#77ee77)); /* Chrome,Safari4+ */
|
||||
background: -webkit-linear-gradient(top, #00dd00 0%,#77ee77 100%); /* Chrome10+,Safari5.1+ */ background: -o-linear-gradient(top, #00dd00 0%,#77ee77 100%); /* Opera 11.10+ */
|
||||
background: -ms-linear-gradient(top, #00dd00 0%,#77ee77 100%); /* IE10+ */ background: linear-gradient(to bottom, #00dd00 0%,#77ee77 100%); /* W3C */
|
||||
filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#00dd00', endColorstr='#77ee77',GradientType=0 ); /* IE6-9 */ }
|
||||
#tooltab { clear: both; padding: 3px; width: 100%; border-collapse:collapse; margin-bottom:3px; }
|
||||
#tooltab .cell_content { } #tooltab .cell_buttons { text-align: right; width:300px; } #options ul { text-align: left; }
|
||||
#tooltab input[type="text"], textarea { width: 75%; margin-left: 5px; } #tooltab tr { height: 50px; } #currentDir { height: 30px; }
|
||||
td.cell_buttons > div { position: fixed; top: 10px; right: 10px; background: #FFF; padding: 10px; border-radius: 5px; box-shadow: 1px 1px 10px #000; }
|
||||
#options { display: inline-block; position: relative; } #options img { width: 20px; } #options ul { display: none; }
|
||||
#options:hover>ul { display: inline-block; position: absolute; padding: 3px; border: 1px solid #CCC; background-color: #FFF; list-style-type:none; margin:0;
|
||||
min-width:200px; right:0; top:0; } #options ul li:before { content: '\2192\00A0' }
|
||||
#filetable { padding:5px; width:100%; border-collapse: collapse;} #filetable thead { border: 1px solid #00A3A3; }
|
||||
#filetable td { text-align:left; } #filetable tbody { border: 1px solid #EEE; }
|
||||
#filetable td:not(:first-child):last-child { text-align: right; } #filetable td:not(:first-child):not(:last-child) { text-align: center; }
|
||||
#filetable th { background-color: #00A3A3; color: #FFF; padding: 5px; text-align:left; text-align: center; font-weight: normal; font-size:0.9em;}
|
||||
#filetable tr:nth-child(2n) { background-color: #EBEBEB;} #filetable tbody tr.selected { border: 1px dashed #777 !important; background-color: #FFE4C4; }
|
||||
#filetable tbody tr:last-child { border-bottom: 1px solid #EBEBEB; }
|
||||
#filetable input[type^=text] { width:90px; } .download-link { font-size: 0.9em; } .overlay { z-index:5; } .overlay fieldset { border: 0; background-color: white;}
|
||||
.overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; text-align: center; background-color: rgba(0,0,0,0.7);}
|
||||
.overlay form { width: 80%; margin: 3em auto; text-align: left; } .overlay form fieldset { position:relative; }
|
||||
.overlay form#showFile { height:80%; } .overlay form#showFile fieldset { height:100%; } .overlay select { float:right; margin-top:10px; padding:10px; }
|
||||
.overlay div#content { height: calc(100% - 100px); }
|
||||
.overlay input[type=text], .overlay textarea { margin: 2px 0; border: solid 1px #ccc; padding: 0.4em; background-color: #f5f5f5; width: 100%;}
|
||||
#waitqueue{display:inline-block;position:fixed;padding:0;font-size:0.8em;bottom:5px; right:5px;border-radius:5px;background-color:#CCC;box-shadow:1px 1px 10px #000;}
|
||||
#waitqueue>div{padding:5px 10px 5px 5px;}#waitqueue>div>div{width:120px;height:14px;display:inline-block;vertical-align:bottom;} #waitqueue.left { left:5px !important; right:auto; }
|
||||
#multiseloptions { border:1px solid #000; box-shadow:#333 3px 3px 10px; position:fixed; top:25%; left:60%; background:#FFF; padding:0; z-index:1; overflow:hidden; }
|
||||
#multiseloptions p { margin:0; padding:2px; background-color:#DDD; } #multiseloptions ul { margin:0;padding:2px 2px 2px 10px;list-style:none; }
|
||||
#mdPicPreview{position:absolute;border:1px solid #ccc;background:#333;padding:5px;display:none;color:#fff;max-width:30em;}
|
||||
#mdPicPreview img { max-width:100%;}
|
||||
|
||||
#loading { position:fixed; top:calc(50% - 49px); left:calc(50% - 45px); border-radius:5px; border:1px solid #EEE; padding:12px 15px; background:#FFF; }
|
||||
#savequestion { position: absolute; top:40%; left:50%; margin-left:-125px;margin-top:-55px; background-color:#FFF; border: 1px solid #CCC; z-index:10; padding:20px; text-align: center; }
|
||||
#loadingAnim{position:relative;width:30px;height:36px}.blockG{position:absolute;background-color:#FFF;width:5px;height:11px;-moz-border-radius:4px 4px 0 0;-moz-transform:scale(0.4);-moz-animation-name:fadeG;-moz-animation-duration:1.04s;-moz-animation-iteration-count:infinite;-moz-animation-direction:linear;-webkit-border-radius:4px 4px 0 0;-webkit-transform:scale(0.4);-webkit-animation-name:fadeG;-webkit-animation-duration:1.04s;-webkit-animation-iteration-count:infinite;-webkit-animation-direction:linear;-ms-border-radius:4px 4px 0 0;-ms-transform:scale(0.4);-ms-animation-name:fadeG;-ms-animation-duration:1.04s;-ms-animation-iteration-count:infinite;-ms-animation-direction:linear;-o-border-radius:4px 4px 0 0;-o-transform:scale(0.4);-o-animation-name:fadeG;-o-animation-duration:1.04s;-o-animation-iteration-count:infinite;-o-animation-direction:linear;border-radius:4px 4px 0 0;transform:scale(0.4);animation-name:fadeG;animation-duration:1.04s;animation-iteration-count:infinite;animation-direction:linear}#rotateG_01{left:0;top:13px;-moz-animation-delay:.39s;-moz-transform:rotate(-90deg);-webkit-animation-delay:.39s;-webkit-transform:rotate(-90deg);-ms-animation-delay:.39s;-ms-transform:rotate(-90deg);-o-animation-delay:.39s;-o-transform:rotate(-90deg);animation-delay:.39s;transform:rotate(-90deg)}#rotateG_02{left:4px;top:5px;-moz-animation-delay:.52s;-moz-transform:rotate(-45deg);-webkit-animation-delay:.52s;-webkit-transform:rotate(-45deg);-ms-animation-delay:.52s;-ms-transform:rotate(-45deg);-o-animation-delay:.52s;-o-transform:rotate(-45deg);animation-delay:.52s;transform:rotate(-45deg)}#rotateG_03{left:13px;top:1px;-moz-animation-delay:.65s;-moz-transform:rotate(0deg);-webkit-animation-delay:.65s;-webkit-transform:rotate(0deg);-ms-animation-delay:.65s;-ms-transform:rotate(0deg);-o-animation-delay:.65s;-o-transform:rotate(0deg);animation-delay:.65s;transform:rotate(0deg)}#rotateG_04{right:4px;top:5px;-moz-animation-delay:.78s;-moz-transform:rotate(45deg);-webkit-animation-delay:.78s;-webkit-transform:rotate(45deg);-ms-animation-delay:.78s;-ms-transform:rotate(45deg);-o-animation-delay:.78s;-o-transform:rotate(45deg);animation-delay:.78s;transform:rotate(45deg)}#rotateG_05{right:0;top:13px;-moz-animation-delay:.9099999999999999s;-moz-transform:rotate(90deg);-webkit-animation-delay:.9099999999999999s;-webkit-transform:rotate(90deg);-ms-animation-delay:.9099999999999999s;-ms-transform:rotate(90deg);-o-animation-delay:.9099999999999999s;-o-transform:rotate(90deg);animation-delay:.9099999999999999s;transform:rotate(90deg)}#rotateG_06{right:4px;bottom:3px;-moz-animation-delay:1.04s;-moz-transform:rotate(135deg);-webkit-animation-delay:1.04s;-webkit-transform:rotate(135deg);-ms-animation-delay:1.04s;-ms-transform:rotate(135deg);-o-animation-delay:1.04s;-o-transform:rotate(135deg);animation-delay:1.04s;transform:rotate(135deg)}#rotateG_07{bottom:0;left:13px;-moz-animation-delay:1.1700000000000002s;-moz-transform:rotate(180deg);-webkit-animation-delay:1.1700000000000002s;-webkit-transform:rotate(180deg);-ms-animation-delay:1.1700000000000002s;-ms-transform:rotate(180deg);-o-animation-delay:1.1700000000000002s;-o-transform:rotate(180deg);animation-delay:1.1700000000000002s;transform:rotate(180deg)}#rotateG_08{left:4px;bottom:3px;-moz-animation-delay:1.3s;-moz-transform:rotate(-135deg);-webkit-animation-delay:1.3s;-webkit-transform:rotate(-135deg);-ms-animation-delay:1.3s;-ms-transform:rotate(-135deg);-o-animation-delay:1.3s;-o-transform:rotate(-135deg);animation-delay:1.3s;transform:rotate(-135deg)}@-moz-keyframes fadeG{0%{background-color:#000}100%{background-color:#FFF}}@-webkit-keyframes fadeG{0%{background-color:#000}100%{background-color:#FFF}}@-ms-keyframes fadeG{0%{background-color:#000}100%{background-color:#FFF}}@-o-keyframes fadeG{0%{background-color:#000}100%{background-color:#FFF}}@keyframes fadeG{0%{background-color:#000}100%{background-color:#FFF}}
|
400
src/zip_file.php
Normal file
400
src/zip_file.php
Normal file
@@ -0,0 +1,400 @@
|
||||
<?php
|
||||
|
||||
// zip class (for zipnload)
|
||||
class archive {
|
||||
function __construct($name) {
|
||||
$this->options = array(
|
||||
'basedir' => ".",
|
||||
'name' => $name,
|
||||
'prepend' => "",
|
||||
'inmemory' => 0,
|
||||
'overwrite' => 0,
|
||||
'recurse' => 1,
|
||||
'storepaths' => 1,
|
||||
'followlinks' => 0,
|
||||
'level' => 3,
|
||||
'method' => 1,
|
||||
'sfx' => "",
|
||||
'type' => "",
|
||||
'comment' => "",
|
||||
'maxsize' => memory_get_usage()
|
||||
);
|
||||
$this->files = array();
|
||||
$this->exclude = array();
|
||||
$this->exclude_regexp = array();
|
||||
$this->storeonly = array();
|
||||
$this->error = array();
|
||||
}
|
||||
function set_options($options) {
|
||||
foreach ($options as $key => $value)
|
||||
$this->options[$key] = $value;
|
||||
if (!empty($this->options['basedir'])) {
|
||||
$this->options['basedir'] = str_replace("\\", "/", $this->options['basedir']);
|
||||
$this->options['basedir'] = preg_replace("/\/+/", "/", $this->options['basedir']);
|
||||
$this->options['basedir'] = preg_replace("/\/$/", "", $this->options['basedir']);
|
||||
}
|
||||
if (!empty($this->options['name'])) {
|
||||
$this->options['name'] = str_replace("\\", "/", $this->options['name']);
|
||||
$this->options['name'] = preg_replace("/\/+/", "/", $this->options['name']);
|
||||
}
|
||||
if (!empty($this->options['prepend'])) {
|
||||
$this->options['prepend'] = str_replace("\\", "/", $this->options['prepend']);
|
||||
$this->options['prepend'] = preg_replace("/^(\.*\/+)+/", "", $this->options['prepend']);
|
||||
$this->options['prepend'] = preg_replace("/\/+/", "/", $this->options['prepend']);
|
||||
$this->options['prepend'] = preg_replace("/\/$/", "", $this->options['prepend']) . "/";
|
||||
}
|
||||
}
|
||||
function create_archive() {
|
||||
$this->make_list();
|
||||
if ($this->options['inmemory'] == 0) {
|
||||
$pwd = getcwd();
|
||||
chdir($this->options['basedir']);
|
||||
if ($this->options['overwrite'] == 0 && file_exists($this->options['name'] . ($this->options['type'] == "gzip" || $this->options['type'] == "bzip" ? ".tmp" : ""))) {
|
||||
$this->error[] = "File {$this->options['name']} already exists.";
|
||||
chdir($pwd);
|
||||
return 0;
|
||||
} else
|
||||
if ($this->archive = @fopen($this->options['name'] . ($this->options['type'] == "gzip" || $this->options['type'] == "bzip" ? ".tmp" : ""), "wb+"))
|
||||
chdir($pwd);
|
||||
else {
|
||||
$this->error[] = "Could not open {$this->options['name']} for writing.";
|
||||
chdir($pwd);
|
||||
return 0;
|
||||
}
|
||||
} else
|
||||
$this->archive = "";
|
||||
switch ($this->options['type']) {
|
||||
case "zip" :
|
||||
if (!$this->create_zip()) {
|
||||
$this->error[] = "Could not create zip file.";
|
||||
return 0;
|
||||
}
|
||||
break;
|
||||
case "bzip" :
|
||||
if (!$this->create_tar()) {
|
||||
$this->error[] = "Could not create tar file.";
|
||||
return 0;
|
||||
}
|
||||
if (!$this->create_bzip()) {
|
||||
$this->error[] = "Could not create bzip2 file.";
|
||||
return 0;
|
||||
}
|
||||
break;
|
||||
case "gzip" :
|
||||
if (!$this->create_tar()) {
|
||||
$this->error[] = "Could not create tar file.";
|
||||
return 0;
|
||||
}
|
||||
if (!$this->create_gzip()) {
|
||||
$this->error[] = "Could not create gzip file.";
|
||||
return 0;
|
||||
}
|
||||
break;
|
||||
case "tar" :
|
||||
if (!$this->create_tar()) {
|
||||
$this->error[] = "Could not create tar file.";
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
if ($this->options['inmemory'] == 0) {
|
||||
fclose($this->archive);
|
||||
if ($this->options['type'] == "gzip" || $this->options['type'] == "bzip")
|
||||
unlink($this->options['basedir'] . "/" . $this->options['name'] . ".tmp");
|
||||
}
|
||||
}
|
||||
function add_data($data) {
|
||||
if ($this->options['inmemory'] == 0)
|
||||
fwrite($this->archive, $data);
|
||||
else
|
||||
$this->archive .= $data;
|
||||
}
|
||||
function exclude_file($name) {
|
||||
$ret = false;
|
||||
// Check for exclusion regexps
|
||||
if (!empty($this->exclude_regexp)) {
|
||||
foreach ($this->exclude_regexp as $pattern) {
|
||||
if (preg_match('|^' . $pattern . '$|', $name)) {
|
||||
$ret = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
if ($this->options['maxsize'] > 0) {
|
||||
if (filesize($this->options['basedir'] . "/" . $name) > $this->options['maxsize']) {
|
||||
$ret = true;
|
||||
}
|
||||
}
|
||||
return $ret;
|
||||
}
|
||||
function make_list() {
|
||||
foreach ($this->files as $key => $value) {
|
||||
if ($this->exclude_file($value['name'])) {
|
||||
unset($this->files[$key]);
|
||||
}
|
||||
}
|
||||
if (!empty($this->exclude)) {
|
||||
foreach ($this->files as $key => $value) {
|
||||
foreach ($this->exclude as $current) {
|
||||
if ($value['name'] == $current['name']) {
|
||||
unset($this->files[$key]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!empty($this->storeonly))
|
||||
foreach ($this->files as $key => $value)
|
||||
foreach ($this->storeonly as $current)
|
||||
if ($value['name'] == $current['name'])
|
||||
$this->files[$key]['method'] = 0;
|
||||
unset($this->exclude, $this->storeonly);
|
||||
}
|
||||
function add_files($list) {
|
||||
$temp = $this->list_files($list);
|
||||
foreach ($temp as $current)
|
||||
$this->files[] = $current;
|
||||
}
|
||||
function exclude_regexp_files($patterns) {
|
||||
$this->exclude_regexp = explode("|", $patterns);
|
||||
}
|
||||
function exclude_files($list) {
|
||||
$temp = $this->list_files($list);
|
||||
foreach ($temp as $current)
|
||||
$this->exclude[] = $current;
|
||||
}
|
||||
function store_files($list) {
|
||||
$temp = $this->list_files($list);
|
||||
foreach ($temp as $current)
|
||||
$this->storeonly[] = $current;
|
||||
}
|
||||
function list_files($list) {
|
||||
if (!is_array($list)) {
|
||||
$temp = $list;
|
||||
$list = array(
|
||||
$temp
|
||||
);
|
||||
unset($temp);
|
||||
}
|
||||
$files = array();
|
||||
$pwd = getcwd();
|
||||
chdir($this->options['basedir']);
|
||||
foreach ($list as $current) {
|
||||
$current = str_replace("\\", "/", $current);
|
||||
$current = preg_replace("/\/+/", "/", $current);
|
||||
$current = preg_replace("/\/$/", "", $current);
|
||||
if (strstr($current, "*")) {
|
||||
$regex = preg_replace("/([\\\^\$\.\[\]\|\(\)\?\+\{\}\/])/", "\\\\\\1", $current);
|
||||
$regex = str_replace("*", ".*", $regex);
|
||||
$dir = strstr($current, "/") ? substr($current, 0, strrpos($current, "/")) : ".";
|
||||
$temp = $this->parse_dir($dir);
|
||||
//print "dir: ".$dir; print_r($temp);print "EOL<br>\n";
|
||||
foreach ($temp as $current2) {
|
||||
//print "$regex : ".$current2['name']." <br>\n";
|
||||
if (preg_match("/^{$regex}$/i", $current2['name']))
|
||||
$files[] = $current2;
|
||||
}
|
||||
unset($regex, $dir, $temp, $current);
|
||||
} else
|
||||
if (@ is_dir($current)) {
|
||||
$temp = $this->parse_dir($current);
|
||||
foreach ($temp as $file)
|
||||
$files[] = $file;
|
||||
unset($temp, $file);
|
||||
} else
|
||||
if (@ file_exists($current))
|
||||
$files[] = array(
|
||||
'name' => $current,
|
||||
'name2' => $this->options['prepend'] .
|
||||
preg_replace("/(\.+\/+)+/", "", ($this->options['storepaths'] == 0 && strstr($current, "/")) ? substr($current, strrpos($current, "/") + 1) : $current),
|
||||
'type' => @ is_link($current) && $this->options['followlinks'] == 0 ? 2 : 0,
|
||||
'ext' => substr($current, strrpos($current, ".")),
|
||||
'stat' => stat($current)
|
||||
);
|
||||
}
|
||||
chdir($pwd);
|
||||
unset($current, $pwd);
|
||||
usort($files, array(
|
||||
"archive",
|
||||
"sort_files"
|
||||
));
|
||||
return $files;
|
||||
}
|
||||
function parse_dir($dirname) {
|
||||
if ($this->options['storepaths'] == 1 && !preg_match("/^(\.+\/*)+$/", $dirname))
|
||||
$files = array(
|
||||
array(
|
||||
'name' => $dirname,
|
||||
'name2' => $this->options['prepend'] .
|
||||
preg_replace("/(\.+\/+)+/", "", ($this->options['storepaths'] == 0 && strstr($dirname, "/")) ? substr($dirname, strrpos($dirname, "/") + 1) : $dirname),
|
||||
'type' => 5,
|
||||
'stat' => stat($dirname)
|
||||
)
|
||||
);
|
||||
else
|
||||
$files= array();
|
||||
$dir = @ opendir($dirname);
|
||||
while ($file = @ readdir($dir)) {
|
||||
$fullname = $dirname . "/" . $file;
|
||||
if ($file == "." || $file == "..")
|
||||
continue;
|
||||
else
|
||||
if (@ is_dir($fullname)) {
|
||||
if (empty($this->options['recurse']))
|
||||
continue;
|
||||
$temp = $this->parse_dir($fullname);
|
||||
foreach ($temp as $file2)
|
||||
$files[] = $file2;
|
||||
} else
|
||||
if (@ file_exists($fullname))
|
||||
$files[] = array(
|
||||
'name' => $fullname,
|
||||
'name2' => $this->options['prepend'] .
|
||||
preg_replace("/(\.+\/+)+/", "", ($this->options['storepaths'] == 0 && strstr($fullname, "/")) ? substr($fullname, strrpos($fullname, "/") + 1) : $fullname),
|
||||
'type' => @ is_link($fullname) && $this->options['followlinks'] == 0 ? 2 : 0,
|
||||
'ext' => substr($file, strrpos($file, ".")),
|
||||
'stat' => stat($fullname)
|
||||
);
|
||||
}
|
||||
@ closedir($dir);
|
||||
return $files;
|
||||
}
|
||||
function sort_files($a, $b) {
|
||||
if ($a['type'] != $b['type'])
|
||||
if ($a['type'] == 5 || $b['type'] == 2)
|
||||
return -1;
|
||||
else
|
||||
if ($a['type'] == 2 || $b['type'] == 5)
|
||||
return 1;
|
||||
else
|
||||
if ($a['type'] == 5)
|
||||
return strcmp(strtolower($a['name']), strtolower($b['name']));
|
||||
else
|
||||
if ($a['ext'] != $b['ext'])
|
||||
return strcmp($a['ext'], $b['ext']);
|
||||
else
|
||||
if ($a['stat'][7] != $b['stat'][7])
|
||||
return $a['stat'][7] > $b['stat'][7] ? -1 : 1;
|
||||
else
|
||||
return strcmp(strtolower($a['name']), strtolower($b['name']));
|
||||
return 0;
|
||||
}
|
||||
function download_file() {
|
||||
if ($this->options['inmemory'] == 0) {
|
||||
$this->error[] = "Can only use download_file() if archive is in memory. Redirect to file otherwise, it is faster.";
|
||||
return;
|
||||
}
|
||||
switch ($this->options['type']) {
|
||||
case "zip" :
|
||||
header("Content-Type: application/zip");
|
||||
break;
|
||||
case "bzip" :
|
||||
header("Content-Type: application/x-bzip2");
|
||||
break;
|
||||
case "gzip" :
|
||||
header("Content-Type: application/x-gzip");
|
||||
break;
|
||||
case "tar" :
|
||||
header("Content-Type: application/x-tar");
|
||||
}
|
||||
$header = "Content-Disposition: attachment; filename=\"";
|
||||
$header .= strstr($this->options['name'], "/") ? substr($this->options['name'], strrpos($this->options['name'], "/") + 1) : $this->options['name'];
|
||||
$header .= "\"";
|
||||
header($header);
|
||||
header("Content-Length: " . strlen($this->archive));
|
||||
header("Content-Transfer-Encoding: binary");
|
||||
header("Cache-Control: no-cache, must-revalidate, max-age=60");
|
||||
header("Expires: Sat, 01 Jan 2000 12:00:00 GMT");
|
||||
print ($this->archive);
|
||||
}
|
||||
function save_file($filename, $path = '') {
|
||||
if ($this->options['inmemory'] == 0) {
|
||||
$this->error[] = "Can only use save_file() if archive is in memory.";
|
||||
return;
|
||||
}
|
||||
if ($fhandle = @fopen($path . $filename, 'w')) {
|
||||
if (@fwrite($fhandle, $this->archive)) {
|
||||
if (@fclose($fhandle))
|
||||
return 1;
|
||||
else
|
||||
return 0;
|
||||
}else {
|
||||
return -2;
|
||||
}
|
||||
} else {
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
class zip_file extends archive {
|
||||
function __construct($name) {
|
||||
$this->archive($name);
|
||||
$this->options['type'] = "zip";
|
||||
}
|
||||
function create_zip() {
|
||||
$files = 0;
|
||||
$offset = 0;
|
||||
$central = "";
|
||||
if (!empty($this->options['sfx']))
|
||||
if ($fp = @ fopen($this->options['sfx'], "rb")) {
|
||||
$temp = fread($fp, filesize($this->options['sfx']));
|
||||
fclose($fp);
|
||||
$this->add_data($temp);
|
||||
$offset += strlen($temp);
|
||||
unset($temp);
|
||||
} else
|
||||
$this->error[] = "Could not open sfx module from {$this->options['sfx']}.";
|
||||
$pwd = getcwd();
|
||||
chdir($this->options['basedir']);
|
||||
foreach ($this->files as $current) {
|
||||
if ($current['name'] == $this->options['name'])
|
||||
continue;
|
||||
$timedate = explode(" ", date("Y n j G i s", $current['stat'][9]));
|
||||
$timedate = ($timedate[0] - 1980 << 25) | ($timedate[1] << 21) | ($timedate[2] << 16) | ($timedate[3] << 11) | ($timedate[4] << 5) | ($timedate[5]);
|
||||
$block = pack("VvvvV", 0x04034b50, 0x000A, 0x0000, (isset($current['method']) || $this->options['method'] == 0) ? 0x0000 : 0x0008, $timedate);
|
||||
//if ($current['stat'][7] == 0 && $current['type'] == 5) {
|
||||
if ($current['type'] == 5) {
|
||||
$block .= pack("VVVvv", 0x00000000, 0x00000000, 0x00000000, strlen($current['name2']) + 1, 0x0000);
|
||||
$block .= $current['name2'] . "/";
|
||||
$this->add_data($block);
|
||||
$central .= pack("VvvvvVVVVvvvvvVV", 0x02014b50, 0x0014, $this->options['method'] == 0 ? 0x0000 : 0x000A, 0x0000, (isset($current['method']) || $this->options['method'] == 0) ? 0x0000 : 0x0008, $timedate, 0x00000000, 0x00000000, 0x00000000, strlen($current['name2']) + 1, 0x0000, 0x0000, 0x0000, 0x0000, $current['type'] == 5 ? 0x00000010 : 0x00000000, $offset);
|
||||
$central .= $current['name2'] . "/";
|
||||
$files++;
|
||||
$offset += ( 31 + strlen($current['name2']));
|
||||
} else
|
||||
if ($current['stat'][7] == 0) {
|
||||
$block .= pack("VVVvv", 0x00000000, 0x00000000, 0x00000000, strlen($current['name2']), 0x0000);
|
||||
$block .= $current['name2'];
|
||||
$this->add_data($block);
|
||||
$central .= pack("VvvvvVVVVvvvvvVV", 0x02014b50, 0x0014, $this->options['method'] == 0 ? 0x0000 : 0x000A, 0x0000, (isset($current['method']) || $this->options['method'] == 0) ? 0x0000 : 0x0008, $timedate, 0x00000000, 0x00000000, 0x00000000, strlen($current['name2']), 0x0000, 0x0000, 0x0000, 0x0000, $current['type'] == 5 ? 0x00000010 : 0x00000000, $offset);
|
||||
$central .= $current['name2'];
|
||||
$files++;
|
||||
$offset += ( 30 + strlen($current['name2']));
|
||||
} else
|
||||
if ($fp = @ fopen($current['name'], "rb")) {
|
||||
$temp = fread($fp, $current['stat'][7]);
|
||||
fclose($fp);
|
||||
$crc32 = crc32($temp);
|
||||
if (!isset($current['method']) && $this->options['method'] == 1) {
|
||||
$temp = gzcompress($temp, $this->options['level']);
|
||||
$size = strlen($temp) - 6;
|
||||
$temp = substr($temp, 2, $size);
|
||||
} else
|
||||
$size= strlen($temp);
|
||||
$block .= pack("VVVvv", $crc32, $size, $current['stat'][7], strlen($current['name2']), 0x0000);
|
||||
$block .= $current['name2'];
|
||||
$this->add_data($block);
|
||||
$this->add_data($temp);
|
||||
unset($temp);
|
||||
$central .= pack("VvvvvVVVVvvvvvVV", 0x02014b50, 0x0014, $this->options['method'] == 0 ? 0x0000 : 0x000A, 0x0000, (isset($current['method']) || $this->options['method'] == 0) ? 0x0000 : 0x0008, $timedate, $crc32, $size, $current['stat'][7], strlen($current['name2']), 0x0000, 0x0000, 0x0000, 0x0000, 0x00000000, $offset);
|
||||
$central .= $current['name2'];
|
||||
$files++;
|
||||
$offset += ( 30 + strlen($current['name2']) + $size);
|
||||
} else
|
||||
$this->error[] = "Could not open file {$current['name']} for reading. It was not added.";
|
||||
}
|
||||
$this->add_data($central);
|
||||
$this->add_data(pack("VvvvvVVv", 0x06054b50, 0x0000, 0x0000, $files, $files, strlen($central), $offset, !empty($this->options['comment']) ? strlen($this->options['comment']) : 0x0000));
|
||||
if (!empty($this->options['comment']))
|
||||
$this->add_data($this->options['comment']);
|
||||
chdir($pwd);
|
||||
return 1;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user