blob: dedf6a6600eecc23969e648145f40cd4f711afbd [file] [log] [blame]
//! Support for parsing advisories from a Markdown file
//! (a.k.a. "V3 advisory format")
use crate::error::{Error, ErrorKind};
/// Parts of a parsed advisory
#[derive(Copy, Clone, Debug)]
pub struct Parts<'a> {
/// TOML front matter
pub front_matter: &'a str,
/// Markdown without front matter
pub markdown: &'a str,
/// Advisory title
pub title: &'a str,
/// Advisory description (markdown)
pub description: &'a str,
}
impl<'a> Parts<'a> {
/// Parse a Markdown advisory into its component parts
pub fn parse(advisory_data: &'a str) -> Result<Self, Error> {
if !advisory_data.starts_with("```toml") {
let context = if advisory_data.len() > 20 {
&advisory_data[..20]
} else {
advisory_data
};
fail!(
ErrorKind::Parse,
"unexpected start of advisory: \"{}\"",
context
)
}
let toml_end = advisory_data.find("\n```").ok_or_else(|| {
format_err!(
ErrorKind::Parse,
"couldn't find end of TOML front matter in advisory"
)
})?;
let front_matter = advisory_data[7..toml_end].trim_start().trim_end();
let markdown = advisory_data[(toml_end + 4)..].trim_start();
if !markdown.starts_with("# ") {
fail!(
ErrorKind::Parse,
"Expected # header after TOML front matter"
);
}
let next_newline = markdown.find('\n').ok_or_else(|| {
format_err!(
ErrorKind::Parse,
"no Markdown body (i.e. description) found"
)
})?;
let title = markdown[2..next_newline].trim_end();
let description = markdown[(next_newline + 1)..].trim_start().trim_end();
Ok(Self {
front_matter,
markdown,
title,
description,
})
}
}