use clap::{App, AppSettings, Arg, SubCommand}; use std::env; use std::fs; use std::path::Path; use std::process::exit; use std::usize; use tree_sitter_cli::{error, generate, highlight, loader, logger, parse, properties, test}; fn main() { if let Err(e) = run() { eprintln!("{}", e.0); exit(1); } } fn run() -> error::Result<()> { let matches = App::new("tree-sitter") .version(concat!( env!("CARGO_PKG_VERSION"), " (", env!("BUILD_SHA"), ")" )) .setting(AppSettings::SubcommandRequiredElseHelp) .author("Max Brunsfeld ") .about("Generates and tests parsers") .subcommand( SubCommand::with_name("generate") .about("Generate a parser") .arg(Arg::with_name("grammar-path").index(1)) .arg(Arg::with_name("log").long("log")) .arg(Arg::with_name("properties-only").long("properties")) .arg( Arg::with_name("state-ids-to-log") .long("log-state") .takes_value(true), ) .arg(Arg::with_name("no-minimize").long("no-minimize")), ) .subcommand( SubCommand::with_name("parse") .about("Parse a file") .arg( Arg::with_name("path") .index(1) .multiple(true) .required(true), ) .arg(Arg::with_name("debug").long("debug").short("d")) .arg(Arg::with_name("debug-graph").long("debug-graph").short("D")) .arg(Arg::with_name("quiet").long("quiet").short("q")) .arg(Arg::with_name("time").long("time").short("t")), ) .subcommand( SubCommand::with_name("test") .about("Run a parser's tests") .arg( Arg::with_name("filter") .long("filter") .short("f") .takes_value(true), ) .arg(Arg::with_name("debug").long("debug").short("d")) .arg(Arg::with_name("debug-graph").long("debug-graph").short("D")), ) .subcommand( SubCommand::with_name("highlight") .about("Highlight a file") .arg( Arg::with_name("path") .index(1) .multiple(true) .required(true), ) .arg(Arg::with_name("scope").long("scope").takes_value(true)) .arg(Arg::with_name("html").long("html").short("h")), ) .get_matches(); let home_dir = dirs::home_dir().unwrap(); let current_dir = env::current_dir().unwrap(); let config_dir = home_dir.join(".tree-sitter"); let theme_path = config_dir.join("theme.json"); let parsers_dir = config_dir.join("parsers"); // TODO - make configurable let parser_repo_paths = vec![home_dir.join("github")]; fs::create_dir_all(&parsers_dir).unwrap(); let mut loader = loader::Loader::new(config_dir); if let Some(matches) = matches.subcommand_matches("generate") { if matches.is_present("log") { logger::init(); } let grammar_path = matches.value_of("grammar-path"); let minimize = !matches.is_present("no-minimize"); let properties_only = matches.is_present("properties-only"); let state_ids_to_log = matches .values_of("state-ids-to-log") .map_or(Vec::new(), |ids| { ids.filter_map(|id| usize::from_str_radix(id, 10).ok()) .collect() }); if !properties_only { generate::generate_parser_in_directory( ¤t_dir, grammar_path, minimize, state_ids_to_log, )?; } properties::generate_property_sheets_in_directory(¤t_dir)?; } else if let Some(matches) = matches.subcommand_matches("test") { let debug = matches.is_present("debug"); let debug_graph = matches.is_present("debug-graph"); let filter = matches.value_of("filter"); let corpus_path = current_dir.join("corpus"); if let Some(language) = loader.language_at_path(¤t_dir)? { test::run_tests_at_path(language, &corpus_path, debug, debug_graph, filter)?; } else { eprintln!("No language found"); } } else if let Some(matches) = matches.subcommand_matches("parse") { let debug = matches.is_present("debug"); let debug_graph = matches.is_present("debug-graph"); let quiet = matches.is_present("quiet"); let time = matches.is_present("time"); loader.find_all_languages(&parser_repo_paths)?; let paths = matches .values_of("path") .unwrap() .into_iter() .collect::>(); let max_path_length = paths.iter().map(|p| p.chars().count()).max().unwrap(); let mut has_error = false; for path in paths { let path = Path::new(path); let language = if let Some((l, _)) = loader.language_configuration_for_file_name(path)? { l } else if let Some(l) = loader.language_at_path(¤t_dir)? { l } else { eprintln!("No language found"); return Ok(()); }; has_error |= parse::parse_file_at_path( language, path, max_path_length, quiet, time, debug, debug_graph, )?; } if has_error { return Err(error::Error(String::new())); } } else if let Some(matches) = matches.subcommand_matches("highlight") { loader.find_all_languages(&parser_repo_paths)?; let theme = highlight::Theme::load(&theme_path).unwrap_or_default(); let paths = matches.values_of("path").unwrap().into_iter(); let html_mode = matches.is_present("html"); if html_mode { println!("{}", highlight::HTML_HEADER); } let language_config; if let Some(scope) = matches.value_of("scope") { language_config = loader.language_configuration_for_scope(scope)?; if language_config.is_none() { return Err(error::Error(format!("Unknown scope '{}'", scope))); } } else { language_config = None; } for path in paths { let path = Path::new(path); let (language, config) = match language_config { Some(v) => v, None => match loader.language_configuration_for_file_name(path)? { Some(v) => v, None => { eprintln!("No language found for path {:?}", path); continue; } }, }; if let Some(sheet) = config.highlight_property_sheet(language)? { let source = fs::read(path)?; if html_mode { highlight::html(&loader, &theme, &source, language, sheet)?; } else { highlight::ansi(&loader, &theme, &source, language, sheet)?; } } else { return Err(error::Error(format!( "No syntax highlighting property sheet specified" ))); } } } Ok(()) }