-
Notifications
You must be signed in to change notification settings - Fork 4.1k
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
Make fix-all code action more parallel #73356
Changes from 5 commits
479790e
83971bf
84947a2
03fd9b8
8d62cd8
b32db78
afd60a8
8470ca1
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 |
---|---|---|
|
@@ -9,7 +9,6 @@ | |
using System.Threading.Tasks; | ||
using Microsoft.CodeAnalysis.CodeActions; | ||
using Microsoft.CodeAnalysis.CodeFixesAndRefactorings; | ||
using Microsoft.CodeAnalysis.PooledObjects; | ||
using Microsoft.CodeAnalysis.Shared.Extensions; | ||
using Microsoft.CodeAnalysis.Shared.Utilities; | ||
using Microsoft.CodeAnalysis.Text; | ||
|
@@ -28,20 +27,15 @@ namespace Microsoft.CodeAnalysis.CodeFixes; | |
/// project and then appropriately bucketed by document. These are then passed to <see | ||
/// cref="FixAllAsync(FixAllContext, Document, ImmutableArray{Diagnostic})"/> for implementors to process. | ||
/// </remarks> | ||
public abstract class DocumentBasedFixAllProvider : FixAllProvider | ||
public abstract class DocumentBasedFixAllProvider(ImmutableArray<FixAllScope> supportedFixAllScopes) : FixAllProvider | ||
{ | ||
private readonly ImmutableArray<FixAllScope> _supportedFixAllScopes; | ||
private readonly ImmutableArray<FixAllScope> _supportedFixAllScopes = supportedFixAllScopes; | ||
|
||
protected DocumentBasedFixAllProvider() | ||
: this(DefaultSupportedFixAllScopes) | ||
{ | ||
} | ||
|
||
protected DocumentBasedFixAllProvider(ImmutableArray<FixAllScope> supportedFixAllScopes) | ||
{ | ||
_supportedFixAllScopes = supportedFixAllScopes; | ||
} | ||
|
||
/// <summary> | ||
/// Produce a suitable title for the fix-all <see cref="CodeAction"/> this type creates in <see | ||
/// cref="GetFixAsync(FixAllContext)"/>. Override this if customizing that title is desired. | ||
|
@@ -73,55 +67,33 @@ public sealed override IEnumerable<FixAllScope> GetSupportedFixAllScopes() | |
fixAllContext.GetDefaultFixAllTitle(), fixAllContext, FixAllContextsHelperAsync); | ||
|
||
private Task<Solution?> FixAllContextsHelperAsync(FixAllContext originalFixAllContext, ImmutableArray<FixAllContext> fixAllContexts) | ||
=> DocumentBasedFixAllProviderHelpers.FixAllContextsAsync(originalFixAllContext, fixAllContexts, | ||
originalFixAllContext.Progress, | ||
this.GetFixAllTitle(originalFixAllContext), | ||
DetermineDiagnosticsAndGetFixedDocumentsAsync); | ||
|
||
private async Task<Dictionary<DocumentId, (SyntaxNode? node, SourceText? text)>> DetermineDiagnosticsAndGetFixedDocumentsAsync( | ||
FixAllContext fixAllContext, | ||
IProgress<CodeAnalysisProgress> progressTracker) | ||
{ | ||
// First, determine the diagnostics to fix. | ||
var diagnostics = await DetermineDiagnosticsAsync(fixAllContext, progressTracker).ConfigureAwait(false); | ||
|
||
// Second, get the fixes for all the diagnostics, and apply them to determine the new root/text for each doc. | ||
return await GetFixedDocumentsAsync(fixAllContext, progressTracker, diagnostics).ConfigureAwait(false); | ||
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. this method was inlined. |
||
} | ||
|
||
/// <summary> | ||
/// Determines all the diagnostics we should be fixing for the given <paramref name="fixAllContext"/>. | ||
/// </summary> | ||
private static async Task<ImmutableDictionary<Document, ImmutableArray<Diagnostic>>> DetermineDiagnosticsAsync(FixAllContext fixAllContext, IProgress<CodeAnalysisProgress> progressTracker) | ||
{ | ||
using var _ = progressTracker.ItemCompletedScope(); | ||
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. this went away. the caller is responsible for this now. |
||
return await FixAllContextHelper.GetDocumentDiagnosticsToFixAsync(fixAllContext).ConfigureAwait(false); | ||
} | ||
|
||
/// <summary> | ||
/// Attempts to fix all the provided <paramref name="diagnostics"/> returning, for each updated document, either | ||
/// the new syntax root for that document or its new text. Syntax roots are returned for documents that support | ||
/// them, and are used to perform a final cleanup pass for formatting/simplication/etc. Text is returned for | ||
/// documents that don't support syntax. | ||
/// </summary> | ||
private async Task<Dictionary<DocumentId, (SyntaxNode? node, SourceText? text)>> GetFixedDocumentsAsync( | ||
FixAllContext fixAllContext, IProgress<CodeAnalysisProgress> progressTracker, ImmutableDictionary<Document, ImmutableArray<Diagnostic>> diagnostics) | ||
=> DocumentBasedFixAllProviderHelpers.FixAllContextsAsync( | ||
originalFixAllContext, | ||
fixAllContexts, | ||
originalFixAllContext.Progress, | ||
this.GetFixAllTitle(originalFixAllContext), | ||
DetermineDiagnosticsAndGetFixedDocumentsAsync); | ||
|
||
private async Task DetermineDiagnosticsAndGetFixedDocumentsAsync( | ||
FixAllContext fixAllContext, Action<(DocumentId documentId, (SyntaxNode? node, SourceText? text))> callback) | ||
{ | ||
var cancellationToken = fixAllContext.CancellationToken; | ||
|
||
using var _1 = progressTracker.ItemCompletedScope(); | ||
|
||
if (diagnostics.IsEmpty) | ||
return []; | ||
|
||
// Then, process all documents in parallel to get the change for each doc. | ||
return await ProducerConsumer<(DocumentId, (SyntaxNode? node, SourceText? text))>.RunParallelAsync( | ||
source: diagnostics.Where(kvp => !kvp.Value.IsDefaultOrEmpty), | ||
produceItems: static async (kvp, callback, args, cancellationToken) => | ||
// First, determine the diagnostics to fix. | ||
var docuementToDiagnostics = await FixAllContextHelper.GetDocumentDiagnosticsToFixAsync(fixAllContext).ConfigureAwait(false); | ||
|
||
// Second, get the fixes for each document+diagnostics pair in parallel, and apply them to determine the new | ||
// root/text for each doc. | ||
await RoslynParallel.ForEachAsync( | ||
source: docuementToDiagnostics, | ||
cancellationToken, | ||
async (kvp, cancellationToken) => | ||
{ | ||
var (document, documentDiagnostics) = kvp; | ||
if (documentDiagnostics.IsDefaultOrEmpty) | ||
return; | ||
|
||
var newDocument = await args.@this.FixAllAsync(args.fixAllContext, document, documentDiagnostics).ConfigureAwait(false); | ||
var newDocument = await this.FixAllAsync(fixAllContext, document, documentDiagnostics).ConfigureAwait(false); | ||
if (newDocument == null || newDocument == document) | ||
return; | ||
|
||
|
@@ -131,16 +103,6 @@ private static async Task<ImmutableDictionary<Document, ImmutableArray<Diagnosti | |
var text = newDocument.SupportsSyntaxTree ? null : await newDocument.GetValueTextAsync(cancellationToken).ConfigureAwait(false); | ||
|
||
callback((document.Id, (node, text))); | ||
}, | ||
consumeItems: static async (results, args, cancellationToken) => | ||
{ | ||
var docIdToNewRootOrText = new Dictionary<DocumentId, (SyntaxNode? node, SourceText? text)>(); | ||
await foreach (var (docId, nodeOrText) in results) | ||
docIdToNewRootOrText[docId] = nodeOrText; | ||
|
||
return docIdToNewRootOrText; | ||
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. no need for a producerconsumer here anymore. we're not making the intermediary dictionary. we're just calling back into |
||
}, | ||
args: (@this: this, fixAllContext), | ||
cancellationToken).ConfigureAwait(false); | ||
}).ConfigureAwait(false); | ||
} | ||
} |
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.
this method was inlined.