mirror of
https://gitlab.com/gabmus/envision.git
synced 2025-08-03 06:38:52 +00:00
feat: log level filtering based on monado json output
This commit is contained in:
parent
43c05070f2
commit
c90b24b554
7 changed files with 163 additions and 57 deletions
|
@ -11,6 +11,7 @@ pub static ENV_VAR_DESCRIPTIONS: Map<&str, &str> = phf_map! {
|
||||||
"LD_LIBRARY_PATH" =>
|
"LD_LIBRARY_PATH" =>
|
||||||
"Colon-separated list of directories where the dynamic linker will search for shared object libraries.",
|
"Colon-separated list of directories where the dynamic linker will search for shared object libraries.",
|
||||||
"XRT_DEBUG_GUI" => "Set to 1 to enable the Monado debug UI",
|
"XRT_DEBUG_GUI" => "Set to 1 to enable the Monado debug UI",
|
||||||
|
"XRT_JSON_LOG" => "Set to 1 to enable JSON logging for Monado. This enables better log visualization and log level filtering.",
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn env_var_descriptions_as_paragraph() -> String {
|
pub fn env_var_descriptions_as_paragraph() -> String {
|
||||||
|
|
102
src/log_level.rs
Normal file
102
src/log_level.rs
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
use serde::{de::Visitor, Deserialize, Serialize};
|
||||||
|
use std::{fmt::Display, slice::Iter};
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Eq, Clone, Copy, Serialize)]
|
||||||
|
pub enum LogLevel {
|
||||||
|
Trace,
|
||||||
|
Debug,
|
||||||
|
Info,
|
||||||
|
Warning,
|
||||||
|
Error,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct LogLevelStringVisitor;
|
||||||
|
impl<'de> Visitor<'de> for LogLevelStringVisitor {
|
||||||
|
type Value = LogLevel;
|
||||||
|
|
||||||
|
fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
|
||||||
|
formatter.write_str(
|
||||||
|
"A case-insensitive string among trace, debug, info, warning, warn, error, err",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_borrowed_str<E>(self, v: &'de str) -> Result<Self::Value, E>
|
||||||
|
where
|
||||||
|
E: serde::de::Error, {
|
||||||
|
Ok(LogLevel::from_string(v.to_string()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
|
||||||
|
where
|
||||||
|
E: serde::de::Error, {
|
||||||
|
Ok(LogLevel::from_string(v))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de> Deserialize<'de> for LogLevel {
|
||||||
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
||||||
|
where
|
||||||
|
D: serde::Deserializer<'de> {
|
||||||
|
deserializer.deserialize_string(LogLevelStringVisitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LogLevel {
|
||||||
|
pub fn from_string(s: String) -> Self {
|
||||||
|
match s.to_lowercase().as_str() {
|
||||||
|
"trace" => Self::Trace,
|
||||||
|
"debug" => Self::Debug,
|
||||||
|
"info" => Self::Info,
|
||||||
|
"warning" => Self::Warning,
|
||||||
|
"warn" => Self::Warning,
|
||||||
|
"error" => Self::Error,
|
||||||
|
"err" => Self::Error,
|
||||||
|
_ => Self::Debug,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn iter() -> Iter<'static, LogLevel> {
|
||||||
|
[
|
||||||
|
Self::Trace,
|
||||||
|
Self::Debug,
|
||||||
|
Self::Info,
|
||||||
|
Self::Warning,
|
||||||
|
Self::Error,
|
||||||
|
]
|
||||||
|
.iter()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn as_number(&self) -> u32 {
|
||||||
|
match self {
|
||||||
|
Self::Trace => 0,
|
||||||
|
Self::Debug => 1,
|
||||||
|
Self::Info => 2,
|
||||||
|
Self::Warning => 3,
|
||||||
|
Self::Error => 99,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialOrd for LogLevel {
|
||||||
|
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
||||||
|
Some(self.as_number().cmp(&other.as_number()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Ord for LogLevel {
|
||||||
|
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
|
||||||
|
self.partial_cmp(other).unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for LogLevel {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
f.write_str(match self {
|
||||||
|
LogLevel::Trace => "Trace",
|
||||||
|
LogLevel::Debug => "Debug",
|
||||||
|
LogLevel::Info => "Info",
|
||||||
|
LogLevel::Warning => "Warning",
|
||||||
|
LogLevel::Error => "Error",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
16
src/log_parser.rs
Normal file
16
src/log_parser.rs
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
use crate::log_level::LogLevel;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
|
||||||
|
pub struct MonadoLog {
|
||||||
|
pub level: LogLevel,
|
||||||
|
pub file: String,
|
||||||
|
pub func: String,
|
||||||
|
pub message: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MonadoLog {
|
||||||
|
pub fn from_str(s: &str) -> Option<Self> {
|
||||||
|
serde_json::from_str::<Self>(s).ok()
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,6 +24,8 @@ pub mod ui;
|
||||||
pub mod adb;
|
pub mod adb;
|
||||||
pub mod downloader;
|
pub mod downloader;
|
||||||
pub mod env_var_descriptions;
|
pub mod env_var_descriptions;
|
||||||
|
pub mod log_parser;
|
||||||
|
pub mod log_level;
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
fn main() -> Result<()> {
|
||||||
// Prepare i18n
|
// Prepare i18n
|
||||||
|
|
|
@ -7,6 +7,7 @@ use std::collections::HashMap;
|
||||||
|
|
||||||
pub fn system_valve_index_profile() -> Profile {
|
pub fn system_valve_index_profile() -> Profile {
|
||||||
let mut environment: HashMap<String, String> = HashMap::new();
|
let mut environment: HashMap<String, String> = HashMap::new();
|
||||||
|
environment.insert("XRT_JSON_LOG".into(), "1".into());
|
||||||
environment.insert("XRT_COMPOSITOR_SCALE_PERCENTAGE".into(), "140".into());
|
environment.insert("XRT_COMPOSITOR_SCALE_PERCENTAGE".into(), "140".into());
|
||||||
environment.insert("XRT_COMPOSITOR_COMPUTE".into(), "1".into());
|
environment.insert("XRT_COMPOSITOR_COMPUTE".into(), "1".into());
|
||||||
environment.insert("SURVIVE_GLOBALSCENESOLVER".into(), "0".into());
|
environment.insert("SURVIVE_GLOBALSCENESOLVER".into(), "0".into());
|
||||||
|
|
|
@ -10,6 +10,7 @@ pub fn valve_index_profile() -> Profile {
|
||||||
let data_dir = get_data_dir();
|
let data_dir = get_data_dir();
|
||||||
let prefix = format!("{data}/prefixes/valve_index_default", data = data_dir);
|
let prefix = format!("{data}/prefixes/valve_index_default", data = data_dir);
|
||||||
let mut environment: HashMap<String, String> = HashMap::new();
|
let mut environment: HashMap<String, String> = HashMap::new();
|
||||||
|
environment.insert("XRT_JSON_LOG".into(), "1".into());
|
||||||
environment.insert("XRT_COMPOSITOR_SCALE_PERCENTAGE".into(), "140".into());
|
environment.insert("XRT_COMPOSITOR_SCALE_PERCENTAGE".into(), "140".into());
|
||||||
environment.insert("XRT_COMPOSITOR_COMPUTE".into(), "1".into());
|
environment.insert("XRT_COMPOSITOR_COMPUTE".into(), "1".into());
|
||||||
environment.insert("SURVIVE_GLOBALSCENESOLVER".into(), "0".into());
|
environment.insert("SURVIVE_GLOBALSCENESOLVER".into(), "0".into());
|
||||||
|
|
|
@ -1,55 +1,10 @@
|
||||||
|
use crate::log_level::LogLevel;
|
||||||
|
use crate::log_parser::MonadoLog;
|
||||||
use expect_dialog::ExpectDialog;
|
use expect_dialog::ExpectDialog;
|
||||||
use gtk::prelude::*;
|
use gtk::prelude::*;
|
||||||
use relm4::prelude::*;
|
use relm4::prelude::*;
|
||||||
use relm4::{ComponentSender, SimpleComponent};
|
use relm4::{ComponentSender, SimpleComponent};
|
||||||
use sourceview5::prelude::*;
|
use sourceview5::prelude::*;
|
||||||
use std::fmt::Display;
|
|
||||||
use std::slice::Iter;
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
|
||||||
pub enum LogLevel {
|
|
||||||
Trace,
|
|
||||||
Debug,
|
|
||||||
Info,
|
|
||||||
Warning,
|
|
||||||
Error,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl LogLevel {
|
|
||||||
pub fn from_string(s: String) -> Self {
|
|
||||||
match s.to_lowercase().as_str() {
|
|
||||||
"trace" => Self::Trace,
|
|
||||||
"debug" => Self::Debug,
|
|
||||||
"info" => Self::Info,
|
|
||||||
"warning" => Self::Warning,
|
|
||||||
"error" => Self::Error,
|
|
||||||
_ => Self::Debug,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn iter() -> Iter<'static, LogLevel> {
|
|
||||||
[
|
|
||||||
Self::Trace,
|
|
||||||
Self::Debug,
|
|
||||||
Self::Info,
|
|
||||||
Self::Warning,
|
|
||||||
Self::Error,
|
|
||||||
]
|
|
||||||
.iter()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Display for LogLevel {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
f.write_str(match self {
|
|
||||||
LogLevel::Trace => "Trace",
|
|
||||||
LogLevel::Debug => "Debug",
|
|
||||||
LogLevel::Info => "Info",
|
|
||||||
LogLevel::Warning => "Warning",
|
|
||||||
LogLevel::Error => "Error",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum SearchDirection {
|
pub enum SearchDirection {
|
||||||
|
@ -63,6 +18,7 @@ pub enum DebugViewMsg {
|
||||||
ClearLog,
|
ClearLog,
|
||||||
EnableDebugViewChanged(bool),
|
EnableDebugViewChanged(bool),
|
||||||
FilterLog(SearchDirection),
|
FilterLog(SearchDirection),
|
||||||
|
LogLevelChanged(LogLevel),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracker::track]
|
#[tracker::track]
|
||||||
|
@ -87,6 +43,8 @@ pub struct DebugView {
|
||||||
search_settings: sourceview5::SearchSettings,
|
search_settings: sourceview5::SearchSettings,
|
||||||
#[tracker::do_not_track]
|
#[tracker::do_not_track]
|
||||||
search_mark: Option<gtk::TextMark>,
|
search_mark: Option<gtk::TextMark>,
|
||||||
|
#[tracker::do_not_track]
|
||||||
|
log_level: LogLevel,
|
||||||
|
|
||||||
enable_debug_view: bool,
|
enable_debug_view: bool,
|
||||||
}
|
}
|
||||||
|
@ -189,16 +147,17 @@ impl SimpleComponent for DebugView {
|
||||||
self.reset();
|
self.reset();
|
||||||
|
|
||||||
match message {
|
match message {
|
||||||
|
Self::Input::LogLevelChanged(lvl) => {
|
||||||
|
self.log_level = lvl;
|
||||||
|
let log = self.log.clone();
|
||||||
|
self.log = vec![];
|
||||||
|
self.textbuf.set_text("");
|
||||||
|
sender.input(Self::Input::LogUpdated(log));
|
||||||
|
}
|
||||||
Self::Input::FilterLog(direction) => {
|
Self::Input::FilterLog(direction) => {
|
||||||
let searchbar = self.searchbar.as_ref().unwrap().clone();
|
let searchbar = self.searchbar.as_ref().unwrap().clone();
|
||||||
let search_entry = self.search_entry.as_ref().unwrap().clone();
|
let search_entry = self.search_entry.as_ref().unwrap().clone();
|
||||||
let search_text = search_entry.text().to_string();
|
let search_text = search_entry.text().to_string();
|
||||||
// TODO: add log level filtering
|
|
||||||
let log_level = LogLevel::iter()
|
|
||||||
.as_slice()
|
|
||||||
.get(self.dropdown.as_ref().unwrap().selected() as usize)
|
|
||||||
.unwrap();
|
|
||||||
println!("log level: {}", log_level.to_string());
|
|
||||||
if searchbar.is_search_mode() && !search_text.is_empty() {
|
if searchbar.is_search_mode() && !search_text.is_empty() {
|
||||||
self.search_settings
|
self.search_settings
|
||||||
.set_search_text(Some(search_text.as_str()));
|
.set_search_text(Some(search_text.as_str()));
|
||||||
|
@ -240,8 +199,25 @@ impl SimpleComponent for DebugView {
|
||||||
(adj.upper() - adj.page_size() - adj.value()) <= 15.0
|
(adj.upper() - adj.page_size() - adj.value()) <= 15.0
|
||||||
};
|
};
|
||||||
self.log.extend(n_log.clone());
|
self.log.extend(n_log.clone());
|
||||||
self.textbuf
|
for row in n_log {
|
||||||
.insert(&mut self.textbuf.end_iter(), n_log.concat().as_str());
|
let txt = match MonadoLog::from_str(row.as_str()) {
|
||||||
|
Some(o) => match o.level >= self.log_level {
|
||||||
|
false => None,
|
||||||
|
true => Some(format!(
|
||||||
|
"{lvl}\t[{file}:{func}]\n\t{msg}\n",
|
||||||
|
lvl = o.level.to_string(),
|
||||||
|
file = o.file,
|
||||||
|
func = o.func,
|
||||||
|
msg = o.message
|
||||||
|
)),
|
||||||
|
},
|
||||||
|
None => Some(row),
|
||||||
|
};
|
||||||
|
if txt.is_some() {
|
||||||
|
self.textbuf
|
||||||
|
.insert(&mut self.textbuf.end_iter(), txt.unwrap().as_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
let textbuf = self.textbuf.clone();
|
let textbuf = self.textbuf.clone();
|
||||||
let textview = self.textview.as_ref().unwrap().clone();
|
let textview = self.textview.as_ref().unwrap().clone();
|
||||||
if is_at_bottom && !self.searchbar.as_ref().unwrap().is_search_mode() {
|
if is_at_bottom && !self.searchbar.as_ref().unwrap().is_search_mode() {
|
||||||
|
@ -293,6 +269,7 @@ impl SimpleComponent for DebugView {
|
||||||
search_settings,
|
search_settings,
|
||||||
search_ctx,
|
search_ctx,
|
||||||
search_mark: None,
|
search_mark: None,
|
||||||
|
log_level: LogLevel::Trace,
|
||||||
};
|
};
|
||||||
|
|
||||||
let widgets = view_output!();
|
let widgets = view_output!();
|
||||||
|
@ -306,8 +283,14 @@ impl SimpleComponent for DebugView {
|
||||||
let dd_sender = sender.clone();
|
let dd_sender = sender.clone();
|
||||||
widgets
|
widgets
|
||||||
.log_level_dropdown
|
.log_level_dropdown
|
||||||
.connect_selected_notify(move |_| {
|
.connect_selected_notify(move |dd| {
|
||||||
dd_sender.input(Self::Input::FilterLog(SearchDirection::Forward));
|
dd_sender.input(Self::Input::LogLevelChanged(
|
||||||
|
LogLevel::iter()
|
||||||
|
.as_slice()
|
||||||
|
.get(dd.selected() as usize)
|
||||||
|
.unwrap()
|
||||||
|
.clone(),
|
||||||
|
));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue