4909: Anchor file-system operations to the file, and not to the source root.  r=matklad a=matklad



bors r+
🤖

Co-authored-by: Aleksey Kladov <aleksey.kladov@gmail.com>
This commit is contained in:
bors[bot] 2020-06-16 16:58:41 +00:00 committed by GitHub
commit b64d0a3659
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 67 additions and 81 deletions

View file

@ -3,7 +3,6 @@
use std::any::Any; use std::any::Any;
use hir_expand::diagnostics::Diagnostic; use hir_expand::diagnostics::Diagnostic;
use ra_db::RelativePathBuf;
use ra_syntax::{ast, AstPtr, SyntaxNodePtr}; use ra_syntax::{ast, AstPtr, SyntaxNodePtr};
use hir_expand::{HirFileId, InFile}; use hir_expand::{HirFileId, InFile};
@ -12,7 +11,7 @@ use hir_expand::{HirFileId, InFile};
pub struct UnresolvedModule { pub struct UnresolvedModule {
pub file: HirFileId, pub file: HirFileId,
pub decl: AstPtr<ast::Module>, pub decl: AstPtr<ast::Module>,
pub candidate: RelativePathBuf, pub candidate: String,
} }
impl Diagnostic for UnresolvedModule { impl Diagnostic for UnresolvedModule {

View file

@ -296,7 +296,6 @@ pub enum ModuleSource {
mod diagnostics { mod diagnostics {
use hir_expand::diagnostics::DiagnosticSink; use hir_expand::diagnostics::DiagnosticSink;
use ra_db::RelativePathBuf;
use ra_syntax::{ast, AstPtr}; use ra_syntax::{ast, AstPtr};
use crate::{db::DefDatabase, diagnostics::UnresolvedModule, nameres::LocalModuleId, AstId}; use crate::{db::DefDatabase, diagnostics::UnresolvedModule, nameres::LocalModuleId, AstId};
@ -306,7 +305,7 @@ mod diagnostics {
UnresolvedModule { UnresolvedModule {
module: LocalModuleId, module: LocalModuleId,
declaration: AstId<ast::Module>, declaration: AstId<ast::Module>,
candidate: RelativePathBuf, candidate: String,
}, },
} }

View file

@ -44,7 +44,7 @@ impl ModDir {
file_id: HirFileId, file_id: HirFileId,
name: &Name, name: &Name,
attr_path: Option<&SmolStr>, attr_path: Option<&SmolStr>,
) -> Result<(FileId, ModDir), RelativePathBuf> { ) -> Result<(FileId, ModDir), String> {
let file_id = file_id.original_file(db.upcast()); let file_id = file_id.original_file(db.upcast());
let mut candidate_files = Vec::new(); let mut candidate_files = Vec::new();
@ -52,11 +52,11 @@ impl ModDir {
Some(attr_path) => { Some(attr_path) => {
let base = let base =
if self.root_non_dir_owner { self.path.parent().unwrap() } else { &self.path }; if self.root_non_dir_owner { self.path.parent().unwrap() } else { &self.path };
candidate_files.push(base.join(attr_path)) candidate_files.push(base.join(attr_path).to_string())
} }
None => { None => {
candidate_files.push(self.path.join(&format!("{}.rs", name))); candidate_files.push(self.path.join(&format!("{}.rs", name)).to_string());
candidate_files.push(self.path.join(&format!("{}/mod.rs", name))); candidate_files.push(self.path.join(&format!("{}/mod.rs", name)).to_string());
} }
}; };

View file

@ -11,7 +11,7 @@ use hir::{
Semantics, Semantics,
}; };
use itertools::Itertools; use itertools::Itertools;
use ra_db::{RelativePath, SourceDatabase, SourceDatabaseExt}; use ra_db::SourceDatabase;
use ra_ide_db::RootDatabase; use ra_ide_db::RootDatabase;
use ra_prof::profile; use ra_prof::profile;
use ra_syntax::{ use ra_syntax::{
@ -57,14 +57,10 @@ pub(crate) fn diagnostics(db: &RootDatabase, file_id: FileId) -> Vec<Diagnostic>
}) })
.on::<hir::diagnostics::UnresolvedModule, _>(|d| { .on::<hir::diagnostics::UnresolvedModule, _>(|d| {
let original_file = d.source().file_id.original_file(db); let original_file = d.source().file_id.original_file(db);
let source_root = db.file_source_root(original_file); let fix = Fix::new(
let path = db "Create module",
.file_relative_path(original_file) FileSystemEdit::CreateFile { anchor: original_file, dst: d.candidate.clone() }.into(),
.parent() );
.unwrap_or_else(|| RelativePath::new(""))
.join(&d.candidate);
let fix =
Fix::new("Create module", FileSystemEdit::CreateFile { source_root, path }.into());
res.borrow_mut().push(Diagnostic { res.borrow_mut().push(Diagnostic {
range: sema.diagnostics_range(d).range, range: sema.diagnostics_range(d).range,
message: d.message(), message: d.message(),
@ -612,10 +608,10 @@ mod tests {
source_file_edits: [], source_file_edits: [],
file_system_edits: [ file_system_edits: [
CreateFile { CreateFile {
source_root: SourceRootId( anchor: FileId(
0, 1,
), ),
path: "foo.rs", dst: "foo.rs",
}, },
], ],
is_snippet: false, is_snippet: false,

View file

@ -1,7 +1,7 @@
//! FIXME: write short doc here //! FIXME: write short doc here
use hir::{ModuleSource, Semantics}; use hir::{ModuleSource, Semantics};
use ra_db::{RelativePath, RelativePathBuf, SourceDatabaseExt}; use ra_db::{RelativePathBuf, SourceDatabaseExt};
use ra_ide_db::RootDatabase; use ra_ide_db::RootDatabase;
use ra_syntax::{ use ra_syntax::{
algo::find_node_at_offset, ast, ast::TypeAscriptionOwner, lex_single_valid_syntax_kind, algo::find_node_at_offset, ast, ast::TypeAscriptionOwner, lex_single_valid_syntax_kind,
@ -92,23 +92,14 @@ fn rename_mod(
ModuleSource::SourceFile(..) => { ModuleSource::SourceFile(..) => {
let mod_path: RelativePathBuf = sema.db.file_relative_path(file_id); let mod_path: RelativePathBuf = sema.db.file_relative_path(file_id);
// mod is defined in path/to/dir/mod.rs // mod is defined in path/to/dir/mod.rs
let dst_path = if mod_path.file_stem() == Some("mod") { let dst = if mod_path.file_stem() == Some("mod") {
mod_path format!("../{}/mod.rs", new_name)
.parent()
.and_then(|p| p.parent())
.or_else(|| Some(RelativePath::new("")))
.map(|p| p.join(new_name).join("mod.rs"))
} else { } else {
Some(mod_path.with_file_name(new_name).with_extension("rs")) format!("{}.rs", new_name)
}; };
if let Some(path) = dst_path { let move_file =
let move_file = FileSystemEdit::MoveFile { FileSystemEdit::MoveFile { src: file_id, anchor: position.file_id, dst };
src: file_id, file_system_edits.push(move_file);
dst_source_root: sema.db.file_source_root(position.file_id),
dst_path: path,
};
file_system_edits.push(move_file);
}
} }
ModuleSource::Module(..) => {} ModuleSource::Module(..) => {}
} }
@ -623,16 +614,16 @@ mod tests {
#[test] #[test]
fn test_rename_mod() { fn test_rename_mod() {
let (analysis, position) = analysis_and_position( let (analysis, position) = analysis_and_position(
" r#"
//- /lib.rs //- /lib.rs
mod bar; mod bar;
//- /bar.rs //- /bar.rs
mod foo<|>; mod foo<|>;
//- /bar/foo.rs //- /bar/foo.rs
// emtpy // emtpy
", "#,
); );
let new_name = "foo2"; let new_name = "foo2";
let source_change = analysis.rename(position, new_name).unwrap(); let source_change = analysis.rename(position, new_name).unwrap();
@ -662,10 +653,10 @@ mod tests {
src: FileId( src: FileId(
3, 3,
), ),
dst_source_root: SourceRootId( anchor: FileId(
0, 2,
), ),
dst_path: "bar/foo2.rs", dst: "foo2.rs",
}, },
], ],
is_snippet: false, is_snippet: false,
@ -678,12 +669,12 @@ mod tests {
#[test] #[test]
fn test_rename_mod_in_dir() { fn test_rename_mod_in_dir() {
let (analysis, position) = analysis_and_position( let (analysis, position) = analysis_and_position(
" r#"
//- /lib.rs //- /lib.rs
mod fo<|>o; mod fo<|>o;
//- /foo/mod.rs //- /foo/mod.rs
// emtpy // emtpy
", "#,
); );
let new_name = "foo2"; let new_name = "foo2";
let source_change = analysis.rename(position, new_name).unwrap(); let source_change = analysis.rename(position, new_name).unwrap();
@ -713,10 +704,10 @@ mod tests {
src: FileId( src: FileId(
2, 2,
), ),
dst_source_root: SourceRootId( anchor: FileId(
0, 1,
), ),
dst_path: "foo2/mod.rs", dst: "../foo2/mod.rs",
}, },
], ],
is_snippet: false, is_snippet: false,
@ -753,19 +744,19 @@ mod tests {
#[test] #[test]
fn test_rename_mod_filename_and_path() { fn test_rename_mod_filename_and_path() {
let (analysis, position) = analysis_and_position( let (analysis, position) = analysis_and_position(
" r#"
//- /lib.rs //- /lib.rs
mod bar; mod bar;
fn f() { fn f() {
bar::foo::fun() bar::foo::fun()
} }
//- /bar.rs //- /bar.rs
pub mod foo<|>; pub mod foo<|>;
//- /bar/foo.rs //- /bar/foo.rs
// pub fn fun() {} // pub fn fun() {}
", "#,
); );
let new_name = "foo2"; let new_name = "foo2";
let source_change = analysis.rename(position, new_name).unwrap(); let source_change = analysis.rename(position, new_name).unwrap();
@ -808,10 +799,10 @@ mod tests {
src: FileId( src: FileId(
3, 3,
), ),
dst_source_root: SourceRootId( anchor: FileId(
0, 2,
), ),
dst_path: "bar/foo2.rs", dst: "foo2.rs",
}, },
], ],
is_snippet: false, is_snippet: false,

View file

@ -3,7 +3,7 @@
//! //!
//! It can be viewed as a dual for `AnalysisChange`. //! It can be viewed as a dual for `AnalysisChange`.
use ra_db::{FileId, RelativePathBuf, SourceRootId}; use ra_db::FileId;
use ra_text_edit::TextEdit; use ra_text_edit::TextEdit;
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -44,8 +44,8 @@ impl From<Vec<SourceFileEdit>> for SourceChange {
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub enum FileSystemEdit { pub enum FileSystemEdit {
CreateFile { source_root: SourceRootId, path: RelativePathBuf }, CreateFile { anchor: FileId, dst: String },
MoveFile { src: FileId, dst_source_root: SourceRootId, dst_path: RelativePathBuf }, MoveFile { src: FileId, anchor: FileId, dst: String },
} }
impl From<FileSystemEdit> for SourceChange { impl From<FileSystemEdit> for SourceChange {

View file

@ -16,7 +16,7 @@ use ra_ide::{
Analysis, AnalysisChange, AnalysisHost, CrateGraph, FileId, LibraryData, SourceRootId, Analysis, AnalysisChange, AnalysisHost, CrateGraph, FileId, LibraryData, SourceRootId,
}; };
use ra_project_model::{ProcMacroClient, ProjectWorkspace}; use ra_project_model::{ProcMacroClient, ProjectWorkspace};
use ra_vfs::{LineEndings, RootEntry, Vfs, VfsChange, VfsFile, VfsRoot, VfsTask, Watch}; use ra_vfs::{LineEndings, RootEntry, Vfs, VfsChange, VfsFile, VfsTask, Watch};
use relative_path::RelativePathBuf; use relative_path::RelativePathBuf;
use stdx::format_to; use stdx::format_to;
@ -298,9 +298,10 @@ impl GlobalStateSnapshot {
self.vfs.read().file_line_endings(VfsFile(id.0)) self.vfs.read().file_line_endings(VfsFile(id.0))
} }
pub fn path_to_url(&self, root: SourceRootId, path: &RelativePathBuf) -> Url { pub fn anchored_path(&self, file_id: FileId, path: &str) -> Url {
let base = self.vfs.read().root2path(VfsRoot(root.0)); let mut base = self.vfs.read().file2path(VfsFile(file_id.0));
let path = path.to_path(base); base.pop();
let path = base.join(path);
url_from_abs_path(&path) url_from_abs_path(&path)
} }

View file

@ -528,13 +528,13 @@ pub(crate) fn resource_op(
file_system_edit: FileSystemEdit, file_system_edit: FileSystemEdit,
) -> lsp_types::ResourceOp { ) -> lsp_types::ResourceOp {
match file_system_edit { match file_system_edit {
FileSystemEdit::CreateFile { source_root, path } => { FileSystemEdit::CreateFile { anchor, dst } => {
let uri = snap.path_to_url(source_root, &path); let uri = snap.anchored_path(anchor, &dst);
lsp_types::ResourceOp::Create(lsp_types::CreateFile { uri, options: None }) lsp_types::ResourceOp::Create(lsp_types::CreateFile { uri, options: None })
} }
FileSystemEdit::MoveFile { src, dst_source_root, dst_path } => { FileSystemEdit::MoveFile { src, anchor, dst } => {
let old_uri = snap.file_id_to_url(src); let old_uri = snap.file_id_to_url(src);
let new_uri = snap.path_to_url(dst_source_root, &dst_path); let new_uri = snap.anchored_path(anchor, &dst);
lsp_types::ResourceOp::Rename(lsp_types::RenameFile { old_uri, new_uri, options: None }) lsp_types::ResourceOp::Rename(lsp_types::RenameFile { old_uri, new_uri, options: None })
} }
} }