diff --git a/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingConnection.cs b/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingConnection.cs index 40b3af14da..d9effa885f 100644 --- a/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingConnection.cs +++ b/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingConnection.cs @@ -43,6 +43,14 @@ protected StreamingConnection(ILogger logger) /// A for the streaming connection. protected ILogger Logger { get; } + /// + /// Gets a value indicating whether this is currently connected. + /// + /// + /// True if this is currently connected, otherwise false. + /// + protected bool IsConnected { get; private set; } = false; + /// /// Sends a streaming request through the connection. /// @@ -65,7 +73,20 @@ public virtual async Task SendStreamingRequestAsync(StreamingRe throw new InvalidOperationException("Cannot send streaming request since the session is not set up."); } - return await _session.SendRequestAsync(request, cancellationToken).ConfigureAwait(false); + try + { + return await _session.SendRequestAsync(request, cancellationToken).ConfigureAwait(false); + } + catch (TimeoutException ex) + { + var timeoutMessage = $"The connection to the client has been disconnected, and the request has timed out after waiting {TaskExtensions.DefaultTimeout.Seconds} seconds for a response."; + if (IsConnected) + { + timeoutMessage = $"The request sent to the client has timed out after waiting {TaskExtensions.DefaultTimeout.Seconds} seconds for a response."; + } + + throw new OperationCanceledException(timeoutMessage, ex, cancellationToken); + } } /// @@ -96,7 +117,7 @@ public virtual async Task ListenAsync(RequestHandler requestHandler, Cancellatio _session = new StreamingSession(requestHandler, _application, Logger, cancellationToken); // Start transport and application - var transportTask = _transport.ConnectAsync(default, cancellationToken); + var transportTask = _transport.ConnectAsync((connected) => IsConnected = connected, cancellationToken); var applicationTask = _application.ListenAsync(cancellationToken); var tasks = new List { transportTask, applicationTask }; diff --git a/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingTransportClient.cs b/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingTransportClient.cs index 16097d1a7a..73b8153504 100644 --- a/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingTransportClient.cs +++ b/libraries/Microsoft.Bot.Connector.Streaming/Application/StreamingTransportClient.cs @@ -105,7 +105,20 @@ public async Task SendAsync(StreamingRequest message, Cancellat throw new ArgumentNullException(nameof(message)); } - return await _session.SendRequestAsync(message, cancellationToken).ConfigureAwait(false); + try + { + return await _session.SendRequestAsync(message, cancellationToken).ConfigureAwait(false); + } + catch (TimeoutException ex) + { + var timeoutMessage = $"The underlying connection has been disconnected, and the request has timed out after waiting {TaskExtensions.DefaultTimeout.Seconds} seconds for a response."; + if (IsConnected) + { + timeoutMessage = $"The request sent to the underlying connection has timed out after waiting {TaskExtensions.DefaultTimeout.Seconds} seconds for a response."; + } + + throw new OperationCanceledException(timeoutMessage, ex, cancellationToken); + } } /// diff --git a/libraries/Microsoft.Bot.Connector.Streaming/TaskExtensions.cs b/libraries/Microsoft.Bot.Connector.Streaming/TaskExtensions.cs index 8f39d051a8..0940a437a9 100644 --- a/libraries/Microsoft.Bot.Connector.Streaming/TaskExtensions.cs +++ b/libraries/Microsoft.Bot.Connector.Streaming/TaskExtensions.cs @@ -6,14 +6,14 @@ using System.Threading.Tasks; namespace Microsoft.Bot.Connector.Streaming -{ +{ internal static class TaskExtensions { - private static readonly TimeSpan _defaultTimeout = TimeSpan.FromSeconds(30); + public static readonly TimeSpan DefaultTimeout = TimeSpan.FromSeconds(30); public static async Task DefaultTimeOutAsync(this Task task) { - return await task.TimeoutAfterAsync(_defaultTimeout).ConfigureAwait(false); + return await task.TimeoutAfterAsync(DefaultTimeout).ConfigureAwait(false); } public static async Task TimeoutAfterAsync(this Task task, TimeSpan timeout) @@ -41,7 +41,7 @@ public static async Task TimeoutAfterAsync(this Task task, TimeSpan tim public static async Task DefaultTimeOutAsync(this Task task) { - await task.TimeoutAfterAsync(_defaultTimeout).ConfigureAwait(false); + await task.TimeoutAfterAsync(DefaultTimeout).ConfigureAwait(false); } public static async Task TimeoutAfterAsync(this Task task, TimeSpan timeout)