Files
NodeBB/src/meta/templates.js
Opliko 23cb67a112 feat: Allow defining active plugins in config (#10767)
* Revert "Revert "feat: cross origin opener policy options (#10710)""

This reverts commit 46050ace1a.

* Revert "Revert "chore(i18n): fallback strings for new resources: nodebb.admin-settings-advanced""

This reverts commit 9f291c07d3.

* feat: closes #10719, don't trim children if category is marked section

* feat: fire hook to allow plugins to filter the pids returned in a user profile

/cc julianlam/nodebb-plugin-support-forum#14

* fix: use `user.hidePrivateData();` more consistently across user retrieval endpoints

* feat: Allow defining active plugins in config

resolves #10766

* fix: assign the db result to files properly

* test: add tests with plugins in config

* feat: better theme change handling

* feat: add visual indication that plugins can't be activated

* test: correct hooks

* test: fix test definitions

* test: remove instead of resetting nconf to avoid affecting other tests

* test: ... I forgot how nconf worked

* fix: remove negation

* docs: improve wording of error message

* feat: reduce code duplication

* style: remove a redundant space

* fix: remove unused imports

* fix: use nconf instead of requiring config.json

* fix: await...

* fix: second missed await

* fix: move back from getActiveIds to getActive

* fix: use paths again?

* fix: typo

* fix: move require into the function

* fix: forgot to change back to getActive

* test: getActive returns only id

* test: accedently commented out some stuff

* feat: added note to top of plugins page if \!canChangeState

Co-authored-by: Julian Lam <julian@nodebb.org>
Co-authored-by: Barış Soner Uşaklı <barisusakli@gmail.com>
2022-07-26 14:27:17 -04:00

140 lines
4.1 KiB
JavaScript

'use strict';
const util = require('util');
let mkdirp = require('mkdirp');
mkdirp = mkdirp.hasOwnProperty('native') ? mkdirp : util.promisify(mkdirp);
const rimraf = require('rimraf');
const winston = require('winston');
const path = require('path');
const fs = require('fs');
const nconf = require('nconf');
const _ = require('lodash');
const Benchpress = require('benchpressjs');
const plugins = require('../plugins');
const file = require('../file');
const { themeNamePattern, paths } = require('../constants');
const viewsPath = nconf.get('views_dir');
const Templates = module.exports;
async function processImports(paths, templatePath, source) {
const regex = /<!-- IMPORT (.+?) -->/;
const matches = source.match(regex);
if (!matches) {
return source;
}
const partial = matches[1];
if (paths[partial] && templatePath !== partial) {
const partialSource = await fs.promises.readFile(paths[partial], 'utf8');
source = source.replace(regex, partialSource);
return await processImports(paths, templatePath, source);
}
winston.warn(`[meta/templates] Partial not loaded: ${matches[1]}`);
source = source.replace(regex, '');
return await processImports(paths, templatePath, source);
}
Templates.processImports = processImports;
async function getTemplateDirs(activePlugins) {
const pluginTemplates = activePlugins.map((id) => {
if (themeNamePattern.test(id)) {
return nconf.get('theme_templates_path');
}
if (!plugins.pluginsData[id]) {
return '';
}
return path.join(paths.nodeModules, id, plugins.pluginsData[id].templates || 'templates');
}).filter(Boolean);
let themeConfig = require(nconf.get('theme_config'));
let theme = themeConfig.baseTheme;
let themePath;
let themeTemplates = [];
while (theme) {
themePath = path.join(nconf.get('themes_path'), theme);
themeConfig = require(path.join(themePath, 'theme.json'));
themeTemplates.push(path.join(themePath, themeConfig.templates || 'templates'));
theme = themeConfig.baseTheme;
}
themeTemplates.push(nconf.get('base_templates_path'));
themeTemplates = _.uniq(themeTemplates.reverse());
const coreTemplatesPath = nconf.get('core_templates_path');
let templateDirs = _.uniq([coreTemplatesPath].concat(themeTemplates, pluginTemplates));
templateDirs = await Promise.all(templateDirs.map(async path => (await file.exists(path) ? path : false)));
return templateDirs.filter(Boolean);
}
async function getTemplateFiles(dirs) {
const buckets = await Promise.all(dirs.map(async (dir) => {
let files = await file.walk(dir);
files = files.filter(path => path.endsWith('.tpl')).map(file => ({
name: path.relative(dir, file).replace(/\\/g, '/'),
path: file,
}));
return files;
}));
const dict = {};
buckets.forEach((files) => {
files.forEach((file) => {
dict[file.name] = file.path;
});
});
return dict;
}
async function compileTemplate(filename, source) {
let paths = await file.walk(viewsPath);
paths = _.fromPairs(paths.map((p) => {
const relative = path.relative(viewsPath, p).replace(/\\/g, '/');
return [relative, p];
}));
source = await processImports(paths, filename, source);
const compiled = await Benchpress.precompile(source, { filename });
return await fs.promises.writeFile(path.join(viewsPath, filename.replace(/\.tpl$/, '.js')), compiled);
}
Templates.compileTemplate = compileTemplate;
async function compile() {
const _rimraf = util.promisify(rimraf);
await _rimraf(viewsPath);
await mkdirp(viewsPath);
let files = await plugins.getActive();
files = await getTemplateDirs(files);
files = await getTemplateFiles(files);
await Promise.all(Object.keys(files).map(async (name) => {
const filePath = files[name];
let imported = await fs.promises.readFile(filePath, 'utf8');
imported = await processImports(files, name, imported);
await mkdirp(path.join(viewsPath, path.dirname(name)));
await fs.promises.writeFile(path.join(viewsPath, name), imported);
const compiled = await Benchpress.precompile(imported, { filename: name });
await fs.promises.writeFile(path.join(viewsPath, name.replace(/\.tpl$/, '.js')), compiled);
}));
winston.verbose('[meta/templates] Successfully compiled templates.');
}
Templates.compile = compile;