Skip to content
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

[API Proposal]: Add APIs to WebSocket which allow it to be read as a Stream #111217

Open
christothes opened this issue Jan 8, 2025 · 4 comments
Assignees
Labels
api-suggestion Early API idea and discussion, it is NOT ready for implementation area-System.Net
Milestone

Comments

@christothes
Copy link

christothes commented Jan 8, 2025

Background and motivation

Utilizing WebSockets is a convenient approach to writing real-time audio processing code for ASP.NET applications. One such scenario is implementing a real-time conversation with Open AI.

OpenAI's real-time API SendInputAudioAsync accept a Stream as input which leaves it up to the developer to write a custom Stream implementation that reads from an underlying WebSocket. It would be a nice enhancement to the WebSocket APIs if one could wrap read operations in a Stream.

API Proposal

public class WebSocket
{
    public Stream AsStream();
}

API Usage

using var webSocket = await context.WebSockets.AcceptWebSocketAsync();
using RealtimeConversationSession session = await InitSession(realtime);
// <...>
using var stream = webSocket.AsStream();
await session.SendInputAudioAsync(stream);

Alternative Designs

No response

Risks

WebSocket doesn’t provide synchronous methods for wire-based operations, so all of the Stream sync APIs (including Dispose, which presumably would need to not just Dispose the WebSocket but also CloseAsync it) would be sync-over-async.

@christothes christothes added the api-suggestion Early API idea and discussion, it is NOT ready for implementation label Jan 8, 2025
@dotnet-policy-service dotnet-policy-service bot added the untriaged New issue has not been triaged by the area owner label Jan 8, 2025
Copy link
Contributor

Tagging subscribers to this area: @dotnet/ncl
See info in area-owners.md if you want to be subscribed.

@MihaZupan
Copy link
Member

MihaZupan commented Jan 9, 2025

It's an interesting idea, though its use seems limited only to cases where you know that only the binary content is being transmitted (e.g. only the audio, no control data, no extra framing).
There's also the question of what happens with close messages -- does the user not care about the data, who's responsible for responding to them, do you send them during disposal.

I can see it being useful in cases where you just need an opaque Stream and happen to be using WebSocket as the transport.

Sample code if someone needed such a Stream could be something like this (untested):

public sealed class WebSocketStream : Stream
{
    private readonly WebSocket _webSocket;

    public WebSocketStream(WebSocket webSocket) => _webSocket = webSocket;

    public override bool CanRead => _webSocket.State is WebSocketState.Open or WebSocketState.CloseSent;
    public override bool CanWrite => _webSocket.State is WebSocketState.Open or WebSocketState.CloseReceived;
    public override bool CanSeek => false;

    public override void Flush() { }
    public override Task FlushAsync(CancellationToken cancellationToken) => Task.CompletedTask;

    public override Task<int> ReadAsync(byte[] buffer, int offset, int count, CancellationToken cancellationToken) =>
        ReadAsync(buffer.AsMemory(offset, count), cancellationToken).AsTask();

    public override Task WriteAsync(byte[] buffer, int offset, int count, CancellationToken cancellationToken) =>
        WriteAsync(buffer.AsMemory(offset, count), cancellationToken).AsTask();

    public override async ValueTask<int> ReadAsync(Memory<byte> buffer, CancellationToken cancellationToken = default)
    {
        ValueWebSocketReceiveResult result = await _webSocket.ReceiveAsync(buffer, cancellationToken);

        if (result.MessageType != WebSocketMessageType.Binary)
        {
            if (result.MessageType == WebSocketMessageType.Close)
            {
                await _webSocket.SendAsync(ReadOnlyMemory<byte>.Empty, WebSocketMessageType.Close, endOfMessage: true, cancellationToken);
                return 0;
            }

            throw new Exception("Expected binary messages");
        }

        return result.Count;
    }

    public override ValueTask WriteAsync(ReadOnlyMemory<byte> buffer, CancellationToken cancellationToken = default) =>
        _webSocket.SendAsync(buffer, WebSocketMessageType.Binary, endOfMessage: true, cancellationToken);

    public override ValueTask DisposeAsync()
    {
        Dispose(true);
        return default;
    }

    protected override void Dispose(bool disposing) => _webSocket.Dispose();

    public override int Read(byte[] buffer, int offset, int count) => ReadAsync(buffer, offset, count).GetAwaiter().GetResult();
    public override void Write(byte[] buffer, int offset, int count) => WriteAsync(buffer, offset, count).GetAwaiter().GetResult();

    public override long Length => throw new NotSupportedException();
    public override long Position { get => throw new NotSupportedException(); set => throw new NotSupportedException(); }
    public override long Seek(long offset, SeekOrigin origin) => throw new NotSupportedException();
    public override void SetLength(long value) => throw new NotSupportedException();
}

@MihaZupan
Copy link
Member

MihaZupan commented Jan 16, 2025

Triage: We see the value and it should be a relatively low amount of work to implement. Even a simple GH search shows many users implementing similar wrappers themselves, moving to 10.0.

We'll have to figure out a default for how we handle things like close messages, but it should otherwise be straightforward.

@MihaZupan MihaZupan removed the untriaged New issue has not been triaged by the area owner label Jan 16, 2025
@MihaZupan MihaZupan added this to the 10.0.0 milestone Jan 16, 2025
@CarnaViire
Copy link
Member

I think we can also take inspiration from the NetworkStream, which does similar thing for a Socket

@antonfirsov antonfirsov self-assigned this Jan 21, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
api-suggestion Early API idea and discussion, it is NOT ready for implementation area-System.Net
Projects
None yet
Development

No branches or pull requests

4 participants