use crate::clean; use crate::docfs::PathError; use crate::error::Error; use crate::fold::DocFolder; use crate::html::format::Buffer; use crate::html::highlight; use crate::html::layout; use crate::html::render::{SharedContext, BASIC_KEYWORDS}; use rustc_hir::def_id::LOCAL_CRATE; use rustc_session::Session; use rustc_span::edition::Edition; use rustc_span::source_map::FileName; use std::ffi::OsStr; use std::fs; use std::path::{Component, Path, PathBuf}; crate fn render( dst: &Path, scx: &mut SharedContext<'_>, krate: clean::Crate, ) -> Result { info!("emitting source files"); let dst = dst.join("src").join(&*krate.name.as_str()); scx.ensure_dir(&dst)?; let mut folder = SourceCollector { dst, scx }; Ok(folder.fold_crate(krate)) } /// Helper struct to render all source code to HTML pages struct SourceCollector<'a, 'tcx> { scx: &'a mut SharedContext<'tcx>, /// Root destination to place all HTML output into dst: PathBuf, } impl DocFolder for SourceCollector<'_, '_> { fn fold_item(&mut self, item: clean::Item) -> Option { // If we're not rendering sources, there's nothing to do. // If we're including source files, and we haven't seen this file yet, // then we need to render it out to the filesystem. if self.scx.include_sources // skip all synthetic "files" && item.span(self.scx.tcx).filename(self.sess()).is_real() // skip non-local files && item.span(self.scx.tcx).cnum(self.sess()) == LOCAL_CRATE { let filename = item.span(self.scx.tcx).filename(self.sess()); // If it turns out that we couldn't read this file, then we probably // can't read any of the files (generating html output from json or // something like that), so just don't include sources for the // entire crate. The other option is maintaining this mapping on a // per-file basis, but that's probably not worth it... self.scx.include_sources = match self.emit_source(&filename) { Ok(()) => true, Err(e) => { self.scx.tcx.sess.span_err( item.span(self.scx.tcx).inner(), &format!( "failed to render source code for `{}`: {}", filename.prefer_local(), e ), ); false } }; } // FIXME: if `include_sources` isn't set and DocFolder didn't require consuming the crate by value, // we could return None here without having to walk the rest of the crate. Some(self.fold_item_recur(item)) } } impl SourceCollector<'_, 'tcx> { fn sess(&self) -> &'tcx Session { &self.scx.tcx.sess } /// Renders the given filename into its corresponding HTML source file. fn emit_source(&mut self, filename: &FileName) -> Result<(), Error> { let p = match *filename { FileName::Real(ref file) => { if let Some(local_path) = file.local_path() { local_path.to_path_buf() } else { unreachable!("only the current crate should have sources emitted"); } } _ => return Ok(()), }; if self.scx.local_sources.contains_key(&*p) { // We've already emitted this source return Ok(()); } let contents = match fs::read_to_string(&p) { Ok(contents) => contents, Err(e) => { return Err(Error::new(e, &p)); } }; // Remove the utf-8 BOM if any let contents = if contents.starts_with('\u{feff}') { &contents[3..] } else { &contents }; // Create the intermediate directories let mut cur = self.dst.clone(); let mut root_path = String::from("../../"); let mut href = String::new(); clean_path(&self.scx.src_root, &p, false, |component| { cur.push(component); root_path.push_str("../"); href.push_str(&component.to_string_lossy()); href.push('/'); }); self.scx.ensure_dir(&cur)?; let src_fname = p.file_name().expect("source has no filename").to_os_string(); let mut fname = src_fname.clone(); fname.push(".html"); cur.push(&fname); href.push_str(&fname.to_string_lossy()); let title = format!("{} - source", src_fname.to_string_lossy()); let desc = format!("Source of the Rust file `{}`.", filename.prefer_remapped()); let page = layout::Page { title: &title, css_class: "source", root_path: &root_path, static_root_path: self.scx.static_root_path.as_deref(), description: &desc, keywords: BASIC_KEYWORDS, resource_suffix: &self.scx.resource_suffix, extra_scripts: &[&format!("source-files{}", self.scx.resource_suffix)], static_extra_scripts: &[&format!("source-script{}", self.scx.resource_suffix)], }; let v = layout::render( &self.scx.layout, &page, "", |buf: &mut _| print_src(buf, contents, self.scx.edition()), &self.scx.style_files, ); self.scx.fs.write(&cur, v.as_bytes())?; self.scx.local_sources.insert(p, href); Ok(()) } } /// Takes a path to a source file and cleans the path to it. This canonicalizes /// things like ".." to components which preserve the "top down" hierarchy of a /// static HTML tree. Each component in the cleaned path will be passed as an /// argument to `f`. The very last component of the path (ie the file name) will /// be passed to `f` if `keep_filename` is true, and ignored otherwise. crate fn clean_path(src_root: &Path, p: &Path, keep_filename: bool, mut f: F) where F: FnMut(&OsStr), { // make it relative, if possible let p = p.strip_prefix(src_root).unwrap_or(p); let mut iter = p.components().peekable(); while let Some(c) = iter.next() { if !keep_filename && iter.peek().is_none() { break; } match c { Component::ParentDir => f("up".as_ref()), Component::Normal(c) => f(c), _ => continue, } } } /// Wrapper struct to render the source code of a file. This will do things like /// adding line numbers to the left-hand side. fn print_src(buf: &mut Buffer, s: &str, edition: Edition) { let lines = s.lines().count(); let mut line_numbers = Buffer::empty_from(buf); let mut cols = 0; let mut tmp = lines; while tmp > 0 { cols += 1; tmp /= 10; } line_numbers.write_str("
");
    for i in 1..=lines {
        writeln!(line_numbers, "{0:1$}", i, cols);
    }
    line_numbers.write_str("
"); highlight::render_with_highlighting(s, buf, None, None, None, edition, Some(line_numbers)); }