use anyhow::Result;
use std::rc::Rc;
use thiserror::Error;
use crate::arg_parser::ParseArgsError;
use crate::commands::CheckError;
use crate::configuration::ResolveConfigError;
use crate::environment::Environment;
use crate::paths::NoFilesFoundError;
use crate::plugins::PluginResolver;
use crate::arg_parser::CliArgs;
use crate::arg_parser::ConfigSubCommand;
use crate::arg_parser::SubCommand;
use crate::commands;
use crate::resolution::NoPluginsFoundError;
use crate::resolution::ResolvePluginsError;
#[derive(Debug, Error)]
#[error("{inner:#}")]
pub struct AppError {
pub inner: anyhow::Error,
pub exit_code: i32,
}
impl From<anyhow::Error> for AppError {
fn from(inner: anyhow::Error) -> Self {
let inner = match inner.downcast::<ParseArgsError>() {
Ok(err) => return err.into(),
Err(err) => err,
};
let inner = match inner.downcast::<ResolveConfigError>() {
Ok(err) => return err.into(),
Err(err) => err,
};
let inner = match inner.downcast::<ResolvePluginsError>() {
Ok(err) => return err.into(),
Err(err) => err,
};
let inner = match inner.downcast::<NoPluginsFoundError>() {
Ok(err) => return err.into(),
Err(err) => err,
};
let inner = match inner.downcast::<NoFilesFoundError>() {
Ok(err) => return err.into(),
Err(err) => err,
};
let inner = match inner.downcast::<CheckError>() {
Ok(err) => return err.into(),
Err(err) => err,
};
AppError { inner, exit_code: 1 }
}
}
impl From<ParseArgsError> for AppError {
fn from(inner: ParseArgsError) -> Self {
AppError {
inner: inner.into(),
exit_code: 10,
}
}
}
impl From<ResolveConfigError> for AppError {
fn from(inner: ResolveConfigError) -> Self {
AppError {
inner: inner.into(),
exit_code: 11,
}
}
}
impl From<ResolvePluginsError> for AppError {
fn from(inner: ResolvePluginsError) -> Self {
AppError {
inner: inner.into(),
exit_code: 12,
}
}
}
impl From<NoPluginsFoundError> for AppError {
fn from(inner: NoPluginsFoundError) -> Self {
AppError {
inner: inner.into(),
exit_code: 13,
}
}
}
impl From<NoFilesFoundError> for AppError {
fn from(inner: NoFilesFoundError) -> Self {
AppError {
inner: inner.into(),
exit_code: 14,
}
}
}
impl From<CheckError> for AppError {
fn from(inner: CheckError) -> Self {
AppError {
inner: inner.into(),
exit_code: 20,
}
}
}
pub async fn run_cli<TEnvironment: Environment>(args: &CliArgs, environment: &TEnvironment, plugin_resolver: &Rc<PluginResolver<TEnvironment>>) -> Result<()> {
match &args.sub_command {
SubCommand::Help(help_text) => commands::output_help(args, environment, plugin_resolver, help_text).await,
SubCommand::License => commands::output_license(args, environment, plugin_resolver).await,
SubCommand::EditorInfo => commands::output_editor_info(args, environment, plugin_resolver).await,
SubCommand::EditorService(cmd) => commands::run_editor_service(args, environment, plugin_resolver, cmd).await,
SubCommand::Lsp => commands::run_language_server(args, environment, plugin_resolver).await,
SubCommand::ClearCache => commands::clear_cache(environment),
SubCommand::Config(cmd) => match cmd {
ConfigSubCommand::Init => commands::init_config_file(environment, &args.config).await,
ConfigSubCommand::Add(plugin_name_or_url) => commands::add_plugin_config_file(args, plugin_name_or_url.as_ref(), environment, plugin_resolver).await,
ConfigSubCommand::Update { yes } => commands::update_plugins_config_file(args, environment, plugin_resolver, *yes).await,
},
SubCommand::Version => commands::output_version(environment),
SubCommand::StdInFmt(cmd) => commands::stdin_fmt(cmd, args, environment, plugin_resolver).await,
SubCommand::OutputResolvedConfig => commands::output_resolved_config(args, environment, plugin_resolver).await,
SubCommand::OutputFilePaths(cmd) => commands::output_file_paths(cmd, args, environment, plugin_resolver).await,
SubCommand::OutputFormatTimes(cmd) => commands::output_format_times(cmd, args, environment, plugin_resolver).await,
SubCommand::Check(cmd) => commands::check(cmd, args, environment, plugin_resolver).await,
SubCommand::Fmt(cmd) => commands::format(cmd, args, environment, plugin_resolver).await,
SubCommand::Completions(shell) => commands::completions(shell.to_owned(), environment),
SubCommand::Upgrade => commands::upgrade(environment).await,
#[cfg(target_os = "windows")]
SubCommand::Hidden(hidden_command) => match hidden_command {
crate::arg_parser::HiddenSubCommand::WindowsInstall(install_path) => commands::handle_windows_install(environment, install_path),
crate::arg_parser::HiddenSubCommand::WindowsUninstall(install_path) => commands::handle_windows_uninstall(environment, install_path),
},
}
}