using System.Buffers.Binary; using Google.Protobuf; using MxGateway.Contracts.Proto; namespace MxGateway.Server.Workers; public sealed class WorkerFrameWriter { private readonly WorkerFrameProtocolOptions _options; private readonly Stream _stream; public WorkerFrameWriter( Stream stream, WorkerFrameProtocolOptions options) { _stream = stream ?? throw new ArgumentNullException(nameof(stream)); _options = options ?? throw new ArgumentNullException(nameof(options)); } public async ValueTask WriteAsync( WorkerEnvelope envelope, CancellationToken cancellationToken = default) { ArgumentNullException.ThrowIfNull(envelope); WorkerEnvelopeValidator.Validate(envelope, _options); int payloadLength = envelope.CalculateSize(); if (payloadLength == 0) { throw new WorkerFrameProtocolException( WorkerFrameProtocolErrorCode.InvalidEnvelope, "Worker envelope cannot serialize to an empty payload."); } if (payloadLength > _options.MaxMessageBytes) { throw new WorkerFrameProtocolException( WorkerFrameProtocolErrorCode.MessageTooLarge, $"Worker envelope payload length {payloadLength} exceeds the configured maximum of {_options.MaxMessageBytes} bytes."); } byte[] lengthPrefix = new byte[sizeof(uint)]; BinaryPrimitives.WriteUInt32LittleEndian(lengthPrefix, (uint)payloadLength); await _stream.WriteAsync(lengthPrefix, cancellationToken).ConfigureAwait(false); await _stream.WriteAsync(envelope.ToByteArray(), cancellationToken).ConfigureAwait(false); } }