mirror of
https://github.com/denoland/deno
synced 2024-11-05 18:45:24 +00:00
166 lines
4.3 KiB
Rust
166 lines
4.3 KiB
Rust
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use chrono::DateTime;
|
|
use chrono::Utc;
|
|
use deno_core::parking_lot::Mutex;
|
|
use std::fs;
|
|
use std::io::prelude::*;
|
|
use std::path::Path;
|
|
use std::sync::atomic::AtomicBool;
|
|
use std::sync::atomic::AtomicUsize;
|
|
use std::sync::atomic::Ordering;
|
|
use std::thread;
|
|
use std::time::SystemTime;
|
|
|
|
static LSP_DEBUG_FLAG: AtomicBool = AtomicBool::new(false);
|
|
static LSP_LOG_LEVEL: AtomicUsize = AtomicUsize::new(log::Level::Info as usize);
|
|
static LSP_WARN_LEVEL: AtomicUsize =
|
|
AtomicUsize::new(log::Level::Warn as usize);
|
|
static LOG_FILE: LogFile = LogFile {
|
|
enabled: AtomicBool::new(true),
|
|
buffer: Mutex::new(String::new()),
|
|
};
|
|
|
|
pub struct LogFile {
|
|
enabled: AtomicBool,
|
|
buffer: Mutex<String>,
|
|
}
|
|
|
|
impl LogFile {
|
|
pub fn write_line(&self, s: &str) {
|
|
if LOG_FILE.enabled.load(Ordering::Relaxed) {
|
|
let mut buffer = self.buffer.lock();
|
|
buffer.push_str(s);
|
|
buffer.push('\n');
|
|
}
|
|
}
|
|
|
|
fn commit(&self, path: &Path) {
|
|
let unbuffered = {
|
|
let mut buffer = self.buffer.lock();
|
|
if buffer.is_empty() {
|
|
return;
|
|
}
|
|
// We clone here rather than take so the buffer can retain its capacity.
|
|
let unbuffered = buffer.clone();
|
|
buffer.clear();
|
|
unbuffered
|
|
};
|
|
if let Ok(file) = fs::OpenOptions::new().append(true).open(path) {
|
|
write!(&file, "{}", unbuffered).ok();
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn init_log_file(enabled: bool) {
|
|
let prepare_path = || {
|
|
if !enabled {
|
|
return None;
|
|
}
|
|
let cwd = std::env::current_dir().ok()?;
|
|
let now = SystemTime::now();
|
|
let now: DateTime<Utc> = now.into();
|
|
let now = now.to_rfc3339().replace(':', "_");
|
|
let path = cwd.join(format!(".deno_lsp/log_{}.txt", now));
|
|
fs::create_dir_all(path.parent()?).ok()?;
|
|
fs::write(&path, "").ok()?;
|
|
Some(path)
|
|
};
|
|
let Some(path) = prepare_path() else {
|
|
LOG_FILE.enabled.store(false, Ordering::Relaxed);
|
|
LOG_FILE.buffer.lock().clear();
|
|
return;
|
|
};
|
|
thread::spawn(move || loop {
|
|
LOG_FILE.commit(&path);
|
|
thread::sleep(std::time::Duration::from_secs(1));
|
|
});
|
|
}
|
|
|
|
pub fn write_line_to_log_file(s: &str) {
|
|
LOG_FILE.write_line(s);
|
|
}
|
|
|
|
pub fn set_lsp_debug_flag(value: bool) {
|
|
LSP_DEBUG_FLAG.store(value, Ordering::SeqCst)
|
|
}
|
|
|
|
pub fn lsp_debug_enabled() -> bool {
|
|
LSP_DEBUG_FLAG.load(Ordering::SeqCst)
|
|
}
|
|
|
|
/// Change the lsp to log at the provided level.
|
|
pub fn set_lsp_log_level(level: log::Level) {
|
|
LSP_LOG_LEVEL.store(level as usize, Ordering::SeqCst)
|
|
}
|
|
|
|
pub fn lsp_log_level() -> log::Level {
|
|
let level = LSP_LOG_LEVEL.load(Ordering::SeqCst);
|
|
// TODO(bartlomieju):
|
|
#[allow(clippy::undocumented_unsafe_blocks)]
|
|
unsafe {
|
|
std::mem::transmute(level)
|
|
}
|
|
}
|
|
|
|
/// Change the lsp to warn at the provided level.
|
|
pub fn set_lsp_warn_level(level: log::Level) {
|
|
LSP_WARN_LEVEL.store(level as usize, Ordering::SeqCst)
|
|
}
|
|
|
|
pub fn lsp_warn_level() -> log::Level {
|
|
let level = LSP_LOG_LEVEL.load(Ordering::SeqCst);
|
|
// TODO(bartlomieju):
|
|
#[allow(clippy::undocumented_unsafe_blocks)]
|
|
unsafe {
|
|
std::mem::transmute(level)
|
|
}
|
|
}
|
|
|
|
/// Use this macro to do "info" logs in the lsp code. This allows
|
|
/// for downgrading these logs to another log level in the REPL.
|
|
macro_rules! lsp_log {
|
|
($($arg:tt)+) => (
|
|
let lsp_log_level = $crate::lsp::logging::lsp_log_level();
|
|
if lsp_log_level == log::Level::Debug {
|
|
$crate::lsp::logging::lsp_debug!($($arg)+)
|
|
} else {
|
|
let s = std::format!($($arg)+);
|
|
$crate::lsp::logging::write_line_to_log_file(&s);
|
|
log::log!(lsp_log_level, "{}", s)
|
|
}
|
|
)
|
|
}
|
|
|
|
/// Use this macro to do "warn" logs in the lsp code. This allows
|
|
/// for downgrading these logs to another log level in the REPL.
|
|
macro_rules! lsp_warn {
|
|
($($arg:tt)+) => (
|
|
{
|
|
let lsp_log_level = $crate::lsp::logging::lsp_warn_level();
|
|
if lsp_log_level == log::Level::Debug {
|
|
$crate::lsp::logging::lsp_debug!($($arg)+)
|
|
} else {
|
|
let s = std::format!($($arg)+);
|
|
$crate::lsp::logging::write_line_to_log_file(&s);
|
|
log::log!(lsp_log_level, "{}", s)
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
macro_rules! lsp_debug {
|
|
($($arg:tt)+) => (
|
|
{
|
|
let s = std::format!($($arg)+);
|
|
$crate::lsp::logging::write_line_to_log_file(&s);
|
|
if $crate::lsp::logging::lsp_debug_enabled() {
|
|
log::debug!("{}", s)
|
|
}
|
|
}
|
|
)
|
|
}
|
|
|
|
pub(super) use lsp_debug;
|
|
pub(super) use lsp_log;
|
|
pub(super) use lsp_warn;
|