2011-07-21 21:13:58 +02:00
|
|
|
/**
|
|
|
|
* Handles the import requests
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2011-08-11 16:26:41 +02:00
|
|
|
* 2011 Peter 'Pita' Martischka (Primary Technology Ltd)
|
2012-11-27 00:11:45 +01:00
|
|
|
* 2012 Iván Eixarch
|
2014-12-30 00:12:26 +01:00
|
|
|
* 2014 John McLear (Etherpad Foundation / McLear Ltd)
|
2011-07-21 21:13:58 +02:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS-IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2012-11-23 19:06:18 +01:00
|
|
|
var ERR = require("async-stacktrace")
|
|
|
|
, padManager = require("../db/PadManager")
|
|
|
|
, padMessageHandler = require("./PadMessageHandler")
|
|
|
|
, async = require("async")
|
|
|
|
, fs = require("fs")
|
2012-11-23 22:55:25 +01:00
|
|
|
, path = require("path")
|
2012-11-23 19:06:18 +01:00
|
|
|
, settings = require('../utils/Settings')
|
|
|
|
, formidable = require('formidable')
|
|
|
|
, os = require("os")
|
2013-09-27 15:47:34 +02:00
|
|
|
, importHtml = require("../utils/ImportHtml")
|
2014-12-29 20:57:58 +01:00
|
|
|
, importEtherpad = require("../utils/ImportEtherpad")
|
2013-11-27 01:23:11 +01:00
|
|
|
, log4js = require("log4js")
|
|
|
|
, hooks = require("ep_etherpad-lite/static/js/pluginfw/hooks.js");
|
2012-11-23 18:18:03 +01:00
|
|
|
|
2018-03-08 12:55:00 +01:00
|
|
|
var convertor = null;
|
|
|
|
var exportExtension = "htm";
|
|
|
|
|
|
|
|
//load abiword only if its enabled and if soffice is disabled
|
|
|
|
if(settings.abiword != null && settings.soffice === null)
|
|
|
|
convertor = require("../utils/Abiword");
|
|
|
|
|
|
|
|
//load soffice only if its enabled
|
|
|
|
if(settings.soffice != null) {
|
|
|
|
convertor = require("../utils/LibreOffice");
|
|
|
|
exportExtension = "html";
|
|
|
|
}
|
2012-12-06 11:49:04 +01:00
|
|
|
|
|
|
|
//for node 0.6 compatibily, os.tmpDir() only works from 0.8
|
|
|
|
var tmpDirectory = process.env.TEMP || process.env.TMPDIR || process.env.TMP || '/tmp';
|
2011-07-21 21:13:58 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* do a requested import
|
|
|
|
*/
|
|
|
|
exports.doImport = function(req, res, padId)
|
|
|
|
{
|
2013-09-27 15:47:34 +02:00
|
|
|
var apiLogger = log4js.getLogger("ImportHandler");
|
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
//pipe to a file
|
2018-03-08 12:55:00 +01:00
|
|
|
//convert file to html via abiword or soffice
|
2012-11-23 19:06:18 +01:00
|
|
|
//set html in the pad
|
2011-07-21 21:13:58 +02:00
|
|
|
|
2012-11-23 19:06:18 +01:00
|
|
|
var srcFile, destFile
|
2012-11-23 22:55:25 +01:00
|
|
|
, pad
|
2013-11-27 01:23:11 +01:00
|
|
|
, text
|
2014-12-29 20:57:58 +01:00
|
|
|
, importHandledByPlugin
|
2017-07-28 14:58:23 +02:00
|
|
|
, directDatabaseAccess
|
2018-03-08 12:55:00 +01:00
|
|
|
, useConvertor;
|
2013-11-27 01:23:11 +01:00
|
|
|
|
|
|
|
var randNum = Math.floor(Math.random()*0xFFFFFFFF);
|
2011-07-21 21:13:58 +02:00
|
|
|
|
2018-03-08 12:55:00 +01:00
|
|
|
// setting flag for whether to use convertor or not
|
|
|
|
useConvertor = (convertor != null);
|
2017-07-28 14:58:23 +02:00
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
async.series([
|
|
|
|
//save the uploaded file to /tmp
|
2012-11-23 19:06:18 +01:00
|
|
|
function(callback) {
|
2011-07-21 21:13:58 +02:00
|
|
|
var form = new formidable.IncomingForm();
|
|
|
|
form.keepExtensions = true;
|
2012-12-06 11:49:04 +01:00
|
|
|
form.uploadDir = tmpDirectory;
|
2011-07-21 21:13:58 +02:00
|
|
|
|
2012-11-23 19:06:18 +01:00
|
|
|
form.parse(req, function(err, fields, files) {
|
2011-10-19 21:46:28 +02:00
|
|
|
//the upload failed, stop at this point
|
2012-11-23 19:06:18 +01:00
|
|
|
if(err || files.file === undefined) {
|
2013-03-19 18:40:39 +01:00
|
|
|
if(err) console.warn("Uploading Error: " + err.stack);
|
2011-10-19 21:46:28 +02:00
|
|
|
callback("uploadFailed");
|
|
|
|
}
|
|
|
|
//everything ok, continue
|
2012-11-23 19:06:18 +01:00
|
|
|
else {
|
2011-10-19 21:46:28 +02:00
|
|
|
//save the path of the uploaded file
|
|
|
|
srcFile = files.file.path;
|
|
|
|
callback();
|
|
|
|
}
|
2011-07-21 21:13:58 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2011-07-24 22:07:42 +02:00
|
|
|
//ensure this is a file ending we know, else we change the file ending to .txt
|
|
|
|
//this allows us to accept source code files like .c or .java
|
2012-11-23 19:06:18 +01:00
|
|
|
function(callback) {
|
2012-11-23 22:55:25 +01:00
|
|
|
var fileEnding = path.extname(srcFile).toLowerCase()
|
2017-08-03 23:53:12 +02:00
|
|
|
, knownFileEndings = [".txt", ".doc", ".docx", ".pdf", ".odt", ".html", ".htm", ".etherpad", ".rtf"]
|
2012-11-23 22:55:25 +01:00
|
|
|
, fileEndingKnown = (knownFileEndings.indexOf(fileEnding) > -1);
|
2011-07-24 22:07:42 +02:00
|
|
|
|
|
|
|
//if the file ending is known, continue as normal
|
2012-11-23 19:06:18 +01:00
|
|
|
if(fileEndingKnown) {
|
2011-07-24 22:07:42 +02:00
|
|
|
callback();
|
|
|
|
}
|
|
|
|
//we need to rename this file with a .txt ending
|
2012-11-23 19:06:18 +01:00
|
|
|
else {
|
2014-07-22 16:46:31 +02:00
|
|
|
if(settings.allowUnknownFileEnds === true){
|
|
|
|
var oldSrcFile = srcFile;
|
|
|
|
srcFile = path.join(path.dirname(srcFile),path.basename(srcFile, fileEnding)+".txt");
|
|
|
|
fs.rename(oldSrcFile, srcFile, callback);
|
|
|
|
}else{
|
|
|
|
console.warn("Not allowing unknown file type to be imported", fileEnding);
|
|
|
|
callback("uploadFailed");
|
|
|
|
}
|
2011-07-24 22:07:42 +02:00
|
|
|
}
|
|
|
|
},
|
2013-11-27 01:23:11 +01:00
|
|
|
function(callback){
|
2018-03-08 12:55:00 +01:00
|
|
|
destFile = path.join(tmpDirectory, "etherpad_import_" + randNum + "." + exportExtension);
|
2012-11-23 22:55:25 +01:00
|
|
|
|
2013-11-27 01:23:11 +01:00
|
|
|
// Logic for allowing external Import Plugins
|
|
|
|
hooks.aCallAll("import", {srcFile: srcFile, destFile: destFile}, function(err, result){
|
|
|
|
if(ERR(err, callback)) return callback();
|
|
|
|
if(result.length > 0){ // This feels hacky and wrong..
|
|
|
|
importHandledByPlugin = true;
|
|
|
|
}
|
2015-04-11 18:55:36 +02:00
|
|
|
callback();
|
2013-11-27 01:23:11 +01:00
|
|
|
});
|
|
|
|
},
|
2014-12-29 20:57:58 +01:00
|
|
|
function(callback) {
|
|
|
|
var fileEnding = path.extname(srcFile).toLowerCase()
|
2018-10-31 23:00:45 +01:00
|
|
|
var fileIsNotEtherpad = (fileEnding !== ".etherpad");
|
2014-12-29 22:02:24 +01:00
|
|
|
|
2018-10-31 23:00:45 +01:00
|
|
|
if (fileIsNotEtherpad) {
|
|
|
|
callback();
|
2018-10-31 22:55:58 +01:00
|
|
|
|
2018-10-31 23:00:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// we do this here so we can see if the pad has quit ea few edits
|
|
|
|
padManager.getPad(padId, function(err, _pad){
|
|
|
|
var headCount = _pad.head;
|
|
|
|
if(headCount >= 10){
|
|
|
|
apiLogger.warn("Direct database Import attempt of a pad that already has content, we wont be doing this")
|
|
|
|
return callback("padHasData");
|
|
|
|
}
|
|
|
|
|
|
|
|
fs.readFile(srcFile, "utf8", function(err, _text){
|
|
|
|
directDatabaseAccess = true;
|
|
|
|
importEtherpad.setPadRaw(padId, _text, function(err){
|
|
|
|
callback();
|
2018-10-31 22:55:58 +01:00
|
|
|
});
|
2014-12-29 20:57:58 +01:00
|
|
|
});
|
2018-10-31 23:00:45 +01:00
|
|
|
});
|
2014-12-29 20:57:58 +01:00
|
|
|
},
|
2013-11-27 01:23:11 +01:00
|
|
|
//convert file to html
|
|
|
|
function(callback) {
|
2018-10-31 23:09:27 +01:00
|
|
|
if (importHandledByPlugin || directDatabaseAccess) {
|
2013-11-27 01:23:11 +01:00
|
|
|
callback();
|
2018-10-31 23:09:27 +01:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var fileEnding = path.extname(srcFile).toLowerCase();
|
|
|
|
var fileIsHTML = (fileEnding === ".html" || fileEnding === ".htm");
|
|
|
|
var fileIsTXT = (fileEnding === ".txt");
|
|
|
|
if (fileIsTXT) useConvertor = false; // Don't use convertor for text files
|
|
|
|
// See https://github.com/ether/etherpad-lite/issues/2572
|
|
|
|
if (useConvertor && !fileIsHTML) {
|
|
|
|
convertor.convertFile(srcFile, destFile, exportExtension, function(err) {
|
|
|
|
//catch convert errors
|
|
|
|
if(err) {
|
|
|
|
console.warn("Converting Error:", err);
|
|
|
|
return callback("convertFailed");
|
|
|
|
} else {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// if no convertor only rename
|
|
|
|
fs.rename(srcFile, destFile, callback);
|
2012-11-23 22:55:25 +01:00
|
|
|
}
|
2011-07-21 21:13:58 +02:00
|
|
|
},
|
|
|
|
|
2013-09-10 21:58:28 +02:00
|
|
|
function(callback) {
|
2018-03-08 12:55:00 +01:00
|
|
|
if (!useConvertor && !directDatabaseAccess){
|
2015-04-11 18:55:36 +02:00
|
|
|
// Read the file with no encoding for raw buffer access.
|
|
|
|
fs.readFile(destFile, function(err, buf) {
|
|
|
|
if (err) throw err;
|
|
|
|
var isAscii = true;
|
|
|
|
// Check if there are only ascii chars in the uploaded file
|
|
|
|
for (var i=0, len=buf.length; i<len; i++) {
|
|
|
|
if (buf[i] > 240) {
|
|
|
|
isAscii=false;
|
|
|
|
break;
|
2014-12-29 21:35:10 +01:00
|
|
|
}
|
2015-04-11 18:55:36 +02:00
|
|
|
}
|
|
|
|
if (isAscii) {
|
|
|
|
callback();
|
|
|
|
} else {
|
|
|
|
callback("uploadFailed");
|
|
|
|
}
|
|
|
|
});
|
2013-09-10 21:58:28 +02:00
|
|
|
} else {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
//get the pad object
|
2012-11-23 19:06:18 +01:00
|
|
|
function(callback) {
|
|
|
|
padManager.getPad(padId, function(err, _pad){
|
2011-12-04 16:50:02 +01:00
|
|
|
if(ERR(err, callback)) return;
|
2011-07-21 21:13:58 +02:00
|
|
|
pad = _pad;
|
2011-12-04 16:50:02 +01:00
|
|
|
callback();
|
2011-07-21 21:13:58 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
//read the text
|
2012-11-23 19:06:18 +01:00
|
|
|
function(callback) {
|
2014-12-29 20:57:58 +01:00
|
|
|
if(!directDatabaseAccess){
|
|
|
|
fs.readFile(destFile, "utf8", function(err, _text){
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
text = _text;
|
|
|
|
// Title needs to be stripped out else it appends it to the pad..
|
|
|
|
text = text.replace("<title>", "<!-- <title>");
|
|
|
|
text = text.replace("</title>","</title>-->");
|
2015-04-01 01:19:42 +02:00
|
|
|
|
2014-12-29 20:57:58 +01:00
|
|
|
//node on windows has a delay on releasing of the file lock.
|
|
|
|
//We add a 100ms delay to work around this
|
|
|
|
if(os.type().indexOf("Windows") > -1){
|
|
|
|
setTimeout(function() {callback();}, 100);
|
|
|
|
} else {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}else{
|
|
|
|
callback();
|
|
|
|
}
|
2011-07-21 21:13:58 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
//change text of the pad and broadcast the changeset
|
2012-11-23 19:06:18 +01:00
|
|
|
function(callback) {
|
2014-12-29 20:57:58 +01:00
|
|
|
if(!directDatabaseAccess){
|
|
|
|
var fileEnding = path.extname(srcFile).toLowerCase();
|
2018-03-08 12:55:00 +01:00
|
|
|
if (importHandledByPlugin || useConvertor || fileEnding == ".htm" || fileEnding == ".html") {
|
2015-01-19 03:51:32 +01:00
|
|
|
importHtml.setPadHTML(pad, text, function(e){
|
|
|
|
if(e) apiLogger.warn("Error importing, possibly caused by malformed HTML");
|
|
|
|
});
|
2014-12-29 20:57:58 +01:00
|
|
|
} else {
|
|
|
|
pad.setText(text);
|
2013-09-27 15:47:34 +02:00
|
|
|
}
|
2012-11-23 22:55:25 +01:00
|
|
|
}
|
2014-12-29 20:57:58 +01:00
|
|
|
|
|
|
|
// Load the Pad into memory then brodcast updates to all clients
|
|
|
|
padManager.unloadPad(padId);
|
|
|
|
padManager.getPad(padId, function(err, _pad){
|
|
|
|
var pad = _pad;
|
|
|
|
padManager.unloadPad(padId);
|
2014-12-30 12:12:24 +01:00
|
|
|
// direct Database Access means a pad user should perform a switchToPad
|
|
|
|
// and not attempt to recieve updated pad data..
|
|
|
|
if(!directDatabaseAccess){
|
|
|
|
padMessageHandler.updatePadClients(pad, function(){
|
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}else{
|
2014-12-29 20:57:58 +01:00
|
|
|
callback();
|
2014-12-30 12:12:24 +01:00
|
|
|
}
|
2014-12-29 20:57:58 +01:00
|
|
|
});
|
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
//clean up temporary files
|
2012-11-23 19:06:18 +01:00
|
|
|
function(callback) {
|
2014-12-29 20:57:58 +01:00
|
|
|
if(!directDatabaseAccess){
|
|
|
|
//for node < 0.7 compatible
|
|
|
|
var fileExists = fs.exists || path.exists;
|
|
|
|
async.parallel([
|
|
|
|
function(callback){
|
|
|
|
fileExists (srcFile, function(exist) { (exist)? fs.unlink(srcFile, callback): callback(); });
|
|
|
|
},
|
|
|
|
function(callback){
|
|
|
|
fileExists (destFile, function(exist) { (exist)? fs.unlink(destFile, callback): callback(); });
|
|
|
|
}
|
|
|
|
], callback);
|
|
|
|
}else{
|
|
|
|
callback();
|
|
|
|
}
|
2011-07-21 21:13:58 +02:00
|
|
|
}
|
2012-11-23 19:06:18 +01:00
|
|
|
], function(err) {
|
2012-02-27 00:22:53 +01:00
|
|
|
var status = "ok";
|
|
|
|
|
|
|
|
//check for known errors and replace the status
|
2014-12-29 22:02:24 +01:00
|
|
|
if(err == "uploadFailed" || err == "convertFailed" || err == "padHasData")
|
2011-10-19 21:46:28 +02:00
|
|
|
{
|
2012-02-27 00:22:53 +01:00
|
|
|
status = err;
|
|
|
|
err = null;
|
2011-10-19 21:46:28 +02:00
|
|
|
}
|
|
|
|
|
2011-12-04 16:50:02 +01:00
|
|
|
ERR(err);
|
2014-12-30 12:12:24 +01:00
|
|
|
|
2011-07-21 21:13:58 +02:00
|
|
|
//close the connection
|
2014-11-25 23:14:48 +01:00
|
|
|
res.send(
|
|
|
|
"<head> \
|
|
|
|
<script type='text/javascript' src='../../static/js/jquery.js'></script> \
|
|
|
|
</head> \
|
|
|
|
<script> \
|
|
|
|
$(window).load(function(){ \
|
2014-12-30 12:12:24 +01:00
|
|
|
var impexp = window.parent.padimpexp.handleFrameCall('" + directDatabaseAccess +"', '" + status + "'); \
|
2014-11-25 23:14:48 +01:00
|
|
|
}) \
|
|
|
|
</script>"
|
2015-04-10 21:10:55 +02:00
|
|
|
);
|
2011-07-21 21:13:58 +02:00
|
|
|
});
|
|
|
|
}
|
2012-12-10 12:34:16 +01:00
|
|
|
|