File: CodeRefactorings\AddMissingImports\AbstractAddMissingImportsFeatureService.cs
Web Access
Project: src\src\Features\Core\Portable\Microsoft.CodeAnalysis.Features.csproj (Microsoft.CodeAnalysis.Features)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Diagnostics;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.AddImport;
using Microsoft.CodeAnalysis.CodeActions;
using Microsoft.CodeAnalysis.Formatting;
using Microsoft.CodeAnalysis.Formatting.Rules;
using Microsoft.CodeAnalysis.OrganizeImports;
using Microsoft.CodeAnalysis.Packaging;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.SymbolSearch;
using Microsoft.CodeAnalysis.Text;
using Microsoft.CodeAnalysis.Utilities;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.AddMissingImports;
 
internal abstract class AbstractAddMissingImportsFeatureService : IAddMissingImportsFeatureService
{
    protected abstract ImmutableArray<string> FixableDiagnosticIds { get; }
 
    protected abstract ImmutableArray<AbstractFormattingRule> GetFormatRules(SourceText text);
 
    /// <inheritdoc/>
    public async Task<Document> AddMissingImportsAsync(Document document, TextSpan textSpan, IProgress<CodeAnalysisProgress> progressTracker, CancellationToken cancellationToken)
    {
        var analysisResult = await AnalyzeAsync(document, textSpan, cancellationToken).ConfigureAwait(false);
        return await AddMissingImportsAsync(
            document, analysisResult, progressTracker, cancellationToken).ConfigureAwait(false);
    }
 
    /// <inheritdoc/>
    public async Task<ImmutableArray<AddImportFixData>> AnalyzeAsync(Document document, TextSpan textSpan, CancellationToken cancellationToken)
    {
        // Get the diagnostics that indicate a missing import.
        var addImportFeatureService = document.GetRequiredLanguageService<IAddImportFeatureService>();
 
        var solution = document.Project.Solution;
        var symbolSearchService = solution.Services.GetRequiredService<ISymbolSearchService>();
 
        // Since we are not currently considering NuGet packages, pass an empty array
        var packageSources = ImmutableArray<PackageSource>.Empty;
 
        var addImportOptions = await document.GetAddImportOptionsAsync(
            searchOptions: new() { SearchReferenceAssemblies = true, SearchNuGetPackages = false },
            cancellationToken).ConfigureAwait(false);
 
        var unambiguousFixes = await addImportFeatureService.GetUniqueFixesAsync(
            document, textSpan, FixableDiagnosticIds, symbolSearchService,
            addImportOptions, packageSources, cancellationToken).ConfigureAwait(false);
 
        // We do not want to add project or framework references without the user's input, so filter those out.
        var usableFixes = unambiguousFixes.WhereAsArray(fixData => DoesNotAddReference(fixData, document.Project.Id));
 
        return usableFixes;
    }
 
    private static bool DoesNotAddReference(AddImportFixData fixData, ProjectId currentProjectId)
    {
        return (fixData.ProjectReferenceToAdd is null || fixData.ProjectReferenceToAdd == currentProjectId)
            && (fixData.PortableExecutableReferenceProjectId is null || fixData.PortableExecutableReferenceProjectId == currentProjectId)
            && string.IsNullOrEmpty(fixData.AssemblyReferenceAssemblyName);
    }
 
