2014-02-05 03:40:30 -05:00
|
|
|
var _ = require('lodash'),
|
2013-12-06 03:51:35 -05:00
|
|
|
dataProvider = require('../models'),
|
|
|
|
when = require('when'),
|
|
|
|
errors = require('../errorHandling'),
|
|
|
|
config = require('../config'),
|
|
|
|
settings,
|
|
|
|
settingsCollection,
|
|
|
|
settingsFilter,
|
|
|
|
updateSettingsCache,
|
|
|
|
readSettingsResult,
|
2014-02-23 07:32:35 -05:00
|
|
|
filterPaths,
|
2014-04-27 18:28:50 -05:00
|
|
|
settingsResult,
|
2013-12-06 03:51:35 -05:00
|
|
|
// Holds cached settings
|
|
|
|
settingsCache = {};
|
|
|
|
|
|
|
|
// ### Helpers
|
|
|
|
// Turn an object into a collection
|
|
|
|
settingsCollection = function (settings) {
|
|
|
|
return _.map(settings, function (value, key) {
|
|
|
|
return { key: key, value: value };
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
// Filters an object based on a given filter object
|
|
|
|
settingsFilter = function (settings, filter) {
|
|
|
|
return _.object(_.filter(_.pairs(settings), function (setting) {
|
|
|
|
if (filter) {
|
|
|
|
return _.some(filter.split(','), function (f) {
|
|
|
|
return setting[1].type === f;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
|
|
|
|
// Maintain the internal cache of the settings object
|
|
|
|
updateSettingsCache = function (settings) {
|
|
|
|
settings = settings || {};
|
|
|
|
|
|
|
|
if (!_.isEmpty(settings)) {
|
|
|
|
_.map(settings, function (setting, key) {
|
2014-04-27 18:28:50 -05:00
|
|
|
settingsCache[key] = setting;
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
2014-04-27 18:28:50 -05:00
|
|
|
|
|
|
|
return when(settingsCache);
|
2013-12-06 03:51:35 -05:00
|
|
|
}
|
|
|
|
|
2014-04-27 18:28:50 -05:00
|
|
|
return dataProvider.Settings.findAll()
|
|
|
|
.then(function (result) {
|
|
|
|
settingsCache = readSettingsResult(result.models);
|
|
|
|
|
|
|
|
return settingsCache;
|
2014-02-23 07:32:35 -05:00
|
|
|
});
|
2014-04-27 18:28:50 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
readSettingsResult = function (settingsModels) {
|
|
|
|
var settings = _.reduce(settingsModels, function (memo, member) {
|
|
|
|
if (!memo.hasOwnProperty(member.attributes.key)) {
|
|
|
|
memo[member.attributes.key] = member.attributes;
|
|
|
|
}
|
|
|
|
|
|
|
|
return memo;
|
|
|
|
}, {}),
|
|
|
|
themes = config().paths.availableThemes,
|
|
|
|
apps = config().paths.availableApps,
|
|
|
|
res;
|
|
|
|
|
|
|
|
if (settings.activeTheme) {
|
|
|
|
res = filterPaths(themes, settings.activeTheme.value);
|
|
|
|
|
|
|
|
settings.availableThemes = {
|
|
|
|
key: 'availableThemes',
|
|
|
|
value: res,
|
|
|
|
type: 'theme'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (settings.activeApps) {
|
|
|
|
res = filterPaths(apps, JSON.parse(settings.activeApps.value));
|
Consistency in model method naming
- The API has the BREAD naming for methods
- The model now has findAll, findOne, findPage (where needed), edit, add and destroy, meaning it is similar but with a bit more flexibility
- browse, read, update, create, and delete, which were effectively just aliases, have all been removed.
- added jsDoc for the model methods
2014-05-05 10:18:38 -05:00
|
|
|
|
2014-04-27 18:28:50 -05:00
|
|
|
settings.availableApps = {
|
|
|
|
key: 'availableApps',
|
|
|
|
value: res,
|
|
|
|
type: 'app'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return settings;
|
2014-02-23 07:32:35 -05:00
|
|
|
};
|
|
|
|
|
2014-04-21 13:04:20 -05:00
|
|
|
|
|
|
|
// Normalizes paths read by require-tree so that the apps and themes modules can use them.
|
|
|
|
// Creates an empty array (res), and populates it with useful info about the read packages
|
|
|
|
// like name, whether they're active (comparison with the second argument), and if they
|
|
|
|
// have a package.json, that, otherwise false
|
|
|
|
// @param {object} paths as returned by require-tree()
|
|
|
|
// @param {array/string} active as read from the settings object
|
|
|
|
// @return {array} of objects with useful info about apps / themes
|
|
|
|
|
2014-02-23 07:32:35 -05:00
|
|
|
filterPaths = function (paths, active) {
|
|
|
|
var pathKeys = Object.keys(paths),
|
|
|
|
res = [],
|
|
|
|
item;
|
|
|
|
|
|
|
|
// turn active into an array (so themes and apps can be checked the same)
|
|
|
|
if (!Array.isArray(active)) {
|
|
|
|
active = [active];
|
|
|
|
}
|
|
|
|
|
|
|
|
_.each(pathKeys, function (key) {
|
|
|
|
//do not include hidden files or _messages
|
2014-04-27 18:28:50 -05:00
|
|
|
if (key.indexOf('.') !== 0 &&
|
|
|
|
key !== '_messages' &&
|
|
|
|
key !== 'README.md'
|
2014-02-23 07:32:35 -05:00
|
|
|
) {
|
|
|
|
item = {
|
|
|
|
name: key
|
|
|
|
};
|
|
|
|
if (paths[key].hasOwnProperty('package.json')) {
|
|
|
|
item.package = paths[key]['package.json'];
|
|
|
|
} else {
|
|
|
|
item.package = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_.indexOf(active, key) !== -1) {
|
|
|
|
item.active = true;
|
|
|
|
}
|
|
|
|
res.push(item);
|
|
|
|
}
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
2014-02-23 07:32:35 -05:00
|
|
|
return res;
|
2013-12-06 03:51:35 -05:00
|
|
|
};
|
|
|
|
|
2014-04-27 18:28:50 -05:00
|
|
|
settingsResult = function (settings, type) {
|
|
|
|
var filteredSettings = _.values(settingsFilter(settings, type)),
|
|
|
|
result = {
|
|
|
|
settings: filteredSettings
|
|
|
|
};
|
|
|
|
|
|
|
|
if (type) {
|
|
|
|
result.meta = {
|
|
|
|
filters: {
|
|
|
|
type: type
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
2013-12-06 03:51:35 -05:00
|
|
|
settings = {
|
|
|
|
// #### Browse
|
|
|
|
|
|
|
|
// **takes:** options object
|
|
|
|
browse: function browse(options) {
|
2014-04-27 18:28:50 -05:00
|
|
|
//TODO: omit where type==core
|
2013-12-06 03:51:35 -05:00
|
|
|
// **returns:** a promise for a settings json object
|
2014-04-27 18:28:50 -05:00
|
|
|
return when(settingsResult(settingsCache, options.type));
|
2013-12-06 03:51:35 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
// #### Read
|
|
|
|
|
|
|
|
// **takes:** either a json object containing a key, or a single key string
|
|
|
|
read: function read(options) {
|
|
|
|
if (_.isString(options)) {
|
|
|
|
options = { key: options };
|
|
|
|
}
|
|
|
|
|
2014-04-27 18:28:50 -05:00
|
|
|
var setting = settingsCache[options.key],
|
|
|
|
result = {};
|
|
|
|
|
|
|
|
if (!setting) {
|
2014-05-05 08:51:21 -05:00
|
|
|
return when.reject({type: 'NotFound', message: 'Unable to find setting: ' + options.key});
|
2013-12-06 03:51:35 -05:00
|
|
|
}
|
Consistency in model method naming
- The API has the BREAD naming for methods
- The model now has findAll, findOne, findPage (where needed), edit, add and destroy, meaning it is similar but with a bit more flexibility
- browse, read, update, create, and delete, which were effectively just aliases, have all been removed.
- added jsDoc for the model methods
2014-05-05 10:18:38 -05:00
|
|
|
|
2014-04-27 18:28:50 -05:00
|
|
|
result[options.key] = setting;
|
|
|
|
|
|
|
|
return when(settingsResult(result));
|
2013-12-06 03:51:35 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
// #### Edit
|
|
|
|
|
|
|
|
// **takes:** either a json object representing a collection of settings, or a key and value pair
|
|
|
|
edit: function edit(key, value) {
|
2014-04-03 08:03:09 -05:00
|
|
|
var self = this,
|
|
|
|
type;
|
|
|
|
|
2013-12-06 03:51:35 -05:00
|
|
|
// Check for passing a collection of settings first
|
|
|
|
if (_.isObject(key)) {
|
|
|
|
//clean data
|
2014-04-03 08:03:09 -05:00
|
|
|
type = key.type;
|
2013-12-06 03:51:35 -05:00
|
|
|
delete key.type;
|
|
|
|
delete key.availableThemes;
|
2014-02-23 07:32:35 -05:00
|
|
|
delete key.availableApps;
|
2013-12-06 03:51:35 -05:00
|
|
|
|
|
|
|
key = settingsCollection(key);
|
2014-04-03 08:03:09 -05:00
|
|
|
return dataProvider.Settings.edit(key, {user: self.user}).then(function (result) {
|
2014-04-27 18:28:50 -05:00
|
|
|
var readResult = readSettingsResult(result);
|
|
|
|
|
|
|
|
return updateSettingsCache(readResult).then(function () {
|
|
|
|
return config.theme.update(settings, config().url);
|
2013-12-09 23:41:58 -05:00
|
|
|
}).then(function () {
|
2014-04-27 18:28:50 -05:00
|
|
|
return settingsResult(readResult, type);
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
|
|
|
}).otherwise(function (error) {
|
Consistency in model method naming
- The API has the BREAD naming for methods
- The model now has findAll, findOne, findPage (where needed), edit, add and destroy, meaning it is similar but with a bit more flexibility
- browse, read, update, create, and delete, which were effectively just aliases, have all been removed.
- added jsDoc for the model methods
2014-05-05 10:18:38 -05:00
|
|
|
return dataProvider.Settings.findOne(key.key).then(function (result) {
|
2013-12-06 03:51:35 -05:00
|
|
|
if (!result) {
|
2014-05-05 08:51:21 -05:00
|
|
|
return when.reject({type: 'NotFound', message: 'Unable to find setting: ' + key + '.'});
|
2013-12-06 03:51:35 -05:00
|
|
|
}
|
2014-05-05 08:51:21 -05:00
|
|
|
return when.reject({type: 'InternalServerError', message: error.message});
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2014-04-27 18:28:50 -05:00
|
|
|
|
Consistency in model method naming
- The API has the BREAD naming for methods
- The model now has findAll, findOne, findPage (where needed), edit, add and destroy, meaning it is similar but with a bit more flexibility
- browse, read, update, create, and delete, which were effectively just aliases, have all been removed.
- added jsDoc for the model methods
2014-05-05 10:18:38 -05:00
|
|
|
return dataProvider.Settings.findOne(key).then(function (setting) {
|
2013-12-06 03:51:35 -05:00
|
|
|
if (!setting) {
|
2014-05-05 08:51:21 -05:00
|
|
|
return when.reject({type: 'NotFound', message: 'Unable to find setting: ' + key + '.'});
|
2013-12-06 03:51:35 -05:00
|
|
|
}
|
|
|
|
if (!_.isString(value)) {
|
|
|
|
value = JSON.stringify(value);
|
|
|
|
}
|
|
|
|
setting.set('value', value);
|
2014-04-03 08:03:09 -05:00
|
|
|
return dataProvider.Settings.edit(setting, {user: self.user}).then(function (result) {
|
2014-04-27 18:28:50 -05:00
|
|
|
var updatedSetting = _.first(result).attributes;
|
|
|
|
settingsCache[updatedSetting.key].value = updatedSetting.value;
|
|
|
|
|
|
|
|
return updatedSetting;
|
|
|
|
}).then(function (updatedSetting) {
|
2013-12-26 07:15:10 -05:00
|
|
|
return config.theme.update(settings, config().url).then(function () {
|
2014-04-27 18:28:50 -05:00
|
|
|
return updatedSetting;
|
2013-12-09 23:41:58 -05:00
|
|
|
});
|
2014-04-27 18:28:50 -05:00
|
|
|
}).then(function (updatedSetting) {
|
|
|
|
var result = {};
|
|
|
|
result[updatedSetting.key] = updatedSetting;
|
|
|
|
|
|
|
|
return settingsResult(result);
|
2013-12-09 23:41:58 -05:00
|
|
|
}).otherwise(errors.logAndThrowError);
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = settings;
|
|
|
|
module.exports.updateSettingsCache = updateSettingsCache;
|