mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-28 04:45:05 +00:00
v2
This commit is contained in:
parent
886eaa0a7d
commit
7e4aad5ba5
6 changed files with 282 additions and 238 deletions
|
@ -329,12 +329,10 @@ pub struct CrateData {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CrateData {
|
impl CrateData {
|
||||||
/**
|
/// Check if [`other`] is almost equal to [`self`] ignoring `CrateOrigin` value.
|
||||||
Check if [`other`] is almost equal to [`self`].
|
pub fn eq_ignoring_origin(&self, other: &CrateData) -> bool {
|
||||||
This method has some obscure bits. These are mostly there to be compliant with
|
// This method has some obscure bits. These are mostly there to be compliant with
|
||||||
some patches. References to the patches are given.
|
// some patches. References to the patches are given.
|
||||||
*/
|
|
||||||
pub fn almost_eq(&self, other: &CrateData) -> bool {
|
|
||||||
if self.root_file_id != other.root_file_id {
|
if self.root_file_id != other.root_file_id {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -356,16 +354,16 @@ impl CrateData {
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut opts = self.cfg_options.clone();
|
let mut opts = self.cfg_options.clone();
|
||||||
opts.apply_diff(CfgDiff {
|
opts.apply_diff(
|
||||||
disable: other.cfg_options.clone().into_iter().collect(),
|
CfgDiff::new(vec![], other.cfg_options.clone().into_iter().collect())
|
||||||
enable: vec![],
|
.expect("CfgOptions were expected to contain no duplicates."),
|
||||||
});
|
);
|
||||||
|
|
||||||
let mut cfgs = opts.into_iter();
|
let mut cfgs = opts.into_iter();
|
||||||
if let Some(cfg) = cfgs.next() {
|
if let Some(cfg) = cfgs.next() {
|
||||||
// Don't care if rust_analyzer CfgAtom is the only cfg in the difference set of self's and other's cfgs.
|
// Don't care if rust_analyzer CfgAtom is the only cfg in the difference set of self's and other's cfgs.
|
||||||
// https://github.com/rust-lang/rust-analyzer/blob/0840038f02daec6ba3238f05d8caa037d28701a0/crates/project-model/src/workspace.rs#L894
|
// https://github.com/rust-lang/rust-analyzer/blob/0840038f02daec6ba3238f05d8caa037d28701a0/crates/project-model/src/workspace.rs#L894
|
||||||
if !cfgs.next().is_none() || cfg.to_string() != "rust_analyzer" {
|
if cfgs.next().is_some() || cfg.to_string() != "rust_analyzer" {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -686,41 +684,35 @@ impl CrateGraph {
|
||||||
/// Note that for deduplication to fully work, `self`'s crate dependencies must be sorted by crate id.
|
/// Note that for deduplication to fully work, `self`'s crate dependencies must be sorted by crate id.
|
||||||
/// If the crate dependencies were sorted, the resulting graph from this `extend` call will also have the crate dependencies sorted.
|
/// If the crate dependencies were sorted, the resulting graph from this `extend` call will also have the crate dependencies sorted.
|
||||||
pub fn extend(&mut self, mut other: CrateGraph, proc_macros: &mut ProcMacroPaths) {
|
pub fn extend(&mut self, mut other: CrateGraph, proc_macros: &mut ProcMacroPaths) {
|
||||||
enum ExtendStrategy {
|
|
||||||
Dedup(CrateId),
|
|
||||||
Replace(CrateId),
|
|
||||||
}
|
|
||||||
|
|
||||||
let topo = other.crates_in_topological_order();
|
let topo = other.crates_in_topological_order();
|
||||||
let mut id_map: FxHashMap<CrateId, CrateId> = FxHashMap::default();
|
let mut id_map: FxHashMap<CrateId, CrateId> = FxHashMap::default();
|
||||||
for topo in topo {
|
for topo in topo {
|
||||||
let crate_data = &mut other.arena[topo];
|
let crate_data = &mut other.arena[topo];
|
||||||
|
|
||||||
crate_data.dependencies.iter_mut().for_each(|dep| {
|
crate_data.dependencies.iter_mut().for_each(|dep| dep.crate_id = id_map[&dep.crate_id]);
|
||||||
dep.crate_id = id_map[&dep.crate_id];
|
|
||||||
});
|
|
||||||
crate_data.dependencies.sort_by_key(|dep| dep.crate_id);
|
crate_data.dependencies.sort_by_key(|dep| dep.crate_id);
|
||||||
let res = self.arena.iter().find_map(|(id, data)| {
|
let res = self.arena.iter().find_map(|(id, data)| {
|
||||||
if data.almost_eq(crate_data) {
|
if data.eq_ignoring_origin(crate_data) {
|
||||||
if data.origin.is_lib() && crate_data.origin.is_local() {
|
if data.origin.is_lib() && crate_data.origin.is_local() {
|
||||||
// See #15656 for a relevant example.
|
// See #15656 for a relevant example.
|
||||||
return Some(ExtendStrategy::Replace(id));
|
return Some((id, true));
|
||||||
}
|
}
|
||||||
|
|
||||||
return Some(ExtendStrategy::Dedup(id));
|
return Some((id, false));
|
||||||
}
|
}
|
||||||
None
|
None
|
||||||
});
|
});
|
||||||
|
|
||||||
if let Some(res) = res {
|
if let Some((res, should_update_lib_to_local)) = res {
|
||||||
match res {
|
id_map.insert(topo, res);
|
||||||
ExtendStrategy::Dedup(res) => id_map.insert(topo, res),
|
if should_update_lib_to_local {
|
||||||
ExtendStrategy::Replace(res) => {
|
let origin_old = self.arena[res].origin.clone();
|
||||||
let id = self.arena.alloc(crate_data.clone());
|
assert!(origin_old.is_lib());
|
||||||
let _ = self.remove_and_replace(res, id);
|
|
||||||
id_map.insert(topo, id)
|
if let CrateOrigin::Library { repo, name } = origin_old {
|
||||||
|
self.arena[res].origin = CrateOrigin::Local { repo, name: Some(name) };
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
} else {
|
} else {
|
||||||
let id = self.arena.alloc(crate_data.clone());
|
let id = self.arena.alloc(crate_data.clone());
|
||||||
id_map.insert(topo, id);
|
id_map.insert(topo, id);
|
||||||
|
|
|
@ -115,8 +115,8 @@ impl<'a> IntoIterator for &'a CfgOptions {
|
||||||
#[derive(Default, Clone, Debug, PartialEq, Eq)]
|
#[derive(Default, Clone, Debug, PartialEq, Eq)]
|
||||||
pub struct CfgDiff {
|
pub struct CfgDiff {
|
||||||
// Invariants: No duplicates, no atom that's both in `enable` and `disable`.
|
// Invariants: No duplicates, no atom that's both in `enable` and `disable`.
|
||||||
pub enable: Vec<CfgAtom>,
|
enable: Vec<CfgAtom>,
|
||||||
pub disable: Vec<CfgAtom>,
|
disable: Vec<CfgAtom>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CfgDiff {
|
impl CfgDiff {
|
||||||
|
|
|
@ -249,3 +249,53 @@ fn crate_graph_dedup() {
|
||||||
crate_graph.extend(regex_crate_graph, &mut regex_proc_macros);
|
crate_graph.extend(regex_crate_graph, &mut regex_proc_macros);
|
||||||
assert_eq!(crate_graph.iter().count(), 118);
|
assert_eq!(crate_graph.iter().count(), 118);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_deduplicate_crate_differing_in_origin() {
|
||||||
|
let path_map = &mut Default::default();
|
||||||
|
let (mut crate_graph, _proc_macros) =
|
||||||
|
load_cargo_with_sysroot(path_map, "deduplication_crate_graph_A.json");
|
||||||
|
crate_graph.sort_deps();
|
||||||
|
let (crate_graph_1, mut _proc_macros_2) =
|
||||||
|
load_cargo_with_sysroot(path_map, "deduplication_crate_graph_B.json");
|
||||||
|
|
||||||
|
crate_graph.extend(crate_graph_1, &mut _proc_macros_2);
|
||||||
|
|
||||||
|
let mut crates_named_p1 = vec![];
|
||||||
|
for id in crate_graph.iter() {
|
||||||
|
let krate = &crate_graph[id];
|
||||||
|
if let Some(name) = krate.display_name.as_ref() {
|
||||||
|
if name.to_string() == "p1" {
|
||||||
|
crates_named_p1.push(krate);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert!(crates_named_p1.len() == 1);
|
||||||
|
assert!(crates_named_p1[0].origin.is_local());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_deduplicate_crate_differing_in_origin_in_rev_resolution_order() {
|
||||||
|
let path_map = &mut Default::default();
|
||||||
|
let (mut crate_graph, _proc_macros) =
|
||||||
|
load_cargo_with_sysroot(path_map, "deduplication_crate_graph_B.json");
|
||||||
|
crate_graph.sort_deps();
|
||||||
|
let (crate_graph_1, mut _proc_macros_2) =
|
||||||
|
load_cargo_with_sysroot(path_map, "deduplication_crate_graph_A.json");
|
||||||
|
|
||||||
|
crate_graph.extend(crate_graph_1, &mut _proc_macros_2);
|
||||||
|
|
||||||
|
let mut crates_named_p1 = vec![];
|
||||||
|
for id in crate_graph.iter() {
|
||||||
|
let krate = &crate_graph[id];
|
||||||
|
if let Some(name) = krate.display_name.as_ref() {
|
||||||
|
if name.to_string() == "p1" {
|
||||||
|
crates_named_p1.push(krate);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert!(crates_named_p1.len() == 1);
|
||||||
|
assert!(crates_named_p1[0].origin.is_local());
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
{
|
||||||
|
"packages": [
|
||||||
|
{
|
||||||
|
"name": "p1",
|
||||||
|
"version": "0.1.0",
|
||||||
|
"id": "p1 0.1.0 (path+file:///path/to/project/projects/p1)",
|
||||||
|
"license": null,
|
||||||
|
"license_file": null,
|
||||||
|
"description": null,
|
||||||
|
"source": null,
|
||||||
|
"dependencies": [],
|
||||||
|
"targets": [
|
||||||
|
{
|
||||||
|
"kind": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"crate_types": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"name": "p1",
|
||||||
|
"src_path": "/path/to/project/projects/p1/src/lib.rs",
|
||||||
|
"edition": "2021",
|
||||||
|
"doc": true,
|
||||||
|
"doctest": true,
|
||||||
|
"test": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"features": {},
|
||||||
|
"manifest_path": "/path/to/project/projects/p1/Cargo.toml",
|
||||||
|
"metadata": null,
|
||||||
|
"publish": null,
|
||||||
|
"authors": [],
|
||||||
|
"categories": [],
|
||||||
|
"keywords": [],
|
||||||
|
"readme": null,
|
||||||
|
"repository": null,
|
||||||
|
"homepage": null,
|
||||||
|
"documentation": null,
|
||||||
|
"edition": "2021",
|
||||||
|
"links": null,
|
||||||
|
"default_run": null,
|
||||||
|
"rust_version": null
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"workspace_members": [
|
||||||
|
"p1 0.1.0 (path+file:///path/to/project/projects/p1)"
|
||||||
|
],
|
||||||
|
"workspace_default_members": [
|
||||||
|
"p1 0.1.0 (path+file:///path/to/project/projects/p1)"
|
||||||
|
],
|
||||||
|
"resolve": {
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"id": "p1 0.1.0 (path+file:///path/to/project/projects/p1)",
|
||||||
|
"dependencies": [],
|
||||||
|
"deps": [],
|
||||||
|
"features": []
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"root": "p1 0.1.0 (path+file:///path/to/project/projects/p1)"
|
||||||
|
},
|
||||||
|
"target_directory": "/path/to/project/projects/p1/target",
|
||||||
|
"version": 1,
|
||||||
|
"workspace_root": "/path/to/project/projects/p1",
|
||||||
|
"metadata": null
|
||||||
|
}
|
140
crates/project-model/test_data/deduplication_crate_graph_B.json
Normal file
140
crates/project-model/test_data/deduplication_crate_graph_B.json
Normal file
|
@ -0,0 +1,140 @@
|
||||||
|
{
|
||||||
|
"packages": [
|
||||||
|
{
|
||||||
|
"name": "p1",
|
||||||
|
"version": "0.1.0",
|
||||||
|
"id": "p1 0.1.0 (path+file:///path/to/project/projects/p1)",
|
||||||
|
"license": null,
|
||||||
|
"license_file": null,
|
||||||
|
"description": null,
|
||||||
|
"source": null,
|
||||||
|
"dependencies": [],
|
||||||
|
"targets": [
|
||||||
|
{
|
||||||
|
"kind": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"crate_types": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"name": "p1",
|
||||||
|
"src_path": "/path/to/project/projects/p1/src/lib.rs",
|
||||||
|
"edition": "2021",
|
||||||
|
"doc": true,
|
||||||
|
"doctest": true,
|
||||||
|
"test": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"features": {},
|
||||||
|
"manifest_path": "/path/to/project/projects/p1/Cargo.toml",
|
||||||
|
"metadata": null,
|
||||||
|
"publish": null,
|
||||||
|
"authors": [],
|
||||||
|
"categories": [],
|
||||||
|
"keywords": [],
|
||||||
|
"readme": null,
|
||||||
|
"repository": null,
|
||||||
|
"homepage": null,
|
||||||
|
"documentation": null,
|
||||||
|
"edition": "2021",
|
||||||
|
"links": null,
|
||||||
|
"default_run": null,
|
||||||
|
"rust_version": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "p2",
|
||||||
|
"version": "0.1.0",
|
||||||
|
"id": "p2 0.1.0 (path+file:///path/to/project/projects/p2)",
|
||||||
|
"license": null,
|
||||||
|
"license_file": null,
|
||||||
|
"description": null,
|
||||||
|
"source": null,
|
||||||
|
"dependencies": [
|
||||||
|
{
|
||||||
|
"name": "p1",
|
||||||
|
"source": null,
|
||||||
|
"req": "*",
|
||||||
|
"kind": null,
|
||||||
|
"rename": null,
|
||||||
|
"optional": false,
|
||||||
|
"uses_default_features": true,
|
||||||
|
"features": [],
|
||||||
|
"target": null,
|
||||||
|
"registry": null,
|
||||||
|
"path": "/path/to/project/projects/p1"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"targets": [
|
||||||
|
{
|
||||||
|
"kind": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"crate_types": [
|
||||||
|
"lib"
|
||||||
|
],
|
||||||
|
"name": "p2",
|
||||||
|
"src_path": "/path/to/project/projects/p2/src/lib.rs",
|
||||||
|
"edition": "2021",
|
||||||
|
"doc": true,
|
||||||
|
"doctest": true,
|
||||||
|
"test": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"features": {},
|
||||||
|
"manifest_path": "/path/to/project/projects/p2/Cargo.toml",
|
||||||
|
"metadata": null,
|
||||||
|
"publish": null,
|
||||||
|
"authors": [],
|
||||||
|
"categories": [],
|
||||||
|
"keywords": [],
|
||||||
|
"readme": null,
|
||||||
|
"repository": null,
|
||||||
|
"homepage": null,
|
||||||
|
"documentation": null,
|
||||||
|
"edition": "2021",
|
||||||
|
"links": null,
|
||||||
|
"default_run": null,
|
||||||
|
"rust_version": null
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"workspace_members": [
|
||||||
|
"p2 0.1.0 (path+file:///path/to/project/projects/p2)"
|
||||||
|
],
|
||||||
|
"workspace_default_members": [
|
||||||
|
"p2 0.1.0 (path+file:///path/to/project/projects/p2)"
|
||||||
|
],
|
||||||
|
"resolve": {
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"id": "p1 0.1.0 (path+file:///path/to/project/projects/p1)",
|
||||||
|
"dependencies": [],
|
||||||
|
"deps": [],
|
||||||
|
"features": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "p2 0.1.0 (path+file:///path/to/project/projects/p2)",
|
||||||
|
"dependencies": [
|
||||||
|
"p1 0.1.0 (path+file:///path/to/project/projects/p1)"
|
||||||
|
],
|
||||||
|
"deps": [
|
||||||
|
{
|
||||||
|
"name": "p1",
|
||||||
|
"pkg": "p1 0.1.0 (path+file:///path/to/project/projects/p1)",
|
||||||
|
"dep_kinds": [
|
||||||
|
{
|
||||||
|
"kind": null,
|
||||||
|
"target": null
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"features": []
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"root": "p2 0.1.0 (path+file:///path/to/project/projects/p2)"
|
||||||
|
},
|
||||||
|
"target_directory": "/path/to/project/projects/p2/target",
|
||||||
|
"version": 1,
|
||||||
|
"workspace_root": "/path/to/project/projects/p2",
|
||||||
|
"metadata": null
|
||||||
|
}
|
|
@ -21,12 +21,12 @@ use std::{collections::HashMap, path::PathBuf, time::Instant};
|
||||||
use lsp_types::{
|
use lsp_types::{
|
||||||
notification::DidOpenTextDocument,
|
notification::DidOpenTextDocument,
|
||||||
request::{
|
request::{
|
||||||
CodeActionRequest, Completion, Formatting, GotoTypeDefinition, HoverRequest, Rename,
|
CodeActionRequest, Completion, Formatting, GotoTypeDefinition, HoverRequest,
|
||||||
WillRenameFiles, WorkspaceSymbolRequest,
|
WillRenameFiles, WorkspaceSymbolRequest,
|
||||||
},
|
},
|
||||||
CodeActionContext, CodeActionParams, CompletionParams, DidOpenTextDocumentParams,
|
CodeActionContext, CodeActionParams, CompletionParams, DidOpenTextDocumentParams,
|
||||||
DocumentFormattingParams, FileRename, FormattingOptions, GotoDefinitionParams, HoverParams,
|
DocumentFormattingParams, FileRename, FormattingOptions, GotoDefinitionParams, HoverParams,
|
||||||
PartialResultParams, Position, Range, RenameFilesParams, RenameParams, TextDocumentItem,
|
PartialResultParams, Position, Range, RenameFilesParams, TextDocumentItem,
|
||||||
TextDocumentPositionParams, WorkDoneProgressParams,
|
TextDocumentPositionParams, WorkDoneProgressParams,
|
||||||
};
|
};
|
||||||
use rust_analyzer::lsp::ext::{OnEnter, Runnables, RunnablesParams};
|
use rust_analyzer::lsp::ext::{OnEnter, Runnables, RunnablesParams};
|
||||||
|
@ -1131,207 +1131,3 @@ version = "0.0.0"
|
||||||
|
|
||||||
server.request::<WorkspaceSymbolRequest>(Default::default(), json!([]));
|
server.request::<WorkspaceSymbolRequest>(Default::default(), json!([]));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_deduplicate_crate_differing_in_origin() {
|
|
||||||
let fixture = r#"
|
|
||||||
//- /projects/p1/Cargo.toml
|
|
||||||
[package]
|
|
||||||
name = "p1"
|
|
||||||
version = "0.0.0"
|
|
||||||
|
|
||||||
//- /projects/p1/src/lib.rs
|
|
||||||
pub fn add2(left: usize, right: usize) -> usize {
|
|
||||||
left + right
|
|
||||||
}
|
|
||||||
|
|
||||||
//- /projects/p2/Cargo.toml
|
|
||||||
[package]
|
|
||||||
name = "p2"
|
|
||||||
version = "0.0.0"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
p1 = { path = "../p1" }
|
|
||||||
|
|
||||||
//- /projects/p2/src/lib.rs
|
|
||||||
use p1::add2;
|
|
||||||
|
|
||||||
pub fn bar() {}
|
|
||||||
"#;
|
|
||||||
|
|
||||||
let server = Project::with_fixture(fixture)
|
|
||||||
.with_config(serde_json::json!({
|
|
||||||
"linkedProjects" : [
|
|
||||||
"./projects/p1/Cargo.toml",
|
|
||||||
"./projects/p2/Cargo.toml"
|
|
||||||
],
|
|
||||||
}
|
|
||||||
))
|
|
||||||
.with_config(serde_json::json!({
|
|
||||||
"cargo": { "sysroot": null },
|
|
||||||
}))
|
|
||||||
.server()
|
|
||||||
.wait_until_workspace_is_loaded();
|
|
||||||
|
|
||||||
let doc_id = server.doc_id("./projects/p2/src/lib.rs");
|
|
||||||
let doc2_id = server.doc_id("./projects/p1/src/lib.rs");
|
|
||||||
|
|
||||||
server.request::<Rename>(
|
|
||||||
RenameParams {
|
|
||||||
text_document_position: TextDocumentPositionParams {
|
|
||||||
text_document: doc_id.clone(),
|
|
||||||
position: Position { line: 0, character: 8 },
|
|
||||||
},
|
|
||||||
new_name: "ABC".to_owned(),
|
|
||||||
work_done_progress_params: WorkDoneProgressParams { work_done_token: None },
|
|
||||||
},
|
|
||||||
json!({
|
|
||||||
"documentChanges": [
|
|
||||||
{
|
|
||||||
"textDocument": {
|
|
||||||
"uri": doc2_id.uri,
|
|
||||||
"version": null
|
|
||||||
},
|
|
||||||
"edits": [
|
|
||||||
{
|
|
||||||
"range": {
|
|
||||||
"start": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 7
|
|
||||||
},
|
|
||||||
"end": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 11
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"newText": "ABC"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"textDocument": {
|
|
||||||
"uri": doc_id.uri,
|
|
||||||
"version": null
|
|
||||||
},
|
|
||||||
"edits": [
|
|
||||||
{
|
|
||||||
"range": {
|
|
||||||
"start": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 8
|
|
||||||
},
|
|
||||||
"end": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 12
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"newText": "ABC"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_deduplicate_crate_differing_in_origin_in_rev_resolution_order() {
|
|
||||||
let fixture = r#"
|
|
||||||
//- /projects/p1/Cargo.toml
|
|
||||||
[package]
|
|
||||||
name = "p1"
|
|
||||||
version = "0.0.0"
|
|
||||||
|
|
||||||
//- /projects/p1/src/lib.rs
|
|
||||||
pub fn add2(left: usize, right: usize) -> usize {
|
|
||||||
left + right
|
|
||||||
}
|
|
||||||
|
|
||||||
//- /projects/p2/Cargo.toml
|
|
||||||
[package]
|
|
||||||
name = "p2"
|
|
||||||
version = "0.0.0"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
p1 = { path = "../p1" }
|
|
||||||
|
|
||||||
//- /projects/p2/src/lib.rs
|
|
||||||
use p1::add2;
|
|
||||||
|
|
||||||
pub fn bar() {}
|
|
||||||
"#;
|
|
||||||
|
|
||||||
let server = Project::with_fixture(fixture)
|
|
||||||
.with_config(serde_json::json!({
|
|
||||||
"linkedProjects" : [
|
|
||||||
"./projects/p2/Cargo.toml",
|
|
||||||
"./projects/p1/Cargo.toml",
|
|
||||||
],
|
|
||||||
}
|
|
||||||
))
|
|
||||||
.with_config(serde_json::json!({
|
|
||||||
"cargo": { "sysroot": null },
|
|
||||||
}))
|
|
||||||
.server()
|
|
||||||
.wait_until_workspace_is_loaded();
|
|
||||||
|
|
||||||
let doc_id = server.doc_id("./projects/p2/src/lib.rs");
|
|
||||||
let doc2_id = server.doc_id("./projects/p1/src/lib.rs");
|
|
||||||
|
|
||||||
server.request::<Rename>(
|
|
||||||
RenameParams {
|
|
||||||
text_document_position: TextDocumentPositionParams {
|
|
||||||
text_document: doc_id.clone(),
|
|
||||||
position: Position { line: 0, character: 8 },
|
|
||||||
},
|
|
||||||
new_name: "ABC".to_owned(),
|
|
||||||
work_done_progress_params: WorkDoneProgressParams { work_done_token: None },
|
|
||||||
},
|
|
||||||
json!({
|
|
||||||
"documentChanges": [
|
|
||||||
{
|
|
||||||
"textDocument": {
|
|
||||||
"uri": doc2_id.uri,
|
|
||||||
"version": null
|
|
||||||
},
|
|
||||||
"edits": [
|
|
||||||
{
|
|
||||||
"range": {
|
|
||||||
"start": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 7
|
|
||||||
},
|
|
||||||
"end": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 11
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"newText": "ABC"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"textDocument": {
|
|
||||||
"uri": doc_id.uri,
|
|
||||||
"version": null
|
|
||||||
},
|
|
||||||
"edits": [
|
|
||||||
{
|
|
||||||
"range": {
|
|
||||||
"start": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 8
|
|
||||||
},
|
|
||||||
"end": {
|
|
||||||
"line": 0,
|
|
||||||
"character": 12
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"newText": "ABC"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in a new issue