haste-server/lib/static_handler.js

85 lines
2.7 KiB
JavaScript
Raw Normal View History

2011-11-19 04:51:38 +08:00
var path = require('path');
var fs = require('fs');
var winston = require('winston');
// For serving static assets
var StaticHandler = function(path) {
2011-11-19 05:00:05 +08:00
this.basePath = path;
2011-11-19 04:51:38 +08:00
this.defaultPath = '/index.html';
// Grab the list of available files - and move into hash for quick lookup
var available = fs.readdirSync(this.basePath);
this.availablePaths = {};
for (var i = 0; i < available.length; i++) {
this.availablePaths['/' + available[i]] = true;
}
2011-11-19 04:51:38 +08:00
};
2011-11-22 11:19:43 +08:00
StaticHandler.cache = {};
2011-11-19 04:51:38 +08:00
// Determine the content type for a given extension
StaticHandler.contentTypeFor = function(ext) {
if (ext == '.js') return 'text/javascript';
else if (ext == '.css') return 'text/css';
else if (ext == '.html') return 'text/html';
else if (ext == '.ico') return 'image/ico';
else {
winston.error('unable to determine content type for static asset with extension: ' + ext);
return 'text/plain';
}
};
// Handle a request, and serve back the asset if it exists
2011-11-19 05:00:05 +08:00
StaticHandler.prototype.handle = function(incPath, response) {
// Go to index if not found or /
if (!this.availablePaths[incPath]) incPath = this.defaultPath;
2011-11-19 05:00:05 +08:00
var filePath = this.basePath + (incPath == '/' ? this.defaultPath : incPath);
2011-11-22 11:53:50 +08:00
// And then stream the file back - either from the cache or from source
var cached = this.isCached(filePath);
var method = cached ? this.serveCached : this.retrieve;
// Run!
method(filePath, function(error, content) {
// Get the content
if (content) {
var contentType = StaticHandler.contentTypeFor(path.extname(filePath));
response.writeHead(200, { 'content-type': contentType });
response.end(content, 'utf-8');
// Stick it in the cache if its not in there
if (!cached) {
StaticHandler.cache[filePath] = content;
}
}
// Or break down and cry
else {
winston.error('unable to read file', { path: filePath, error: error });
response.writeHead(500, { 'content-type': 'application/json' });
response.end(JSON.stringify({ message: 'IO: Unable to read file' }));
// If it was cached, bust the cache
if (cached) {
StaticHandler.cache[filePath] = null;
}
}
});
2011-11-22 11:19:43 +08:00
};
// Retrieve from the file
2011-11-22 11:53:50 +08:00
StaticHandler.prototype.retrieve = function(filePath, callback) {
2011-11-19 05:22:00 +08:00
var _this = this;
fs.readFile(filePath, function(error, content) {
2011-11-22 11:53:50 +08:00
callback(error, content);
});
2011-11-19 04:51:38 +08:00
};
2011-11-22 11:19:43 +08:00
// Retrieve from memory cache
2011-11-22 11:53:50 +08:00
StaticHandler.prototype.serveCached = function(filePath, callback) {
callback(undefined, StaticHandler.cache[filePath]);
2011-11-22 11:19:43 +08:00
};
2011-11-22 11:53:50 +08:00
// Determine if a given filePath is cached or not
StaticHandler.prototype.isCached = function(filePath) {
return !!StaticHandler.cache[filePath];
};
2011-11-22 11:19:43 +08:00
2011-11-19 04:51:38 +08:00
module.exports = StaticHandler;