filter by time

This commit is contained in:
Aleksey Kladov 2019-04-14 23:18:58 +03:00
parent b228947b68
commit 5b7012318c
3 changed files with 44 additions and 29 deletions

View file

@ -8,31 +8,14 @@ use ra_prof;
fn main() -> Result<()> { fn main() -> Result<()> {
std::env::set_var("RUST_BACKTRACE", "short"); std::env::set_var("RUST_BACKTRACE", "short");
let logger = Logger::with_env_or_str("error").duplicate_to_stderr(Duplicate::All); let logger = Logger::with_env_or_str("error").duplicate_to_stderr(Duplicate::All);
match std::env::var("RA_INTERNAL_MODE") { match std::env::var("RA_LOG_DIR") {
Ok(ref v) if v == "1" => logger.log_to_file().directory("log").start()?, Ok(ref v) if v == "1" => logger.log_to_file().directory("log").start()?,
_ => logger.start()?, _ => logger.start()?,
}; };
// Filtering syntax ra_prof::set_filter(match std::env::var("RA_PROFILE") {
// env RA_PROFILE=* // dump everything Ok(spec) => ra_prof::Filter::from_spec(&spec),
// env RA_PROFILE=foo|bar|baz // enabled only selected entries
// env RA_PROFILE=*@3 // dump everything, up to depth 3
let filter = match std::env::var("RA_PROFILE") {
Ok(p) => {
let mut p = p.as_str();
let depth = if let Some(idx) = p.rfind("@") {
let depth: usize = p[idx + 1..].parse().expect("invalid profile depth");
p = &p[..idx];
depth
} else {
999
};
let allowed =
if p == "*" { Vec::new() } else { p.split(";").map(String::from).collect() };
ra_prof::Filter::new(depth, allowed)
}
Err(_) => ra_prof::Filter::disabled(), Err(_) => ra_prof::Filter::disabled(),
}; });
ra_prof::set_filter(filter);
log::info!("lifecycle: server started"); log::info!("lifecycle: server started");
match ::std::panic::catch_unwind(main_inner) { match ::std::panic::catch_unwind(main_inner) {
Ok(res) => { Ok(res) => {

View file

@ -27,7 +27,8 @@ pub fn set_filter(f: Filter) {
PROFILING_ENABLED.store(f.depth > 0, Ordering::SeqCst); PROFILING_ENABLED.store(f.depth > 0, Ordering::SeqCst);
let set = HashSet::from_iter(f.allowed.iter().cloned()); let set = HashSet::from_iter(f.allowed.iter().cloned());
let mut old = FILTER.write().unwrap(); let mut old = FILTER.write().unwrap();
let filter_data = FilterData { depth: f.depth, allowed: set, version: old.version + 1 }; let filter_data =
FilterData { depth: f.depth, allowed: set, cutoff: f.cutoff, version: old.version + 1 };
*old = filter_data; *old = filter_data;
} }
@ -101,15 +102,41 @@ pub struct Profiler {
pub struct Filter { pub struct Filter {
depth: usize, depth: usize,
allowed: Vec<String>, allowed: Vec<String>,
cutoff: Duration,
} }
impl Filter { impl Filter {
pub fn disabled() -> Filter { // Filtering syntax
Filter::new(0, Vec::new()) // env RA_PROFILE=* // dump everything
// env RA_PROFILE=foo|bar|baz // enabled only selected entries
// env RA_PROFILE=*@3>10 // dump everything, up to depth 3, if it takes more than 10 ms
pub fn from_spec(mut spec: &str) -> Filter {
let cutoff = if let Some(idx) = spec.rfind(">") {
let cutoff = spec[idx + 1..].parse().expect("invalid profile cutoff");
spec = &spec[..idx];
Duration::from_millis(cutoff)
} else {
Duration::new(0, 0)
};
let depth = if let Some(idx) = spec.rfind("@") {
let depth: usize = spec[idx + 1..].parse().expect("invalid profile depth");
spec = &spec[..idx];
depth
} else {
999
};
let allowed =
if spec == "*" { Vec::new() } else { spec.split("|").map(String::from).collect() };
Filter::new(depth, allowed, cutoff)
} }
pub fn new(depth: usize, allowed: Vec<String>) -> Filter { pub fn disabled() -> Filter {
Filter { depth, allowed } Filter::new(0, Vec::new(), Duration::new(0, 0))
}
pub fn new(depth: usize, allowed: Vec<String>, cutoff: Duration) -> Filter {
Filter { depth, allowed, cutoff }
} }
} }
@ -136,6 +163,7 @@ struct FilterData {
depth: usize, depth: usize,
version: usize, version: usize,
allowed: HashSet<String>, allowed: HashSet<String>,
cutoff: Duration,
} }
static PROFILING_ENABLED: AtomicBool = AtomicBool::new(false); static PROFILING_ENABLED: AtomicBool = AtomicBool::new(false);
@ -159,7 +187,9 @@ impl Drop for Profiler {
stack.messages.push(Message { level, duration, message }); stack.messages.push(Message { level, duration, message });
if level == 0 { if level == 0 {
let stdout = stderr(); let stdout = stderr();
if duration >= stack.filter_data.cutoff {
print(0, &stack.messages, &mut stdout.lock()); print(0, &stack.messages, &mut stdout.lock());
}
stack.messages.clear(); stack.messages.clear();
} }
}); });

View file

@ -105,7 +105,7 @@ figure out where logs go.
Inside rust-analyzer, we use the standard `log` crate for logging, and Inside rust-analyzer, we use the standard `log` crate for logging, and
`flexi_logger` for logging frotend. By default, log goes to stderr (the same as `flexi_logger` for logging frotend. By default, log goes to stderr (the same as
with `env_logger`), but the stderr itself is processed by VS Code. To mirror with `env_logger`), but the stderr itself is processed by VS Code. To mirror
logs to a `./log` directory, set `RA_INTERNAL_MODE=1` environmental variable. logs to a `./log` directory, set `RA_LOG_DIR=1` environmental variable.
To see stderr in the running VS Code instance, go to the "Output" tab of the To see stderr in the running VS Code instance, go to the "Output" tab of the
panel and select `rust-analyzer`. This shows `eprintln!` as well. Note that panel and select `rust-analyzer`. This shows `eprintln!` as well. Note that
@ -143,5 +143,7 @@ We have a built-in hierarchical profiler, you can enable it by using `RA_PROF` e
``` ```
RA_PROFILE=* // dump everything RA_PROFILE=* // dump everything
RA_PROFILE=foo|bar|baz // enabled only selected entries RA_PROFILE=foo|bar|baz // enabled only selected entries
RA_PROFILE=*@3 // dump everything, up to depth 3 RA_PROFILE=*@3>10 // dump everything, up to depth 3, if it takes more than 10 ms
``` ```
In particular, I have `export RA_PROFILE='*>10' in my shell profile.