use std::borrow::Cow::Owned; use pulldown_cmark as cmark; use self::cmark::{Parser, Event, Tag, Options, OPTION_ENABLE_TABLES, OPTION_ENABLE_FOOTNOTES}; use slug::slugify; use syntect::easy::HighlightLines; use syntect::html::{start_coloured_html_snippet, styles_to_coloured_html, IncludeBackground}; use errors::Result; use utils::site::resolve_internal_link; use highlighting::{get_highlighter, THEME_SET}; use link_checker::check_url; use table_of_contents::{TempHeader, Header, make_table_of_contents}; use context::RenderContext; // We might have cases where the slug is already present in our list of anchor // for example an article could have several titles named Example // We add a counter after the slug if the slug is already present, which // means we will have example, example-1, example-2 etc fn find_anchor(anchors: &[String], name: String, level: u8) -> String { if level == 0 && !anchors.contains(&name) { return name.to_string(); } let new_anchor = format!("{}-{}", name, level + 1); if !anchors.contains(&new_anchor) { return new_anchor; } find_anchor(anchors, name, level + 1) } fn is_colocated_asset_link(link: &str) -> bool { !link.contains("/") // http://, ftp://, ../ etc && !link.starts_with("mailto:") } pub fn markdown_to_html(content: &str, context: &RenderContext) -> Result<(String, Vec
)> { // the rendered html let mut html = String::with_capacity(content.len()); // Set while parsing let mut error = None; let mut highlighter: Option = None; // If we get text in header, we need to insert the id and a anchor let mut in_header = false; // pulldown_cmark can send several text events for a title if there are markdown // specific characters like `!` in them. We only want to insert the anchor the first time let mut header_created = false; let mut anchors: Vec = vec![]; let mut headers = vec![]; // Defaults to a 0 level so not a real header // It should be an Option ideally but not worth the hassle to update let mut temp_header = TempHeader::default(); let mut opts = Options::empty(); opts.insert(OPTION_ENABLE_TABLES); opts.insert(OPTION_ENABLE_FOOTNOTES); { let parser = Parser::new_ext(content, opts).map(|event| { match event { Event::Text(text) => { // Header first if in_header { if header_created { temp_header.push(&text); return Event::Html(Owned(String::new())); } let id = find_anchor(&anchors, slugify(&text), 0); anchors.push(id.clone()); // update the header and add it to the list temp_header.permalink = format!("{}#{}", context.current_page_permalink, id); temp_header.id = id; // += as we might have some or other things already there temp_header.title += &text; header_created = true; return Event::Html(Owned(String::new())); } // if we are in the middle of a code block if let Some(ref mut highlighter) = highlighter { let highlighted = &highlighter.highlight(&text); let html = styles_to_coloured_html(highlighted, IncludeBackground::Yes); return Event::Html(Owned(html)); } // Business as usual Event::Text(text) } Event::Start(Tag::CodeBlock(ref info)) => { if !context.config.highlight_code { return Event::Html(Owned("
".to_string()));
                    }

                    let theme = &THEME_SET.themes[&context.config.highlight_theme];
                    match get_highlighter(&theme, info, context.base_path, &context.config.extra_syntaxes) {
                        Ok(h) => highlighter = Some(h),
                        Err(err) => {
                            error = Some(format!("Could not load syntax: {}", err).into());
                            return Event::Html(Owned(String::new()));
                        }
                    }
                    let snippet = start_coloured_html_snippet(theme);
                    Event::Html(Owned(snippet))
                }
                Event::End(Tag::CodeBlock(_)) => {
                    if !context.config.highlight_code {
                        return Event::Html(Owned("
\n".to_string())); } // reset highlight and close the code block highlighter = None; Event::Html(Owned("".to_string())) } Event::Start(Tag::Image(src, title)) => { if is_colocated_asset_link(&src) { return Event::Start( Tag::Image( Owned(format!("{}{}", context.current_page_permalink, src)), title, ) ); } Event::Start(Tag::Image(src, title)) } Event::Start(Tag::Link(link, title)) => { // A few situations here: // - it could be a relative link (starting with `./`) // - it could be a link to a co-located asset // - it could be a normal link // - any of those can be in a header or not: if it's in a header // we need to append to a string let fixed_link = if link.starts_with("./") { match resolve_internal_link(&link, context.permalinks) { Ok(url) => url, Err(_) => { error = Some(format!("Relative link {} not found.", link).into()); return Event::Html(Owned(String::new())); } } } else if is_colocated_asset_link(&link) { format!("{}{}", context.current_page_permalink, link) } else { if context.config.check_external_links && !link.starts_with('#') { let res = check_url(&link); if res.is_valid() { link.to_string() } else { error = Some( format!("Link {} is not valid: {}", link, res.message()).into() ); String::new() } } else { link.to_string() } }; if in_header { let html = if title.is_empty() { format!("", fixed_link) } else { format!("", fixed_link, title) }; temp_header.push(&html); return Event::Html(Owned(String::new())); } Event::Start(Tag::Link(Owned(fixed_link), title)) } Event::End(Tag::Link(_, _)) => { if in_header { temp_header.push(""); return Event::Html(Owned(String::new())); } event } Event::Start(Tag::Code) => { if in_header { temp_header.push(""); return Event::Html(Owned(String::new())); } event } Event::End(Tag::Code) => { if in_header { temp_header.push(""); return Event::Html(Owned(String::new())); } event } Event::Start(Tag::Header(num)) => { in_header = true; temp_header = TempHeader::new(num); Event::Html(Owned(String::new())) } Event::End(Tag::Header(_)) => { // End of a header, reset all the things and return the stringified // version of the header in_header = false; header_created = false; let val = temp_header.to_string(context.tera, context.insert_anchor); headers.push(temp_header.clone()); temp_header = TempHeader::default(); Event::Html(Owned(val)) } _ => event, } }); cmark::html::push_html(&mut html, parser); } match error { Some(e) => Err(e), None => Ok(( html.replace("

", "").replace("

", "

"), make_table_of_contents(&headers) )), } }