1
0
mirror of https://github.com/Jermolene/TiddlyWiki5 synced 2025-01-16 04:05:40 +00:00
TiddlyWiki5/js/FileRetriever.js
Jeremy Ruston efd194141c Major refactoring of async code
The result is correct handling of tiddlers being overwritten within
recipes.
2011-12-02 14:40:18 +00:00

81 lines
2.5 KiB
JavaScript

/*
FileRetriever can asynchronously retrieve files from HTTP URLs or the local file system. It incorporates
throttling so that we don't get error EMFILE "Too many open files".
*/
/*global require: false, exports: false */
"use strict";
var fs = require("fs"),
path = require("path"),
url = require("url"),
util = require("util"),
http = require("http"),
https = require("https");
var FileRetriever = exports;
var fileRequest = function fileRequest(filepath,callback) {
fs.readFile(filepath,"utf8", callback);
};
var httpRequest = function(fileurl,callback) {
var opts = url.parse(fileurl);
var httpLib = opts.protocol === "http:" ? http : https;
var request = httpLib.get(opts,function(res) {
if(res.statusCode != 200) {
var err = new Error("HTTP error");
err.code = res.statusCode.toString();
callback(err);
} else {
var data = [];
res.on("data", function(chunk) {
data.push(chunk);
});
res.on("end", function() {
callback(null,data.join(""));
});
}
});
request.addListener("error", function(err) {
callback(err);
});
request.end();
};
// Retrieve a file given a filepath specifier and a context path. If the filepath isn't an absolute
// filepath or an absolute URL, then it is interpreted relative to the context path, which can also be
// a filepath or a URL. On completion, the callback function is called as callback(err,data). The
// data hashmap is as follows:
// text: full text of file
// path: full path used to reach the file
// basename: the basename of the file
// extname: the extension of the file
FileRetriever.retrieveFile = function(filepath,contextPath,callback) {
var httpRegExp = /^(https?:\/\/)/gi,
result = {},
filepathIsHttp = httpRegExp.test(filepath),
contextPathIsHttp = httpRegExp.test(contextPath),
requester;
if(contextPathIsHttp || filepathIsHttp) {
// If we've got a full HTTP URI then we're good to go
result.path = url.resolve(contextPath,filepath);
var parsedPath = url.parse(result.path);
result.extname = path.extname(parsedPath.pathname);
result.basename = path.basename(parsedPath.extname);
requester = httpRequest;
} else {
// It's a file requested in a file context
result.path = path.resolve(path.dirname(contextPath),filepath);
result.extname = path.extname(result.path);
result.basename = path.basename(result.path,result.extname);
requester = fileRequest;
}
requester(result.path,function(err,data) {
if(!err) {
result.text = data;
}
callback(err,result);
});
};