2020-11-04 05:55:47 -05:00
|
|
|
const Promise = require('bluebird');
|
2019-06-20 06:19:22 -05:00
|
|
|
const moment = require('moment-timezone');
|
|
|
|
const fs = require('fs-extra');
|
|
|
|
const path = require('path');
|
|
|
|
const urlService = require('../url');
|
|
|
|
|
2020-05-22 13:22:20 -05:00
|
|
|
const errors = require('@tryghost/errors');
|
2020-05-27 12:47:53 -05:00
|
|
|
const config = require('../../../shared/config');
|
2021-06-27 07:38:48 -05:00
|
|
|
const bridge = require('../../../bridge');
|
2019-06-20 06:19:22 -05:00
|
|
|
|
2019-06-20 06:34:22 -05:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2021-06-27 07:38:48 -05:00
|
|
|
|
2019-06-21 06:58:26 -05:00
|
|
|
const setFromFilePath = (filePath) => {
|
2019-06-20 06:19:22 -05:00
|
|
|
const settingsPath = config.getContentPath('settings');
|
|
|
|
const backupRoutesPath = path.join(settingsPath, `routes-${moment().format('YYYY-MM-DD-HH-mm-ss')}.yaml`);
|
|
|
|
|
|
|
|
return fs.copy(`${settingsPath}/routes.yaml`, backupRoutesPath)
|
|
|
|
.then(() => {
|
|
|
|
return fs.copy(filePath, `${settingsPath}/routes.yaml`);
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
urlService.resetGenerators({releaseResourcesOnly: true});
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
const bringBackValidRoutes = () => {
|
|
|
|
urlService.resetGenerators({releaseResourcesOnly: true});
|
|
|
|
|
|
|
|
return fs.copy(backupRoutesPath, `${settingsPath}/routes.yaml`)
|
|
|
|
.then(() => {
|
2021-06-27 07:38:48 -05:00
|
|
|
return bridge.reloadFrontend();
|
2019-06-20 06:19:22 -05:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
try {
|
2021-06-27 07:38:48 -05:00
|
|
|
bridge.reloadFrontend();
|
2019-06-20 06:19:22 -05:00
|
|
|
} catch (err) {
|
|
|
|
return bringBackValidRoutes()
|
|
|
|
.finally(() => {
|
|
|
|
throw err;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let tries = 0;
|
|
|
|
|
|
|
|
function isBlogRunning() {
|
|
|
|
return Promise.delay(1000)
|
|
|
|
.then(() => {
|
|
|
|
if (!urlService.hasFinished()) {
|
|
|
|
if (tries > 5) {
|
2020-05-22 13:22:20 -05:00
|
|
|
throw new errors.InternalServerError({
|
2019-06-20 06:19:22 -05:00
|
|
|
message: 'Could not load routes.yaml file.'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
tries = tries + 1;
|
|
|
|
return isBlogRunning();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return isBlogRunning()
|
|
|
|
.catch((err) => {
|
|
|
|
return bringBackValidRoutes()
|
|
|
|
.finally(() => {
|
|
|
|
throw err;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-06-21 06:58:26 -05:00
|
|
|
const get = () => {
|
2019-06-20 06:23:58 -05:00
|
|
|
const routesPath = path.join(config.getContentPath('settings'), 'routes.yaml');
|
|
|
|
|
|
|
|
return fs.readFile(routesPath, 'utf-8')
|
|
|
|
.catch((err) => {
|
|
|
|
if (err.code === 'ENOENT') {
|
|
|
|
return Promise.resolve([]);
|
|
|
|
}
|
|
|
|
|
2020-05-22 13:22:20 -05:00
|
|
|
if (errors.utils.isIgnitionError(err)) {
|
2019-06-20 06:23:58 -05:00
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
|
2020-05-22 13:22:20 -05:00
|
|
|
throw new errors.NotFoundError({
|
2019-06-20 06:23:58 -05:00
|
|
|
err: err
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-06-21 06:58:26 -05:00
|
|
|
module.exports.setFromFilePath = setFromFilePath;
|
|
|
|
module.exports.get = get;
|