File: Remote\VisualStudioWorkspaceServiceHubConnector.cs
Web Access
Project: src\src\VisualStudio\Core\Def\Microsoft.VisualStudio.LanguageServices_pxr0p0dn_wpftmp.csproj (Microsoft.VisualStudio.LanguageServices)
// 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.Composition;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.Host;
using Microsoft.CodeAnalysis.Host.Mef;
using Microsoft.CodeAnalysis.Remote;
using Microsoft.CodeAnalysis.Shared.TestHooks;
 
namespace Microsoft.VisualStudio.LanguageServices.Remote;
 
/// <summary>
/// Connects <see cref="VisualStudioWorkspace"/> to the ServiceHub services.
/// Launches ServiceHub if it is not running yet and starts services that push information from <see cref="VisualStudioWorkspace"/> to the ServiceHub process.
/// </summary>
[ExportEventListener(WellKnownEventListeners.Workspace, WorkspaceKind.Host), Shared]
[method: ImportingConstructor]
[method: Obsolete(MefConstruction.ImportingConstructorMessage, error: true)]
internal sealed class VisualStudioWorkspaceServiceHubConnector(
    IAsynchronousOperationListenerProvider listenerProvider) : IEventListener<object>, IEventListenerStoppable
{
    private readonly IAsynchronousOperationListenerProvider _listenerProvider = listenerProvider;
    private readonly CancellationTokenSource _disposalCancellationSource = new();
 
    private Task<RemoteHostClient?>? _remoteClientInitializationTask;
    private SolutionChecksumUpdater? _checksumUpdater;
 
    public void StartListening(Workspace workspace, object serviceOpt)
    {
        if (workspace is not VisualStudioWorkspace)
        {
            return;
        }
 
        // only push solution snapshot from primary (VS) workspace:
        _checksumUpdater = new SolutionChecksumUpdater(workspace, _listenerProvider, _disposalCancellationSource.Token);
 
        // start launching remote process, so that the first service that needs it doesn't need to wait for it:
        var service = workspace.Services.GetRequiredService<IRemoteHostClientProvider>();
        _remoteClientInitializationTask = service.TryGetRemoteHostClientAsync(_disposalCancellationSource.Token);
    }
 
    public void StopListening(Workspace workspace)
    {
        if (workspace is not VisualStudioWorkspace)
        {
            return;
        }
 
        _disposalCancellationSource.Cancel();
        _disposalCancellationSource.Dispose();
 
        _checksumUpdater?.Shutdown();
 
        // dispose remote client if its initialization has completed:
        _remoteClientInitializationTask?.ContinueWith(
            previousTask => previousTask.Result?.Dispose(),
            CancellationToken.None,
            TaskContinuationOptions.OnlyOnRanToCompletion,
            TaskScheduler.Default);
    }
}