-
Notifications
You must be signed in to change notification settings - Fork 0
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Analyzing progress notifications #44
base: main
Are you sure you want to change the base?
Changes from 3 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
@@ -0,0 +1,154 @@ | ||||||
use std::collections::HashSet; | ||||||
use std::sync::{Arc, Mutex}; | ||||||
|
||||||
use lsp_types::notification::Notification; | ||||||
|
||||||
use crate::id_generator::IdGenerator; | ||||||
use crate::server::client::Notifier; | ||||||
use crate::state::Beacon; | ||||||
|
||||||
/// Controller used to send notifications to the client about analysis progress. | ||||||
/// Uses information provided from other controllers (diagnostics controller, procmacro controller) | ||||||
/// to assess if diagnostics are in fact calculated. | ||||||
#[derive(Debug, Clone)] | ||||||
pub struct AnalysisProgressController { | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we get flow explanation here? like in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can do although i think it's nicely explained here in the docstrings There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. TBD Still - i would like to finalize business logic first if that's OK so i don't have to update this flowchart |
||||||
notifier: Notifier, | ||||||
/// ID of the diagnostics "generation" - the scheduled diagnostics jobs set. | ||||||
/// Used to filter out stale threads finishing when new ones (from newer "generation") | ||||||
/// are already in progress and being tracked by the controller. | ||||||
generation_id: Arc<Mutex<u64>>, | ||||||
/// Sequential IDs of state snapshots from the current generation, used to track their status | ||||||
/// (present meaning it's still being used) | ||||||
active_snapshots: Arc<Mutex<HashSet<usize>>>, | ||||||
id_generator: Arc<IdGenerator>, | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. TBH you'd be better off not touching the IdGenerator here and just peeing that AtomicU64 in generation_id and incrementing it directly. you're just doing this in a fancy way now |
||||||
/// If `true` - a request to procmacro server was submitted, meaning that analysis will extend | ||||||
/// beyond the current generation of diagnostics. | ||||||
did_submit_procmacro_request: Arc<Mutex<bool>>, | ||||||
/// Indicates that a notification was sent and analysis (i.e. macro expansion) is taking place. | ||||||
analysis_in_progress: Arc<Mutex<bool>>, | ||||||
/// Loaded asynchronously from config - unset if config was not loaded yet. | ||||||
/// Has to be set in order for analysis to finish. | ||||||
procmacros_enabled: Arc<Mutex<Option<bool>>>, | ||||||
} | ||||||
|
||||||
impl AnalysisProgressController { | ||||||
pub fn new(notifier: Notifier) -> Self { | ||||||
let id_generator = Arc::new(IdGenerator::default()); | ||||||
Self { | ||||||
notifier, | ||||||
id_generator: id_generator.clone(), | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why clone here, just generate unique_id before the struct constructor expr |
||||||
active_snapshots: Arc::new(Mutex::new(HashSet::default())), | ||||||
did_submit_procmacro_request: Arc::new(Mutex::new(true)), | ||||||
analysis_in_progress: Arc::new(Mutex::new(false)), | ||||||
procmacros_enabled: Arc::new(Mutex::new(None)), | ||||||
generation_id: Arc::new(Mutex::new(id_generator.unique_id())), | ||||||
} | ||||||
} | ||||||
|
||||||
/// Signals that a request to proc macro server was made during the current generation of | ||||||
/// diagnostics. | ||||||
pub fn register_procmacro_request(&self) { | ||||||
let mut write_guard = self.did_submit_procmacro_request.lock().unwrap(); | ||||||
*write_guard = true; | ||||||
} | ||||||
|
||||||
/// Allows to set the procmacro configuration to whatever is in the config, upon loading it. | ||||||
pub fn set_procmacros_enabled(&self, value: bool) { | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
More future proof and it is clear where this info is coming from. |
||||||
let mut guard = self.procmacros_enabled.lock().unwrap(); | ||||||
*guard = Some(value); | ||||||
} | ||||||
|
||||||
/// Sets handlers for tracking beacons sent to threads. | ||||||
/// The beacons are wrapping snapshots, which are signalling when diagnostics finished | ||||||
/// calculating for a given snapshot (used for calculating files diagnostics or removing | ||||||
/// stale ones) | ||||||
pub fn track_analysis<T: Send>(&self, beacons: &mut [Beacon<T>]) { | ||||||
let gen_id = self.next_generation_id(); | ||||||
|
||||||
self.clear_active_snapshots(); | ||||||
beacons.iter_mut().enumerate().for_each(|(i, beacon)| { | ||||||
self.insert_active_snapshot(i); | ||||||
|
||||||
let self_ref: AnalysisProgressController = self.clone(); | ||||||
beacon.on_signal(move || { | ||||||
let current_gen = self_ref.get_generation_id(); | ||||||
if current_gen == gen_id { | ||||||
self_ref.remove_active_snapshot(i); | ||||||
self_ref.try_stop_analysis(); | ||||||
} | ||||||
}); | ||||||
}); | ||||||
|
||||||
self.start_analysis(); | ||||||
} | ||||||
|
||||||
fn insert_active_snapshot(&self, snapshot_id: usize) { | ||||||
let mut active_snapshots = self.active_snapshots.lock().unwrap(); | ||||||
active_snapshots.insert(snapshot_id); | ||||||
} | ||||||
|
||||||
fn next_generation_id(&self) -> u64 { | ||||||
let mut generation_id_guard = self.generation_id.lock().unwrap(); | ||||||
*generation_id_guard = self.id_generator.unique_id(); | ||||||
*generation_id_guard | ||||||
} | ||||||
|
||||||
fn get_generation_id(&self) -> u64 { | ||||||
*self.generation_id.lock().unwrap() | ||||||
} | ||||||
|
||||||
fn remove_active_snapshot(&self, snapshot_id: usize) { | ||||||
let mut active_snapshots = self.active_snapshots.lock().unwrap(); | ||||||
active_snapshots.remove(&snapshot_id); | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. An assertion could be made here to make sure sth was removed |
||||||
} | ||||||
|
||||||
fn clear_active_snapshots(&self) { | ||||||
let active_snapshots_ref = self.active_snapshots.clone(); | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do you need to clone here? |
||||||
active_snapshots_ref.lock().unwrap().clear(); | ||||||
} | ||||||
|
||||||
/// Starts a next generation of diagnostics, sends a notification | ||||||
fn start_analysis(&self) { | ||||||
let mut analysis_in_progress = self.analysis_in_progress.lock().unwrap(); | ||||||
if !(*analysis_in_progress) { | ||||||
*analysis_in_progress = true; | ||||||
self.notifier.notify::<DiagnosticsCalculationStart>(()); | ||||||
} | ||||||
} | ||||||
|
||||||
/// Checks a bunch of conditions and if they are fulfilled, sends stop notification | ||||||
/// and resets the state back to start of generation defaults. | ||||||
fn try_stop_analysis(&self) { | ||||||
let mut did_submit_procmacro_request = self.did_submit_procmacro_request.lock().unwrap(); | ||||||
let snapshots_empty = self.active_snapshots.lock().unwrap().is_empty(); | ||||||
let mut analysis_in_progress = self.analysis_in_progress.lock().unwrap(); | ||||||
let procmacros_enabled = *self.procmacros_enabled.lock().unwrap(); | ||||||
|
||||||
if snapshots_empty | ||||||
&& (!*did_submit_procmacro_request || (procmacros_enabled == Some(false))) | ||||||
&& *analysis_in_progress | ||||||
{ | ||||||
*analysis_in_progress = false; | ||||||
*did_submit_procmacro_request = false; | ||||||
self.notifier.notify::<DiagnosticsCalculationFinish>(()); | ||||||
} | ||||||
} | ||||||
} | ||||||
|
||||||
/// Notifies about diagnostics generation which is beginning to calculate | ||||||
#[derive(Debug)] | ||||||
pub struct DiagnosticsCalculationStart; | ||||||
|
||||||
impl Notification for DiagnosticsCalculationStart { | ||||||
type Params = (); | ||||||
const METHOD: &'static str = "cairo/diagnosticsCalculationStart"; | ||||||
} | ||||||
|
||||||
/// Notifies about diagnostics generation which ended calculating | ||||||
#[derive(Debug)] | ||||||
pub struct DiagnosticsCalculationFinish; | ||||||
|
||||||
impl Notification for DiagnosticsCalculationFinish { | ||||||
type Params = (); | ||||||
const METHOD: &'static str = "cairo/diagnosticsCalculationFinish"; | ||||||
} | ||||||
Comment on lines
+164
to
+180
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Idk if it shouldn't go to There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't know why not place it near the usage tbh There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Both are fine for me tbh, I guess we should get rid of There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. let's place it in |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
pub mod analysis_progress; | ||
pub mod code_actions; | ||
pub mod completion; | ||
pub mod formatter; | ||
|
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
|
@@ -26,7 +26,7 @@ pub struct Client<'s> { | |||||
pub(super) requester: Requester<'s>, | ||||||
} | ||||||
|
||||||
#[derive(Clone)] | ||||||
#[derive(Clone, Debug)] | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why?
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's included in the new Controller which needs to be There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe custom impl on There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not done |
||||||
pub struct Notifier(ClientSender); | ||||||
|
||||||
#[derive(Clone)] | ||||||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A lot of pubs are unnecessary here, some methods could be inlined