bat/src/main.rs

187 lines
4.9 KiB
Rust
Raw Normal View History

2018-04-30 11:09:24 +02:00
// `error_chain!` can recurse deeply
#![recursion_limit = "1024"]
#[macro_use]
extern crate error_chain;
#[macro_use]
extern crate clap;
#[macro_use]
extern crate lazy_static;
2018-04-21 12:51:43 +02:00
extern crate ansi_term;
extern crate atty;
extern crate console;
extern crate directories;
2018-04-21 17:12:25 +02:00
extern crate git2;
2018-04-21 12:51:43 +02:00
extern crate syntect;
2018-05-10 23:39:13 +02:00
mod app;
mod assets;
2018-08-23 22:37:27 +02:00
mod controller;
mod decorations;
mod diff;
2018-08-22 22:29:12 +02:00
mod line_range;
2018-05-21 14:59:42 +02:00
mod output;
mod printer;
2018-05-11 02:32:31 +02:00
mod style;
2018-04-23 23:56:47 +02:00
mod terminal;
2018-05-21 21:51:41 +02:00
use std::io;
use std::path::Path;
2018-05-21 14:59:42 +02:00
use std::process;
2018-04-21 12:51:43 +02:00
2018-08-23 22:37:27 +02:00
use ansi_term::Colour::Green;
use app::{App, Config};
2018-05-21 22:29:03 +02:00
use assets::{clear_assets, config_dir, HighlightingAssets};
2018-08-23 22:37:27 +02:00
use controller::Controller;
2018-04-23 23:56:47 +02:00
2018-04-30 11:09:24 +02:00
mod errors {
error_chain! {
2018-04-30 11:09:24 +02:00
foreign_links {
Clap(::clap::Error);
2018-04-30 11:09:24 +02:00
Io(::std::io::Error);
2018-05-25 22:12:30 +02:00
SyntectError(::syntect::LoadingError);
2018-06-05 01:07:01 +02:00
ParseIntError(::std::num::ParseIntError);
2018-04-30 11:09:24 +02:00
}
}
2018-04-21 12:51:43 +02:00
2018-05-21 21:51:41 +02:00
pub fn handle_error(error: &Error) {
match error {
&Error(ErrorKind::Io(ref io_error), _)
if io_error.kind() == super::io::ErrorKind::BrokenPipe =>
{
super::process::exit(0);
}
_ => {
use ansi_term::Colour::Red;
eprintln!("{}: {}", Red.paint("[bat error]"), error);
}
};
2018-04-21 12:51:43 +02:00
}
}
2018-05-21 21:51:41 +02:00
use errors::*;
2018-08-22 22:29:12 +02:00
fn run_cache_subcommand(matches: &clap::ArgMatches) -> Result<()> {
if matches.is_present("init") {
let source_dir = matches.value_of("source").map(Path::new);
let target_dir = matches.value_of("target").map(Path::new);
let blank = matches.is_present("blank");
let assets = HighlightingAssets::from_files(source_dir, blank)?;
assets.save(target_dir)?;
} else if matches.is_present("clear") {
clear_assets();
} else if matches.is_present("config-dir") {
println!("{}", config_dir());
}
Ok(())
}
2018-08-23 22:37:27 +02:00
pub fn list_languages(assets: &HighlightingAssets, term_width: usize) {
let mut languages = assets
.syntax_set
.syntaxes()
.iter()
.filter(|syntax| !syntax.hidden && !syntax.file_extensions.is_empty())
.collect::<Vec<_>>();
languages.sort_by_key(|lang| lang.name.to_uppercase());
let longest = languages
.iter()
.map(|syntax| syntax.name.len())
.max()
.unwrap_or(32); // Fallback width if they have no language definitions.
let comma_separator = ", ";
let separator = " ";
// Line-wrapping for the possible file extension overflow.
let desired_width = term_width - longest - separator.len();
for lang in languages {
print!("{:width$}{}", lang.name, separator, width = longest);
// Number of characters on this line so far, wrap before `desired_width`
let mut num_chars = 0;
let mut extension = lang.file_extensions.iter().peekable();
while let Some(word) = extension.next() {
// If we can't fit this word in, then create a line break and align it in.
let new_chars = word.len() + comma_separator.len();
if num_chars + new_chars >= desired_width {
num_chars = 0;
print!("\n{:width$}{}", "", separator, width = longest);
}
num_chars += new_chars;
print!("{}", Green.paint(&word[..]));
if extension.peek().is_some() {
print!("{}", comma_separator);
}
}
println!();
}
}
pub fn list_themes(assets: &HighlightingAssets, config: &mut Config) {
2018-08-23 22:37:27 +02:00
let themes = &assets.theme_set.themes;
config.files = vec![Some("assets/hello.rs")];
2018-08-23 22:37:27 +02:00
for (theme, _) in themes.iter() {
println!("{}", theme);
config.theme = theme.to_string();
let _controller = Controller::new(&config, &assets).run();
2018-08-23 22:37:27 +02:00
}
}
2018-05-19 11:46:41 +02:00
/// Returns `Err(..)` upon fatal errors. Otherwise, returns `Some(true)` on full success and
2018-05-21 21:51:41 +02:00
/// `Some(false)` if any intermediate errors occurred (were printed).
2018-05-19 11:46:41 +02:00
fn run() -> Result<bool> {
2018-05-10 23:39:13 +02:00
let app = App::new();
2018-04-21 12:51:43 +02:00
2018-05-10 23:39:13 +02:00
match app.matches.subcommand() {
("cache", Some(cache_matches)) => {
2018-08-22 22:29:12 +02:00
run_cache_subcommand(cache_matches)?;
Ok(true)
}
_ => {
let mut config = app.config()?;
let assets = HighlightingAssets::new();
2018-05-10 23:39:13 +02:00
if app.matches.is_present("list-languages") {
2018-05-21 21:51:41 +02:00
list_languages(&assets, config.term_width);
2018-08-22 22:29:12 +02:00
Ok(true)
} else if app.matches.is_present("list-themes") {
list_themes(&assets, &mut config);
2018-05-11 13:53:17 +02:00
2018-08-22 22:29:12 +02:00
Ok(true)
} else {
2018-08-23 22:37:27 +02:00
let controller = Controller::new(&config, &assets);
controller.run()
2018-08-22 22:29:12 +02:00
}
}
}
2018-05-19 11:46:41 +02:00
}
fn main() {
let result = run();
2018-04-21 12:51:43 +02:00
2018-05-19 11:46:41 +02:00
match result {
Err(error) => {
handle_error(&error);
process::exit(1);
}
Ok(false) => {
process::exit(1);
}
Ok(true) => {
process::exit(0);
}
2018-04-21 12:51:43 +02:00
}
}