-
-
Notifications
You must be signed in to change notification settings - Fork 326
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
Fix possible result update issue for plugins with IResultUpdate interface #3115
Conversation
This comment has been minimized.
This comment has been minimized.
Be a legend 🏆 by adding a before and after screenshot of the changes you made, especially if they are around UI/UX. |
📝 WalkthroughWalkthroughThe changes in this pull request primarily focus on the Changes
Possibly related PRs
Suggested labels
Suggested reviewers
Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
CodeRabbit Configuration File (
|
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.
Actionable comments posted: 1
🧹 Outside diff range and nitpick comments (2)
Flow.Launcher/ViewModel/MainViewModel.cs (2)
1475-1476
: Fix typo in error message comment- // Plugin with IResultUpdate interface can somtimes throw this exception - // Collection was modified; enumeration operation may not execute + // Plugin with IResultUpdate interface can sometimes throw this exception + // Collection was modified; enumeration operation may not execute
1450-1478
: Consider implementing proper synchronization mechanismsThe current implementation could benefit from proper synchronization mechanisms to handle concurrent access to shared collections.
Consider these architectural improvements:
- Use a synchronization mechanism (like
lock
orSemaphoreSlim
) when accessing shared collections- Implement the producer-consumer pattern using
BlockingCollection<T>
for thread-safe queuing of results- Consider using immutable collections (
ImmutableList<T>
) to prevent modification during enumeration- Add documentation about thread-safety requirements for plugin developers implementing
IResultUpdate
Would you like me to provide example implementations for any of these approaches?
I do not know why I have caught exception in
Seems that Solution 2 fails to work. So I reverted fix in |
This comment has been minimized.
This comment has been minimized.
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.
Actionable comments posted: 2
🧹 Outside diff range and nitpick comments (2)
Plugins/Flow.Launcher.Plugin.WebSearch/Main.cs (2)
Line range hint
88-95
: Add cancellation check before raising eventThe event is raised without checking the cancellation token, which could cause unnecessary work when cancellation is requested.
+ token.ThrowIfCancellationRequested(); ResultsUpdated?.Invoke(this, new ResultUpdatedEventArgs { Results = results.ToList(), Query = query });
Line range hint
15-24
: Consider improving thread-safety and separation of concernsThe class manages shared state (
_settings
,_viewModel
) without apparent thread-safety mechanisms while implementing multiple interfaces. This could lead to race conditions under concurrent access.Consider these architectural improvements:
- Implement the Repository pattern for thread-safe settings access
- Use a dedicated event aggregator for result updates
- Split the class into smaller, focused components:
- SearchService (core search logic)
- ResultAggregator (result collection and event handling)
- SettingsManager (thread-safe settings access)
Would you like me to provide a detailed implementation proposal for any of these improvements?
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
📒 Files selected for processing (1)
Plugins/Flow.Launcher.Plugin.WebSearch/Main.cs
(1 hunks)
🔇 Additional comments (1)
Plugins/Flow.Launcher.Plugin.WebSearch/Main.cs (1)
Line range hint 27-91
: Verify alignment with PR objectives
The PR description mentions implementing solution #2 (error handling with try-catch), but the changes show implementation of solution #1 (defensive copy). Additionally, the comments indicate ongoing exceptions despite error handling.
Let's verify the presence of error handling in the codebase:
Please clarify which solution is being implemented and ensure the changes align with the chosen approach.
However, what I am confused is that IResultsUpdate should only append a job to the channel Flow.Launcher/Flow.Launcher/ViewModel/MainViewModel.cs Lines 229 to 241 in 304f98e
and then the consumer should be a single thread? Flow.Launcher/Flow.Launcher/ViewModel/MainViewModel.cs Lines 188 to 215 in 304f98e
I want to understand the root courses of this synchronization problem. |
on the other hand looks like you didn't commit the correct change? |
Sorry, I don not know your meaning. |
Here I describe my finding about the issue of |
Sorry that I am not quite familiar with such synchronization problem and I just happen to find it. |
Oh now I understand what's going wrong. The ResultsUpdateEvent didn't return a new list, which causes issue when the plugin is modifying the same underlying list. |
A better solution is to create a copy here |
Now the only change of this PR is all about |
Let me have a try. |
@check-spelling-bot Report🔴 Please reviewSee the 📂 files view, the 📜action log, or 📝 job summary for details.
See ❌ Event descriptions for more information. If the flagged items are 🤯 false positivesIf items relate to a ...
|
🥷 Code experts: no user matched threshold 10 Yusyuriv has most 👩💻 activity in the files. See details
Activity based on git-commit:
Knowledge based on git-blame: To learn more about /:\ gitStream - Visit our Docs |
@taooceros Now handle the copy operation in |
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.
Actionable comments posted: 0
🧹 Outside diff range and nitpick comments (1)
Flow.Launcher/ViewModel/MainViewModel.cs (1)
236-240
: Consider enhancing thread safety with ConcurrentBag.While creating a defensive copy works, consider implementing Solution 3 from the PR objectives by using
ConcurrentBag<Result>
forResultUpdatedEventArgs
. This would provide stronger thread-safety guarantees and potentially better performance by avoiding list copying.Example implementation:
- var resultsCopy = e.Results.ToList(); + var resultsCopy = new ConcurrentBag<Result>(e.Results);
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
📒 Files selected for processing (1)
Flow.Launcher/ViewModel/MainViewModel.cs
(1 hunks)
🔇 Additional comments (1)
Flow.Launcher/ViewModel/MainViewModel.cs (1)
236-240
: Good implementation of defensive copying to prevent concurrent modification.
The implementation correctly addresses the concurrency issue by creating a defensive copy of the results list before updating the view model. This prevents the race condition where the plugin might modify the same list that's being processed by the main view model.
However, let's verify that this change is consistently applied across all result update paths:
#!/bin/bash
# Description: Check for other potential places where results list might be modified
# Look for patterns where Results collection is modified without defensive copying
# Search for direct result list modifications
rg -A 5 'Results\.(Add|Clear|Remove)'
# Search for result list assignments
ast-grep --pattern 'Results = $_'
@taooceros PTAL? |
@Jack251970 have you tested whether it works? I can merge this now if that's the case! |
@taooceros I have tested, and it can address this issue. |
Sounds good! Thanks for looking into this! |
Issue
This issue could happen when calling
ResultsUpdated.Invoke
event inIResultUpdate
interface so fast that main view model cannot complete update event (still execute foreach sentence).Example
Take
WebSearch
plugin for example, I change itsQueryAsync
function to this:And I can catch the exception in
UpdateResultView
function ofMainViewModel
:Solution
The key of the problem is that this plugin has not tell the developer to do thread-safe operation in
QueryAsync
(like return a copy of the list as theResults
inResultUpdatedEventArgs
). So the developer will possibly return the sameList
to theResultUpdatedEventArgs
and the result of theQueryAsync
function. I think there are two solutions for this.Solution 1
Update the
IResultUpdate
documents and ask developers to return the copy of the list as the parameter ofResultUpdatedEventArgs
.Take the above example, I can change its
QueryAsync
function to this:Solution 2
Wrap the update codes in
UpdateResultView
with try catch, it is what my PR does.Solution 3
For example, you can change the definition of the
ResultUpdatedEventArgs
so that developer won't return the sameList
like this: