2021-05-04 14:02:23 +03:00
|
|
|
//! Fully integrated benchmarks for rust-analyzer, which load real cargo
|
|
|
|
//! projects.
|
|
|
|
//!
|
|
|
|
//! The benchmark here is used to debug specific performance regressions. If you
|
|
|
|
//! notice that, eg, completion is slow in some specific case, you can modify
|
|
|
|
//! code here exercise this specific completion, and thus have a fast
|
|
|
|
//! edit/compile/test cycle.
|
|
|
|
//!
|
2022-08-01 13:47:09 +02:00
|
|
|
//! Note that "rust-analyzer: Run" action does not allow running a single test
|
|
|
|
//! in release mode in VS Code. There's however "rust-analyzer: Copy Run Command Line"
|
2021-05-04 14:02:23 +03:00
|
|
|
//! which you can use to paste the command in terminal and add `--release` manually.
|
|
|
|
|
2023-12-18 12:09:54 +01:00
|
|
|
use hir::Change;
|
|
|
|
use ide::{CallableSnippets, CompletionConfig, FilePosition, TextSize};
|
2022-03-06 19:01:30 +01:00
|
|
|
use ide_db::{
|
|
|
|
imports::insert_use::{ImportGranularity, InsertUseConfig},
|
2021-05-18 19:49:15 +02:00
|
|
|
SnippetCap,
|
|
|
|
};
|
2021-07-18 11:29:22 +03:00
|
|
|
use project_model::CargoConfig;
|
2021-05-04 14:02:23 +03:00
|
|
|
use test_utils::project_root;
|
2023-05-02 17:12:22 +03:00
|
|
|
use triomphe::Arc;
|
2021-05-04 14:02:23 +03:00
|
|
|
use vfs::{AbsPathBuf, VfsPath};
|
|
|
|
|
2023-07-03 17:40:31 +02:00
|
|
|
use load_cargo::{load_workspace_at, LoadCargoConfig, ProcMacroServerChoice};
|
2021-05-04 14:02:23 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn integrated_highlighting_benchmark() {
|
2021-05-04 18:21:36 +03:00
|
|
|
if std::env::var("RUN_SLOW_BENCHES").is_err() {
|
2021-05-04 14:02:23 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load rust-analyzer itself.
|
|
|
|
let workspace_to_load = project_root();
|
2023-12-02 13:34:40 +01:00
|
|
|
let file = "./crates/rust-analyzer/src/config.rs";
|
2021-05-04 14:02:23 +03:00
|
|
|
|
2023-12-12 15:44:27 +01:00
|
|
|
let cargo_config = CargoConfig {
|
|
|
|
sysroot: Some(project_model::RustLibSource::Discover),
|
|
|
|
..CargoConfig::default()
|
|
|
|
};
|
2021-05-04 14:02:23 +03:00
|
|
|
let load_cargo_config = LoadCargoConfig {
|
|
|
|
load_out_dirs_from_check: true,
|
2023-02-08 11:05:34 +01:00
|
|
|
with_proc_macro_server: ProcMacroServerChoice::None,
|
2021-06-11 09:27:25 +03:00
|
|
|
prefill_caches: false,
|
2021-05-04 14:02:23 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
let (mut host, vfs, _proc_macro) = {
|
|
|
|
let _it = stdx::timeit("workspace loading");
|
|
|
|
load_workspace_at(&workspace_to_load, &cargo_config, &load_cargo_config, &|_| {}).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let file_id = {
|
|
|
|
let file = workspace_to_load.join(file);
|
|
|
|
let path = VfsPath::from(AbsPathBuf::assert(file));
|
2022-12-23 13:42:58 -05:00
|
|
|
vfs.file_id(&path).unwrap_or_else(|| panic!("can't find virtual file for {path}"))
|
2021-05-04 14:02:23 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
{
|
|
|
|
let _it = stdx::timeit("initial");
|
|
|
|
let analysis = host.analysis();
|
|
|
|
analysis.highlight_as_html(file_id, false).unwrap();
|
|
|
|
}
|
|
|
|
|
2023-11-28 16:28:51 +01:00
|
|
|
profile::init_from("*>100");
|
2021-05-04 14:02:23 +03:00
|
|
|
|
|
|
|
{
|
|
|
|
let _it = stdx::timeit("change");
|
|
|
|
let mut text = host.analysis().file_text(file_id).unwrap().to_string();
|
|
|
|
text.push_str("\npub fn _dummy() {}\n");
|
|
|
|
let mut change = Change::new();
|
2023-04-22 09:48:37 +02:00
|
|
|
change.change_file(file_id, Some(Arc::from(text)));
|
2021-05-04 14:02:23 +03:00
|
|
|
host.apply_change(change);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
let _it = stdx::timeit("after change");
|
|
|
|
let _span = profile::cpu_span();
|
|
|
|
let analysis = host.analysis();
|
|
|
|
analysis.highlight_as_html(file_id, false).unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn integrated_completion_benchmark() {
|
2021-05-04 18:21:36 +03:00
|
|
|
if std::env::var("RUN_SLOW_BENCHES").is_err() {
|
2021-05-04 14:02:23 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load rust-analyzer itself.
|
|
|
|
let workspace_to_load = project_root();
|
|
|
|
let file = "./crates/hir/src/lib.rs";
|
|
|
|
|
2023-12-12 15:44:27 +01:00
|
|
|
let cargo_config = CargoConfig {
|
|
|
|
sysroot: Some(project_model::RustLibSource::Discover),
|
|
|
|
..CargoConfig::default()
|
|
|
|
};
|
2021-05-04 14:02:23 +03:00
|
|
|
let load_cargo_config = LoadCargoConfig {
|
|
|
|
load_out_dirs_from_check: true,
|
2023-02-08 11:05:34 +01:00
|
|
|
with_proc_macro_server: ProcMacroServerChoice::None,
|
2021-06-11 09:27:25 +03:00
|
|
|
prefill_caches: true,
|
2021-05-04 14:02:23 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
let (mut host, vfs, _proc_macro) = {
|
|
|
|
let _it = stdx::timeit("workspace loading");
|
|
|
|
load_workspace_at(&workspace_to_load, &cargo_config, &load_cargo_config, &|_| {}).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let file_id = {
|
|
|
|
let file = workspace_to_load.join(file);
|
|
|
|
let path = VfsPath::from(AbsPathBuf::assert(file));
|
2022-12-23 13:42:58 -05:00
|
|
|
vfs.file_id(&path).unwrap_or_else(|| panic!("can't find virtual file for {path}"))
|
2021-05-04 14:02:23 +03:00
|
|
|
};
|
|
|
|
|
2023-12-12 22:43:33 +01:00
|
|
|
// kick off parsing and index population
|
|
|
|
|
|
|
|
let completion_offset = {
|
|
|
|
let _it = stdx::timeit("change");
|
|
|
|
let mut text = host.analysis().file_text(file_id).unwrap().to_string();
|
|
|
|
let completion_offset =
|
|
|
|
patch(&mut text, "db.struct_data(self.id)", "sel;\ndb.struct_data(self.id)")
|
|
|
|
+ "sel".len();
|
|
|
|
let mut change = Change::new();
|
|
|
|
change.change_file(file_id, Some(Arc::from(text)));
|
|
|
|
host.apply_change(change);
|
|
|
|
completion_offset
|
|
|
|
};
|
|
|
|
|
2021-05-04 14:02:23 +03:00
|
|
|
{
|
2023-12-12 22:43:33 +01:00
|
|
|
let _span = profile::cpu_span();
|
2021-05-04 14:02:23 +03:00
|
|
|
let analysis = host.analysis();
|
2023-12-12 22:43:33 +01:00
|
|
|
let config = CompletionConfig {
|
|
|
|
enable_postfix_completions: true,
|
|
|
|
enable_imports_on_the_fly: true,
|
|
|
|
enable_self_on_the_fly: true,
|
|
|
|
enable_private_editable: true,
|
|
|
|
full_function_signatures: false,
|
|
|
|
callable: Some(CallableSnippets::FillArguments),
|
|
|
|
snippet_cap: SnippetCap::new(true),
|
|
|
|
insert_use: InsertUseConfig {
|
|
|
|
granularity: ImportGranularity::Crate,
|
|
|
|
prefix_kind: hir::PrefixKind::ByCrate,
|
|
|
|
enforce_granularity: true,
|
|
|
|
group: true,
|
|
|
|
skip_glob_imports: true,
|
|
|
|
},
|
|
|
|
snippets: Vec::new(),
|
|
|
|
prefer_no_std: false,
|
|
|
|
prefer_prelude: true,
|
|
|
|
limit: None,
|
|
|
|
};
|
|
|
|
let position =
|
|
|
|
FilePosition { file_id, offset: TextSize::try_from(completion_offset).unwrap() };
|
|
|
|
analysis.completions(&config, position, None).unwrap();
|
2021-05-04 14:02:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
profile::init_from("*>5");
|
|
|
|
// let _s = profile::heartbeat_span();
|
|
|
|
|
|
|
|
let completion_offset = {
|
|
|
|
let _it = stdx::timeit("change");
|
|
|
|
let mut text = host.analysis().file_text(file_id).unwrap().to_string();
|
|
|
|
let completion_offset =
|
2023-12-12 22:43:33 +01:00
|
|
|
patch(&mut text, "sel;\ndb.struct_data(self.id)", ";sel;\ndb.struct_data(self.id)")
|
|
|
|
+ ";sel".len();
|
2021-05-04 14:02:23 +03:00
|
|
|
let mut change = Change::new();
|
2023-04-22 09:48:37 +02:00
|
|
|
change.change_file(file_id, Some(Arc::from(text)));
|
2021-05-04 14:02:23 +03:00
|
|
|
host.apply_change(change);
|
|
|
|
completion_offset
|
|
|
|
};
|
|
|
|
|
|
|
|
{
|
2021-05-14 20:23:29 +03:00
|
|
|
let _p = profile::span("unqualified path completion");
|
2021-05-04 14:02:23 +03:00
|
|
|
let _span = profile::cpu_span();
|
|
|
|
let analysis = host.analysis();
|
|
|
|
let config = CompletionConfig {
|
|
|
|
enable_postfix_completions: true,
|
|
|
|
enable_imports_on_the_fly: true,
|
2021-05-30 16:41:33 +02:00
|
|
|
enable_self_on_the_fly: true,
|
2022-02-23 16:02:54 +01:00
|
|
|
enable_private_editable: true,
|
2023-09-04 00:02:08 -03:00
|
|
|
full_function_signatures: false,
|
2022-05-13 19:52:44 +02:00
|
|
|
callable: Some(CallableSnippets::FillArguments),
|
2021-05-04 14:02:23 +03:00
|
|
|
snippet_cap: SnippetCap::new(true),
|
|
|
|
insert_use: InsertUseConfig {
|
2021-05-18 19:49:15 +02:00
|
|
|
granularity: ImportGranularity::Crate,
|
2021-05-04 14:02:23 +03:00
|
|
|
prefix_kind: hir::PrefixKind::ByCrate,
|
2021-05-18 20:21:47 +02:00
|
|
|
enforce_granularity: true,
|
2021-05-04 14:02:23 +03:00
|
|
|
group: true,
|
2021-06-18 23:11:56 +02:00
|
|
|
skip_glob_imports: true,
|
2021-05-04 14:02:23 +03:00
|
|
|
},
|
2021-10-04 19:22:41 +02:00
|
|
|
snippets: Vec::new(),
|
2022-09-13 15:09:40 +02:00
|
|
|
prefer_no_std: false,
|
2023-11-11 14:52:11 +01:00
|
|
|
prefer_prelude: true,
|
2023-01-19 18:33:47 -08:00
|
|
|
limit: None,
|
2021-05-04 14:02:23 +03:00
|
|
|
};
|
|
|
|
let position =
|
|
|
|
FilePosition { file_id, offset: TextSize::try_from(completion_offset).unwrap() };
|
2022-05-30 00:06:48 +09:00
|
|
|
analysis.completions(&config, position, None).unwrap();
|
2021-05-04 14:02:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
let completion_offset = {
|
|
|
|
let _it = stdx::timeit("change");
|
|
|
|
let mut text = host.analysis().file_text(file_id).unwrap().to_string();
|
|
|
|
let completion_offset =
|
|
|
|
patch(&mut text, "sel;\ndb.struct_data(self.id)", "self.;\ndb.struct_data(self.id)")
|
|
|
|
+ "self.".len();
|
|
|
|
let mut change = Change::new();
|
2023-04-22 09:48:37 +02:00
|
|
|
change.change_file(file_id, Some(Arc::from(text)));
|
2021-05-04 14:02:23 +03:00
|
|
|
host.apply_change(change);
|
|
|
|
completion_offset
|
|
|
|
};
|
|
|
|
|
|
|
|
{
|
2021-05-14 20:23:29 +03:00
|
|
|
let _p = profile::span("dot completion");
|
2021-05-04 14:02:23 +03:00
|
|
|
let _span = profile::cpu_span();
|
|
|
|
let analysis = host.analysis();
|
|
|
|
let config = CompletionConfig {
|
|
|
|
enable_postfix_completions: true,
|
|
|
|
enable_imports_on_the_fly: true,
|
2021-05-30 16:41:33 +02:00
|
|
|
enable_self_on_the_fly: true,
|
2022-02-23 16:02:54 +01:00
|
|
|
enable_private_editable: true,
|
2023-09-04 00:02:08 -03:00
|
|
|
full_function_signatures: false,
|
2022-05-13 19:52:44 +02:00
|
|
|
callable: Some(CallableSnippets::FillArguments),
|
2021-05-04 14:02:23 +03:00
|
|
|
snippet_cap: SnippetCap::new(true),
|
|
|
|
insert_use: InsertUseConfig {
|
2021-05-18 19:49:15 +02:00
|
|
|
granularity: ImportGranularity::Crate,
|
2021-05-04 14:02:23 +03:00
|
|
|
prefix_kind: hir::PrefixKind::ByCrate,
|
2021-05-18 20:21:47 +02:00
|
|
|
enforce_granularity: true,
|
2021-05-04 14:02:23 +03:00
|
|
|
group: true,
|
2021-06-18 23:11:56 +02:00
|
|
|
skip_glob_imports: true,
|
2021-05-04 14:02:23 +03:00
|
|
|
},
|
2021-10-04 19:22:41 +02:00
|
|
|
snippets: Vec::new(),
|
2022-09-13 15:09:40 +02:00
|
|
|
prefer_no_std: false,
|
2023-11-11 14:52:11 +01:00
|
|
|
prefer_prelude: true,
|
2023-01-19 18:33:47 -08:00
|
|
|
limit: None,
|
2021-05-04 14:02:23 +03:00
|
|
|
};
|
|
|
|
let position =
|
|
|
|
FilePosition { file_id, offset: TextSize::try_from(completion_offset).unwrap() };
|
2022-05-30 00:06:48 +09:00
|
|
|
analysis.completions(&config, position, None).unwrap();
|
2021-05-04 14:02:23 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn patch(what: &mut String, from: &str, to: &str) -> usize {
|
|
|
|
let idx = what.find(from).unwrap();
|
|
|
|
*what = what.replacen(from, to, 1);
|
|
|
|
idx
|
|
|
|
}
|