rolldown_plugin_utils 0.1.0

Shared utilities for Rolldown plugins
Documentation
use std::{borrow::Cow, path::Path};

use cow_utils::CowUtils as _;
use rolldown_common::OutputFormat;
use sugar_path::SugarPath as _;

use super::to_output_file_path::AssetUrlResult;

const CURRENT_SCRIPT_URL_OR_BASE_URI: &str = "typeof document === 'undefined' ? location.href : document.currentScript && document.currentScript.tagName.toUpperCase() === 'SCRIPT' && document.currentScript.src || document.baseURI";

pub fn create_to_import_meta_url_based_relative_runtime(
  format: OutputFormat,
  is_worker: bool,
) -> impl Fn(&Path, &Path) -> AssetUrlResult {
  let to_relative_path = match format {
    OutputFormat::Esm => es,
    OutputFormat::Cjs => cjs,
    OutputFormat::Umd => umd,
    OutputFormat::Iife => {
      if is_worker {
        worker_iife
      } else {
        iife
      }
    }
  };
  move |filename: &Path, importer: &Path| -> AssetUrlResult {
    let path = filename.relative(importer.parent().unwrap_or(importer));
    AssetUrlResult::WithoutRuntime(to_relative_path(&path.to_slash_lossy()))
  }
}

fn cjs(path: &str) -> String {
  format!(
    "(typeof document === 'undefined' ? {} : {})",
    get_file_url_from_relative_path(path),
    get_relative_url_from_document(path, false)
  )
}

fn es(path: &str) -> String {
  format!("new URL('{}', import.meta.url).href", escape_id(&partial_encode_url_path(path)))
}

fn iife(path: &str) -> String {
  get_relative_url_from_document(path, false)
}

fn umd(path: &str) -> String {
  format!(
    "(typeof document === 'undefined' && typeof location === 'undefined' ? {} : {})",
    get_file_url_from_relative_path(path),
    get_relative_url_from_document(path, true)
  )
}

fn worker_iife(path: &str) -> String {
  format!("new URL('{}', self.location.href).href", escape_id(&partial_encode_url_path(path)))
}

pub fn partial_encode_url_path(url: &str) -> Cow<'_, str> {
  if url.starts_with("data:") {
    return Cow::Borrowed(url);
  }
  let file_path = rolldown_utils::url::clean_url(url);
  Cow::Owned(format!("{}{}", file_path.cow_replace('%', "%25"), &url[file_path.len()..]))
}

fn get_file_url_from_relative_path(path: &str) -> String {
  format!("require('u' + 'rl').pathToFileURL(__dirname + '/{}').href", escape_id(path))
}

fn get_relative_url_from_document(path: &str, is_umd: bool) -> String {
  format!(
    "new URL({}, {}).href",
    escape_id(&partial_encode_url_path(path)),
    if is_umd { CURRENT_SCRIPT_URL_OR_BASE_URI } else { &CURRENT_SCRIPT_URL_OR_BASE_URI[50..] }
  )
}

fn escape_id(id: &str) -> Cow<'_, str> {
  if id.contains(['\n', '\r', '\\', '\u{2028}', '\u{2029}']) {
    let mut result = String::with_capacity(id.len() + 2);
    for c in id.chars() {
      match c {
        '\\' => result.push_str("\\\\"),
        '\n' => result.push_str("\\\n"),
        '\r' => result.push_str("\\\r"),
        '\u{2028}' => result.push_str("\\\u{2028}"),
        '\u{2029}' => result.push_str("\\\u{2029}"),
        _ => result.push(c),
      }
    }
    Cow::Owned(result)
  } else {
    Cow::Borrowed(id)
  }
}