    public async Task<Document> AddMissingImportsAsync(
        Document document,
        ImmutableArray<AddImportFixData> fixes,
        IProgress<CodeAnalysisProgress> progressTracker,
        CancellationToken cancellationToken)
    {
        if (fixes.IsEmpty)
            return document;
 
        var solution = document.Project.Solution;
        var textDiffingService = solution.Services.GetRequiredService<IDocumentTextDifferencingService>();
        var packageInstallerService = solution.Services.GetService<IPackageInstallerService>();
 
        var addImportService = document.GetRequiredLanguageService<IAddImportFeatureService>();
        var organizeImportsService = document.GetRequiredLanguageService<IOrganizeImportsService>();
 
        var formattingOptions = await document.GetSyntaxFormattingOptionsAsync(cancellationToken).ConfigureAwait(false);
        var organizeImportsOptions = await document.GetOrganizeImportsOptionsAsync(cancellationToken).ConfigureAwait(false);
 
        // Do not limit the results since we plan to fix all the reported issues.
        var codeActions = addImportService.GetCodeActionsForFixes(document, fixes, packageInstallerService, maxResults: int.MaxValue);
        var getChangesTasks = codeActions.Select(
            action => GetChangesForCodeActionAsync(document, action, textDiffingService, progressTracker, cancellationToken));
 
        // Using Sets allows us to accumulate only the distinct changes. Only consider insertion changes to reduce the
        // chance of producing a badly merged final document.
        var insertionOnlyChanges = new HashSet<TextChange>();
 
        // Some fixes require adding missing references.
        var allAddedProjectReferences = new HashSet<ProjectReference>();
        var allAddedMetaDataReferences = new HashSet<MetadataReference>();
 
        foreach (var getChangesTask in getChangesTasks)
        {
            var (projectChanges, textChanges) = await getChangesTask.ConfigureAwait(false);
 
            foreach (var textChange in textChanges)
            {
                if (textChange.Span.IsEmpty)
                    insertionOnlyChanges.Add(textChange);
            }
 
            allAddedProjectReferences.UnionWith(projectChanges.GetAddedProjectReferences());
            allAddedMetaDataReferences.UnionWith(projectChanges.GetAddedMetadataReferences());
        }
 
        // Apply changes to both the project and document.
        var newProject = document.Project;
        newProject = newProject.AddMetadataReferences(allAddedMetaDataReferences);
        newProject = newProject.AddProjectReferences(allAddedProjectReferences);
 
        // Capture each location where we are inserting imports as well as the total
        // length of the text we are inserting so that we can format the span afterwards.
        var insertSpans = insertionOnlyChanges
            .GroupBy(change => change.Span)
            .Select(changes => new TextSpan(changes.Key.Start, changes.Sum(change => change.NewText!.Length)));
 
        var text = await document.GetValueTextAsync(cancellationToken).ConfigureAwait(false);
        var newText = text.WithChanges(insertionOnlyChanges);
        var newDocument = newProject.GetRequiredDocument(document.Id).WithText(newText);
 
        // When imports are added to a code file that has no previous imports, extra newlines are generated between each
        // import because the fix is expecting to separate the imports from the rest of the code file. We need to format
        // the imports to remove these extra newlines.
        var cleanedDocument = await CleanUpNewLinesAsync(
            newDocument, insertSpans, formattingOptions, cancellationToken).ConfigureAwait(false);
 
        // Finally, organize the imports to ensure they are in the correct order.  Normally, the underling add-import
        // service will already ensure this.  However, this takes care of the case where we want to insert two or more
        // usings into the same location in an existing using-list.  In that case, there are many possible outcomes we 
        // could get depending on what order we processed the fixes in.  This ensures that no matter what order we do 
        // things in, the final result is organized properly.
        var organizedDocument = await organizeImportsService.OrganizeImportsAsync(
            cleanedDocument, organizeImportsOptions, cancellationToken).ConfigureAwait(false);
 
        return organizedDocument;
    }
 
    private async Task<Document> CleanUpNewLinesAsync(Document document, IEnumerable<TextSpan> insertSpans, SyntaxFormattingOptions formattingOptions, CancellationToken cancellationToken)
    {
        var newDocument = document;
 
        // Since imports can be added at both the CompilationUnit and the Namespace level,
        // format each span individually so that we can retain each newline that was intended
        // to separate the import section from the other content.
        foreach (var insertSpan in insertSpans)
        {
            newDocument = await CleanUpNewLinesAsync(newDocument, insertSpan, formattingOptions, cancellationToken).ConfigureAwait(false);
        }
 
        return newDocument;
    }
 
