mirror of
https://github.com/discourse/discourse.git
synced 2025-01-18 13:43:16 +08:00
DEV: add loader.js shims for packages used across bundles (#22703)
This adds a new `loaderShim()` function to ensure certain modules are present in the `loader.js` registry and therefore runtime `require()`-able. Currently, the classic build pipeline puts a lot of things in the runtime `loader.js` registry automatically. For example, all of the ember-auto-import packages are in there. Going forward, and especially as we switch to the Embroider build pipeline, this will not be guarenteed. We need to keep an eye on what modules (packages) our "external" bundles (admin, wizard, markdown-it, plugins, etc) are expecting to be present and put them into the registry proactively.
This commit is contained in:
parent
a846226e92
commit
923b51ad25
|
@ -0,0 +1,58 @@
|
||||||
|
// Webpack has bugs, using globalThis is the safest
|
||||||
|
// https://github.com/embroider-build/embroider/issues/1545
|
||||||
|
let { define: __define__, require: __require__ } = globalThis;
|
||||||
|
|
||||||
|
// Traditionally, Ember compiled ES modules into AMD modules, which are then
|
||||||
|
// made usable in the browser at runtime via loader.js. In a classic build, all
|
||||||
|
// the modules, including any external ember-auto-imported dependencies, are
|
||||||
|
// added to the loader.js registry and therefore require()-able at runtime.
|
||||||
|
//
|
||||||
|
// Overtime, the AMD-ness of the modules, the ability to define arbitrarily
|
||||||
|
// named modules and the ability to require any modules and even enumerate the
|
||||||
|
// known modules at runtime (require.entries/_eak_seen) became heavily relied
|
||||||
|
// upon, which is problematic. For one thing, these features don't align well
|
||||||
|
// with ES modules semantics, and it is also impossible to perform tree-shaking
|
||||||
|
// as the presence of a particular module could end up being important even if
|
||||||
|
// it appears to be unused in the static analysis.
|
||||||
|
//
|
||||||
|
// For Discourse, the AMD/loader.js mechanism is an important glue. It is what
|
||||||
|
// allows Discourse core/admin/wizard/plugins to all be separate .js bundlers
|
||||||
|
// and be "glued back together" as full module graph in the browser.
|
||||||
|
//
|
||||||
|
// For instance, a plugin module can `import Post from "discourse/models/post";
|
||||||
|
// because the babel plugin compiled discourse/models/post.js into an AMD
|
||||||
|
// module into app.js (`define("discourse/models/post", ...)`), which makes
|
||||||
|
// it available in the runtime loader.js registry, and the plugin module itself
|
||||||
|
// is also compiled into AMD with a dependency on the core module.
|
||||||
|
//
|
||||||
|
// This has similar drawbacks as the general problem in the ecosystem, but in
|
||||||
|
// addition, it has a particular bad side-effect that any external dependencies
|
||||||
|
// (NPM packages) we use in core will automatically become a defacto public API
|
||||||
|
// for plugins to use as well, making it difficult for core to upgrade/remove
|
||||||
|
// dependencies (and thus so as introducing them in the first place).
|
||||||
|
//
|
||||||
|
// Ember is aggressively moving away from AMD modules and there are active RFCs
|
||||||
|
// to explore the path to deprecating AMD/loader.js. While it would still be
|
||||||
|
// fine (in the medium term at least) for us to use AMD/loader.js as an interop
|
||||||
|
// mechanism between our bundles, we will have to be more conscious about what
|
||||||
|
// to make available to plugins via this mechanism.
|
||||||
|
//
|
||||||
|
// In the meantime Embroider no longer automatically add AMD shims for external
|
||||||
|
// dependencies. In order to preserve compatibility for plugins, this utility
|
||||||
|
// allows us to manually force a particular module to be included in loader.js
|
||||||
|
// and available to plugins. Overtime we should review this list and start
|
||||||
|
// deprecating any accidental leakages.
|
||||||
|
//
|
||||||
|
// The general way to use it is:
|
||||||
|
//
|
||||||
|
// import { importSync } from "@embroider/macros";
|
||||||
|
//
|
||||||
|
// loaderShim("some-npm-pkg", () => importSync("some-npm-pkg"));
|
||||||
|
//
|
||||||
|
// Note that `importSync` is a macro which must be passed a string
|
||||||
|
// literal, therefore cannot be abstracted away.
|
||||||
|
export default function loaderShim(pkg, callback) {
|
||||||
|
if (!__require__.has(pkg)) {
|
||||||
|
__define__(pkg, callback);
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
import "./global-compat";
|
import "./global-compat";
|
||||||
|
import "./loader-shims";
|
||||||
|
|
||||||
import require from "require";
|
import require from "require";
|
||||||
import Application from "@ember/application";
|
import Application from "@ember/application";
|
||||||
|
|
32
app/assets/javascripts/discourse/app/loader-shims.js
Normal file
32
app/assets/javascripts/discourse/app/loader-shims.js
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
import { importSync } from "@embroider/macros";
|
||||||
|
import loaderShim from "discourse-common/lib/loader-shim";
|
||||||
|
|
||||||
|
// AMD shims for the app bunndle, see the comment in loader-shim.js
|
||||||
|
// These effectively become public APIs for plugins, so add/remove them carefully
|
||||||
|
loaderShim("@discourse/itsatrap", () => importSync("@discourse/itsatrap"));
|
||||||
|
loaderShim("@ember-compat/tracked-built-ins", () =>
|
||||||
|
importSync("@ember-compat/tracked-built-ins")
|
||||||
|
);
|
||||||
|
loaderShim("@popperjs/core", () => importSync("@popperjs/core"));
|
||||||
|
loaderShim("@uppy/aws-s3", () => importSync("@uppy/aws-s3"));
|
||||||
|
loaderShim("@uppy/aws-s3-multipart", () =>
|
||||||
|
importSync("@uppy/aws-s3-multipart")
|
||||||
|
);
|
||||||
|
loaderShim("@uppy/core", () => importSync("@uppy/core"));
|
||||||
|
loaderShim("@uppy/drop-target", () => importSync("@uppy/drop-target"));
|
||||||
|
loaderShim("@uppy/utils/lib/AbortController", () =>
|
||||||
|
importSync("@uppy/utils/lib/AbortController")
|
||||||
|
);
|
||||||
|
loaderShim("@uppy/utils/lib/delay", () => importSync("@uppy/utils/lib/delay"));
|
||||||
|
loaderShim("@uppy/utils/lib/EventTracker", () =>
|
||||||
|
importSync("@uppy/utils/lib/EventTracker")
|
||||||
|
);
|
||||||
|
loaderShim("@uppy/xhr-upload", () => importSync("@uppy/xhr-upload"));
|
||||||
|
loaderShim("a11y-dialog", () => importSync("a11y-dialog"));
|
||||||
|
loaderShim("ember-modifier", () => importSync("ember-modifier"));
|
||||||
|
loaderShim("handlebars", () => importSync("handlebars"));
|
||||||
|
loaderShim("js-yaml", () => importSync("js-yaml"));
|
||||||
|
loaderShim("message-bus-client", () => importSync("message-bus-client"));
|
||||||
|
loaderShim("tippy.js", () => importSync("tippy.js"));
|
||||||
|
loaderShim("virtual-dom", () => importSync("virtual-dom"));
|
||||||
|
loaderShim("xss", () => importSync("xss"));
|
|
@ -1,7 +1,7 @@
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
const EmberApp = require("ember-cli/lib/broccoli/ember-app");
|
const EmberApp = require("ember-cli/lib/broccoli/ember-app");
|
||||||
const resolve = require("path").resolve;
|
const { resolve, join } = require("path");
|
||||||
const mergeTrees = require("broccoli-merge-trees");
|
const mergeTrees = require("broccoli-merge-trees");
|
||||||
const concat = require("broccoli-concat");
|
const concat = require("broccoli-concat");
|
||||||
const { createI18nTree } = require("./lib/translation-plugin");
|
const { createI18nTree } = require("./lib/translation-plugin");
|
||||||
|
@ -138,6 +138,7 @@ module.exports = function (defaults) {
|
||||||
|
|
||||||
const testHelpers = concat(appTestTrees, {
|
const testHelpers = concat(appTestTrees, {
|
||||||
inputFiles: [
|
inputFiles: [
|
||||||
|
"**/tests/loader-shims.js",
|
||||||
"**/tests/test-boot-ember-cli.js",
|
"**/tests/test-boot-ember-cli.js",
|
||||||
"**/tests/helpers/**/*.js",
|
"**/tests/helpers/**/*.js",
|
||||||
"**/tests/fixtures/**/*.js",
|
"**/tests/fixtures/**/*.js",
|
||||||
|
@ -178,6 +179,10 @@ module.exports = function (defaults) {
|
||||||
"/app/assets/javascripts/discourse/public/assets/scripts/module-shims.js"
|
"/app/assets/javascripts/discourse/public/assets/scripts/module-shims.js"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// See: https://github.com/embroider-build/embroider/issues/1574
|
||||||
|
// Specifically, markdownItBundleTree is triggering the MacrosConfig error
|
||||||
|
finalizeEmbroiderMacrosConfigs(app, resolve("."), app.project);
|
||||||
|
|
||||||
const discoursePluginsTree = app.project
|
const discoursePluginsTree = app.project
|
||||||
.findAddonByName("discourse-plugins")
|
.findAddonByName("discourse-plugins")
|
||||||
.generatePluginsTree();
|
.generatePluginsTree();
|
||||||
|
@ -215,3 +220,21 @@ module.exports = function (defaults) {
|
||||||
discoursePluginsTree,
|
discoursePluginsTree,
|
||||||
]);
|
]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// See: https://github.com/embroider-build/embroider/issues/1574
|
||||||
|
function finalizeEmbroiderMacrosConfigs(appInstance, appRoot, parent) {
|
||||||
|
parent.initializeAddons?.();
|
||||||
|
|
||||||
|
for (let addon of parent.addons) {
|
||||||
|
if (addon.name === "@embroider/macros") {
|
||||||
|
const MacrosConfig = require(join(
|
||||||
|
addon.packageRoot,
|
||||||
|
"src",
|
||||||
|
"macros-config"
|
||||||
|
)).default;
|
||||||
|
MacrosConfig.for(appInstance, appRoot).finalize();
|
||||||
|
} else {
|
||||||
|
finalizeEmbroiderMacrosConfigs(appInstance, appRoot, addon);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -35,6 +35,7 @@
|
||||||
"@ember/render-modifiers": "^2.1.0",
|
"@ember/render-modifiers": "^2.1.0",
|
||||||
"@ember/string": "^3.1.1",
|
"@ember/string": "^3.1.1",
|
||||||
"@ember/test-helpers": "^2.9.4",
|
"@ember/test-helpers": "^2.9.4",
|
||||||
|
"@embroider/macros": "^1.13.1",
|
||||||
"@glimmer/component": "^1.1.2",
|
"@glimmer/component": "^1.1.2",
|
||||||
"@glimmer/tracking": "^1.1.2",
|
"@glimmer/tracking": "^1.1.2",
|
||||||
"@popperjs/core": "^2.11.8",
|
"@popperjs/core": "^2.11.8",
|
||||||
|
|
7
app/assets/javascripts/discourse/tests/loader-shims.js
Normal file
7
app/assets/javascripts/discourse/tests/loader-shims.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import { importSync } from "@embroider/macros";
|
||||||
|
import loaderShim from "discourse-common/lib/loader-shim";
|
||||||
|
|
||||||
|
// AMD shims for the test bunndle, see the comment in loader-shim.js
|
||||||
|
loaderShim("pretender", () => importSync("pretender"));
|
||||||
|
loaderShim("qunit", () => importSync("qunit"));
|
||||||
|
loaderShim("sinon", () => importSync("sinon"));
|
|
@ -1,3 +1,4 @@
|
||||||
|
import "./loader-shims";
|
||||||
import {
|
import {
|
||||||
applyPretender,
|
applyPretender,
|
||||||
exists,
|
exists,
|
||||||
|
|
7
app/assets/javascripts/pretty-text/addon/loader-shims.js
Normal file
7
app/assets/javascripts/pretty-text/addon/loader-shims.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import { importSync } from "@embroider/macros";
|
||||||
|
import loaderShim from "discourse-common/lib/loader-shim";
|
||||||
|
|
||||||
|
// AMD shims for the addon, see the comment in loader-shim.js
|
||||||
|
// These effectively become public APIs for plugins, so add/remove them carefully
|
||||||
|
// Note that this is included into the main app bundle for core
|
||||||
|
loaderShim("xss", () => importSync("xss"));
|
|
@ -1,3 +1,4 @@
|
||||||
|
import "./loader-shims";
|
||||||
import {
|
import {
|
||||||
cook as cookIt,
|
cook as cookIt,
|
||||||
setup as setupIt,
|
setup as setupIt,
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
"start": "ember serve"
|
"start": "ember serve"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@embroider/macros": "^1.13.1",
|
||||||
"discourse-common": "1.0.0",
|
"discourse-common": "1.0.0",
|
||||||
"ember-auto-import": "^2.6.3",
|
"ember-auto-import": "^2.6.3",
|
||||||
"ember-cli-babel": "^7.26.11",
|
"ember-cli-babel": "^7.26.11",
|
||||||
|
|
|
@ -1244,6 +1244,20 @@
|
||||||
resolve "^1.20.0"
|
resolve "^1.20.0"
|
||||||
semver "^7.3.2"
|
semver "^7.3.2"
|
||||||
|
|
||||||
|
"@embroider/macros@^1.13.1":
|
||||||
|
version "1.13.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/@embroider/macros/-/macros-1.13.1.tgz#aee17e5af0e0086bd36873bdb4e49ea346bab3fa"
|
||||||
|
integrity sha512-4htraP/rNIht8uCxXoc59Bw2EsBFfc4YUQD9XSpzJ4xUr1V0GQf9wL/noeSuYSxIhwRfZOErnJhsdyf1hH+I/A==
|
||||||
|
dependencies:
|
||||||
|
"@embroider/shared-internals" "2.4.0"
|
||||||
|
assert-never "^1.2.1"
|
||||||
|
babel-import-util "^2.0.0"
|
||||||
|
ember-cli-babel "^7.26.6"
|
||||||
|
find-up "^5.0.0"
|
||||||
|
lodash "^4.17.21"
|
||||||
|
resolve "^1.20.0"
|
||||||
|
semver "^7.3.2"
|
||||||
|
|
||||||
"@embroider/shared-internals@2.2.0", "@embroider/shared-internals@^2.0.0", "@embroider/shared-internals@^2.1.0":
|
"@embroider/shared-internals@2.2.0", "@embroider/shared-internals@^2.0.0", "@embroider/shared-internals@^2.1.0":
|
||||||
version "2.2.0"
|
version "2.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/@embroider/shared-internals/-/shared-internals-2.2.0.tgz#7c4f5cf8f7289b36ffbfc04f6b4c71876713c869"
|
resolved "https://registry.yarnpkg.com/@embroider/shared-internals/-/shared-internals-2.2.0.tgz#7c4f5cf8f7289b36ffbfc04f6b4c71876713c869"
|
||||||
|
@ -1258,6 +1272,21 @@
|
||||||
semver "^7.3.5"
|
semver "^7.3.5"
|
||||||
typescript-memoize "^1.0.1"
|
typescript-memoize "^1.0.1"
|
||||||
|
|
||||||
|
"@embroider/shared-internals@2.4.0":
|
||||||
|
version "2.4.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@embroider/shared-internals/-/shared-internals-2.4.0.tgz#0e9fdb0b2df9bad45fab8c54cbb70d8a2cbf01fc"
|
||||||
|
integrity sha512-pFE05ebenWMC9XAPRjadYCXXb6VmqjkhYN5uqkhPo+VUmMHnx7sZYYxqGjxfVuhC/ghS/BNlOffOCXDOoE7k7g==
|
||||||
|
dependencies:
|
||||||
|
babel-import-util "^2.0.0"
|
||||||
|
debug "^4.3.2"
|
||||||
|
ember-rfc176-data "^0.3.17"
|
||||||
|
fs-extra "^9.1.0"
|
||||||
|
js-string-escape "^1.0.1"
|
||||||
|
lodash "^4.17.21"
|
||||||
|
resolve-package-path "^4.0.1"
|
||||||
|
semver "^7.3.5"
|
||||||
|
typescript-memoize "^1.0.1"
|
||||||
|
|
||||||
"@embroider/test-setup@^3.0.1":
|
"@embroider/test-setup@^3.0.1":
|
||||||
version "3.0.1"
|
version "3.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/@embroider/test-setup/-/test-setup-3.0.1.tgz#603b21a809708ac928fe9f002905ee3711dc6864"
|
resolved "https://registry.yarnpkg.com/@embroider/test-setup/-/test-setup-3.0.1.tgz#603b21a809708ac928fe9f002905ee3711dc6864"
|
||||||
|
@ -2278,6 +2307,11 @@ babel-import-util@^1.1.0, babel-import-util@^1.2.2, babel-import-util@^1.3.0, ba
|
||||||
resolved "https://registry.yarnpkg.com/babel-import-util/-/babel-import-util-1.4.1.tgz#1df6fd679845df45494bac9ca12461d49497fdd4"
|
resolved "https://registry.yarnpkg.com/babel-import-util/-/babel-import-util-1.4.1.tgz#1df6fd679845df45494bac9ca12461d49497fdd4"
|
||||||
integrity sha512-TNdiTQdPhXlx02pzG//UyVPSKE7SNWjY0n4So/ZnjQpWwaM5LvWBLkWa1JKll5u06HNscHD91XZPuwrMg1kadQ==
|
integrity sha512-TNdiTQdPhXlx02pzG//UyVPSKE7SNWjY0n4So/ZnjQpWwaM5LvWBLkWa1JKll5u06HNscHD91XZPuwrMg1kadQ==
|
||||||
|
|
||||||
|
babel-import-util@^2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/babel-import-util/-/babel-import-util-2.0.0.tgz#99a2e7424bcde01898bc61bb19700ff4c74379a3"
|
||||||
|
integrity sha512-pkWynbLwru0RZmA9iKeQL63+CkkW0RCP3kL5njCtudd6YPUKb5Pa0kL4fb3bmuKn2QDBFwY5mvvhEK/+jv2Ynw==
|
||||||
|
|
||||||
babel-loader@^8.0.6:
|
babel-loader@^8.0.6:
|
||||||
version "8.3.0"
|
version "8.3.0"
|
||||||
resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.3.0.tgz#124936e841ba4fe8176786d6ff28add1f134d6a8"
|
resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.3.0.tgz#124936e841ba4fe8176786d6ff28add1f134d6a8"
|
||||||
|
|
|
@ -1,3 +1,21 @@
|
||||||
|
define("@embroider/macros", ["exports", "require"], function (
|
||||||
|
__require__,
|
||||||
|
__exports__
|
||||||
|
) {
|
||||||
|
__exports__.importSync = __require__;
|
||||||
|
});
|
||||||
|
|
||||||
|
define("discourse-common/lib/loader-shim", ["exports", "require"], function (
|
||||||
|
__exports__,
|
||||||
|
__require__
|
||||||
|
) {
|
||||||
|
__exports__.default = (id, callback) => {
|
||||||
|
if (!__require__.has(id)) {
|
||||||
|
define(id, callback);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
define("xss", ["exports"], function (__exports__) {
|
define("xss", ["exports"], function (__exports__) {
|
||||||
__exports__.default = window.filterXSS;
|
__exports__.default = window.filterXSS;
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue
Block a user