2013-05-11 11:44:25 -05:00
|
|
|
|
// # Ghost Module
|
|
|
|
|
// Defines core methods required to build the frontend
|
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
// ## Setup Prerequisites
|
|
|
|
|
var config = require('./../config'),
|
|
|
|
|
when = require('when'),
|
|
|
|
|
express = require('express'),
|
2013-07-11 14:02:18 -05:00
|
|
|
|
errors = require('./server/errorHandling'),
|
2013-07-10 17:45:13 -05:00
|
|
|
|
fs = require('fs'),
|
2013-06-25 06:43:15 -05:00
|
|
|
|
path = require('path'),
|
|
|
|
|
hbs = require('express-hbs'),
|
2013-07-10 17:45:13 -05:00
|
|
|
|
nodefn = require('when/node/function'),
|
2013-06-25 06:43:15 -05:00
|
|
|
|
_ = require('underscore'),
|
|
|
|
|
Polyglot = require('node-polyglot'),
|
|
|
|
|
|
2013-07-11 14:02:18 -05:00
|
|
|
|
models = require('./server/models'),
|
2013-06-25 06:43:15 -05:00
|
|
|
|
|
2013-07-11 14:02:18 -05:00
|
|
|
|
requireTree = require('./server/require-tree'),
|
2013-07-01 14:24:48 -05:00
|
|
|
|
themePath = path.resolve(__dirname + '../../content/themes'),
|
|
|
|
|
pluginPath = path.resolve(__dirname + '../../content/plugins'),
|
|
|
|
|
themeDirectories = requireTree(themePath),
|
|
|
|
|
pluginDirectories = requireTree(pluginPath),
|
2013-06-25 06:43:15 -05:00
|
|
|
|
|
|
|
|
|
Ghost,
|
|
|
|
|
instance,
|
|
|
|
|
defaults,
|
|
|
|
|
statuses;
|
|
|
|
|
|
|
|
|
|
// ## Default values
|
|
|
|
|
/**
|
|
|
|
|
* A hash of default values to use instead of 'magic' numbers/strings.
|
|
|
|
|
* @type {Object}
|
|
|
|
|
*/
|
|
|
|
|
defaults = {
|
|
|
|
|
filterPriority: 5,
|
|
|
|
|
maxPriority: 9
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ## Article Statuses
|
|
|
|
|
/**
|
|
|
|
|
* A list of atricle status types
|
|
|
|
|
* @type {Object}
|
|
|
|
|
*/
|
|
|
|
|
statuses = {
|
|
|
|
|
'draft': 'draft',
|
|
|
|
|
'complete': 'complete',
|
|
|
|
|
'approved': 'approved',
|
|
|
|
|
'scheduled': 'scheduled',
|
|
|
|
|
'published': 'published'
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ## Module Methods
|
|
|
|
|
/**
|
|
|
|
|
* @method Ghost
|
|
|
|
|
* @returns {*}
|
|
|
|
|
* @constructor
|
|
|
|
|
*/
|
|
|
|
|
Ghost = function () {
|
|
|
|
|
var app,
|
|
|
|
|
polyglot;
|
|
|
|
|
|
|
|
|
|
if (!instance) {
|
|
|
|
|
instance = this;
|
|
|
|
|
|
|
|
|
|
// Holds the filters
|
|
|
|
|
instance.filterCallbacks = [];
|
|
|
|
|
|
|
|
|
|
// Holds the filter hooks (that are built in to Ghost Core)
|
|
|
|
|
instance.filters = [];
|
|
|
|
|
|
|
|
|
|
// Holds the theme directories temporarily
|
|
|
|
|
instance.themeDirectories = {};
|
|
|
|
|
|
|
|
|
|
// Holds the plugin directories temporarily
|
|
|
|
|
instance.pluginDirectories = {};
|
|
|
|
|
|
|
|
|
|
app = express();
|
|
|
|
|
|
|
|
|
|
polyglot = new Polyglot();
|
|
|
|
|
|
|
|
|
|
// functionality
|
|
|
|
|
// load Plugins...
|
|
|
|
|
// var f = new FancyFirstChar(ghost).init();
|
|
|
|
|
|
|
|
|
|
_.extend(instance, {
|
|
|
|
|
app: function () { return app; },
|
|
|
|
|
config: function () { return config; },
|
|
|
|
|
|
|
|
|
|
// there's no management here to be sure this has loaded
|
|
|
|
|
settings: function () { return instance.settingsCache; },
|
|
|
|
|
dataProvider: models,
|
|
|
|
|
statuses: function () { return statuses; },
|
|
|
|
|
polyglot: function () { return polyglot; },
|
|
|
|
|
getPaths: function () {
|
|
|
|
|
return when.all([themeDirectories, pluginDirectories]).then(function (paths) {
|
|
|
|
|
instance.themeDirectories = paths[0];
|
|
|
|
|
instance.pluginDirectories = paths[1];
|
|
|
|
|
return;
|
2013-06-16 11:14:01 -05:00
|
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
|
},
|
|
|
|
|
paths: function () {
|
|
|
|
|
return {
|
2013-07-11 14:02:18 -05:00
|
|
|
|
'activeTheme': __dirname + '/../content/' + config.themeDir + '/' + config.activeTheme + '/',
|
|
|
|
|
'adminViews': __dirname + '/server/views/',
|
|
|
|
|
'helperTemplates': __dirname + '/server/helpers/tpl/',
|
|
|
|
|
'lang': __dirname + '/shared/lang/',
|
|
|
|
|
'availableThemes': instance.themeDirectories,
|
2013-06-25 06:43:15 -05:00
|
|
|
|
'availablePlugins': instance.pluginDirectories
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return instance;
|
|
|
|
|
};
|
2013-06-16 11:14:01 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
Ghost.prototype.init = function () {
|
|
|
|
|
var self = this;
|
2013-05-11 11:44:25 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
return when.join(instance.dataProvider.init(), instance.getPaths()).then(function () {
|
2013-07-01 14:24:48 -05:00
|
|
|
|
return self.loadPlugins();
|
|
|
|
|
}, errors.logAndThrowError).then(function () {
|
2013-06-25 06:43:15 -05:00
|
|
|
|
return self.updateSettingsCache();
|
|
|
|
|
}, errors.logAndThrowError);
|
|
|
|
|
};
|
2013-05-11 11:44:25 -05:00
|
|
|
|
|
2013-07-01 14:24:48 -05:00
|
|
|
|
Ghost.prototype.loadPlugins = function () {
|
|
|
|
|
var self = this,
|
|
|
|
|
pluginPaths = _.values(self.paths().availablePlugins),
|
|
|
|
|
pluginPromises = [];
|
|
|
|
|
|
|
|
|
|
_.each(self.config().activePlugins, function (plugin) {
|
|
|
|
|
var match = _.find(pluginPaths, function (path) {
|
|
|
|
|
return new RegExp(plugin + '$').test(path);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (match) {
|
|
|
|
|
pluginPromises.push(require(path.join(pluginPath, plugin)));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return when.all(pluginPromises).then(function (plugins) {
|
|
|
|
|
_.each(plugins, function (plugin) {
|
|
|
|
|
if (_.isFunction(plugin.init)) {
|
|
|
|
|
plugin.init(self);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}, errors.logAndThrowError);
|
|
|
|
|
};
|
2013-05-11 11:44:25 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
Ghost.prototype.updateSettingsCache = function (settings) {
|
|
|
|
|
var self = this;
|
2013-06-09 11:16:25 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
settings = settings || {};
|
2013-06-09 11:16:25 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
if (!_.isEmpty(settings)) {
|
|
|
|
|
self.settingsCache = settings;
|
|
|
|
|
} else {
|
|
|
|
|
// TODO: this should use api.browse
|
|
|
|
|
return models.Settings.findAll().then(function (result) {
|
|
|
|
|
var settings = {};
|
|
|
|
|
_.map(result.models, function (member) {
|
|
|
|
|
if (!settings.hasOwnProperty(member.attributes.key)) {
|
|
|
|
|
settings[member.attributes.key] = member.attributes.value;
|
|
|
|
|
}
|
|
|
|
|
});
|
2013-05-11 11:44:25 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
self.settingsCache = settings;
|
|
|
|
|
}, errors.logAndThrowError);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2013-07-10 17:45:13 -05:00
|
|
|
|
// ## Template utils
|
|
|
|
|
|
|
|
|
|
Ghost.prototype.compileTemplate = function (templatePath) {
|
|
|
|
|
return nodefn.call(fs.readFile, templatePath).then(function (templateContents) {
|
|
|
|
|
return hbs.handlebars.compile(templateContents.toString());
|
|
|
|
|
}, errors.logAndThrowError);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Ghost.prototype.loadTemplate = function (name) {
|
|
|
|
|
// TODO: allow themes to override these templates
|
2013-07-11 14:02:18 -05:00
|
|
|
|
var templatePath = path.join(this.paths().helperTemplates, name + '.hbs');
|
2013-07-10 17:45:13 -05:00
|
|
|
|
|
|
|
|
|
return this.compileTemplate(templatePath);
|
|
|
|
|
};
|
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
/**
|
|
|
|
|
* @param {string} name
|
|
|
|
|
* @param {Function} fn
|
|
|
|
|
* @return {method} hbs.registerHelper
|
|
|
|
|
*/
|
|
|
|
|
Ghost.prototype.registerThemeHelper = function (name, fn) {
|
|
|
|
|
hbs.registerHelper(name, fn);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {string} name
|
|
|
|
|
* @param {Function} fn
|
|
|
|
|
* @return {*}
|
|
|
|
|
*/
|
|
|
|
|
Ghost.prototype.registerTheme = function (name, fn) {
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {string} name
|
|
|
|
|
* @param {Function} fn
|
|
|
|
|
* @return {*}
|
|
|
|
|
*/
|
|
|
|
|
Ghost.prototype.registerPlugin = function (name, fn) {
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {string} name
|
|
|
|
|
* @param {integer} priority
|
|
|
|
|
* @param {Function} fn
|
|
|
|
|
*/
|
|
|
|
|
Ghost.prototype.registerFilter = function (name, priority, fn) {
|
|
|
|
|
// Curry the priority optional parameter to a default of 5
|
|
|
|
|
if (_.isFunction(priority)) {
|
|
|
|
|
fn = priority;
|
|
|
|
|
priority = defaults.filterPriority;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.filterCallbacks[name] = this.filterCallbacks[name] || {};
|
|
|
|
|
this.filterCallbacks[name][priority] = this.filterCallbacks[name][priority] || [];
|
|
|
|
|
|
|
|
|
|
this.filterCallbacks[name][priority].push(fn);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {string} name [description]
|
|
|
|
|
* @param {*} args
|
|
|
|
|
* @param {Function} callback
|
|
|
|
|
* @return {method} callback
|
|
|
|
|
*/
|
|
|
|
|
Ghost.prototype.doFilter = function (name, args, callback) {
|
|
|
|
|
var callbacks = this.filterCallbacks[name];
|
|
|
|
|
|
|
|
|
|
// Bug out early if no callbacks by that name
|
|
|
|
|
if (!callbacks) {
|
|
|
|
|
return callback(args);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_.times(defaults.maxPriority + 1, function (priority) {
|
|
|
|
|
// Bug out if no handlers on this priority
|
|
|
|
|
if (!_.isArray(callbacks[priority])) {
|
|
|
|
|
return;
|
2013-05-11 11:44:25 -05:00
|
|
|
|
}
|
2013-05-27 14:27:41 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
// Call each handler for this priority level
|
|
|
|
|
_.each(callbacks[priority], function (filterHandler) {
|
2013-07-07 13:02:26 -05:00
|
|
|
|
try {
|
|
|
|
|
args = filterHandler(args);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
// If a filter causes an error, we log it so that it can be debugged, but do not throw the error
|
|
|
|
|
errors.logError(e);
|
|
|
|
|
}
|
2013-06-09 11:16:25 -05:00
|
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
callback(args);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Initialise Theme
|
|
|
|
|
*
|
|
|
|
|
* @todo Tod (?) Old comment
|
|
|
|
|
* @param {Object} app
|
|
|
|
|
*/
|
|
|
|
|
Ghost.prototype.initTheme = function (app) {
|
|
|
|
|
var self = this;
|
|
|
|
|
return function initTheme(req, res, next) {
|
|
|
|
|
app.set('view engine', 'hbs');
|
2013-06-25 16:47:29 -05:00
|
|
|
|
// return the correct mime type for woff files
|
|
|
|
|
express['static'].mime.define({'application/font-woff': ['woff']});
|
|
|
|
|
|
2013-06-25 16:36:53 -05:00
|
|
|
|
if (!res.isAdmin) {
|
2013-06-25 06:43:15 -05:00
|
|
|
|
app.engine('hbs', hbs.express3(
|
|
|
|
|
{partialsDir: self.paths().activeTheme + 'partials'}
|
|
|
|
|
));
|
|
|
|
|
app.set('views', self.paths().activeTheme);
|
|
|
|
|
} else {
|
|
|
|
|
app.engine('hbs', hbs.express3({partialsDir: self.paths().adminViews + 'partials'}));
|
|
|
|
|
app.set('views', self.paths().adminViews);
|
2013-07-11 14:02:18 -05:00
|
|
|
|
app.use('/public', express['static'](path.join(__dirname, '/client/assets')));
|
|
|
|
|
app.use('/public', express['static'](path.join(__dirname, '/client')));
|
2013-07-19 18:44:22 -05:00
|
|
|
|
app.use('/shared', express['static'](path.join(__dirname, '/shared/')));
|
2013-07-11 14:02:18 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
}
|
|
|
|
|
app.use(express['static'](self.paths().activeTheme));
|
|
|
|
|
app.use('/content/images', express['static'](path.join(__dirname, '/../content/images')));
|
|
|
|
|
next();
|
2013-05-11 11:44:25 -05:00
|
|
|
|
};
|
2013-06-25 06:43:15 -05:00
|
|
|
|
};
|
2013-05-11 11:44:25 -05:00
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
|
// TODO: Expose the defaults for other people to see/manipulate as a static value?
|
|
|
|
|
// Ghost.defaults = defaults;
|
2013-06-09 11:16:25 -05:00
|
|
|
|
|
2013-06-30 09:41:25 -05:00
|
|
|
|
module.exports = Ghost;
|