File: Installer\Windows\PipeStreamMessageDispatcherBase.cs
Web Access
Project: ..\..\..\src\Cli\dotnet\dotnet.csproj (dotnet)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
#nullable disable
 
using System.IO.Pipes;
using System.Runtime.Versioning;
 
namespace Microsoft.DotNet.Cli.Installer.Windows;
 
/// <summary>
/// Base class used for dispatching messages (<see cref="PipeTransmissionMode.Message"/>) over a named pipe.
/// </summary>
/// <remarks>
/// Creates a new <see cref="PipeStreamMessageDispatcherBase"/> instance.
/// </remarks>
/// <param name="pipeStream">The pipe stream to use for reading and writing messages. The pipe must be configured
/// to use <see cref="PipeTransmissionMode.Message"/>.</param>
/// <exception cref="ArgumentNullException" />
[SupportedOSPlatform("windows")]
internal class PipeStreamMessageDispatcherBase(PipeStream pipeStream)
{
    /// <summary>
    /// The maxmimum length of a message.
    /// </summary>
    public const int MaxMessageSize = 2048;
 
    /// <summary>
    /// The backing stream used for reading & writing messages.
    /// </summary>
    private readonly PipeStream _pipeStream = pipeStream ?? throw new ArgumentNullException(nameof(pipeStream));
 
    /// <summary>
    /// The number of milliseconds to wait for a pipe connection to be established. See <see cref="Connect"/>.
    /// </summary>
    public const int ConnectionTimeout = 30000;
 
    /// <summary>
    /// Returns <see langword="true"/> if the dispatcher is using a <see cref="NamedPipeClientStream"/>.
    /// </summary>
    public bool IsClient => _pipeStream is NamedPipeServerStream;
 
    /// <summary>
    /// Gets whether the underlying stream is connected.
    /// </summary>
    public bool IsConnected => _pipeStream.IsConnected;
 
    /// <summary>
    /// Waits for the underlying <see cref="PipeStream"/> to establish a connection. If the stream is a 
    /// <see cref="NamedPipeClientStream"/>, its transmission mode will be set to <see cref="PipeTransmissionMode.Message"/>
    /// once the connection is established.
    /// </summary>
    /// <exception cref="TimeoutException"/>
    public void Connect()
    {
        if (!_pipeStream.IsConnected)
        {
            if (_pipeStream is NamedPipeServerStream)
            {
                Task connectTask = Task.Factory.StartNew(() => ((NamedPipeServerStream)_pipeStream).WaitForConnection());
                connectTask.Wait(ConnectionTimeout);
 
                if (!connectTask.IsCompleted)
                {
                    throw new TimeoutException("Timed out waiting for connection.");
                }
            }
            else if (_pipeStream is NamedPipeClientStream)
            {
                ((NamedPipeClientStream)_pipeStream).Connect(ConnectionTimeout);
                // We cannot set the transmission mode of a pipe client until it's connected.
                _pipeStream.ReadMode = PipeTransmissionMode.Message;
            }
        }
    }
 
    /// <summary>
    /// Reads a message from the underlying pipe stream. This method blocks until
    /// data is available to read.
    /// </summary>
    /// <returns>The message read from the pipe.</returns>
    public byte[] ReadMessage()
    {
        byte[] message = new byte[2048];
        _ = _pipeStream.Read(message, 0, message.Length);
        int messageLength = BitConverter.ToInt32(message, 0);
        byte[] messageBytes = new byte[messageLength];
        Array.Copy(message, 4, messageBytes, 0, messageLength);
 
        return messageBytes;
    }
 
    /// <summary>
    /// Writes a message to the underlying pipe stream.
    /// </summary>
    /// <param name="messageBytes">The message to write.</param>
    public void WriteMessage(byte[] messageBytes)
    {
        byte[] messageLengthBytes = BitConverter.GetBytes(messageBytes.Length);
        byte[] msg = [.. messageLengthBytes, .. messageBytes];
 
        if (msg.Length > MaxMessageSize)
        {
            throw new IOException($"The message is too long.");
        }
 
        _pipeStream.Write(msg, 0, msg.Length);
        _pipeStream.WaitForPipeDrain();
    }
}