rspack_plugin_javascript 0.100.0-rc.3

rspack javascript plugin
Documentation
use rspack_cacheable::{cacheable, cacheable_dyn};
use rspack_core::{
  Compilation, DependencyCodeGeneration, DependencyLocation, DependencyRange, DependencyTemplate,
  DependencyTemplateType, RuntimeGlobals, RuntimeSpec, TemplateContext, TemplateReplaceSource,
};
use rspack_util::ext::DynHash;

#[cacheable]
#[derive(Debug, Clone)]
pub struct ModuleArgumentDependency {
  id: Option<String>,
  range: DependencyRange,
  loc: Option<DependencyLocation>,
}

impl ModuleArgumentDependency {
  pub fn new(id: Option<String>, range: DependencyRange, loc: Option<DependencyLocation>) -> Self {
    Self { id, range, loc }
  }

  pub fn loc(&self) -> Option<DependencyLocation> {
    self.loc.clone()
  }
}

#[cacheable_dyn]
impl DependencyCodeGeneration for ModuleArgumentDependency {
  fn dependency_template(&self) -> Option<DependencyTemplateType> {
    Some(ModuleArgumentDependencyTemplate::template_type())
  }

  fn update_hash(
    &self,
    hasher: &mut dyn std::hash::Hasher,
    _compilation: &Compilation,
    _runtime: Option<&RuntimeSpec>,
  ) {
    self.id.dyn_hash(hasher);
    self.range.dyn_hash(hasher);
  }
}

#[cacheable]
#[derive(Debug, Clone, Default)]
pub struct ModuleArgumentDependencyTemplate;

impl ModuleArgumentDependencyTemplate {
  pub fn template_type() -> DependencyTemplateType {
    DependencyTemplateType::Custom("ModuleArgumentDependency")
  }
}

impl DependencyTemplate for ModuleArgumentDependencyTemplate {
  fn render(
    &self,
    dep: &dyn DependencyCodeGeneration,
    source: &mut TemplateReplaceSource,
    code_generatable_context: &mut TemplateContext,
  ) {
    let dep = dep
      .as_any()
      .downcast_ref::<ModuleArgumentDependency>()
      .expect("ModuleArgumentDependencyTemplate should be used for ModuleArgumentDependency");

    let TemplateContext {
      compilation,
      module,
      runtime_template,
      ..
    } = code_generatable_context;

    let module_argument = runtime_template.render_module_argument(
      compilation
        .get_module_graph()
        .module_by_identifier(&module.identifier())
        .expect("should have mgm")
        .get_module_argument(),
    );

    let content = if let Some(id) = &dep.id {
      match id.as_str() {
        "id" => runtime_template
          .runtime_requirements_mut()
          .insert(RuntimeGlobals::MODULE_ID),
        "loaded" => runtime_template
          .runtime_requirements_mut()
          .insert(RuntimeGlobals::MODULE_LOADED),
        _ => {}
      };

      format!("{module_argument}.{id}")
    } else {
      module_argument
    };

    source.replace(dep.range.start, dep.range.end, content, None);
  }
}