2022-05-21 13:11:05 +01:00
//! Project loading & configuration updates.
//!
//! This is quite tricky. The main problem is time and changes -- there's no
//! fixed "project" rust-analyzer is working with, "current project" is itself
//! mutable state. For example, when the user edits `Cargo.toml` by adding a new
//! dependency, project model changes. What's more, switching project model is
//! not instantaneous -- it takes time to run `cargo metadata` and (for proc
//! macros) `cargo check`.
//!
2022-05-23 10:29:03 +02:00
//! The main guiding principle here is, as elsewhere in rust-analyzer,
2022-05-21 13:11:05 +01:00
//! robustness. We try not to assume that the project model exists or is
//! correct. Instead, we try to provide a best-effort service. Even if the
//! project is currently loading and we don't have a full project model, we
//! still want to respond to various requests.
2023-07-03 17:40:31 +02:00
use std ::{ iter , mem } ;
2020-06-25 23:44:58 +02:00
2020-09-15 18:51:57 -07:00
use flycheck ::{ FlycheckConfig , FlycheckHandle } ;
2021-06-03 16:11:20 +02:00
use hir ::db ::DefDatabase ;
2020-10-02 15:45:09 +02:00
use ide ::Change ;
2023-02-28 12:08:23 +01:00
use ide_db ::{
2023-07-03 17:40:31 +02:00
base_db ::{ salsa ::Durability , CrateGraph , ProcMacroPaths , ProcMacros } ,
2023-02-28 12:08:23 +01:00
FxHashMap ,
2021-08-22 14:05:12 +03:00
} ;
2023-07-03 17:40:31 +02:00
use load_cargo ::{ load_proc_macro , ProjectFolders } ;
use proc_macro_api ::ProcMacroServer ;
use project_model ::{ ProjectWorkspace , WorkspaceBuildScripts } ;
2023-06-13 12:25:04 +02:00
use rustc_hash ::FxHashSet ;
2023-05-27 03:18:17 +10:00
use stdx ::{ format_to , thread ::ThreadIntent } ;
2023-05-02 17:12:22 +03:00
use triomphe ::Arc ;
2023-07-03 17:40:31 +02:00
use vfs ::{ AbsPath , ChangeKind } ;
2020-06-25 23:44:58 +02:00
use crate ::{
config ::{ Config , FilesWatcher , LinkedProject } ,
2021-04-06 14:16:35 +03:00
global_state ::GlobalState ,
2020-07-10 15:27:34 +02:00
lsp_ext ,
2020-07-02 16:47:42 +02:00
main_loop ::Task ,
2022-04-16 13:16:58 +01:00
op_queue ::Cause ,
2020-06-25 23:44:58 +02:00
} ;
2021-01-08 01:08:46 +08:00
#[ derive(Debug) ]
pub ( crate ) enum ProjectWorkspaceProgress {
Begin ,
Report ( String ) ,
2023-06-13 12:25:04 +02:00
End ( Vec < anyhow ::Result < ProjectWorkspace > > , bool ) ,
2021-01-08 01:08:46 +08:00
}
2021-01-28 23:33:02 +08:00
#[ derive(Debug) ]
pub ( crate ) enum BuildDataProgress {
Begin ,
Report ( String ) ,
2021-07-18 13:13:03 +03:00
End ( ( Arc < Vec < ProjectWorkspace > > , Vec < anyhow ::Result < WorkspaceBuildScripts > > ) ) ,
2021-01-28 23:33:02 +08:00
}
2023-03-25 18:06:06 +01:00
#[ derive(Debug) ]
pub ( crate ) enum ProcMacroProgress {
Begin ,
Report ( String ) ,
End ( ProcMacros ) ,
}
2020-06-25 23:44:58 +02:00
impl GlobalState {
2021-04-06 14:16:35 +03:00
pub ( crate ) fn is_quiescent ( & self ) -> bool {
2023-03-10 10:26:02 +01:00
! ( self . last_reported_status . is_none ( )
| | self . fetch_workspaces_queue . op_in_progress ( )
2021-04-06 14:16:35 +03:00
| | self . fetch_build_data_queue . op_in_progress ( )
2023-03-26 08:39:28 +02:00
| | self . fetch_proc_macros_queue . op_in_progress ( )
2021-04-06 14:16:35 +03:00
| | self . vfs_progress_config_version < self . vfs_config_version
| | self . vfs_progress_n_done < self . vfs_progress_n_total )
}
2020-06-26 17:28:04 +02:00
pub ( crate ) fn update_configuration ( & mut self , config : Config ) {
2020-08-12 16:32:36 +02:00
let _p = profile ::span ( " GlobalState::update_configuration " ) ;
2021-01-06 15:46:31 +03:00
let old_config = mem ::replace ( & mut self . config , Arc ::new ( config ) ) ;
2023-03-25 23:03:22 +01:00
if self . config . lru_parse_query_capacity ( ) ! = old_config . lru_parse_query_capacity ( ) {
self . analysis_host . update_lru_capacity ( self . config . lru_parse_query_capacity ( ) ) ;
}
if self . config . lru_query_capacities ( ) ! = old_config . lru_query_capacities ( ) {
self . analysis_host . update_lru_capacities (
& self . config . lru_query_capacities ( ) . cloned ( ) . unwrap_or_default ( ) ,
) ;
2020-06-26 17:28:04 +02:00
}
2021-01-06 13:54:28 +03:00
if self . config . linked_projects ( ) ! = old_config . linked_projects ( ) {
2023-06-13 12:25:04 +02:00
self . fetch_workspaces_queue . request_op ( " linked projects changed " . to_string ( ) , false )
2021-01-06 13:54:28 +03:00
} else if self . config . flycheck ( ) ! = old_config . flycheck ( ) {
2020-06-25 23:44:58 +02:00
self . reload_flycheck ( ) ;
}
2021-06-03 16:11:20 +02:00
2023-03-29 19:13:40 +02:00
if self . analysis_host . raw_database ( ) . expand_proc_attr_macros ( )
2022-05-06 09:46:02 +03:00
! = self . config . expand_proc_attr_macros ( )
{
2023-03-29 19:13:40 +02:00
self . analysis_host . raw_database_mut ( ) . set_expand_proc_attr_macros_with_durability (
self . config . expand_proc_attr_macros ( ) ,
Durability ::HIGH ,
) ;
2022-05-06 09:46:02 +03:00
}
2020-06-25 23:44:58 +02:00
}
2021-08-30 19:35:49 +03:00
pub ( crate ) fn current_status ( & self ) -> lsp_ext ::ServerStatusParams {
2021-04-06 14:16:35 +03:00
let mut status = lsp_ext ::ServerStatusParams {
health : lsp_ext ::Health ::Ok ,
quiescent : self . is_quiescent ( ) ,
message : None ,
} ;
2023-03-15 11:28:37 +01:00
let mut message = String ::new ( ) ;
2021-04-06 18:08:05 +03:00
2021-10-20 15:29:50 +02:00
if self . proc_macro_changed {
status . health = lsp_ext ::Health ::Warning ;
2023-06-07 20:55:19 +02:00
message . push_str ( " Proc-macros have changed and need to be rebuilt. \n \n " ) ;
2021-10-20 15:29:50 +02:00
}
2022-04-11 14:38:30 +02:00
if let Err ( _ ) = self . fetch_build_data_error ( ) {
2021-04-06 18:08:05 +03:00
status . health = lsp_ext ::Health ::Warning ;
2023-03-15 11:28:37 +01:00
message . push_str ( " Failed to run build scripts of some packages. \n \n " ) ;
2021-04-06 18:08:05 +03:00
}
2023-04-04 19:24:50 +02:00
if self . proc_macro_clients . iter ( ) . any ( | it | it . is_err ( ) ) {
status . health = lsp_ext ::Health ::Warning ;
message . push_str ( " Failed to spawn one or more proc-macro servers. \n \n " ) ;
}
2021-04-06 14:16:35 +03:00
if ! self . config . cargo_autoreload ( )
& & self . is_quiescent ( )
& & self . fetch_workspaces_queue . op_requested ( )
{
status . health = lsp_ext ::Health ::Warning ;
2023-03-15 11:35:34 +01:00
message . push_str ( " Auto-reloading is disabled and the workspace has changed, a manual workspace reload is required. \n \n " ) ;
2021-04-06 14:16:35 +03:00
}
2022-11-11 13:00:22 +01:00
if self . config . linked_projects ( ) . is_empty ( )
& & self . config . detached_files ( ) . is_empty ( )
& & self . config . notifications ( ) . cargo_toml_not_found
{
status . health = lsp_ext ::Health ::Warning ;
2023-03-16 16:26:19 +01:00
message . push_str ( " Failed to discover workspace. \n " ) ;
message . push_str ( " Consider adding the `Cargo.toml` of the workspace to the [`linkedProjects`](https://rust-analyzer.github.io/manual.html#rust-analyzer.linkedProjects) setting. \n \n " ) ;
2022-11-11 13:00:22 +01:00
}
2023-05-26 15:21:00 +02:00
if let Some ( err ) = & self . config_errors {
status . health = lsp_ext ::Health ::Warning ;
format_to! ( message , " {err} \n " ) ;
}
2023-05-26 15:37:41 +02:00
if let Some ( err ) = & self . last_flycheck_error {
status . health = lsp_ext ::Health ::Warning ;
message . push_str ( err ) ;
message . push ( '\n' ) ;
}
2023-03-10 09:49:37 +01:00
2023-03-15 11:35:34 +01:00
for ws in self . workspaces . iter ( ) {
let ( ProjectWorkspace ::Cargo { sysroot , .. }
| ProjectWorkspace ::Json { sysroot , .. }
| ProjectWorkspace ::DetachedFiles { sysroot , .. } ) = ws ;
2023-04-13 08:40:14 +02:00
match sysroot {
Err ( None ) = > ( ) ,
Err ( Some ( e ) ) = > {
status . health = lsp_ext ::Health ::Warning ;
message . push_str ( e ) ;
message . push_str ( " \n \n " ) ;
}
Ok ( s ) = > {
if let Some ( e ) = s . loading_warning ( ) {
status . health = lsp_ext ::Health ::Warning ;
message . push_str ( & e ) ;
message . push_str ( " \n \n " ) ;
}
}
2023-03-15 11:35:34 +01:00
}
if let ProjectWorkspace ::Cargo { rustc : Err ( Some ( e ) ) , .. } = ws {
status . health = lsp_ext ::Health ::Warning ;
message . push_str ( e ) ;
message . push_str ( " \n \n " ) ;
}
}
2023-03-15 11:28:37 +01:00
if let Err ( _ ) = self . fetch_workspace_error ( ) {
status . health = lsp_ext ::Health ::Error ;
2023-03-15 11:35:34 +01:00
message . push_str ( " Failed to load workspaces. \n \n " ) ;
2023-03-15 11:28:37 +01:00
}
if ! message . is_empty ( ) {
status . message = Some ( message . trim_end ( ) . to_owned ( ) ) ;
}
2021-08-30 19:35:49 +03:00
status
2020-07-10 15:27:34 +02:00
}
2021-01-10 18:02:02 +03:00
2023-06-13 12:25:04 +02:00
pub ( crate ) fn fetch_workspaces ( & mut self , cause : Cause , force_crate_graph_reload : bool ) {
2022-04-16 13:16:58 +01:00
tracing ::info! ( % cause , " will fetch workspaces " ) ;
2021-01-08 01:08:46 +08:00
2023-06-04 09:30:21 +02:00
self . task_pool . handle . spawn_with_sender ( ThreadIntent ::Worker , {
2021-01-06 13:54:28 +03:00
let linked_projects = self . config . linked_projects ( ) ;
2021-05-23 20:56:54 +03:00
let detached_files = self . config . detached_files ( ) . to_vec ( ) ;
2021-01-06 13:54:28 +03:00
let cargo_config = self . config . cargo ( ) ;
2021-01-08 01:08:46 +08:00
move | sender | {
let progress = {
let sender = sender . clone ( ) ;
move | msg | {
sender
. send ( Task ::FetchWorkspace ( ProjectWorkspaceProgress ::Report ( msg ) ) )
. unwrap ( )
}
} ;
sender . send ( Task ::FetchWorkspace ( ProjectWorkspaceProgress ::Begin ) ) . unwrap ( ) ;
2021-05-23 20:56:54 +03:00
let mut workspaces = linked_projects
2020-07-02 16:47:42 +02:00
. iter ( )
. map ( | project | match project {
LinkedProject ::ProjectManifest ( manifest ) = > {
2021-01-08 01:08:46 +08:00
project_model ::ProjectWorkspace ::load (
manifest . clone ( ) ,
& cargo_config ,
& progress ,
)
2020-07-02 16:47:42 +02:00
}
LinkedProject ::InlineJsonProject ( it ) = > {
2023-01-27 13:49:28 +01:00
Ok ( project_model ::ProjectWorkspace ::load_inline (
2021-01-18 14:52:12 +03:00
it . clone ( ) ,
cargo_config . target . as_deref ( ) ,
2022-09-19 17:31:08 +02:00
& cargo_config . extra_env ,
2023-03-31 09:10:18 +02:00
None ,
2023-01-27 13:49:28 +01:00
) )
2020-07-02 16:47:42 +02:00
}
2020-06-26 16:33:57 +02:00
} )
2020-07-02 16:47:42 +02:00
. collect ::< Vec < _ > > ( ) ;
2021-01-08 01:08:46 +08:00
2023-04-18 14:27:01 +02:00
let mut i = 0 ;
while i < workspaces . len ( ) {
if let Ok ( w ) = & workspaces [ i ] {
2023-04-25 10:47:33 +02:00
let dupes : Vec < _ > = workspaces
2023-04-18 14:27:01 +02:00
. iter ( )
2023-04-25 10:47:33 +02:00
. enumerate ( )
. skip ( i + 1 )
. filter_map ( | ( i , it ) | {
it . as_ref ( ) . ok ( ) . filter ( | ws | ws . eq_ignore_build_data ( w ) ) . map ( | _ | i )
} )
. collect ( ) ;
dupes . into_iter ( ) . rev ( ) . for_each ( | d | {
_ = workspaces . remove ( d ) ;
} ) ;
2023-04-18 14:27:01 +02:00
}
i + = 1 ;
}
2021-05-23 20:56:54 +03:00
if ! detached_files . is_empty ( ) {
2022-11-24 10:21:19 +01:00
workspaces . push ( project_model ::ProjectWorkspace ::load_detached_files (
detached_files ,
& cargo_config ,
) ) ;
2021-05-23 20:56:54 +03:00
}
2021-08-15 20:46:13 +08:00
tracing ::info! ( " did fetch workspaces {:?} " , workspaces ) ;
2021-01-10 18:02:02 +03:00
sender
2023-06-13 12:25:04 +02:00
. send ( Task ::FetchWorkspace ( ProjectWorkspaceProgress ::End (
workspaces ,
force_crate_graph_reload ,
) ) )
2021-01-10 18:02:02 +03:00
. unwrap ( ) ;
2020-07-02 16:47:42 +02:00
}
} ) ;
}
2021-07-18 11:29:22 +03:00
2022-04-16 13:16:58 +01:00
pub ( crate ) fn fetch_build_data ( & mut self , cause : Cause ) {
2022-06-15 14:29:13 +02:00
tracing ::info! ( % cause , " will fetch build data " ) ;
2021-07-18 11:29:22 +03:00
let workspaces = Arc ::clone ( & self . workspaces ) ;
let config = self . config . cargo ( ) ;
2023-06-04 09:30:21 +02:00
self . task_pool . handle . spawn_with_sender ( ThreadIntent ::Worker , move | sender | {
sender . send ( Task ::FetchBuildData ( BuildDataProgress ::Begin ) ) . unwrap ( ) ;
2021-04-06 12:35:40 +03:00
2023-06-04 09:30:21 +02:00
let progress = {
let sender = sender . clone ( ) ;
move | msg | {
sender . send ( Task ::FetchBuildData ( BuildDataProgress ::Report ( msg ) ) ) . unwrap ( )
}
} ;
let res = ProjectWorkspace ::run_all_build_scripts ( & workspaces , & config , & progress ) ;
2022-08-27 18:28:09 +02:00
2023-06-04 09:30:21 +02:00
sender . send ( Task ::FetchBuildData ( BuildDataProgress ::End ( ( workspaces , res ) ) ) ) . unwrap ( ) ;
} ) ;
2021-04-06 12:35:40 +03:00
}
2023-03-26 08:39:28 +02:00
pub ( crate ) fn fetch_proc_macros ( & mut self , cause : Cause , paths : Vec < ProcMacroPaths > ) {
tracing ::info! ( % cause , " will load proc macros " ) ;
2023-03-25 18:06:06 +01:00
let dummy_replacements = self . config . dummy_replacements ( ) . clone ( ) ;
let proc_macro_clients = self . proc_macro_clients . clone ( ) ;
2023-06-04 09:30:21 +02:00
self . task_pool . handle . spawn_with_sender ( ThreadIntent ::Worker , move | sender | {
sender . send ( Task ::LoadProcMacros ( ProcMacroProgress ::Begin ) ) . unwrap ( ) ;
2023-03-25 18:06:06 +01:00
2023-06-04 09:30:21 +02:00
let dummy_replacements = & dummy_replacements ;
let progress = {
let sender = sender . clone ( ) ;
& move | msg | {
sender . send ( Task ::LoadProcMacros ( ProcMacroProgress ::Report ( msg ) ) ) . unwrap ( )
2023-03-25 18:06:06 +01:00
}
2023-06-04 09:30:21 +02:00
} ;
let mut res = FxHashMap ::default ( ) ;
let chain = proc_macro_clients
. iter ( )
. map ( | res | res . as_ref ( ) . map_err ( | e | e . to_string ( ) ) )
. chain ( iter ::repeat_with ( | | Err ( " Proc macros servers are not running " . into ( ) ) ) ) ;
for ( client , paths ) in chain . zip ( paths ) {
res . extend ( paths . into_iter ( ) . map ( move | ( crate_id , res ) | {
(
crate_id ,
res . map_or_else (
| _ | Err ( " proc macro crate is missing dylib " . to_owned ( ) ) ,
| ( crate_name , path ) | {
2023-06-19 13:01:47 +01:00
progress ( path . to_string ( ) ) ;
2023-06-04 09:30:21 +02:00
client . as_ref ( ) . map_err ( Clone ::clone ) . and_then ( | client | {
load_proc_macro (
client ,
& path ,
crate_name
. as_deref ( )
. and_then ( | crate_name | {
dummy_replacements . get ( crate_name ) . map ( | v | & * * v )
} )
. unwrap_or_default ( ) ,
)
} )
} ,
) ,
)
} ) ) ;
}
2023-03-25 18:06:06 +01:00
2023-06-04 09:30:21 +02:00
sender . send ( Task ::LoadProcMacros ( ProcMacroProgress ::End ( res ) ) ) . unwrap ( ) ;
} ) ;
2023-03-25 18:06:06 +01:00
}
pub ( crate ) fn set_proc_macros ( & mut self , proc_macros : ProcMacros ) {
let mut change = Change ::new ( ) ;
change . set_proc_macros ( proc_macros ) ;
self . analysis_host . apply_change ( change ) ;
}
2022-04-18 14:26:00 +08:00
pub ( crate ) fn switch_workspaces ( & mut self , cause : Cause ) {
2020-08-12 16:32:36 +02:00
let _p = profile ::span ( " GlobalState::switch_workspaces " ) ;
2022-04-18 14:26:00 +08:00
tracing ::info! ( % cause , " will switch workspaces " ) ;
2021-04-06 14:16:35 +03:00
2023-06-13 12:25:04 +02:00
let Some ( ( workspaces , force_reload_crate_graph ) ) = self . fetch_workspaces_queue . last_op_result ( ) else { return ; } ;
2023-03-10 09:49:37 +01:00
if let Err ( _ ) = self . fetch_workspace_error ( ) {
2021-04-06 14:16:35 +03:00
if ! self . workspaces . is_empty ( ) {
2023-06-13 12:25:04 +02:00
if * force_reload_crate_graph {
self . recreate_crate_graph ( cause ) ;
}
2021-07-18 13:13:03 +03:00
// It only makes sense to switch to a partially broken workspace
// if we don't have any workspace at all yet.
2021-04-06 14:16:35 +03:00
return ;
}
}
2020-07-10 15:27:34 +02:00
2022-11-11 14:36:27 +01:00
let workspaces =
workspaces . iter ( ) . filter_map ( | res | res . as_ref ( ) . ok ( ) . cloned ( ) ) . collect ::< Vec < _ > > ( ) ;
2020-06-25 23:44:58 +02:00
2023-03-29 15:29:32 -04:00
let same_workspaces = workspaces . len ( ) = = self . workspaces . len ( )
& & workspaces
. iter ( )
. zip ( self . workspaces . iter ( ) )
. all ( | ( l , r ) | l . eq_ignore_build_data ( r ) ) ;
2021-01-28 23:33:02 +08:00
2021-07-18 13:13:03 +03:00
if same_workspaces {
let ( workspaces , build_scripts ) = self . fetch_build_data_queue . last_op_result ( ) ;
2021-10-16 12:32:55 +01:00
if Arc ::ptr_eq ( workspaces , & self . workspaces ) {
2022-04-18 14:26:00 +08:00
tracing ::debug! ( " set build scripts to workspaces " ) ;
2021-07-18 13:13:03 +03:00
let workspaces = workspaces
. iter ( )
. cloned ( )
. zip ( build_scripts )
. map ( | ( mut ws , bs ) | {
ws . set_build_scripts ( bs . as_ref ( ) . ok ( ) . cloned ( ) . unwrap_or_default ( ) ) ;
ws
} )
. collect ::< Vec < _ > > ( ) ;
// Workspaces are the same, but we've updated build data.
self . workspaces = Arc ::new ( workspaces ) ;
} else {
2022-04-29 05:59:32 +00:00
tracing ::info! ( " build scripts do not match the version of the active workspace " ) ;
2023-06-13 12:25:04 +02:00
if * force_reload_crate_graph {
self . recreate_crate_graph ( cause ) ;
}
2021-07-18 13:13:03 +03:00
// Current build scripts do not match the version of the active
// workspace, so there's nothing for us to update.
return ;
}
} else {
2022-04-18 14:26:00 +08:00
tracing ::debug! ( " abandon build scripts for workspaces " ) ;
2021-07-18 13:13:03 +03:00
// Here, we completely changed the workspace (Cargo.toml edit), so
// we don't care about build-script results, they are stale.
2023-01-27 13:49:28 +01:00
// FIXME: can we abort the build scripts here?
self . workspaces = Arc ::new ( workspaces ) ;
2020-07-10 15:27:34 +02:00
}
2021-01-06 13:54:28 +03:00
if let FilesWatcher ::Client = self . config . files ( ) . watcher {
2022-01-01 17:26:54 +03:00
let registration_options = lsp_types ::DidChangeWatchedFilesRegistrationOptions {
watchers : self
. workspaces
. iter ( )
. flat_map ( | ws | ws . to_roots ( ) )
. filter ( | it | it . is_local )
. flat_map ( | root | {
root . include . into_iter ( ) . flat_map ( | it | {
[
2023-06-19 13:01:47 +01:00
format! ( " {it} /**/*.rs " ) ,
format! ( " {it} /**/Cargo.toml " ) ,
format! ( " {it} /**/Cargo.lock " ) ,
2022-01-01 17:26:54 +03:00
]
2021-01-10 22:38:35 +03:00
} )
2022-01-01 17:26:54 +03:00
} )
2023-02-14 14:59:50 +01:00
. map ( | glob_pattern | lsp_types ::FileSystemWatcher {
glob_pattern : lsp_types ::GlobPattern ::String ( glob_pattern ) ,
kind : None ,
} )
2022-01-01 17:26:54 +03:00
. collect ( ) ,
} ;
let registration = lsp_types ::Registration {
id : " workspace/didChangeWatchedFiles " . to_string ( ) ,
method : " workspace/didChangeWatchedFiles " . to_string ( ) ,
register_options : Some ( serde_json ::to_value ( registration_options ) . unwrap ( ) ) ,
} ;
self . send_request ::< lsp_types ::request ::RegisterCapability > (
lsp_types ::RegistrationParams { registrations : vec ! [ registration ] } ,
| _ , _ | ( ) ,
) ;
2020-06-25 23:44:58 +02:00
}
2021-01-26 16:18:01 +03:00
let files_config = self . config . files ( ) ;
2021-07-18 13:13:03 +03:00
let project_folders = ProjectFolders ::new ( & self . workspaces , & files_config . exclude ) ;
2020-06-25 23:44:58 +02:00
2023-03-29 15:29:32 -04:00
if self . proc_macro_clients . is_empty ( ) | | ! same_workspaces {
2023-04-26 08:06:15 +02:00
if self . config . expand_proc_macros ( ) {
2022-08-05 12:06:31 +02:00
tracing ::info! ( " Spawning proc-macro servers " ) ;
2023-04-26 08:06:15 +02:00
2023-05-02 17:12:22 +03:00
// FIXME: use `Arc::from_iter` when it becomes available
self . proc_macro_clients = Arc ::from (
self . workspaces
. iter ( )
. map ( | ws | {
let path = match self . config . proc_macro_srv ( ) {
Some ( path ) = > path ,
None = > ws . find_sysroot_proc_macro_srv ( ) ? ,
} ;
2023-06-19 13:01:47 +01:00
tracing ::info! ( " Using proc-macro server at {path} " ) ;
2023-05-02 17:12:22 +03:00
ProcMacroServer ::spawn ( path . clone ( ) ) . map_err ( | err | {
tracing ::error! (
2023-06-19 13:01:47 +01:00
" Failed to run proc-macro server from path {path}, error: {err:?} " ,
2023-05-02 17:12:22 +03:00
) ;
2023-06-19 13:01:47 +01:00
anyhow ::format_err! (
" Failed to run proc-macro server from path {path}, error: {err:?} " ,
2023-05-02 17:12:22 +03:00
)
} )
2022-07-23 20:24:01 +02:00
} )
2023-05-02 17:12:22 +03:00
. collect ::< Vec < _ > > ( ) ,
)
2022-08-28 12:31:31 +02:00
} ;
2021-02-02 04:55:17 +08:00
}
2020-07-10 23:39:25 +02:00
2021-01-26 16:18:01 +03:00
let watch = match files_config . watcher {
2020-06-25 23:44:58 +02:00
FilesWatcher ::Client = > vec! [ ] ,
2022-07-18 17:50:56 +02:00
FilesWatcher ::Server = > project_folders . watch ,
2020-06-25 23:44:58 +02:00
} ;
2021-02-12 15:58:29 +01:00
self . vfs_config_version + = 1 ;
self . loader . handle . set_config ( vfs ::loader ::Config {
load : project_folders . load ,
watch ,
version : self . vfs_config_version ,
} ) ;
2023-03-25 18:06:06 +01:00
self . source_root_config = project_folders . source_root_config ;
2020-06-25 23:44:58 +02:00
2023-06-13 12:25:04 +02:00
self . recreate_crate_graph ( cause ) ;
tracing ::info! ( " did switch workspaces " ) ;
}
fn recreate_crate_graph ( & mut self , cause : String ) {
2020-06-25 23:44:58 +02:00
// Create crate graph from all the workspaces
2023-06-13 12:25:04 +02:00
let ( crate_graph , proc_macro_paths , crate_graph_file_dependencies ) = {
2020-06-25 23:44:58 +02:00
let vfs = & mut self . vfs . write ( ) . 0 ;
let loader = & mut self . loader ;
2023-06-13 12:25:04 +02:00
// crate graph construction relies on these paths, record them so when one of them gets
// deleted or created we trigger a reconstruction of the crate graph
let mut crate_graph_file_dependencies = FxHashSet ::default ( ) ;
2023-03-31 09:10:18 +02:00
let mut load = | path : & AbsPath | {
2023-01-25 14:46:06 +01:00
let _p = profile ::span ( " switch_workspaces::load " ) ;
2020-09-18 22:15:44 +03:00
let vfs_path = vfs ::VfsPath ::from ( path . to_path_buf ( ) ) ;
2023-06-13 12:25:04 +02:00
crate_graph_file_dependencies . insert ( vfs_path . clone ( ) ) ;
2023-03-31 09:10:18 +02:00
match vfs . file_id ( & vfs_path ) {
Some ( file_id ) = > Some ( file_id ) ,
None = > {
if ! self . mem_docs . contains ( & vfs_path ) {
let contents = loader . handle . load_sync ( path ) ;
vfs . set_file_contents ( vfs_path . clone ( ) , contents ) ;
}
vfs . file_id ( & vfs_path )
}
2020-11-17 12:17:24 +01:00
}
2020-06-25 23:44:58 +02:00
} ;
2021-08-22 13:32:00 +03:00
let mut crate_graph = CrateGraph ::default ( ) ;
2023-03-25 18:06:06 +01:00
let mut proc_macros = Vec ::default ( ) ;
for ws in & * * self . workspaces {
let ( other , mut crate_proc_macros ) =
2023-04-06 10:28:34 +02:00
ws . to_crate_graph ( & mut load , & self . config . extra_env ( ) ) ;
2023-03-25 18:06:06 +01:00
crate_graph . extend ( other , & mut crate_proc_macros ) ;
proc_macros . push ( crate_proc_macros ) ;
2020-06-25 23:44:58 +02:00
}
2023-06-13 12:25:04 +02:00
( crate_graph , proc_macros , crate_graph_file_dependencies )
2020-06-25 23:44:58 +02:00
} ;
2023-03-26 08:39:28 +02:00
2023-03-29 15:29:32 -04:00
if self . config . expand_proc_macros ( ) {
self . fetch_proc_macros_queue . request_op ( cause , proc_macro_paths ) ;
2023-03-26 08:39:28 +02:00
}
2023-06-13 12:25:04 +02:00
let mut change = Change ::new ( ) ;
2020-06-25 23:44:58 +02:00
change . set_crate_graph ( crate_graph ) ;
self . analysis_host . apply_change ( change ) ;
2023-06-13 12:25:04 +02:00
self . crate_graph_file_dependencies = crate_graph_file_dependencies ;
2023-03-25 18:20:42 +01:00
self . process_changes ( ) ;
2023-03-25 18:06:06 +01:00
2020-06-25 23:44:58 +02:00
self . reload_flycheck ( ) ;
}
2023-03-10 09:49:37 +01:00
pub ( super ) fn fetch_workspace_error ( & self ) -> Result < ( ) , String > {
2021-04-06 18:08:05 +03:00
let mut buf = String ::new ( ) ;
2021-04-06 14:16:35 +03:00
2023-06-19 12:32:04 +01:00
let Some ( ( last_op_result , _ ) ) = self . fetch_workspaces_queue . last_op_result ( ) else {
return Ok ( ( ) )
} ;
2022-11-11 13:00:22 +01:00
if last_op_result . is_empty ( ) {
stdx ::format_to! ( buf , " rust-analyzer failed to discover workspace " ) ;
} else {
for ws in last_op_result {
if let Err ( err ) = ws {
stdx ::format_to! ( buf , " rust-analyzer failed to load workspace: {:#} \n " , err ) ;
}
2021-04-06 14:16:35 +03:00
}
}
2021-04-06 18:08:05 +03:00
if buf . is_empty ( ) {
2022-04-11 14:38:30 +02:00
return Ok ( ( ) ) ;
2021-04-06 14:16:35 +03:00
}
2022-04-11 14:38:30 +02:00
Err ( buf )
2021-04-06 18:08:05 +03:00
}
2023-03-10 09:49:37 +01:00
pub ( super ) fn fetch_build_data_error ( & self ) -> Result < ( ) , String > {
2022-04-11 14:38:30 +02:00
let mut buf = String ::new ( ) ;
2021-07-18 11:29:22 +03:00
2021-07-18 13:13:03 +03:00
for ws in & self . fetch_build_data_queue . last_op_result ( ) . 1 {
2021-08-25 17:56:39 +02:00
match ws {
2022-04-11 14:38:30 +02:00
Ok ( data ) = > match data . error ( ) {
Some ( stderr ) = > stdx ::format_to! ( buf , " {:#} \n " , stderr ) ,
_ = > ( ) ,
} ,
// io errors
Err ( err ) = > stdx ::format_to! ( buf , " {:#} \n " , err ) ,
2021-04-06 18:08:05 +03:00
}
}
2021-07-18 11:29:22 +03:00
2022-04-11 14:38:30 +02:00
if buf . is_empty ( ) {
Ok ( ( ) )
2021-08-25 17:56:39 +02:00
} else {
2022-04-11 14:38:30 +02:00
Err ( buf )
2021-07-18 11:29:22 +03:00
}
2021-04-06 14:16:35 +03:00
}
2020-06-25 23:44:58 +02:00
fn reload_flycheck ( & mut self ) {
2021-01-18 13:25:57 +03:00
let _p = profile ::span ( " GlobalState::reload_flycheck " ) ;
2022-12-17 23:26:54 +01:00
let config = self . config . flycheck ( ) ;
2020-07-15 14:37:44 +02:00
let sender = self . flycheck_sender . clone ( ) ;
2022-10-22 23:02:59 +02:00
let invocation_strategy = match config {
FlycheckConfig ::CargoCommand { .. } = > flycheck ::InvocationStrategy ::PerWorkspace ,
FlycheckConfig ::CustomCommand { invocation_strategy , .. } = > invocation_strategy ,
} ;
2022-09-15 13:28:09 +02:00
self . flycheck = match invocation_strategy {
2022-10-19 23:34:36 +02:00
flycheck ::InvocationStrategy ::Once = > vec! [ FlycheckHandle ::spawn (
2022-09-15 13:28:09 +02:00
0 ,
Box ::new ( move | msg | sender . send ( msg ) . unwrap ( ) ) ,
2022-12-23 02:08:08 -05:00
config ,
2022-09-15 13:28:09 +02:00
self . config . root_path ( ) . clone ( ) ,
) ] ,
2022-09-26 15:58:55 +02:00
flycheck ::InvocationStrategy ::PerWorkspace = > {
2022-09-15 13:28:09 +02:00
self . workspaces
. iter ( )
. enumerate ( )
. filter_map ( | ( id , w ) | match w {
ProjectWorkspace ::Cargo { cargo , .. } = > Some ( ( id , cargo . workspace_root ( ) ) ) ,
ProjectWorkspace ::Json { project , .. } = > {
// Enable flychecks for json projects if a custom flycheck command was supplied
// in the workspace configuration.
match config {
FlycheckConfig ::CustomCommand { .. } = > Some ( ( id , project . path ( ) ) ) ,
_ = > None ,
}
}
ProjectWorkspace ::DetachedFiles { .. } = > None ,
} )
. map ( | ( id , root ) | {
let sender = sender . clone ( ) ;
FlycheckHandle ::spawn (
id ,
Box ::new ( move | msg | sender . send ( msg ) . unwrap ( ) ) ,
config . clone ( ) ,
root . to_path_buf ( ) ,
)
} )
. collect ( )
}
2022-10-20 19:28:28 +02:00
}
. into ( ) ;
2020-06-25 23:44:58 +02:00
}
}
2021-09-13 20:58:09 +03:00
pub ( crate ) fn should_refresh_for_change ( path : & AbsPath , change_kind : ChangeKind ) -> bool {
const IMPLICIT_TARGET_FILES : & [ & str ] = & [ " build.rs " , " src/main.rs " , " src/lib.rs " ] ;
const IMPLICIT_TARGET_DIRS : & [ & str ] = & [ " src/bin " , " examples " , " tests " , " benches " ] ;
2022-08-05 12:06:31 +02:00
let file_name = match path . file_name ( ) . unwrap_or_default ( ) . to_str ( ) {
Some ( it ) = > it ,
None = > return false ,
} ;
if let " Cargo.toml " | " Cargo.lock " = file_name {
2021-09-13 20:58:09 +03:00
return true ;
}
if change_kind = = ChangeKind ::Modify {
return false ;
}
2022-08-05 12:06:31 +02:00
// .cargo/config{.toml}
2021-09-13 20:58:09 +03:00
if path . extension ( ) . unwrap_or_default ( ) ! = " rs " {
2022-08-05 12:06:31 +02:00
let is_cargo_config = matches! ( file_name , " config.toml " | " config " )
& & path . parent ( ) . map ( | parent | parent . as_ref ( ) . ends_with ( " .cargo " ) ) . unwrap_or ( false ) ;
return is_cargo_config ;
2021-09-13 20:58:09 +03:00
}
2022-08-05 12:06:31 +02:00
2021-09-13 20:58:09 +03:00
if IMPLICIT_TARGET_FILES . iter ( ) . any ( | it | path . as_ref ( ) . ends_with ( it ) ) {
return true ;
}
let parent = match path . parent ( ) {
Some ( it ) = > it ,
None = > return false ,
} ;
if IMPLICIT_TARGET_DIRS . iter ( ) . any ( | it | parent . as_ref ( ) . ends_with ( it ) ) {
return true ;
}
if file_name = = " main.rs " {
let grand_parent = match parent . parent ( ) {
Some ( it ) = > it ,
None = > return false ,
} ;
if IMPLICIT_TARGET_DIRS . iter ( ) . any ( | it | grand_parent . as_ref ( ) . ends_with ( it ) ) {
return true ;
}
}
false
}