Ghost/core/server/services/route-settings/route-settings.js
Naz 6e075c78bf Moved URL service to backend
refs https://linear.app/tryghost/issue/CORE-104/decouple-frontend-routing-events-from-urlserver-events

- URL module is part of the backend heavily dependent on the model and fits perfectly here. Frontend should get the data it needs by passing a URL manager instance to it
2021-10-19 07:29:09 +13:00

165 lines
4.7 KiB
JavaScript

const Promise = require('bluebird');
const moment = require('moment-timezone');
const fs = require('fs-extra');
const path = require('path');
const crypto = require('crypto');
const urlService = require('../url');
const debug = require('@tryghost/debug')('services:route-settings');
const errors = require('@tryghost/errors');
const tpl = require('@tryghost/tpl');
const config = require('../../../shared/config');
const bridge = require('../../../bridge');
const SettingsLoader = require('./settings-loader');
const parseYaml = require('./yaml-parser');
const settingsLoader = new SettingsLoader({parseYaml});
const messages = {
loadError: 'Could not load {filename} file.'
};
/**
* The `routes.yaml` file offers a way to configure your Ghost blog. It's currently a setting feature
* we have added. That's why the `routes.yaml` file is treated as a "setting" right now.
* If we want to add single permissions for this file (e.g. upload/download routes.yaml), we can add later.
*
* How does it work?
*
* - we first reset all url generators (each url generator belongs to one express router)
* - we don't destroy the resources, we only release them (this avoids reloading all resources from the db again)
* - then we reload the whole site app, which will reset all routers and re-create the url generators
*/
const filename = 'routes';
const ext = 'yaml';
const getSettingsFilePath = () => {
const settingsFolder = config.getContentPath('settings');
return path.join(settingsFolder, `${filename}.${ext}`);
};
const getBackupFilePath = () => {
const settingsFolder = config.getContentPath('settings');
return path.join(settingsFolder, `${filename}-${moment().format('YYYY-MM-DD-HH-mm-ss')}.${ext}`);
};
const createBackupFile = async (settingsPath, backupPath) => {
return await fs.copy(settingsPath, backupPath);
};
const restoreBackupFile = async (settingsPath, backupPath) => {
return await fs.copy(backupPath, settingsPath);
};
const saveFile = async (filePath, settingsPath) => {
return await fs.copy(filePath, settingsPath);
};
const readFile = (settingsFilePath) => {
return fs.readFile(settingsFilePath, 'utf-8')
.catch((err) => {
if (err.code === 'ENOENT') {
return Promise.resolve([]);
}
if (errors.utils.isIgnitionError(err)) {
throw err;
}
throw new errors.NotFoundError({
err: err
});
});
};
const setFromFilePath = async (filePath) => {
const settingsPath = getSettingsFilePath();
const backupPath = getBackupFilePath();
await createBackupFile(settingsPath, backupPath);
await saveFile(filePath, settingsPath);
urlService.resetGenerators({releaseResourcesOnly: true});
const bringBackValidRoutes = async () => {
urlService.resetGenerators({releaseResourcesOnly: true});
await restoreBackupFile(settingsPath, backupPath);
return bridge.reloadFrontend();
};
try {
bridge.reloadFrontend();
} catch (err) {
return bringBackValidRoutes()
.finally(() => {
throw err;
});
}
// @TODO: how can we get rid of this from here?
let tries = 0;
function isBlogRunning() {
debug('waiting for blog running');
return Promise.delay(1000)
.then(() => {
debug('waited for blog running');
if (!urlService.hasFinished()) {
if (tries > 5) {
throw new errors.InternalServerError({
message: tpl(messages.loadError, {filename: `${filename}.${ext}`})
});
}
tries = tries + 1;
return isBlogRunning();
}
});
}
return isBlogRunning()
.catch((err) => {
return bringBackValidRoutes()
.finally(() => {
throw err;
});
});
};
const get = async () => {
const settingsFilePath = await getSettingsFilePath();
return readFile(settingsFilePath);
};
/**
* md5 hashes of default routes settings
*/
const defaultRoutesSettingHash = '3d180d52c663d173a6be791ef411ed01';
const calculateHash = (data) => {
return crypto.createHash('md5')
.update(data, 'binary')
.digest('hex');
};
const getDefaultHash = () => {
return defaultRoutesSettingHash;
};
const getCurrentHash = async () => {
const data = await settingsLoader.loadSettings();
return calculateHash(JSON.stringify(data));
};
module.exports = {
getDefaultHash: getDefaultHash,
setFromFilePath: setFromFilePath,
get: get,
getCurrentHash: getCurrentHash
};