#![deny(missing_docs, missing_debug_implementations, missing_copy_implementations, trivial_casts,
trivial_numeric_casts, unstable_features, unused_import_braces, unused_qualifications)]
#![cfg_attr(feature = "dev", allow(unstable_features))]
#![allow(unknown_lints, doc_markdown, cyclomatic_complexity)]
#[macro_use]
extern crate clap;
extern crate entities;
#[macro_use]
extern crate lazy_static;
extern crate regex;
extern crate twoway;
extern crate typed_arena;
extern crate unicode_categories;
mod arena_tree;
mod html;
mod cm;
mod parser;
mod nodes;
mod ctype;
mod scanners;
mod strings;
mod entity;
use std::collections::BTreeSet;
use std::io::Read;
use std::process;
use typed_arena::Arena;
fn main() {
let matches = clap::App::new(crate_name!())
.version(crate_version!())
.author(crate_authors!())
.about(crate_description!())
.arg(
clap::Arg::with_name("file")
.value_name("FILE")
.multiple(true)
.help("The CommonMark file to parse; or standard input if none passed"),
)
.arg(
clap::Arg::with_name("hardbreaks")
.long("hardbreaks")
.help("Treat newlines as hard line breaks"),
)
.arg(
clap::Arg::with_name("github-pre-lang")
.long("github-pre-lang")
.help("Use GitHub-style <pre lang> for code blocks"),
)
.arg(
clap::Arg::with_name("prepare-regexes")
.long("prepare-regexes")
.help("Prepare regexes up front for benchmarking"),
)
.arg(
clap::Arg::with_name("extension")
.short("e")
.long("extension")
.takes_value(true)
.number_of_values(1)
.multiple(true)
.possible_values(&[
"strikethrough",
"tagfilter",
"table",
"autolink",
"tasklist",
"superscript",
"footnotes",
])
.value_name("EXTENSION")
.help("Specify an extension name to use"),
)
.arg(
clap::Arg::with_name("format")
.short("t")
.long("to")
.takes_value(true)
.possible_values(&["html", "commonmark"])
.default_value("html")
.value_name("FORMAT")
.help("Specify output format"),
)
.arg(
clap::Arg::with_name("width")
.long("width")
.takes_value(true)
.value_name("WIDTH")
.default_value("0")
.help("Specify wrap width (0 = nowrap)"),
)
.arg(
clap::Arg::with_name("header-ids")
.long("header-ids")
.takes_value(true)
.value_name("PREFIX")
.help("Use the Comrak header IDs extension, with the given ID prefix"),
)
.arg(
clap::Arg::with_name("footnotes")
.long("footnotes")
.help("Parse footnotes"),
)
.get_matches();
let mut exts = matches
.values_of("extension")
.map_or(BTreeSet::new(), |vals| vals.collect());
let options = parser::ComrakOptions {
hardbreaks: matches.is_present("hardbreaks"),
github_pre_lang: matches.is_present("github-pre-lang"),
width: matches
.value_of("width")
.unwrap_or("0")
.parse()
.unwrap_or(0),
ext_strikethrough: exts.remove("strikethrough"),
ext_tagfilter: exts.remove("tagfilter"),
ext_table: exts.remove("table"),
ext_autolink: exts.remove("autolink"),
ext_tasklist: exts.remove("tasklist"),
ext_superscript: exts.remove("superscript"),
ext_header_ids: matches.value_of("header-ids").map(|s| s.to_string()),
ext_footnotes: matches.is_present("footnotes"),
};
assert!(exts.is_empty());
let mut s: Vec<u8> = Vec::with_capacity(2048);
match matches.values_of("file") {
None => {
std::io::stdin().read_to_end(&mut s).unwrap();
}
Some(fs) => for f in fs {
let mut io = std::fs::File::open(f).unwrap();
io.read_to_end(&mut s).unwrap();
},
};
if matches.is_present("prepare-regexes") {
init_scanners();
}
let arena = Arena::new();
let root = parser::parse_document(&arena, &String::from_utf8(s).unwrap(), &options);
let formatter = match matches.value_of("format") {
Some("html") => html::format_document,
Some("commonmark") => cm::format_document,
_ => panic!("unknown format"),
};
formatter(root, &options, &mut std::io::stdout()).unwrap();
process::exit(0);
}
fn init_scanners() {
scanners::atx_heading_start(b"");
scanners::html_block_end_1(b"");
scanners::open_code_fence(b"");
scanners::close_code_fence(b"");
scanners::html_block_start(b"");
scanners::html_block_start_7(b"");
scanners::setext_heading_line(b"");
scanners::thematic_break(b"");
scanners::scheme(b"");
scanners::autolink_uri(b"");
scanners::autolink_email(b"");
scanners::html_tag(b"");
scanners::spacechars(b"");
scanners::link_title(b"");
scanners::table_start(b"");
scanners::table_cell(b"");
scanners::table_cell_end(b"");
scanners::table_row_end(b"");
}