    private async Task<Document> CleanUpNewLinesAsync(Document document, TextSpan insertSpan, SyntaxFormattingOptions options, CancellationToken cancellationToken)
    {
        var root = await document.GetRequiredSyntaxRootAsync(cancellationToken).ConfigureAwait(false);
        var text = await document.GetValueTextAsync(cancellationToken).ConfigureAwait(false);
        var services = document.Project.Solution.Services;
 
        var textChanges = Formatter.GetFormattedTextChanges(
            root,
            [insertSpan],
            services,
            options: options,
            rules: GetFormatRules(text),
            cancellationToken);
 
        // If there are no changes then, do less work.
        if (textChanges.Count == 0)
        {
            return document;
        }
 
        // The last text change should include where the insert span ends
        Debug.Assert(textChanges.Last().Span.IntersectsWith(insertSpan.End));
 
        // If there are changes then, this was a case where there were no
        // previous imports statements. We need to retain the final extra
        // newline because that separates the imports section from the rest
        // of the code.
        textChanges.RemoveAt(textChanges.Count - 1);
 
        var newText = text.WithChanges(textChanges);
        return document.WithText(newText);
    }
 
    private static async Task<(ProjectChanges, IEnumerable<TextChange>)> GetChangesForCodeActionAsync(
        Document document,
        CodeAction codeAction,
        IDocumentTextDifferencingService textDiffingService,
        IProgress<CodeAnalysisProgress> progressTracker,
        CancellationToken cancellationToken)
    {
        // CodeAction.GetChangedSolutionAsync is only implemented for code actions that can fully compute the new	            
        // solution without deferred computation or taking a dependency on the main thread. In other cases, the	                
        // implementation of GetChangedSolutionAsync will throw an exception and the code action application is	            
        // expected to apply the changes by executing the operations in GetOperationsAsync (which may have other	
        // side effects). This code cannot assume the input CodeAction supports GetChangedSolutionAsync, so it first	
        // attempts to apply text changes obtained from GetOperationsAsync. Two forms are supported:	
        //	
        // 1. GetOperationsAsync returns an empty list of operations (i.e. no changes are required)	
        // 2. GetOperationsAsync returns a list of operations, where the first change is an ApplyChangesOperation to	
        //    change the text in the solution, and any remaining changes are deferred computation changes.	
        //	
        // If GetOperationsAsync does not adhere to one of these patterns, the code falls back to calling	
        // GetChangedSolutionAsync since there is no clear way to apply the changes otherwise.	
        var operations = await codeAction.GetOperationsAsync(
            document.Project.Solution, progressTracker, cancellationToken).ConfigureAwait(false);
        Solution newSolution;
        if (operations.Length == 0)
        {
            newSolution = document.Project.Solution;
        }
        else if (operations is [ApplyChangesOperation applyChangesOperation])
        {
            newSolution = applyChangesOperation.ChangedSolution;
        }
        else
        {
            newSolution = await codeAction.GetRequiredChangedSolutionAsync(progressTracker, cancellationToken).ConfigureAwait(false);
        }
 
        var newDocument = newSolution.GetRequiredDocument(document.Id);
 
        // Use Line differencing to reduce the possibility of changes that overwrite existing code.
        var textChanges = await textDiffingService.GetTextChangesAsync(
            document, newDocument, TextDifferenceTypes.Line, cancellationToken).ConfigureAwait(false);
        var projectChanges = newDocument.Project.GetChanges(document.Project);
 
        return (projectChanges, textChanges);
    }
 
    protected sealed class CleanUpNewLinesFormatter(SourceText text) : AbstractFormattingRule
    {
        private readonly SourceText _text = text;
 
        public override AdjustNewLinesOperation? GetAdjustNewLinesOperation(in SyntaxToken previousToken, in SyntaxToken currentToken, in NextGetAdjustNewLinesOperation nextOperation)
        {
            // Since we know the general shape of these new import statements, we simply look for where
            // tokens are not on the same line and force them to only be separated by a single newline.
 
            _text.GetLineAndOffset(previousToken.Span.Start, out var previousLine, out _);
            _text.GetLineAndOffset(currentToken.Span.Start, out var currentLine, out _);
 
            if (previousLine != currentLine)
            {
                return FormattingOperations.CreateAdjustNewLinesOperation(1, AdjustNewLinesOption.ForceLines);
            }
 
            return null;
        }
    }
}