use rayon::prelude::*;
use rspack_collections::IdentifierSet;
use rspack_core::{ModuleIdentifier, SourceType};
use super::ModuleGroupMap;
use crate::{
CacheGroup, SplitChunkSizes, SplitChunksPlugin, common::ModuleSizes, module_group::ModuleGroup,
};
pub trait ModulesContainer {
fn get_sizes(&mut self, module_sizes: &ModuleSizes) -> SplitChunkSizes;
fn get_source_types_modules(
&self,
source_types: &[SourceType],
module_sizes: &ModuleSizes,
) -> IdentifierSet;
fn remove_module(&mut self, module: ModuleIdentifier);
fn modules(&self) -> &IdentifierSet;
}
impl ModulesContainer for ModuleGroup {
fn get_sizes(&mut self, module_sizes: &ModuleSizes) -> SplitChunkSizes {
ModuleGroup::get_sizes(self, module_sizes)
}
fn get_source_types_modules(
&self,
source_types: &[SourceType],
module_sizes: &ModuleSizes,
) -> IdentifierSet {
ModuleGroup::get_source_types_modules(self, source_types, module_sizes)
}
fn remove_module(&mut self, module: ModuleIdentifier) {
ModuleGroup::remove_module(self, module);
}
fn modules(&self) -> &IdentifierSet {
&self.modules
}
}
pub(crate) fn remove_min_size_violating_modules(
module_group_key: &str,
module_group: &mut ModuleGroup,
cache_group: &CacheGroup,
module_sizes: &ModuleSizes,
) -> bool {
let violating_source_types: Box<[SourceType]> = module_group
.get_sizes(module_sizes)
.iter()
.filter_map(|(module_group_ty, module_group_ty_size)| {
let cache_group_ty_min_size = cache_group
.min_size
.get(module_group_ty)
.copied()
.unwrap_or_default();
if *module_group_ty_size < cache_group_ty_min_size {
tracing::trace!(
"ModuleGroup({}) have violating SourceType({:?}). Reason: module_group_ty_size({:?}) < CacheGroup({}).min_size({:?})",
module_group_key,
module_group_ty,
module_group_ty_size,
cache_group.key,
cache_group_ty_min_size,
);
Some(*module_group_ty)
} else {
None
}
})
.collect::<Box<[_]>>();
if violating_source_types.is_empty() {
return module_group.modules.is_empty();
}
let violating_modules =
module_group.get_source_types_modules(&violating_source_types, module_sizes);
for violating_module in violating_modules {
module_group.remove_module(violating_module);
}
module_group.modules.is_empty()
}
impl SplitChunksPlugin {
pub(crate) fn check_min_size_reduction(
sizes: &SplitChunkSizes,
min_size_reduction: &SplitChunkSizes,
chunk_count: usize,
) -> bool {
for (ty, min_reduction_size) in min_size_reduction.iter() {
if *min_reduction_size == 0.0f64 {
continue;
}
let Some(size) = sizes.get(ty) else {
continue;
};
if *size == 0.0f64 {
continue;
}
if size * (chunk_count as f64) < *min_reduction_size {
return false;
}
}
true
}
pub(crate) fn ensure_min_size_fit(
&self,
module_group_map: &mut ModuleGroupMap,
module_sizes: &ModuleSizes,
) {
let invalidated_module_groups = module_group_map
.par_iter_mut()
.filter_map(|(module_group_key, module_group)| {
let cache_group = module_group.get_cache_group(&self.cache_groups);
if cache_group.min_size.is_empty() {
let _ = module_group.get_sizes(module_sizes);
tracing::debug!(
"ModuleGroup({}) skips `minSize` checking. Reason: min_size of CacheGroup({}) is empty",
module_group_key,
cache_group.key,
);
return None;
}
if remove_min_size_violating_modules(
module_group_key,
module_group,
cache_group,
module_sizes,
) || !Self::check_min_size_reduction(
&module_group.get_sizes(module_sizes),
&cache_group.min_size_reduction,
module_group.chunks.len(),
) {
Some(module_group_key.clone())
} else {
None
}
})
.collect::<Vec<_>>();
invalidated_module_groups.into_iter().for_each(|key| {
tracing::debug!(
"ModuleGroup({}) is removed. Reason: empty modules cause by `minSize` checking",
key,
);
module_group_map.swap_remove(&key);
});
}
}