2019-07-05 13:54:32 +02:00
|
|
|
import * as express from 'express'
|
2019-07-12 11:39:58 +02:00
|
|
|
import { join } from 'path'
|
2021-03-12 15:20:46 +01:00
|
|
|
import { logger } from '@server/helpers/logger'
|
|
|
|
import { optionalAuthenticate } from '@server/middlewares/auth'
|
2020-12-07 14:32:36 +01:00
|
|
|
import { getCompleteLocale, is18nLocale } from '../../shared/core-utils/i18n'
|
2021-07-16 10:42:24 +02:00
|
|
|
import { HttpStatusCode } from '../../shared/models/http/http-error-codes'
|
2019-07-12 11:39:58 +02:00
|
|
|
import { PluginType } from '../../shared/models/plugins/plugin.type'
|
2019-07-23 09:48:48 +02:00
|
|
|
import { isTestInstance } from '../helpers/core-utils'
|
2021-03-12 15:20:46 +01:00
|
|
|
import { PLUGIN_GLOBAL_CSS_PATH } from '../initializers/constants'
|
|
|
|
import { PluginManager, RegisteredPlugin } from '../lib/plugins/plugin-manager'
|
|
|
|
import { getExternalAuthValidator, getPluginValidator, pluginStaticDirectoryValidator } from '../middlewares/validators/plugins'
|
|
|
|
import { serveThemeCSSValidator } from '../middlewares/validators/themes'
|
2019-07-23 09:48:48 +02:00
|
|
|
|
|
|
|
const sendFileOptions = {
|
|
|
|
maxAge: '30 days',
|
|
|
|
immutable: !isTestInstance()
|
|
|
|
}
|
2019-07-05 13:54:32 +02:00
|
|
|
|
|
|
|
const pluginsRouter = express.Router()
|
|
|
|
|
2019-07-12 11:39:58 +02:00
|
|
|
pluginsRouter.get('/plugins/global.css',
|
2019-07-08 14:02:03 +02:00
|
|
|
servePluginGlobalCSS
|
2019-07-05 13:54:32 +02:00
|
|
|
)
|
|
|
|
|
2019-07-26 14:44:50 +02:00
|
|
|
pluginsRouter.get('/plugins/translations/:locale.json',
|
|
|
|
getPluginTranslations
|
|
|
|
)
|
|
|
|
|
2020-04-28 14:49:03 +02:00
|
|
|
pluginsRouter.get('/plugins/:pluginName/:pluginVersion/auth/:authName',
|
|
|
|
getPluginValidator(PluginType.PLUGIN),
|
|
|
|
getExternalAuthValidator,
|
|
|
|
handleAuthInPlugin
|
|
|
|
)
|
|
|
|
|
2019-07-12 11:39:58 +02:00
|
|
|
pluginsRouter.get('/plugins/:pluginName/:pluginVersion/static/:staticEndpoint(*)',
|
2020-04-10 15:07:54 +02:00
|
|
|
getPluginValidator(PluginType.PLUGIN),
|
|
|
|
pluginStaticDirectoryValidator,
|
2019-07-05 13:54:32 +02:00
|
|
|
servePluginStaticDirectory
|
|
|
|
)
|
|
|
|
|
2019-07-12 11:39:58 +02:00
|
|
|
pluginsRouter.get('/plugins/:pluginName/:pluginVersion/client-scripts/:staticEndpoint(*)',
|
2020-04-10 15:07:54 +02:00
|
|
|
getPluginValidator(PluginType.PLUGIN),
|
|
|
|
pluginStaticDirectoryValidator,
|
2019-07-05 13:54:32 +02:00
|
|
|
servePluginClientScripts
|
|
|
|
)
|
|
|
|
|
2020-04-10 15:07:54 +02:00
|
|
|
pluginsRouter.use('/plugins/:pluginName/router',
|
|
|
|
getPluginValidator(PluginType.PLUGIN, false),
|
2020-12-04 20:56:48 +01:00
|
|
|
optionalAuthenticate,
|
2020-04-10 15:07:54 +02:00
|
|
|
servePluginCustomRoutes
|
|
|
|
)
|
|
|
|
|
|
|
|
pluginsRouter.use('/plugins/:pluginName/:pluginVersion/router',
|
|
|
|
getPluginValidator(PluginType.PLUGIN),
|
2020-12-04 20:56:48 +01:00
|
|
|
optionalAuthenticate,
|
2020-04-10 15:07:54 +02:00
|
|
|
servePluginCustomRoutes
|
|
|
|
)
|
|
|
|
|
2019-07-12 11:39:58 +02:00
|
|
|
pluginsRouter.get('/themes/:pluginName/:pluginVersion/static/:staticEndpoint(*)',
|
2020-04-10 15:07:54 +02:00
|
|
|
getPluginValidator(PluginType.THEME),
|
|
|
|
pluginStaticDirectoryValidator,
|
2019-07-12 11:39:58 +02:00
|
|
|
servePluginStaticDirectory
|
|
|
|
)
|
|
|
|
|
|
|
|
pluginsRouter.get('/themes/:pluginName/:pluginVersion/client-scripts/:staticEndpoint(*)',
|
2020-04-10 15:07:54 +02:00
|
|
|
getPluginValidator(PluginType.THEME),
|
|
|
|
pluginStaticDirectoryValidator,
|
2019-07-12 11:39:58 +02:00
|
|
|
servePluginClientScripts
|
|
|
|
)
|
|
|
|
|
|
|
|
pluginsRouter.get('/themes/:themeName/:themeVersion/css/:staticEndpoint(*)',
|
|
|
|
serveThemeCSSValidator,
|
|
|
|
serveThemeCSSDirectory
|
|
|
|
)
|
|
|
|
|
2019-07-05 13:54:32 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
pluginsRouter
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2019-07-08 14:02:03 +02:00
|
|
|
function servePluginGlobalCSS (req: express.Request, res: express.Response) {
|
2019-07-23 09:48:48 +02:00
|
|
|
// Only cache requests that have a ?hash=... query param
|
|
|
|
const globalCSSOptions = req.query.hash
|
|
|
|
? sendFileOptions
|
|
|
|
: {}
|
|
|
|
|
|
|
|
return res.sendFile(PLUGIN_GLOBAL_CSS_PATH, globalCSSOptions)
|
2019-07-08 14:02:03 +02:00
|
|
|
}
|
|
|
|
|
2019-07-26 14:44:50 +02:00
|
|
|
function getPluginTranslations (req: express.Request, res: express.Response) {
|
|
|
|
const locale = req.params.locale
|
|
|
|
|
|
|
|
if (is18nLocale(locale)) {
|
|
|
|
const completeLocale = getCompleteLocale(locale)
|
|
|
|
const json = PluginManager.Instance.getTranslations(completeLocale)
|
|
|
|
|
|
|
|
return res.json(json)
|
|
|
|
}
|
|
|
|
|
2021-06-01 01:36:53 +02:00
|
|
|
return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-07-26 14:44:50 +02:00
|
|
|
}
|
|
|
|
|
2019-07-05 13:54:32 +02:00
|
|
|
function servePluginStaticDirectory (req: express.Request, res: express.Response) {
|
|
|
|
const plugin: RegisteredPlugin = res.locals.registeredPlugin
|
|
|
|
const staticEndpoint = req.params.staticEndpoint
|
|
|
|
|
2019-07-08 14:02:03 +02:00
|
|
|
const [ directory, ...file ] = staticEndpoint.split('/')
|
|
|
|
|
|
|
|
const staticPath = plugin.staticDirs[directory]
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!staticPath) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-07-05 13:54:32 +02:00
|
|
|
|
2019-07-08 14:02:03 +02:00
|
|
|
const filepath = file.join('/')
|
2019-07-23 09:48:48 +02:00
|
|
|
return res.sendFile(join(plugin.path, staticPath, filepath), sendFileOptions)
|
2019-07-05 13:54:32 +02:00
|
|
|
}
|
|
|
|
|
2020-04-10 15:07:54 +02:00
|
|
|
function servePluginCustomRoutes (req: express.Request, res: express.Response, next: express.NextFunction) {
|
|
|
|
const plugin: RegisteredPlugin = res.locals.registeredPlugin
|
|
|
|
const router = PluginManager.Instance.getRouter(plugin.npmName)
|
|
|
|
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!router) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2020-04-10 15:07:54 +02:00
|
|
|
|
|
|
|
return router(req, res, next)
|
|
|
|
}
|
|
|
|
|
2019-07-05 13:54:32 +02:00
|
|
|
function servePluginClientScripts (req: express.Request, res: express.Response) {
|
|
|
|
const plugin: RegisteredPlugin = res.locals.registeredPlugin
|
|
|
|
const staticEndpoint = req.params.staticEndpoint
|
|
|
|
|
2019-07-08 14:02:03 +02:00
|
|
|
const file = plugin.clientScripts[staticEndpoint]
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!file) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-07-08 14:02:03 +02:00
|
|
|
|
2019-07-23 09:48:48 +02:00
|
|
|
return res.sendFile(join(plugin.path, staticEndpoint), sendFileOptions)
|
2019-07-05 13:54:32 +02:00
|
|
|
}
|
2019-07-12 11:39:58 +02:00
|
|
|
|
|
|
|
function serveThemeCSSDirectory (req: express.Request, res: express.Response) {
|
|
|
|
const plugin: RegisteredPlugin = res.locals.registeredPlugin
|
|
|
|
const staticEndpoint = req.params.staticEndpoint
|
|
|
|
|
|
|
|
if (plugin.css.includes(staticEndpoint) === false) {
|
2021-06-01 01:36:53 +02:00
|
|
|
return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-07-12 11:39:58 +02:00
|
|
|
}
|
|
|
|
|
2019-07-23 09:48:48 +02:00
|
|
|
return res.sendFile(join(plugin.path, staticEndpoint), sendFileOptions)
|
2019-07-12 11:39:58 +02:00
|
|
|
}
|
2020-04-28 14:49:03 +02:00
|
|
|
|
|
|
|
function handleAuthInPlugin (req: express.Request, res: express.Response) {
|
|
|
|
const authOptions = res.locals.externalAuth
|
|
|
|
|
|
|
|
try {
|
|
|
|
logger.debug('Forwarding auth plugin request in %s of plugin %s.', authOptions.authName, res.locals.registeredPlugin.npmName)
|
|
|
|
authOptions.onAuthRequest(req, res)
|
|
|
|
} catch (err) {
|
2020-04-30 08:47:25 +02:00
|
|
|
logger.error('Forward request error in auth %s of plugin %s.', authOptions.authName, res.locals.registeredPlugin.npmName, { err })
|
2020-04-28 14:49:03 +02:00
|
|
|
}
|
|
|
|
}
|