swc_bundler 2.0.0

Very fast ecmascript bundler
Documentation
#![allow(dead_code)]

use std::hash::Hash;

use swc_common::{SyntaxContext, DUMMY_SP};
use swc_ecma_ast::*;
use swc_ecma_utils::ident::IdentLike;
use swc_ecma_visit::{noop_visit_mut_type, VisitMut};

#[cfg(feature = "concurrent")]
pub(crate) type Readonly<T> = std::sync::Arc<T>;

#[cfg(not(feature = "concurrent"))]
pub(crate) type Readonly<T> = T;

const TRACK: bool = false;

pub(crate) trait VarDeclaratorExt: Into<VarDeclarator> {
    fn into_module_item(self, injected_ctxt: SyntaxContext, name: &str) -> ModuleItem {
        VarDecl {
            span: DUMMY_SP,
            ctxt: injected_ctxt,
            kind: VarDeclKind::Const,
            declare: false,
            decls: if TRACK {
                vec![
                    self.into(),
                    Str {
                        span: DUMMY_SP,
                        raw: None,
                        value: name.into(),
                    }
                    .assign_to(Ident::new_no_ctxt("INJECTED_FROM".into(), DUMMY_SP)),
                ]
            } else {
                vec![self.into()]
            },
        }
        .into()
    }
}

impl<T> VarDeclaratorExt for T where T: Into<VarDeclarator> {}

pub(crate) trait ExprExt: Into<Expr> {
    #[track_caller]
    fn assign_to<T>(self, lhs: T) -> VarDeclarator
    where
        T: IdentLike,
    {
        let init = self.into();
        let lhs = lhs.into_id();

        if cfg!(debug_assertions) {
            if let Expr::Ident(rhs) = &init {
                debug_assert_ne!(lhs, rhs.to_id());
            }
        }

        VarDeclarator {
            span: DUMMY_SP,
            name: Ident::new(lhs.0, DUMMY_SP, lhs.1).into(),
            init: Some(Box::new(init)),
            definite: false,
        }
    }
}

impl<T> ExprExt for T where T: Into<Expr> {}

#[derive(Debug)]
pub(crate) struct CHashSet<V>
where
    V: Eq + Hash,
{
    inner: CloneMap<V, ()>,
}

impl<V> CHashSet<V>
where
    V: Eq + Hash,
{
    // pub fn insert(&self, v: V) -> bool {
    //     self.inner.insert(v, ()).is_none()
    // }
}

impl<V> Default for CHashSet<V>
where
    V: Eq + Hash,
{
    fn default() -> Self {
        Self {
            inner: Default::default(),
        }
    }
}

#[derive(Debug)]
pub(crate) struct CloneMap<K, V>
where
    K: Eq + Hash,
    V: Clone,
{
    #[cfg(feature = "concurrent")]
    inner: dashmap::DashMap<K, V, swc_common::collections::ARandomState>,
    #[cfg(not(feature = "concurrent"))]
    inner: std::cell::RefCell<swc_common::collections::AHashMap<K, V>>,
}

impl<K, V> Default for CloneMap<K, V>
where
    K: Eq + Hash,
    V: Clone,
{
    fn default() -> Self {
        Self {
            inner: Default::default(),
        }
    }
}

impl<K, V> CloneMap<K, V>
where
    K: Eq + Hash,
    V: Clone,
{
    #[cfg(feature = "concurrent")]
    pub fn get(&self, k: &K) -> Option<V> {
        self.inner.get(k).map(|v| v.value().clone())
    }

    #[cfg(not(feature = "concurrent"))]
    pub fn get(&self, k: &K) -> Option<V> {
        self.inner.borrow().get(k).map(|v| v.clone())
    }

    #[cfg(feature = "concurrent")]
    pub fn insert(&self, k: K, v: V) -> Option<V> {
        self.inner.insert(k, v)
    }

    #[cfg(not(feature = "concurrent"))]
    pub fn insert(&self, k: K, v: V) -> Option<V> {
        self.inner.borrow_mut().insert(k, v)
    }
}

