File: Shared\Utilities\CompilationAvailableEventSource.cs
Web Access
Project: ..\..\..\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.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.Remote;
using Microsoft.CodeAnalysis.Shared.TestHooks;
using Microsoft.CodeAnalysis.Tagging;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.Shared.Utilities
{
    /// <summary>
    /// Helper type that can be used to ask for a <see cref="Compilation"/> to be produced in our OOP server for a
    /// particular <see cref="Project"/>, asking for a callback to be executed when that has happened.  Each time this
    /// is asked for for a particular project, any existing outstanding work to produce a <see cref="Compilation"/> for
    /// a prior <see cref="Project"/> will be cancelled.
    /// </summary>
    internal class CompilationAvailableEventSource : IDisposable
    {
        private readonly IAsynchronousOperationListener _asyncListener;
 
        /// <summary>
        /// Cancellation tokens controlling background computation of the compilation.
        /// </summary>
        private readonly ReferenceCountedDisposable<CancellationSeries> _cancellationSeries = new(new CancellationSeries());
 
        public CompilationAvailableEventSource(
            IAsynchronousOperationListener asyncListener)
        {
            _asyncListener = asyncListener;
        }
 
        public void Dispose()
            => _cancellationSeries.Dispose();
 
        /// <summary>
        /// Request that the compilation for <see cref="Project"/> be made available in our OOP server, calling back on
        /// <paramref name="onCompilationAvailable"/> once that happens.  Subsequence calls to this method will cancel
        /// any outstanding requests in flight.
        /// </summary>
        public void EnsureCompilationAvailability(Project project, Action onCompilationAvailable)
        {
            if (project == null)
                return;
 
            if (!project.SupportsCompilation)
                return;
 
            using var cancellationSeries = _cancellationSeries.TryAddReference();
            if (cancellationSeries is null)
            {
                // Already in the process of disposing this instance
                return;
            }
 
            // Cancel any existing tasks that are computing the compilation and spawn a new one to compute
            // it and notify any listening clients.
            var cancellationToken = cancellationSeries.Target.CreateNext();
 
            var token = _asyncListener.BeginAsyncOperation(nameof(EnsureCompilationAvailability));
            var task = Task.Run(async () =>
            {
                // Support cancellation without throwing.
                //
                // We choose a long delay here so that we can avoid this work as long as the user is continually making
                // changes to their code.  During that time, features that use this are already kicking off fast work
                // with frozen-partial semantics and we'd like that to not have to contend with more expensive work
                // kicked off in OOP to compute full compilations.
                await _asyncListener.Delay(DelayTimeSpan.NonFocus, cancellationToken).NoThrowAwaitableInternal(captureContext: false);
                if (cancellationToken.IsCancellationRequested)
                    return;
 
                var client = await RemoteHostClient.TryGetClientAsync(project, cancellationToken).ConfigureAwait(false);
                if (client != null)
                {
                    var result = await client.TryInvokeAsync<IRemoteCompilationAvailableService>(
                        project,
                        (service, solutionInfo, cancellationToken) => service.ComputeCompilationAsync(solutionInfo, project.Id, cancellationToken),
                        cancellationToken).ConfigureAwait(false);
 
                    if (!result)
                        return;
                }
                else
                {
                    // if we can't get the client, just compute the compilation locally and fire the event once we have it.
                    await CompilationAvailableHelpers.ComputeCompilationInCurrentProcessAsync(project, cancellationToken).ConfigureAwait(false);
                }
 
                // now that we know we have an full compilation, let the caller know so it can do whatever it needs in
                // response.
                onCompilationAvailable();
            }, cancellationToken);
            task.CompletesAsyncOperation(token);
        }
    }
}