Skip to content

Commit

Permalink
feat: Module Federation, part 2, ContainerReferencePlugin
Browse files Browse the repository at this point in the history
  • Loading branch information
ahabhgk committed Nov 22, 2023
1 parent a69a863 commit 08f61d3
Show file tree
Hide file tree
Showing 22 changed files with 895 additions and 55 deletions.
28 changes: 13 additions & 15 deletions crates/node_binding/binding.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,7 @@ export const enum BuiltinPluginName {
WebWorkerTemplatePlugin = 'WebWorkerTemplatePlugin',
MergeDuplicateChunksPlugin = 'MergeDuplicateChunksPlugin',
ContainerPlugin = 'ContainerPlugin',
ContainerReferencePlugin = 'ContainerReferencePlugin',
HttpExternalsRspackPlugin = 'HttpExternalsRspackPlugin',
CopyRspackPlugin = 'CopyRspackPlugin',
HtmlRspackPlugin = 'HtmlRspackPlugin',
Expand Down Expand Up @@ -486,13 +487,6 @@ export interface JsStatsWarning {
formatted: string
}

export interface NodeFS {
writeFile: (...args: any[]) => any
removeFile: (...args: any[]) => any
mkdir: (...args: any[]) => any
mkdirp: (...args: any[]) => any
}

export interface PathData {
filename?: string
hash?: string
Expand Down Expand Up @@ -621,6 +615,12 @@ export interface RawContainerPluginOptions {
exposes: Array<RawExposeOptions>
}

export interface RawContainerReferencePluginOptions {
remoteType: string
remotes: Array<RawRemoteOptions>
shareScope?: string
}

export interface RawCopyGlobOptions {
caseSensitiveMatch?: boolean
dot?: boolean
Expand Down Expand Up @@ -1019,6 +1019,12 @@ export interface RawRelayConfig {
language: 'javascript' | 'typescript' | 'flow'
}

export interface RawRemoteOptions {
key: string
external: Array<string>
shareScope: string
}

export interface RawResolveOptions {
preferRelative?: boolean
extensions?: Array<string>
Expand Down Expand Up @@ -1128,11 +1134,3 @@ export function registerGlobalTrace(filter: string, layer: "chrome" | "logger",
/** Builtin loader runner */
export function runBuiltinLoader(builtin: string, options: string | undefined | null, loaderContext: JsLoaderContext): Promise<JsLoaderContext>

export interface ThreadsafeNodeFS {
writeFile: (...args: any[]) => any
removeFile: (...args: any[]) => any
mkdir: (...args: any[]) => any
mkdirp: (...args: any[]) => any
removeDirAll: (...args: any[]) => any
}

13 changes: 12 additions & 1 deletion crates/rspack_binding_options/src/options/raw_builtins/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,8 @@ use napi::{
};
use napi_derive::napi;
use rspack_core::{
mf::ContainerPlugin, BoxPlugin, Define, DefinePlugin, PluginExt, Provide, ProvidePlugin,
mf::{ContainerPlugin, ContainerReferencePlugin},
BoxPlugin, Define, DefinePlugin, PluginExt, Provide, ProvidePlugin,
};
use rspack_error::Result;
use rspack_napi_shared::NapiResultExt;
Expand All @@ -38,6 +39,7 @@ use rspack_plugin_swc_js_minimizer::SwcJsMinimizerRspackPlugin;
use rspack_plugin_wasm::enable_wasm_loading_plugin;
use rspack_plugin_web_worker_template::web_worker_template_plugin;

use self::raw_mf::RawContainerReferencePluginOptions;
pub use self::{
raw_banner::RawBannerPluginOptions, raw_copy::RawCopyRspackPluginOptions,
raw_html::RawHtmlRspackPluginOptions, raw_limit_chunk_count::RawLimitChunkCountPluginOptions,
Expand Down Expand Up @@ -72,6 +74,7 @@ pub enum BuiltinPluginName {
WebWorkerTemplatePlugin,
MergeDuplicateChunksPlugin,
ContainerPlugin,
ContainerReferencePlugin,

// rspack specific plugins
HttpExternalsRspackPlugin,
Expand Down Expand Up @@ -184,6 +187,14 @@ impl RawOptionsApply for BuiltinPlugin {
.boxed(),
);
}
BuiltinPluginName::ContainerReferencePlugin => {
plugins.push(
ContainerReferencePlugin::new(
downcast_into::<RawContainerReferencePluginOptions>(self.options)?.into(),
)
.boxed(),
);
}

// rspack specific plugins
BuiltinPluginName::HttpExternalsRspackPlugin => {
Expand Down
42 changes: 41 additions & 1 deletion crates/rspack_binding_options/src/options/raw_builtins/raw_mf.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
use napi_derive::napi;
use rspack_core::mf::{ContainerPluginOptions, ExposeOptions};
use rspack_core::mf::{
ContainerPluginOptions, ContainerReferencePluginOptions, ExposeOptions, RemoteOptions,
};

use crate::RawLibraryOptions;

Expand Down Expand Up @@ -46,3 +48,41 @@ impl From<RawExposeOptions> for (String, ExposeOptions) {
)
}
}

#[derive(Debug)]
#[napi(object)]
pub struct RawContainerReferencePluginOptions {
pub remote_type: String,
pub remotes: Vec<RawRemoteOptions>,
pub share_scope: Option<String>,
}

impl From<RawContainerReferencePluginOptions> for ContainerReferencePluginOptions {
fn from(value: RawContainerReferencePluginOptions) -> Self {
Self {
remote_type: value.remote_type,
remotes: value.remotes.into_iter().map(|e| e.into()).collect(),
share_scope: value.share_scope,
}
}
}

#[derive(Debug)]
#[napi(object)]
pub struct RawRemoteOptions {
pub key: String,
pub external: Vec<String>,
pub share_scope: String,
}

impl From<RawRemoteOptions> for (String, RemoteOptions) {
fn from(value: RawRemoteOptions) -> Self {
(
value.key,
RemoteOptions {
external: value.external,
share_scope: value.share_scope,
},
)
}
}
3 changes: 3 additions & 0 deletions crates/rspack_core/src/dependency/dependency_type.rs
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,8 @@ pub enum DependencyType {
ContainerExposed,
/// container entry,
ContainerEntry,
/// remote to external,
RemoteToExternal,
Custom(Box<str>), // TODO it will increase large layout size
}

Expand Down Expand Up @@ -107,6 +109,7 @@ impl DependencyType {
DependencyType::ImportMetaContext => Cow::Borrowed("import.meta context"),
DependencyType::ContainerExposed => Cow::Borrowed("container exposed"),
DependencyType::ContainerEntry => Cow::Borrowed("container entry"),
DependencyType::RemoteToExternal => Cow::Borrowed("remote to external"),
}
}
}
Expand Down
47 changes: 41 additions & 6 deletions crates/rspack_core/src/external_module.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ use rustc_hash::FxHashMap as HashMap;
use serde::Serialize;

use crate::{
property_access,
extract_url_and_global, property_access,
rspack_sources::{BoxSource, RawSource, Source, SourceExt},
to_identifier, AsyncDependenciesBlockIdentifier, BuildContext, BuildInfo, BuildMetaExportsType,
BuildResult, ChunkInitFragments, ChunkUkey, CodeGenerationDataUrl, CodeGenerationResult,
Expand Down Expand Up @@ -134,12 +134,45 @@ impl ExternalModule {
)
}

fn get_source_for_script_external(
&self,
url_and_global: &ExternalRequestValue,
runtime_requirements: &mut RuntimeGlobals,
) -> Result<String> {
let url_and_global = extract_url_and_global(url_and_global.primary())?;
runtime_requirements.insert(RuntimeGlobals::LOAD_SCRIPT);
Ok(format!(
r#"
var __webpack_error__ = new Error();
module.exports = new Promise(function(resolve, reject) {{
if(typeof {global} !== "undefined") return resolve();
{load_script}({url_str}, function(event) {{
if(typeof {global} !== "undefined") return resolve();
var errorType = event && (event.type === 'load' ? 'missing' : event.type);
var realSrc = event && event.target && event.target.src;
__webpack_error__.message = 'Loading script failed.\n(' + errorType + ': ' + realSrc + ')';
__webpack_error__.name = 'ScriptExternalLoadError';
__webpack_error__.type = errorType;
__webpack_error__.request = realSrc;
reject(__webpack_error__);
}}, {global_str});
}}).then(function() {{ return {global}; }});
"#,
global = url_and_global.global,
global_str =
serde_json::to_string(url_and_global.global).map_err(|e| internal_error!(e.to_string()))?,
url_str =
serde_json::to_string(url_and_global.url).map_err(|e| internal_error!(e.to_string()))?,
load_script = RuntimeGlobals::LOAD_SCRIPT.name()
))
}

fn get_source(
&self,
compilation: &Compilation,
request: Option<&ExternalRequestValue>,
external_type: &ExternalType,
) -> (BoxSource, ChunkInitFragments, RuntimeGlobals) {
) -> Result<(BoxSource, ChunkInitFragments, RuntimeGlobals)> {
let mut chunk_init_fragments: ChunkInitFragments = Default::default();
let mut runtime_requirements: RuntimeGlobals = Default::default();
let source = match self.external_type.as_str() {
Expand Down Expand Up @@ -231,15 +264,17 @@ impl ExternalModule {
self.get_source_for_import(request, compilation)
}
}
// TODO "script"
"script" if let Some(request) = request => {
self.get_source_for_script_external(request, &mut runtime_requirements)?
}
_ => "".to_string(),
};
runtime_requirements.insert(RuntimeGlobals::MODULE);
(
Ok((
RawSource::from(source).boxed(),
chunk_init_fragments,
runtime_requirements,
)
))
}
}

Expand Down Expand Up @@ -379,7 +414,7 @@ impl Module for ExternalModule {
}
_ => {
let (source, chunk_init_fragments, runtime_requirements) =
self.get_source(compilation, request, external_type);
self.get_source(compilation, request, external_type)?;
cgr.add(SourceType::JavaScript, source);
cgr.chunk_init_fragments = chunk_init_fragments;
cgr.runtime_requirements.insert(runtime_requirements);
Expand Down
6 changes: 6 additions & 0 deletions crates/rspack_core/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,8 @@ pub enum SourceType {
Css,
Wasm,
Asset,
Remote,
ShareInit,
#[default]
Unknown,
}
Expand All @@ -106,6 +108,8 @@ impl std::fmt::Display for SourceType {
SourceType::Css => write!(f, "css"),
SourceType::Wasm => write!(f, "wasm"),
SourceType::Asset => write!(f, "asset"),
SourceType::Remote => write!(f, "remote"),
SourceType::ShareInit => write!(f, "share-init"),
SourceType::Unknown => write!(f, "unknown"),
}
}
Expand All @@ -132,6 +136,7 @@ pub enum ModuleType {
AssetSource,
Asset,
Runtime,
Remote,
}

impl ModuleType {
Expand Down Expand Up @@ -216,6 +221,7 @@ impl ModuleType {
ModuleType::AssetResource => "asset/resource",
ModuleType::AssetInline => "asset/inline",
ModuleType::Runtime => "runtime",
ModuleType::Remote => "remote-module",
}
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,12 +52,6 @@ impl ModuleDependency for ContainerEntryDependency {
fn request(&self) -> &str {
&self.resource_identifier
}

fn user_request(&self) -> &str {
&self.resource_identifier
}

fn set_request(&mut self, _request: String) {}
}

impl AsContextDependency for ContainerEntryDependency {}
Expand Down
7 changes: 4 additions & 3 deletions crates/rspack_core/src/mf/container/container_entry_module.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,14 @@ pub struct ContainerEntryModule {
blocks: Vec<AsyncDependenciesBlockIdentifier>,
dependencies: Vec<DependencyId>,
identifier: ModuleIdentifier,
name: String,
lib_ident: String,
exposes: Vec<(String, ExposeOptions)>,
share_scope: String,
}

impl ContainerEntryModule {
pub fn new(name: String, exposes: Vec<(String, ExposeOptions)>, share_scope: String) -> Self {
let lib_ident = format!("webpack/container/entry/{}", &name);
Self {
blocks: Vec::new(),
dependencies: Vec::new(),
Expand All @@ -35,7 +36,7 @@ impl ContainerEntryModule {
share_scope,
serde_json::to_string(&exposes).expect("should able to json to_string")
)),
name,
lib_ident,
exposes,
share_scope,
}
Expand Down Expand Up @@ -89,7 +90,7 @@ impl Module for ContainerEntryModule {
}

fn lib_ident(&self, _options: LibIdentOptions) -> Option<Cow<str>> {
Some(format!("webpack/container/entry/{}", self.name).into())
Some(self.lib_ident.as_str().into())
}

async fn build(
Expand Down
Loading

0 comments on commit 08f61d3

Please sign in to comment.