pub(crate) struct HygieneRemover;

impl VisitMut for HygieneRemover {
    noop_visit_mut_type!(fail);

    fn visit_mut_syntax_context(&mut self, n: &mut SyntaxContext) {
        *n = SyntaxContext::empty();
    }
}

#[cfg(feature = "rayon")]
pub(crate) use rayon::join;

#[cfg(not(feature = "rayon"))]
pub(crate) fn join<A, B, RA, RB>(op_a: A, op_b: B) -> (RA, RB)
where
    A: FnOnce() -> RA,
    B: FnOnce() -> RB,
{
    (op_a(), op_b())
}

#[cfg(feature = "rayon")]
pub(crate) use rayon::iter::IntoParallelIterator;

/// Fake trait
#[cfg(not(feature = "rayon"))]
pub(crate) trait IntoParallelIterator: Sized + IntoIterator {
    fn into_par_iter(self) -> <Self as IntoIterator>::IntoIter {
        self.into_iter()
    }
}

#[cfg(not(feature = "rayon"))]
impl<T> IntoParallelIterator for T where T: IntoIterator {}

fn metadata(key: &str, value: &str) -> PropOrSpread {
    PropOrSpread::Prop(Box::new(Prop::KeyValue(KeyValueProp {
        key: PropName::Ident(IdentName::new(key.into(), DUMMY_SP)),
        value: Lit::Str(Str {
            span: DUMMY_SP,
            value: value.into(),
            raw: None,
        })
        .into(),
    })))
}

#[derive(Debug, Default)]
pub(crate) struct ExportMetadata {
    pub injected: bool,
    pub export_ctxt: Option<SyntaxContext>,
}

impl ExportMetadata {
    pub fn into_with(self) -> Box<ObjectLit> {
        let mut obj = Some(Box::new(ObjectLit {
            span: DUMMY_SP,
            props: Vec::new(),
        }));

        self.encode(&mut obj);

        obj.unwrap()
    }

    pub fn encode(&self, to: &mut Option<Box<ObjectLit>>) {
        let mut props = Vec::new();

        if self.injected {
            props.push(metadata("__swc_bundler__injected__", "1"));
        }

        if let Some(export_ctxt) = self.export_ctxt {
            props.push(metadata(
                "__swc_bundler__export_ctxt__",
                &export_ctxt.as_u32().to_string(),
            ));
        }

        if to.is_none() {
            *to = Some(Box::new(ObjectLit {
                span: DUMMY_SP,
                props,
            }));
        } else {
            let obj = to.as_mut().unwrap();
            obj.props.extend(props);
        }
    }

    pub fn decode(with: Option<&ObjectLit>) -> Self {
        let mut data = ExportMetadata::default();

        if let Some(with) = with {
            for prop in &with.props {
                if let PropOrSpread::Prop(p) = prop {
                    if let Prop::KeyValue(KeyValueProp {
                        key: PropName::Ident(IdentName { sym, .. }),
                        value,
                        ..
                    }) = &**p
                    {
                        if *sym == "__swc_bundler__injected__" {
                            if let Expr::Lit(Lit::Str(Str { value, .. })) = &**value {
                                if value == "1" {
                                    data.injected = true;
                                }
                            }
                        } else if *sym == "__swc_bundler__export_ctxt__" {
                            if let Expr::Lit(Lit::Str(Str { value, .. })) = &**value {
                                if let Ok(v) = value.parse() {
                                    data.export_ctxt = Some(SyntaxContext::from_u32(v));
                                }
                            }
                        }
                    }
                }
            }
        }

        data
    }
}

pub(crate) fn is_injected(with: &ObjectLit) -> bool {
    ExportMetadata::decode(Some(with)).injected
}