2011-07-19 20:48:11 +02:00
|
|
|
/**
|
|
|
|
* Handles the export requests
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2011-08-11 16:26:41 +02:00
|
|
|
* 2011 Peter 'Pita' Martischka (Primary Technology Ltd)
|
2014-12-30 00:13:01 +01:00
|
|
|
* 2014 John McLear (Etherpad Foundation / McLear Ltd)
|
2011-07-19 20:48:11 +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.
|
|
|
|
*/
|
|
|
|
|
2011-12-04 16:50:02 +01:00
|
|
|
var ERR = require("async-stacktrace");
|
2011-07-27 19:52:23 +02:00
|
|
|
var exporthtml = require("../utils/ExportHtml");
|
2013-02-10 18:34:34 +01:00
|
|
|
var exporttxt = require("../utils/ExportTxt");
|
2014-12-29 20:57:58 +01:00
|
|
|
var exportEtherpad = require("../utils/ExportEtherpad");
|
2011-07-19 20:48:11 +02:00
|
|
|
var async = require("async");
|
|
|
|
var fs = require("fs");
|
2011-07-27 19:52:23 +02:00
|
|
|
var settings = require('../utils/Settings');
|
2011-08-04 00:57:11 +02:00
|
|
|
var os = require('os');
|
2014-05-06 22:13:22 +02:00
|
|
|
var hooks = require("ep_etherpad-lite/static/js/pluginfw/hooks");
|
2015-05-18 17:24:41 +02:00
|
|
|
var TidyHtml = require('../utils/TidyHtml');
|
2011-07-19 20:48:11 +02:00
|
|
|
|
|
|
|
//load abiword only if its enabled
|
|
|
|
if(settings.abiword != null)
|
2011-07-27 19:52:23 +02:00
|
|
|
var abiword = require("../utils/Abiword");
|
2011-07-19 20:48:11 +02:00
|
|
|
|
2011-08-04 00:57:11 +02:00
|
|
|
var tempDirectory = "/tmp";
|
|
|
|
|
2015-05-18 17:24:41 +02:00
|
|
|
//tempDirectory changes if the operating system is windows
|
2011-08-04 00:57:11 +02:00
|
|
|
if(os.type().indexOf("Windows") > -1)
|
|
|
|
{
|
2011-08-13 14:11:48 +02:00
|
|
|
tempDirectory = process.env.TEMP;
|
2011-08-04 00:57:11 +02:00
|
|
|
}
|
2015-05-18 17:24:41 +02:00
|
|
|
|
2011-07-19 20:48:11 +02:00
|
|
|
/**
|
|
|
|
* do a requested export
|
2015-05-18 17:24:41 +02:00
|
|
|
*/
|
2011-07-19 20:48:11 +02:00
|
|
|
exports.doExport = function(req, res, padId, type)
|
|
|
|
{
|
2014-05-07 17:27:41 +02:00
|
|
|
var fileName = padId;
|
|
|
|
|
2014-05-06 22:13:22 +02:00
|
|
|
// allow fileName to be overwritten by a hook, the type type is kept static for security reasons
|
2015-05-18 17:24:41 +02:00
|
|
|
hooks.aCallFirst("exportFileName", padId,
|
2014-05-07 17:27:41 +02:00
|
|
|
function(err, hookFileName){
|
|
|
|
// if fileName is set then set it to the padId, note that fileName is returned as an array.
|
2014-09-28 14:53:07 +02:00
|
|
|
if(hookFileName.length) fileName = hookFileName;
|
2013-02-10 18:34:34 +01:00
|
|
|
|
2014-05-15 14:47:28 +02:00
|
|
|
//tell the browser that this is a downloadable file
|
|
|
|
res.attachment(fileName + "." + type);
|
2015-05-18 17:24:41 +02:00
|
|
|
|
2014-05-15 14:47:28 +02:00
|
|
|
//if this is a plain text export, we can do this directly
|
|
|
|
// We have to over engineer this because tabs are stored as attributes and not plain text
|
2014-12-29 20:57:58 +01:00
|
|
|
if(type == "etherpad"){
|
|
|
|
exportEtherpad.getPadRaw(padId, function(err, pad){
|
|
|
|
if(!err){
|
|
|
|
res.send(pad);
|
|
|
|
// return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else if(type == "txt")
|
2013-02-10 18:34:34 +01:00
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
var txt;
|
|
|
|
var randNum;
|
|
|
|
var srcFile, destFile;
|
2015-05-18 17:24:41 +02:00
|
|
|
|
2014-05-15 14:47:28 +02:00
|
|
|
async.series([
|
|
|
|
//render the txt document
|
2013-02-10 18:34:34 +01:00
|
|
|
function(callback)
|
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
exporttxt.getPadTXTDocument(padId, req.params.rev, false, function(err, _txt)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
txt = _txt;
|
|
|
|
callback();
|
|
|
|
});
|
2013-02-10 18:34:34 +01:00
|
|
|
},
|
2014-05-15 14:47:28 +02:00
|
|
|
//decide what to do with the txt export
|
2013-02-10 18:34:34 +01:00
|
|
|
function(callback)
|
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
//if this is a txt export, we can send this from here directly
|
|
|
|
res.send(txt);
|
|
|
|
callback("stop");
|
|
|
|
},
|
|
|
|
//send the convert job to abiword
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//ensure html can be collected by the garbage collector
|
|
|
|
txt = null;
|
2015-05-18 17:24:41 +02:00
|
|
|
|
2014-05-15 14:47:28 +02:00
|
|
|
destFile = tempDirectory + "/etherpad_export_" + randNum + "." + type;
|
|
|
|
abiword.convertFile(srcFile, destFile, type, callback);
|
|
|
|
},
|
|
|
|
//send the file
|
|
|
|
function(callback)
|
|
|
|
{
|
2015-04-07 14:55:05 +02:00
|
|
|
res.sendFile(destFile, null, callback);
|
2014-05-15 14:47:28 +02:00
|
|
|
},
|
|
|
|
//clean up temporary files
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
async.parallel([
|
|
|
|
function(callback)
|
2013-02-10 18:34:34 +01:00
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
fs.unlink(srcFile, callback);
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//100ms delay to accomidate for slow windows fs
|
|
|
|
if(os.type().indexOf("Windows") > -1)
|
|
|
|
{
|
|
|
|
setTimeout(function()
|
|
|
|
{
|
|
|
|
fs.unlink(destFile, callback);
|
|
|
|
}, 100);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fs.unlink(destFile, callback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
], callback);
|
2013-02-10 18:34:34 +01:00
|
|
|
}
|
2014-05-15 14:47:28 +02:00
|
|
|
], function(err)
|
|
|
|
{
|
|
|
|
if(err && err != "stop") ERR(err);
|
|
|
|
})
|
2011-12-18 06:18:35 +01:00
|
|
|
}
|
2014-05-15 14:47:28 +02:00
|
|
|
else
|
2011-07-19 20:48:11 +02:00
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
var html;
|
|
|
|
var randNum;
|
|
|
|
var srcFile, destFile;
|
2015-05-18 17:24:41 +02:00
|
|
|
|
2014-05-15 14:47:28 +02:00
|
|
|
async.series([
|
|
|
|
//render the html document
|
2011-07-19 20:48:11 +02:00
|
|
|
function(callback)
|
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
exporthtml.getPadHTMLDocument(padId, req.params.rev, false, function(err, _html)
|
|
|
|
{
|
|
|
|
if(ERR(err, callback)) return;
|
|
|
|
html = _html;
|
|
|
|
callback();
|
2015-05-18 17:24:41 +02:00
|
|
|
});
|
2011-07-19 20:48:11 +02:00
|
|
|
},
|
2014-05-15 14:47:28 +02:00
|
|
|
//decide what to do with the html export
|
2011-07-19 20:48:11 +02:00
|
|
|
function(callback)
|
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
//if this is a html export, we can send this from here directly
|
|
|
|
if(type == "html")
|
2011-12-04 16:50:02 +01:00
|
|
|
{
|
2015-01-06 17:31:52 +01:00
|
|
|
// do any final changes the plugin might want to make cake
|
|
|
|
hooks.aCallFirst("exportHTMLSend", html, function(err, newHTML){
|
|
|
|
if(newHTML.length) html = newHTML;
|
|
|
|
res.send(html);
|
2015-05-18 17:24:41 +02:00
|
|
|
callback("stop");
|
2015-01-06 17:31:52 +01:00
|
|
|
});
|
2011-12-04 16:50:02 +01:00
|
|
|
}
|
2014-05-15 14:47:28 +02:00
|
|
|
else //write the html export to a file
|
2011-12-04 16:50:02 +01:00
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
randNum = Math.floor(Math.random()*0xFFFFFFFF);
|
|
|
|
srcFile = tempDirectory + "/etherpad_export_" + randNum + ".html";
|
2015-05-18 17:24:41 +02:00
|
|
|
fs.writeFile(srcFile, html, callback);
|
2011-12-04 16:50:02 +01:00
|
|
|
}
|
2014-05-15 14:47:28 +02:00
|
|
|
},
|
2015-05-18 17:24:41 +02:00
|
|
|
|
|
|
|
// Tidy up the exported HTML
|
2014-05-15 14:47:28 +02:00
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//ensure html can be collected by the garbage collector
|
|
|
|
html = null;
|
2015-05-18 17:24:41 +02:00
|
|
|
|
|
|
|
TidyHtml.tidy(srcFile, callback);
|
|
|
|
},
|
|
|
|
|
|
|
|
//send the convert job to abiword
|
|
|
|
function(callback)
|
|
|
|
{
|
2014-05-15 14:47:28 +02:00
|
|
|
destFile = tempDirectory + "/etherpad_export_" + randNum + "." + type;
|
|
|
|
abiword.convertFile(srcFile, destFile, type, callback);
|
|
|
|
},
|
|
|
|
//send the file
|
|
|
|
function(callback)
|
|
|
|
{
|
2015-04-07 14:55:05 +02:00
|
|
|
res.sendFile(destFile, null, callback);
|
2014-05-15 14:47:28 +02:00
|
|
|
},
|
|
|
|
//clean up temporary files
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
async.parallel([
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
fs.unlink(srcFile, callback);
|
|
|
|
},
|
|
|
|
function(callback)
|
|
|
|
{
|
|
|
|
//100ms delay to accomidate for slow windows fs
|
|
|
|
if(os.type().indexOf("Windows") > -1)
|
|
|
|
{
|
2015-05-18 17:24:41 +02:00
|
|
|
setTimeout(function()
|
2014-05-15 14:47:28 +02:00
|
|
|
{
|
|
|
|
fs.unlink(destFile, callback);
|
|
|
|
}, 100);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fs.unlink(destFile, callback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
], callback);
|
2011-07-19 20:48:11 +02:00
|
|
|
}
|
2014-05-15 14:47:28 +02:00
|
|
|
], function(err)
|
|
|
|
{
|
|
|
|
if(err && err != "stop") ERR(err);
|
|
|
|
})
|
2011-07-19 20:48:11 +02:00
|
|
|
}
|
2014-05-15 14:47:28 +02:00
|
|
|
}
|
|
|
|
);
|
2011-07-19 20:48:11 +02:00
|
|
|
};
|