// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. //! This example shows how to use swc to transpile TypeScript and JSX/TSX //! modules. //! //! It will only transpile, not typecheck (like Deno's `--no-check` flag). use std::cell::RefCell; use std::collections::HashMap; use std::rc::Rc; use anyhow::anyhow; use anyhow::bail; use anyhow::Context; use anyhow::Error; use deno_ast::MediaType; use deno_ast::ParseParams; use deno_ast::SourceMapOption; use deno_ast::SourceTextInfo; use deno_core::error::AnyError; use deno_core::resolve_import; use deno_core::resolve_path; use deno_core::JsRuntime; use deno_core::ModuleLoadResponse; use deno_core::ModuleLoader; use deno_core::ModuleSource; use deno_core::ModuleSourceCode; use deno_core::ModuleSpecifier; use deno_core::ModuleType; use deno_core::RequestedModuleType; use deno_core::ResolutionKind; use deno_core::RuntimeOptions; use deno_core::SourceMapGetter; #[derive(Clone)] struct SourceMapStore(Rc>>>); impl SourceMapGetter for SourceMapStore { fn get_source_map(&self, specifier: &str) -> Option> { self.0.borrow().get(specifier).cloned() } fn get_source_line( &self, _file_name: &str, _line_number: usize, ) -> Option { None } } struct TypescriptModuleLoader { source_maps: SourceMapStore, } impl ModuleLoader for TypescriptModuleLoader { fn resolve( &self, specifier: &str, referrer: &str, _kind: ResolutionKind, ) -> Result { Ok(resolve_import(specifier, referrer)?) } fn load( &self, module_specifier: &ModuleSpecifier, _maybe_referrer: Option<&ModuleSpecifier>, _is_dyn_import: bool, _requested_module_type: RequestedModuleType, ) -> ModuleLoadResponse { let source_maps = self.source_maps.clone(); fn load( source_maps: SourceMapStore, module_specifier: &ModuleSpecifier, ) -> Result { let path = module_specifier .to_file_path() .map_err(|_| anyhow!("Only file:// URLs are supported."))?; let media_type = MediaType::from_path(&path); let (module_type, should_transpile) = match MediaType::from_path(&path) { MediaType::JavaScript | MediaType::Mjs | MediaType::Cjs => { (ModuleType::JavaScript, false) } MediaType::Jsx => (ModuleType::JavaScript, true), MediaType::TypeScript | MediaType::Mts | MediaType::Cts | MediaType::Dts | MediaType::Dmts | MediaType::Dcts | MediaType::Tsx => (ModuleType::JavaScript, true), MediaType::Json => (ModuleType::Json, false), _ => bail!("Unknown extension {:?}", path.extension()), }; let code = std::fs::read_to_string(&path)?; let code = if should_transpile { let parsed = deno_ast::parse_module(ParseParams { specifier: module_specifier.clone(), text_info: SourceTextInfo::from_string(code), media_type, capture_tokens: false, scope_analysis: false, maybe_syntax: None, })?; let res = parsed.transpile(&deno_ast::EmitOptions { source_map: SourceMapOption::Separate, inline_sources: true, ..Default::default() })?; let source_map = res.source_map.unwrap(); source_maps .0 .borrow_mut() .insert(module_specifier.to_string(), source_map.into_bytes()); res.text } else { code }; Ok(ModuleSource::new( module_type, ModuleSourceCode::String(code.into()), module_specifier, None, )) } ModuleLoadResponse::Sync(load(source_maps, module_specifier)) } } fn main() -> Result<(), Error> { let args: Vec = std::env::args().collect(); if args.len() < 2 { println!("Usage: target/examples/debug/ts_module_loader "); std::process::exit(1); } let main_url = &args[1]; println!("Run {main_url}"); let source_map_store = SourceMapStore(Rc::new(RefCell::new(HashMap::new()))); let mut js_runtime = JsRuntime::new(RuntimeOptions { module_loader: Some(Rc::new(TypescriptModuleLoader { source_maps: source_map_store.clone(), })), source_map_getter: Some(Rc::new(source_map_store)), ..Default::default() }); let main_module = resolve_path( main_url, &std::env::current_dir().context("Unable to get CWD")?, )?; let future = async move { let mod_id = js_runtime.load_main_es_module(&main_module).await?; let result = js_runtime.mod_evaluate(mod_id); js_runtime.run_event_loop(Default::default()).await?; result.await }; tokio::runtime::Builder::new_current_thread() .enable_all() .build() .unwrap() .block_on(future) }