// # Themes API // RESTful API for Themes const debug = require('ghost-ignition').debug('api:themes'), Promise = require('bluebird'), fs = require('fs-extra'), localUtils = require('./utils'), common = require('../../lib/common'), models = require('../../models'), settingsCache = require('../../services/settings/cache'), themeUtils = require('../../services/themes'), themeList = themeUtils.list; let themes; /** * ## Themes API Methods * * **See:** [API Methods](constants.js.html#api%20methods) */ themes = { /** * Every role can browse all themes. The response contains a list of all available themes in your content folder. * The active theme get's marked as `active:true` and contains an extra object `templates`, which * contains the custom templates of the active theme. These custom templates are used to show a dropdown * in the PSM to be able to choose a custom post template. */ browse(options) { return localUtils // Permissions .handlePermissions('themes', 'browse')(options) // Main action .then(() => { // Return JSON result return themeUtils.toJSON(); }); }, activate(options) { let themeName = options.name, newSettings = [{ key: 'active_theme', value: themeName }], loadedTheme, checkedTheme; return localUtils // Permissions .handlePermissions('themes', 'activate')(options) // Validation .then(() => { loadedTheme = themeList.get(themeName); if (!loadedTheme) { return Promise.reject(new common.errors.ValidationError({ message: common.i18n.t('notices.data.validation.index.themeCannotBeActivated', {themeName: themeName}), context: 'active_theme' })); } return themeUtils.validate.check(loadedTheme); }) // Update setting .then((_checkedTheme) => { checkedTheme = _checkedTheme; // We use the model, not the API here, as we don't want to trigger permissions return models.Settings.edit(newSettings, options); }) // Call activate .then(() => { // Activate! (sort of) debug('Activating theme (method B on API "activate")', themeName); themeUtils.activate(loadedTheme, checkedTheme); // Return JSON result return themeUtils.toJSON(themeName, checkedTheme); }); }, upload(options) { options = options || {}; // consistent filename uploads options.originalname = options.originalname.toLowerCase(); let zip = { path: options.path, name: options.originalname, shortName: themeUtils.storage.getSanitizedFileName(options.originalname.split('.zip')[0]) }, checkedTheme; // check if zip name is casper.zip if (zip.name === 'casper.zip') { throw new common.errors.ValidationError({message: common.i18n.t('errors.api.themes.overrideCasper')}); } return localUtils // Permissions .handlePermissions('themes', 'add')(options) // Validation .then(() => { return themeUtils.validate.check(zip, true); }) // More validation (existence check) .then((_checkedTheme) => { checkedTheme = _checkedTheme; return themeUtils.storage.exists(zip.shortName); }) // If the theme existed we need to delete it .then((themeExists) => { // delete existing theme if (themeExists) { return themeUtils.storage.delete(zip.shortName); } }) .then(() => { // store extracted theme return themeUtils.storage.save({ name: zip.shortName, path: checkedTheme.path }); }) .then(() => { // Loads the theme from the filesystem // Sets the theme on the themeList return themeUtils.loadOne(zip.shortName); }) .then((loadedTheme) => { // If this is the active theme, we are overriding // This is a special case of activation if (zip.shortName === settingsCache.get('active_theme')) { // Activate! (sort of) debug('Activating theme (method C, on API "override")', zip.shortName); themeUtils.activate(loadedTheme, checkedTheme); } // @TODO: unify the name across gscan and Ghost! return themeUtils.toJSON(zip.shortName, checkedTheme); }) .finally(() => { // @TODO we should probably do this as part of saving the theme // remove zip upload from multer // happens in background fs.remove(zip.path) .catch((err) => { common.logging.error(new common.errors.GhostError({err: err})); }); // @TODO we should probably do this as part of saving the theme // remove extracted dir from gscan // happens in background if (checkedTheme) { fs.remove(checkedTheme.path) .catch((err) => { common.logging.error(new common.errors.GhostError({err: err})); }); } }); }, download(options) { let themeName = options.name, theme = themeList.get(themeName); if (!theme) { return Promise.reject(new common.errors.BadRequestError({message: common.i18n.t('errors.api.themes.invalidThemeName')})); } return localUtils // Permissions .handlePermissions('themes', 'read')(options) .then(() => { return themeUtils.storage.serve({ name: themeName }); }); }, /** * remove theme zip * remove theme folder */ destroy(options) { let themeName = options.name, theme; return localUtils // Permissions .handlePermissions('themes', 'destroy')(options) // Validation .then(() => { if (themeName === 'casper') { throw new common.errors.ValidationError({message: common.i18n.t('errors.api.themes.destroyCasper')}); } if (themeName === settingsCache.get('active_theme')) { throw new common.errors.ValidationError({message: common.i18n.t('errors.api.themes.destroyActive')}); } theme = themeList.get(themeName); if (!theme) { throw new common.errors.NotFoundError({message: common.i18n.t('errors.api.themes.themeDoesNotExist')}); } // Actually do the deletion here return themeUtils.storage.delete(themeName); }) // And some extra stuff to maintain state here .then(() => { themeList.del(themeName); // Delete returns an empty 204 response }); } }; module.exports = themes;