Skip to content
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

Draft
wants to merge 1 commit into
base: feature/joined-diagnostics-threads
Choose a base branch
from

Conversation

Arcticae
Copy link
Member

@Arcticae Arcticae commented Dec 18, 2024

Stack :
⬆️ #212
⬇️ #190

Based off this PR we can start implementing tests (like currently we can wait for project update + the diagnostics finish and it should work fine)

@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch 2 times, most recently from ae728b8 to bcd6a77 Compare December 18, 2024 13:33
src/lang/diagnostics/mod.rs Outdated Show resolved Hide resolved
src/server/client.rs Outdated Show resolved Hide resolved
src/state.rs Show resolved Hide resolved
src/state.rs Outdated Show resolved Hide resolved
/// Uses information provided from other controllers (diagnostics controller, procmacro controller)
/// to assess if diagnostics are in fact calculated.
#[derive(Debug, Clone)]
pub struct AnalysisProgressController {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we get flow explanation here? like in ProcMacroController

Copy link
Member Author

Choose a reason for hiding this comment

The 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

Copy link
Member Author

Choose a reason for hiding this comment

The 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

@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from bcd6a77 to 2a2bc04 Compare December 18, 2024 13:49
src/state.rs Outdated Show resolved Hide resolved
src/server/client.rs Outdated Show resolved Hide resolved
src/state.rs Outdated
}
}
/// Struct which allows setting a callback - which can be triggered afterward
/// by the function which has the reference.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reference to what?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reference to the thing that i'm talking about - the struct

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we state that explicitly in the comment? It can be figured out, but it is not immediately obvious on the (English) language level

