use std::{
borrow::Cow,
path::{Path, PathBuf},
sync::{Arc, Mutex, Weak},
};
use anyhow::Context;
use arcstr::ArcStr;
use derive_more::Debug;
use rolldown_common::{
FilenameTemplate, LogLevel, LogWithoutPlugin, ModuleDefFormat, ModuleId, ModuleLoaderMsg,
PackageJson, PluginIdx, ResolvedId, SharedFileEmitter, SharedModuleInfoDashMap,
SharedNormalizedBundlerOptions, side_effects::HookSideEffects,
};
use rolldown_resolver::{ResolveError, Resolver};
use rolldown_utils::dashmap::FxDashSet;
use tracing::Instrument;
use crate::{
PluginDriver,
plugin_context::PluginContextMeta,
types::{
hook_resolve_id_skipped::HookResolveIdSkipped,
plugin_context_resolve_options::PluginContextResolveOptions,
},
utils::resolve_id_check_external::resolve_id_check_external,
};
pub type SharedNativePluginContext = Arc<NativePluginContextImpl>;
#[derive(Debug)]
pub struct NativePluginContextImpl {
pub(crate) plugin_name: Cow<'static, str>,
pub(crate) skipped_resolve_calls: Vec<Arc<HookResolveIdSkipped>>,
pub(crate) plugin_idx: PluginIdx,
pub(crate) resolver: Arc<Resolver>,
pub(crate) meta: Arc<PluginContextMeta>,
pub(crate) plugin_driver: Weak<PluginDriver>,
pub(crate) file_emitter: SharedFileEmitter,
pub(crate) options: SharedNormalizedBundlerOptions,
pub(crate) watch_files: Arc<FxDashSet<ArcStr>>,
pub(crate) module_infos: SharedModuleInfoDashMap,
pub(crate) tx: Arc<Mutex<Option<tokio::sync::mpsc::UnboundedSender<ModuleLoaderMsg>>>>,
pub(crate) session: rolldown_devtools::Session,
pub(crate) bundle_span: Arc<tracing::Span>,
pub(crate) manual_resolve_span: Arc<tracing::Span>,
}
impl NativePluginContextImpl {
pub async fn load(
&self,
specifier: &str,
side_effects: Option<HookSideEffects>,
module_def_format: ModuleDefFormat,
) -> anyhow::Result<()> {
let sender = {
let guard = self.tx.lock().ok().context("Failed to acquire PluginDriver tx lock")?.clone();
guard.context("The `PluginContext.load` only work at `resolveId/load/transform/moduleParsed` hooks. If you using it at resolveId hook, please make sure it could not load the entry module.")?
};
sender
.send(ModuleLoaderMsg::FetchModule(Box::new(ResolvedId {
id: ModuleId::new(specifier),
side_effects,
module_def_format,
..Default::default()
})))
.map_err(|e| anyhow::Error::new(e).context("PluginContext: failed to send FetchModule message - module loader shut down during plugin execution"))?;
let plugin_driver = self
.plugin_driver
.upgrade()
.ok_or_else(|| anyhow::anyhow!("Plugin driver is already dropped."))?;
plugin_driver.wait_for_module_load_completion(specifier).await;
Ok(())
}
pub fn try_get_package_json_or_create(&self, path: &Path) -> anyhow::Result<Arc<PackageJson>> {
self.resolver.try_get_package_json_or_create(path)
}
pub async fn resolve(
&self,
specifier: &str,
importer: Option<&str>,
extra_options: Option<PluginContextResolveOptions>,
) -> anyhow::Result<Result<ResolvedId, ResolveError>> {
let plugin_driver = self
.plugin_driver
.upgrade()
.ok_or_else(|| anyhow::anyhow!("Plugin driver is already dropped."))?;
let normalized_extra_options = extra_options.unwrap_or_default();
let skipped_resolve_calls = if normalized_extra_options.skip_self {
let mut skipped_resolve_calls = Vec::with_capacity(self.skipped_resolve_calls.len() + 1);
skipped_resolve_calls.extend(self.skipped_resolve_calls.clone());
skipped_resolve_calls.push(Arc::new(HookResolveIdSkipped {
plugin_idx: self.plugin_idx,
importer: importer.map(Into::into),
specifier: specifier.into(),
}));
Some(skipped_resolve_calls)
} else if !self.skipped_resolve_calls.is_empty() {
Some(self.skipped_resolve_calls.clone())
} else {
None
};
async {
resolve_id_check_external(
&self.resolver,
&plugin_driver,
specifier,
importer,
normalized_extra_options.is_entry,
normalized_extra_options.import_kind,
skipped_resolve_calls,
normalized_extra_options.custom,
false,
&self.options,
)
.await
}
.instrument(self.manual_resolve_span.as_ref().clone())
.await
}
pub fn emit_chunk(&self, chunk: rolldown_common::EmittedChunk) -> anyhow::Result<ArcStr> {
self.file_emitter.emit_chunk(Arc::new(chunk))
}
pub fn emit_prebuilt_chunk(&self, chunk: rolldown_common::EmittedPrebuiltChunk) -> ArcStr {
self.file_emitter.emit_prebuilt_chunk(chunk)
}
pub fn emit_file(
&self,
file: rolldown_common::EmittedAsset,
fn_asset_filename: Option<String>,
fn_sanitized_file_name: Option<String>,
) -> anyhow::Result<ArcStr> {
let file_name_is_none = file.file_name.is_none();
let asset_filename_template = file_name_is_none.then(|| {
FilenameTemplate::new(self.options.asset_filenames.value(fn_asset_filename), "assetFileNames")
});
let sanitized_file_name = file_name_is_none.then(|| {
self.options.sanitize_filename.value(file.name_for_sanitize(), fn_sanitized_file_name)
});
self.file_emitter.emit_file(file, asset_filename_template, sanitized_file_name)
}
pub async fn emit_file_async(
&self,
file: rolldown_common::EmittedAsset,
) -> anyhow::Result<ArcStr> {
let asset_filename = self.options.asset_filename_with_file(&file).await?;
let sanitized_file_name = self.options.sanitize_file_name_with_file(&file).await?;
self.file_emitter.emit_file(file, asset_filename, sanitized_file_name)
}
pub fn get_file_name(&self, reference_id: &str) -> anyhow::Result<ArcStr> {
self.file_emitter.get_file_name(reference_id)
}
pub fn associate_module_with_file_ref(&self, module_id: &str, reference_id: &str) {
self.file_emitter.associate_module_with_file_ref(module_id, reference_id);
}
pub fn get_module_info(&self, module_id: &str) -> Option<Arc<rolldown_common::ModuleInfo>> {
self.module_infos.get(module_id).map(|v| Arc::<rolldown_common::ModuleInfo>::clone(v.value()))
}
pub fn get_module_ids(&self) -> Vec<ArcStr> {
self.module_infos.iter().map(|v| v.key().clone()).collect()
}
pub fn cwd(&self) -> &PathBuf {
self.resolver.cwd()
}
pub fn add_watch_file(&self, file: &str) {
self.watch_files.insert(file.into());
}
fn log(&self, level: LogLevel, log: LogWithoutPlugin) {
if let Some(on_log) = &self.options.on_log {
let on_log = on_log.clone();
let log = log.into_log(Some(self.plugin_name.to_string()));
rolldown_utils::futures::spawn(async move {
let _ = on_log.call(level, log).await;
});
}
}
#[inline]
pub fn info(&self, log: LogWithoutPlugin) {
self.log(LogLevel::Info, log);
}
#[inline]
pub fn warn(&self, log: LogWithoutPlugin) {
self.log(LogLevel::Warn, log);
}
#[inline]
pub fn debug(&self, log: LogWithoutPlugin) {
self.log(LogLevel::Debug, log);
}
}