mirror of
https://github.com/discourse/discourse.git
synced 2024-11-25 09:42:07 +08:00
121 lines
3.6 KiB
JavaScript
121 lines
3.6 KiB
JavaScript
|
"use strict";
|
||
|
|
||
|
const path = require("path");
|
||
|
const WatchedDir = require("broccoli-source").WatchedDir;
|
||
|
const Funnel = require("broccoli-funnel");
|
||
|
const mergeTrees = require("broccoli-merge-trees");
|
||
|
const fs = require("fs");
|
||
|
const concat = require("broccoli-concat");
|
||
|
const RawHandlebarsCompiler = require("discourse-hbr/raw-handlebars-compiler");
|
||
|
|
||
|
function fixLegacyExtensions(tree) {
|
||
|
return new Funnel(tree, {
|
||
|
getDestinationPath: function (relativePath) {
|
||
|
if (relativePath.endsWith(".es6")) {
|
||
|
return relativePath.slice(0, -4);
|
||
|
} else if (relativePath.endsWith(".raw.hbs")) {
|
||
|
return relativePath.replace(".raw.hbs", ".hbr");
|
||
|
}
|
||
|
return relativePath;
|
||
|
},
|
||
|
});
|
||
|
}
|
||
|
|
||
|
const COLOCATED_CONNECTOR_REGEX =
|
||
|
/^(?<prefix>.*)\/connectors\/(?<outlet>[^\/]+)\/(?<name>[^\/\.]+)\.(?<extension>.+)$/;
|
||
|
|
||
|
// Having connector templates and js in the same directory causes a clash
|
||
|
// when outputting es6 modules. This shim separates colocated connectors
|
||
|
// into separate js / template locations.
|
||
|
function unColocateConnectors(tree) {
|
||
|
return new Funnel(tree, {
|
||
|
getDestinationPath: function (relativePath) {
|
||
|
const match = relativePath.match(COLOCATED_CONNECTOR_REGEX);
|
||
|
if (
|
||
|
match &&
|
||
|
match.groups.extension === "hbs" &&
|
||
|
!match.groups.prefix.endsWith("/templates")
|
||
|
) {
|
||
|
const { prefix, outlet, name } = match.groups;
|
||
|
return `${prefix}/templates/connectors/${outlet}/${name}.hbs`;
|
||
|
}
|
||
|
if (
|
||
|
match &&
|
||
|
match.groups.extension === "js" &&
|
||
|
match.groups.prefix.endsWith("/templates")
|
||
|
) {
|
||
|
// Some plugins are colocating connector JS under `/templates`
|
||
|
const { prefix, outlet, name } = match.groups;
|
||
|
const newPrefix = prefix.slice(0, -"/templates".length);
|
||
|
return `${newPrefix}/connectors/${outlet}/${name}.js`;
|
||
|
}
|
||
|
return relativePath;
|
||
|
},
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function namespaceModules(tree, pluginDirectoryName) {
|
||
|
return new Funnel(tree, {
|
||
|
getDestinationPath: function (relativePath) {
|
||
|
return `discourse/plugins/${pluginDirectoryName}/${relativePath}`;
|
||
|
},
|
||
|
});
|
||
|
}
|
||
|
|
||
|
module.exports = {
|
||
|
name: require("./package").name,
|
||
|
|
||
|
pluginInfos() {
|
||
|
const root = path.resolve("../../../../plugins");
|
||
|
const pluginDirectories = fs
|
||
|
.readdirSync(root, { withFileTypes: true })
|
||
|
.filter(
|
||
|
(dirent) =>
|
||
|
(dirent.isDirectory() || dirent.isSymbolicLink()) &&
|
||
|
!dirent.name.startsWith(".")
|
||
|
);
|
||
|
|
||
|
return pluginDirectories.map((directory) => {
|
||
|
const name = directory.name;
|
||
|
const jsDirectory = path.resolve(root, name, "assets/javascripts");
|
||
|
const hasJs = fs.existsSync(jsDirectory);
|
||
|
return { name, jsDirectory, hasJs };
|
||
|
});
|
||
|
},
|
||
|
|
||
|
generatePluginsTree() {
|
||
|
const trees = this.pluginInfos()
|
||
|
.filter((p) => p.hasJs)
|
||
|
.map(({ name, jsDirectory }) => {
|
||
|
let tree = new WatchedDir(jsDirectory);
|
||
|
|
||
|
tree = fixLegacyExtensions(tree);
|
||
|
tree = unColocateConnectors(tree);
|
||
|
tree = namespaceModules(tree, name);
|
||
|
|
||
|
tree = RawHandlebarsCompiler(tree);
|
||
|
tree = this.compileTemplates(tree);
|
||
|
|
||
|
tree = this.processedAddonJsFiles(tree);
|
||
|
|
||
|
return concat(mergeTrees([tree]), {
|
||
|
inputFiles: ["**/*.js"],
|
||
|
outputFile: `assets/plugins/${name}.js`,
|
||
|
});
|
||
|
});
|
||
|
return mergeTrees(trees);
|
||
|
},
|
||
|
|
||
|
shouldCompileTemplates() {
|
||
|
// The base Addon implementation checks for template
|
||
|
// files in the addon directories. We need to override that
|
||
|
// check so that the template compiler always runs.
|
||
|
return true;
|
||
|
},
|
||
|
|
||
|
treeFor() {
|
||
|
// This addon doesn't contribute any 'real' trees to the app
|
||
|
return;
|
||
|
},
|
||
|
};
|