use anyhow::bail;
use anyhow::Result;
use std::fmt::Write as FmtWrite;
use std::io::Read;
use std::io::Write;
use std::path::Path;
use std::path::PathBuf;
use std::sync::Arc;
use dprint_core::async_runtime::async_trait;
use crate::plugins::CompilationResult;
use crate::utils::LogLevel;
use crate::utils::ProgressBars;
use super::CanonicalizedPathBuf;
#[derive(Debug)]
pub enum DirEntry {
Directory(PathBuf),
File { name: std::ffi::OsString, path: PathBuf },
}
#[derive(Debug, Clone)]
pub enum FilePermissions {
Std(std::fs::Permissions),
#[allow(dead_code)]
Test(TestFilePermissions),
}
impl FilePermissions {
pub fn readonly(&self) -> bool {
match self {
FilePermissions::Std(p) => p.readonly(),
FilePermissions::Test(p) => p.readonly,
}
}
}
#[derive(Default, Debug, Clone)]
pub struct TestFilePermissions {
pub readonly: bool,
}
#[async_trait(?Send)]
pub trait UrlDownloader {
async fn download_file(&self, url: &str) -> Result<Option<Vec<u8>>>;
async fn download_file_err_404(&self, url: &str) -> Result<Vec<u8>> {
match self.download_file(url).await? {
Some(result) => Ok(result),
None => bail!("Error downloading {} - 404 Not Found", url),
}
}
}
#[async_trait]
pub trait Environment: Clone + Send + Sync + UrlDownloader + 'static {
fn is_real(&self) -> bool;
fn read_file(&self, file_path: impl AsRef<Path>) -> Result<String>;
fn read_file_bytes(&self, file_path: impl AsRef<Path>) -> Result<Vec<u8>>;
fn write_file(&self, file_path: impl AsRef<Path>, file_text: &str) -> Result<()> {
self.write_file_bytes(file_path, file_text.as_bytes())
}
fn write_file_bytes(&self, file_path: impl AsRef<Path>, bytes: &[u8]) -> Result<()>;
fn atomic_write_file_bytes(&self, file_path: impl AsRef<Path>, bytes: &[u8]) -> Result<()> {
let rand: String = (0..4).fold(String::new(), |mut output, _| {
let _ = write!(output, "{:02x}", rand::random::<u8>());
output
});
let extension = format!("{rand}.tmp");
let tmp_file = file_path.as_ref().with_extension(extension);
self.write_file_bytes(&tmp_file, bytes)?;
self.rename(tmp_file, file_path)
}
fn rename(&self, path_from: impl AsRef<Path>, path_to: impl AsRef<Path>) -> Result<()>;
fn remove_file(&self, file_path: impl AsRef<Path>) -> Result<()>;
fn remove_dir_all(&self, dir_path: impl AsRef<Path>) -> Result<()>;
fn dir_info(&self, dir_path: impl AsRef<Path>) -> std::io::Result<Vec<DirEntry>>;
fn path_exists(&self, file_path: impl AsRef<Path>) -> bool;
fn canonicalize(&self, path: impl AsRef<Path>) -> Result<CanonicalizedPathBuf>;
fn is_absolute_path(&self, path: impl AsRef<Path>) -> bool;
fn file_permissions(&self, path: impl AsRef<Path>) -> Result<FilePermissions>;
fn set_file_permissions(&self, path: impl AsRef<Path>, permissions: FilePermissions) -> Result<()>;
fn mk_dir_all(&self, path: impl AsRef<Path>) -> Result<()>;
fn cwd(&self) -> CanonicalizedPathBuf;
fn current_exe(&self) -> Result<PathBuf>;
fn __log__(&self, text: &str);
fn __log_stderr__(&self, text: &str) {
self.log_stderr_with_context(text, "dprint");
}
fn log_stderr_with_context(&self, text: &str, context_name: &str);
fn log_machine_readable(&self, text: &str);
fn log_action_with_progress<TResult: Send + Sync, TCreate: FnOnce(Box<dyn Fn(usize)>) -> TResult + Send + Sync>(
&self,
message: &str,
action: TCreate,
total_size: usize,
) -> TResult;
fn get_cache_dir(&self) -> CanonicalizedPathBuf;
fn cpu_arch(&self) -> String;
fn os(&self) -> String;
fn max_threads(&self) -> usize;
fn cli_version(&self) -> String;
fn get_time_secs(&self) -> u64;
fn get_selection(&self, prompt_message: &str, item_indent_width: u16, items: &[String]) -> Result<usize>;
fn get_multi_selection(&self, prompt_message: &str, item_indent_width: u16, items: &[(bool, String)]) -> Result<Vec<usize>>;
fn confirm(&self, prompt_message: &str, default_value: bool) -> Result<bool>;
fn is_ci(&self) -> bool;
fn log_level(&self) -> LogLevel;
fn compile_wasm(&self, wasm_bytes: &[u8]) -> Result<CompilationResult>;
fn wasm_cache_key(&self) -> String;
async fn cpu_usage(&self) -> u8;
fn stdout(&self) -> Box<dyn Write + Send>;
fn stdin(&self) -> Box<dyn Read + Send>;
fn progress_bars(&self) -> Option<&Arc<ProgressBars>> {
None
}
#[cfg(windows)]
fn ensure_system_path(&self, directory_path: &str) -> Result<()>;
#[cfg(windows)]
fn remove_system_path(&self, directory_path: &str) -> Result<()>;
}
macro_rules! log_debug {
($logger:expr, $($arg:tt)*) => {
if $logger.log_level().is_debug() {
let mut text = String::from("[DEBUG] ");
text.push_str(&format!($($arg)*));
$logger.__log_stderr__(&text);
}
}
}
macro_rules! log_stderr_info {
($logger:expr, $single_arg:expr $(,)? ) => {
if $logger.log_level().is_info() {
$logger.__log_stderr__($single_arg);
}
};
($logger:expr, $($arg:tt)*) => {
if $logger.log_level().is_info() {
$logger.__log_stderr__(&format!($($arg)*));
}
}
}
macro_rules! log_stdout_info {
($logger:expr, $single_arg:expr $(,)? ) => {
if $logger.log_level().is_info() {
$logger.__log__($single_arg);
}
};
($logger:expr, $($arg:tt)*) => {
if $logger.log_level().is_info() {
$logger.__log__(&format!($($arg)*));
}
}
}
macro_rules! log_warn {
($logger:expr, $single_arg:expr $(,)? ) => {
if $logger.log_level().is_warn() {
$logger.__log_stderr__($single_arg);
}
};
($logger:expr, $($arg:tt)*) => {
if $logger.log_level().is_warn() {
$logger.__log_stderr__(&format!($($arg)*));
}
}
}
macro_rules! log_error {
($logger:expr, $single_arg:expr $(,)? ) => {
if $logger.log_level().is_error() {
$logger.__log_stderr__($single_arg);
}
};
($logger:expr, $($arg:tt)*) => {
if $logger.log_level().is_error() {
$logger.__log_stderr__(&format!($($arg)*));
}
}
}
macro_rules! log_all {
($logger:expr, $single_arg:expr $(,)? ) => {
$logger.__log_stderr__($single_arg);
};
($logger:expr, $($arg:tt)*) => {
$logger.__log_stderr__(&format!($($arg)*));
}
}