src/state.rs Outdated
/// Struct which allows setting a callback - which can be triggered afterward
/// by the function which has the reference.
#[derive(Default)]
pub struct Beacon {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this a standard name for this pattern? Isn't Callback with set and execute methods better?

src/state.rs Outdated

pub fn signal(&mut self) {
if let Some(hook) = self.signal_hook.take() {
hook(); // call the hook
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This comment is useless

src/state.rs Outdated
self.signal_hook = Some(Box::new(drop_hook));
}

pub fn signal(&mut self) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

signal_and_clear_hook?

Comment on lines 164 to 180
/// 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";
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Idk if it shouldn't go to lsp/ext.rs for consistency

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know why not place it near the usage tbh

Copy link
Contributor

@piotmag769 piotmag769 Dec 21, 2024

Choose a reason for hiding this comment

The 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 lsp/ext.rs altogether #130

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

let's place it in lsp/ext.rs as discussed on meeting

src/state.rs Outdated Show resolved Hide resolved
src/state.rs Outdated
}

impl Beacon {
// Set the drop hook
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

dot.

Btw it doesn't set the drop hook, it is not done on Drop.

src/ide/analysis_progress.rs Outdated Show resolved Hide resolved
.field("id_generator", &self.id_generator)
.field("requests_params", &self.requests_params)
.field("error_channel", &self.error_channel)
.finish()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
.finish()
.finish_non_exhaustive()

src/state.rs Outdated
}

impl Beacon {
// Set the drop hook
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe note here that it is disposable?

src/state.rs Outdated Show resolved Hide resolved
controller: AnalysisProgressController,
}

impl AnalysisProgressTracker {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can split it for proc macros / diagnostics so interface is clear there, also it does not have to store whole controller (and should not if it possible, if not then maybe controller itself is ok).

}

/// Allows to set the procmacro configuration to whatever is in the config, upon loading it.
pub fn set_procmacros_enabled(&self, value: bool) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
pub fn set_procmacros_enabled(&self, value: bool) {
pub fn on_config_change(&self, config: &Config) {

More future proof and it is clear where this info is coming from.

let id_generator = Arc::new(IdGenerator::default());
Self {
notifier,
id_generator: id_generator.clone(),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why clone here, just generate unique_id before the struct constructor expr

Comment on lines 116 to 118
pub fn get_generation_id(&self) -> u64 {
self.generation_id.load(Ordering::SeqCst)
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Used in one place, just inline it imo

}

pub fn clear_active_snapshots(&self) {
let active_snapshots_ref = self.active_snapshots.clone();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you need to clone here?

@@ -142,6 +150,7 @@ impl ProcMacroClient {
match self.send_request_untracked::<M>(id, &params) {
Ok(()) => {
requests_params.insert(id, map(params));
self.analysis_progress_tracker.register_procmacro_request();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe that there is a small chance that a yet alive snapshot from previous diagnostics calculation was already in db.get_attribute_expansion before its cancellation. It will still call this method and possibly register proc macro request in analysis tracker AFTER track_analysis is called to start tracking the new diagnostics batch. It can cause us to never finish waiting for diagnostics in tests, since:

  • the next batch won't be scheduled
  • we will always think there is some proc macro yet to resolve while it may not be

Copy link
Contributor

@piotmag769 piotmag769 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Logic wise seems good besides what I wrote before


pub fn remove_active_snapshot(&self, snapshot_id: usize) {
let mut active_snapshots = self.active_snapshots.lock().unwrap();
active_snapshots.remove(&snapshot_id);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

An assertion could be made here to make sure sth was removed

Copy link
Contributor

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

src/state.rs Outdated
Comment on lines 77 to 80
}
/// Struct which allows setting a callback - which can be triggered afterward
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
}
/// Struct which allows setting a callback - which can be triggered afterward
}
/// Struct which allows setting a callback - which can be triggered afterward

src/state.rs Outdated
/// Struct which allows setting a callback - which can be triggered afterward
/// by the function which has the reference.
#[derive(Default)]
pub struct Beacon {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TBH you have simplified this so much that I would probably drop this type entirely and just stick that Option<Box<dyn FnOnce() + Send>>.

/// 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<'a>(&self, beacons: impl Iterator<Item = &'a mut Beacon>) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can't this just take a mutable reference to StateSnapshots?

/// 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>,
Copy link
Member

Choose a reason for hiding this comment

The 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

Comment on lines 48 to 38
#[derive(Debug, Clone)]
pub struct AnalysisProgressController {
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<AtomicU64>,
/// 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>,
/// 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<AtomicBool>,
/// Indicates that a notification was sent and analysis (i.e. macro expansion) is taking place.
analysis_in_progress: Arc<AtomicBool>,
/// 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>>>,
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

consider just using a mutex for everything. I don't believe you benefit from atomics here over mutexes, and I'm not sure all methods here guarantee consistency in case of races.

struct AnalysisProgressController(Arc<Mutex<AnalysisProgressControllerState>>);

struct AnalysisProgressControllerState {
    generation_id: u64,
    // ...
}

Comment on lines 164 to 180
/// 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";
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

let's place it in lsp/ext.rs as discussed on meeting

@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from 935e0ad to 40d3863 Compare January 21, 2025 15:06
@Arcticae Arcticae changed the base branch from main to feature/joined-diagnostics-threads January 21, 2025 15:06
@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch 2 times, most recently from b3e7fb1 to 85e9edd Compare January 23, 2025 14:14
);

client.start_initialize();

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

new line

@@ -97,10 +97,10 @@ impl Config {
response.pop_front().as_ref().and_then(Value::as_bool).unwrap_or_default();
state.config.enable_proc_macros =
response.pop_front().as_ref().and_then(Value::as_bool).unwrap_or(false);

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

new line

@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from 85e9edd to 3f39b2b Compare January 23, 2025 15:12
@Arcticae Arcticae changed the base branch from feature/joined-diagnostics-threads to chore/move-scarb-notifications-to-lsp-ext January 23, 2025 15:13
@Arcticae Arcticae force-pushed the chore/move-scarb-notifications-to-lsp-ext branch from ab6da3c to c9d9e62 Compare January 23, 2025 15:16
@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from 3f39b2b to 888b845 Compare January 23, 2025 15:18
@Arcticae Arcticae changed the base branch from chore/move-scarb-notifications-to-lsp-ext to feature/joined-diagnostics-threads January 23, 2025 15:32
@Arcticae Arcticae force-pushed the feature/joined-diagnostics-threads branch from 62ad275 to a9bd5c8 Compare January 23, 2025 15:34
@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from 888b845 to 251f488 Compare January 23, 2025 15:39
@Arcticae Arcticae force-pushed the feature/joined-diagnostics-threads branch from a9bd5c8 to b1a0a44 Compare January 23, 2025 15:43
@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from 251f488 to b242543 Compare January 23, 2025 15:44
@Arcticae Arcticae force-pushed the feature/analyzing-progress-notifications branch from b242543 to 029269d Compare January 23, 2025 15:50
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants