Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
/**
|
|
|
|
* The Bridge
|
|
|
|
*
|
|
|
|
* The bridge is responsible for handing communication from the server to the frontend.
|
|
|
|
* Data should only be flowing server -> frontend.
|
|
|
|
* As the architecture improves, the number of cross requires here should go down
|
2021-09-23 11:36:32 -05:00
|
|
|
* Eventually, the aim is to make this a component that is initialized on boot and is either handed to or actively creates the frontend, if the frontend is desired.
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
*
|
|
|
|
* This file is a great place for all the cross-component event handling in lieu of refactoring
|
2021-09-30 09:43:37 -05:00
|
|
|
* NOTE: You may require anything from shared, the frontend or server here - it is the one place (other than boot) that is allowed :)
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
*/
|
2021-07-06 05:02:37 -05:00
|
|
|
|
|
|
|
const debug = require('@tryghost/debug')('bridge');
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
const errors = require('@tryghost/errors');
|
2021-04-26 08:38:57 -05:00
|
|
|
const config = require('./shared/config');
|
2021-06-15 09:36:27 -05:00
|
|
|
const logging = require('@tryghost/logging');
|
2021-06-24 14:01:29 -05:00
|
|
|
const tpl = require('@tryghost/tpl');
|
2021-04-26 08:38:57 -05:00
|
|
|
const themeEngine = require('./frontend/services/theme-engine');
|
2021-10-18 13:01:38 -05:00
|
|
|
const routerManager = require('./frontend/services/routing').routerManager;
|
2021-06-30 08:56:57 -05:00
|
|
|
const settingsCache = require('./shared/settings-cache');
|
2021-04-23 08:19:18 -05:00
|
|
|
|
2021-07-07 09:49:45 -05:00
|
|
|
// Listen to settings.lang.edited, similar to the member service and models/base/listeners
|
|
|
|
const events = require('./server/lib/common/events');
|
|
|
|
|
2021-06-24 14:01:29 -05:00
|
|
|
const messages = {
|
|
|
|
activateFailed: 'Unable to activate the theme "{theme}".'
|
|
|
|
};
|
|
|
|
|
2021-04-23 08:19:18 -05:00
|
|
|
class Bridge {
|
|
|
|
constructor() {
|
2021-04-26 05:53:15 -05:00
|
|
|
/**
|
|
|
|
* When locale changes, we reload theme translations
|
2021-06-09 10:32:48 -05:00
|
|
|
* @deprecated: the term "lang" was deprecated in favor of "locale" publicly in 4.0
|
2021-04-26 05:53:15 -05:00
|
|
|
*/
|
2021-05-04 10:49:35 -05:00
|
|
|
events.on('settings.lang.edited', (model) => {
|
2021-07-06 05:38:52 -05:00
|
|
|
debug('Active theme init18n');
|
2021-05-04 10:49:35 -05:00
|
|
|
this.getActiveTheme().initI18n({locale: model.get('value')});
|
2021-04-26 05:53:15 -05:00
|
|
|
});
|
2021-10-14 12:10:36 -05:00
|
|
|
|
|
|
|
// NOTE: eventually this event should somehow be listened on and handled by the URL Service
|
|
|
|
// for now this eliminates the need for the frontend routing to listen to
|
|
|
|
// server events
|
|
|
|
events.on('settings.timezone.edited', (model) => {
|
2021-10-18 13:01:38 -05:00
|
|
|
routerManager.handleTimezoneEdit(model);
|
2021-10-14 12:10:36 -05:00
|
|
|
});
|
2021-04-23 08:19:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
getActiveTheme() {
|
|
|
|
return themeEngine.getActive();
|
|
|
|
}
|
|
|
|
|
2021-09-23 11:39:43 -05:00
|
|
|
activateTheme(loadedTheme, checkedTheme) {
|
2021-05-04 10:49:35 -05:00
|
|
|
let settings = {
|
|
|
|
locale: settingsCache.get('lang')
|
|
|
|
};
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
// no need to check the score, activation should be used in combination with validate.check
|
|
|
|
// Use the two theme objects to set the current active theme
|
|
|
|
try {
|
|
|
|
let previousGhostAPI;
|
|
|
|
|
|
|
|
if (this.getActiveTheme()) {
|
|
|
|
previousGhostAPI = this.getActiveTheme().engine('ghost-api');
|
|
|
|
}
|
|
|
|
|
2021-09-23 11:39:43 -05:00
|
|
|
themeEngine.setActive(settings, loadedTheme, checkedTheme);
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
const currentGhostAPI = this.getActiveTheme().engine('ghost-api');
|
|
|
|
|
|
|
|
if (previousGhostAPI !== undefined && (previousGhostAPI !== currentGhostAPI)) {
|
|
|
|
events.emit('services.themes.api.changed');
|
2021-06-27 07:38:48 -05:00
|
|
|
this.reloadFrontend();
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
logging.error(new errors.InternalServerError({
|
2021-06-24 14:01:29 -05:00
|
|
|
message: tpl(messages.activateFailed, {theme: loadedTheme.name}),
|
Moved activate from themes to the bridge
refs: https://github.com/TryGhost/Ghost/commit/bf0823c9a2ddbc93ad0ddcec36eed130c8c5a203
- continuing the work of splitting up the theme service into logical components
Theme activations are a trickier piece of the theme split puzzle because they are called from the API and theme service on boot in different ways.
Activations require a theme to have been validated at different levels. Validations are also tricky - do they belong to the theme engine, or the theme service?
There are then several different flows for activations:
- On Boot
- API "activate" call
- API override on upload or install via setFromZip, which is a method in the storage layer
These calls all have quite different logical flows at the moment, and need to be unified
For now, I've moved the existing "activate" function onto the bridge. This allows the theme service to be split from the frontend, and refactoring can start from there.
I hope to move this so there is less code in the actual bridge very soon, but my goal is not to require any server packages in the frontend part of this
I think ideally:
- all activation code, including validation, should probably be part of the theme engine
- the theme engine should offer 3 methods: getActive() canActivate() and activate()
- the theme service is then only responsible for loading themes in and out of storage, JSON responses for the API, and handing themes to the frontend via the bridge at the appropriate moment
2021-04-27 06:49:48 -05:00
|
|
|
err: err
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-23 08:19:18 -05:00
|
|
|
getFrontendApiVersion() {
|
|
|
|
if (this.getActiveTheme()) {
|
|
|
|
return this.getActiveTheme().engine('ghost-api');
|
|
|
|
} else {
|
|
|
|
return config.get('api:versions:default');
|
|
|
|
}
|
|
|
|
}
|
2021-06-27 07:38:48 -05:00
|
|
|
|
2021-10-13 02:45:56 -05:00
|
|
|
getCardAssetConfig() {
|
|
|
|
if (this.getActiveTheme()) {
|
|
|
|
return this.getActiveTheme().config('card_assets');
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-27 07:38:48 -05:00
|
|
|
reloadFrontend() {
|
|
|
|
const apiVersion = this.getFrontendApiVersion();
|
2021-10-13 02:45:56 -05:00
|
|
|
const cardAssetConfig = this.getCardAssetConfig();
|
|
|
|
|
|
|
|
debug('reload card assets config', cardAssetConfig);
|
|
|
|
const cardAssetService = require('./frontend/services/card-assets');
|
|
|
|
cardAssetService.load(cardAssetConfig);
|
|
|
|
|
2021-07-06 05:02:37 -05:00
|
|
|
debug('reload frontend', apiVersion);
|
2021-10-21 13:43:48 -05:00
|
|
|
const siteApp = require('./frontend/web/site');
|
2021-06-27 07:38:48 -05:00
|
|
|
siteApp.reload({apiVersion});
|
|
|
|
}
|
2021-04-23 08:19:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
const bridge = new Bridge();
|
|
|
|
|
|
|
|
module.exports = bridge;
|