|
8 | 8 |
|
9 | 9 | namespace Microsoft.AspNetCore.SignalR.Client
|
10 | 10 | {
|
11 |
| - internal class InvocationRequest : IDisposable |
| 11 | + internal abstract class InvocationRequest : IDisposable |
12 | 12 | {
|
13 |
| - private readonly TaskCompletionSource<object> _completionSource; |
14 | 13 | private readonly CancellationTokenRegistration _cancellationTokenRegistration;
|
15 | 14 | private readonly bool _streaming;
|
16 |
| - private readonly ILogger _logger; |
17 |
| - private readonly InvocationSubject _subject; |
| 15 | + |
| 16 | + protected ILogger Logger { get; } |
18 | 17 |
|
19 | 18 | public Type ResultType { get; }
|
20 | 19 | public CancellationToken CancellationToken { get; }
|
21 | 20 | public string InvocationId { get; }
|
22 | 21 |
|
23 |
| - public Task<object> Result => _completionSource?.Task ?? Task.FromResult<object>(null); |
24 |
| - public IObservable<object> Observable => _subject; |
25 |
| - |
26 |
| - public InvocationRequest(CancellationToken cancellationToken, Type resultType, string invocationId, ILoggerFactory loggerFactory, bool streaming) |
| 22 | + protected InvocationRequest(CancellationToken cancellationToken, Type resultType, string invocationId, ILogger logger) |
27 | 23 | {
|
28 |
| - _logger = loggerFactory.CreateLogger<InvocationRequest>(); |
29 |
| - _cancellationTokenRegistration = cancellationToken.Register(state => (state as TaskCompletionSource<object>)?.TrySetCanceled(), _completionSource); |
30 |
| - _streaming = streaming; |
31 |
| - |
32 |
| - if (_streaming) |
33 |
| - { |
34 |
| - _subject = new InvocationSubject(); |
35 |
| - } |
36 |
| - else |
37 |
| - { |
38 |
| - _completionSource = new TaskCompletionSource<object>(TaskCreationOptions.RunContinuationsAsynchronously); |
39 |
| - } |
| 24 | + _cancellationTokenRegistration = cancellationToken.Register(self => ((InvocationRequest)self).Cancel(), this); |
40 | 25 |
|
41 | 26 | InvocationId = invocationId;
|
42 | 27 | CancellationToken = cancellationToken;
|
43 | 28 | ResultType = resultType;
|
44 | 29 |
|
45 |
| - _logger.LogTrace("Invocation {invocationId} created", InvocationId); |
| 30 | + Logger.LogTrace("Invocation {invocationId} created", InvocationId); |
46 | 31 | }
|
47 | 32 |
|
48 |
| - public void Fail(Exception exception) |
| 33 | + public static InvocationRequest Invoke(CancellationToken cancellationToken, Type resultType, string invocationId, ILoggerFactory loggerFactory, out Task<object> result) |
49 | 34 | {
|
50 |
| - _logger.LogTrace("Invocation {invocationId} marked as failed.", InvocationId); |
51 |
| - if (_streaming) |
52 |
| - { |
53 |
| - _subject.TryOnError(exception); |
54 |
| - } |
55 |
| - else |
56 |
| - { |
57 |
| - _completionSource.TrySetException(exception); |
58 |
| - } |
| 35 | + var req = new NonStreaming(cancellationToken, resultType, invocationId, loggerFactory); |
| 36 | + result = req.Result; |
| 37 | + return req; |
59 | 38 | }
|
60 | 39 |
|
61 |
| - public void Complete(object result) |
| 40 | + |
| 41 | + public static InvocationRequest Stream(CancellationToken cancellationToken, Type resultType, string invocationId, ILoggerFactory loggerFactory, out IObservable<object> result) |
62 | 42 | {
|
63 |
| - _logger.LogTrace("Invocation {invocationId} marked as completed.", InvocationId); |
64 |
| - if (_streaming) |
| 43 | + var req = new Streaming(cancellationToken, resultType, invocationId, loggerFactory); |
| 44 | + result = req.Result; |
| 45 | + return req; |
| 46 | + } |
| 47 | + |
| 48 | + public abstract void Fail(Exception exception); |
| 49 | + public abstract void Complete(object result); |
| 50 | + public abstract void StreamItem(object item); |
| 51 | + |
| 52 | + protected abstract void Cancel(); |
| 53 | + |
| 54 | + public virtual void Dispose() |
| 55 | + { |
| 56 | + Logger.LogTrace("Invocation {invocationId} disposed", InvocationId); |
| 57 | + |
| 58 | + // Just in case it hasn't already been completed |
| 59 | + Cancel(); |
| 60 | + |
| 61 | + _cancellationTokenRegistration.Dispose(); |
| 62 | + } |
| 63 | + |
| 64 | + private class Streaming : InvocationRequest |
| 65 | + { |
| 66 | + private readonly InvocationSubject _subject; |
| 67 | + |
| 68 | + public Streaming(CancellationToken cancellationToken, Type resultType, string invocationId, ILoggerFactory loggerFactory) |
| 69 | + : base(cancellationToken, resultType, invocationId, loggerFactory.CreateLogger<Streaming>()) |
65 | 70 | {
|
| 71 | + } |
| 72 | + |
| 73 | + public IObservable<object> Result => _subject; |
| 74 | + |
| 75 | + public override void Complete(object result) |
| 76 | + { |
| 77 | + Logger.LogTrace("Invocation {invocationId} marked as completed.", InvocationId); |
66 | 78 | if (result != null)
|
67 | 79 | {
|
68 |
| - _logger.LogError("Invocation {invocationId} received a completion result, but was invoked as a streaming invocation.", InvocationId); |
| 80 | + Logger.LogError("Invocation {invocationId} received a completion result, but was invoked as a streaming invocation.", InvocationId); |
69 | 81 | _subject.TryOnError(new InvalidOperationException("Server provided a result in a completion response to a streamed invocation."));
|
70 | 82 | }
|
71 | 83 | else
|
72 | 84 | {
|
73 | 85 | _subject.TryOnCompleted();
|
74 | 86 | }
|
75 | 87 | }
|
76 |
| - else |
| 88 | + |
| 89 | + public override void Fail(Exception exception) |
77 | 90 | {
|
78 |
| - _completionSource.TrySetResult(result); |
| 91 | + Logger.LogTrace("Invocation {invocationId} marked as failed.", InvocationId); |
| 92 | + _subject.TryOnError(exception); |
79 | 93 | }
|
80 |
| - } |
81 | 94 |
|
82 |
| - public void StreamItem(object item) |
83 |
| - { |
84 |
| - if (_streaming) |
| 95 | + public override void StreamItem(object item) |
85 | 96 | {
|
86 |
| - _logger.LogTrace("Invocation {invocationId} received stream item.", InvocationId); |
| 97 | + Logger.LogTrace("Invocation {invocationId} received stream item.", InvocationId); |
87 | 98 | _subject.TryOnNext(item);
|
88 | 99 | }
|
89 |
| - else |
| 100 | + |
| 101 | + protected override void Cancel() |
90 | 102 | {
|
91 |
| - _logger.LogError("Invocation {invocationId} received stream item but was invoked as a non-streamed invocation.", InvocationId); |
92 |
| - _completionSource.TrySetException(new InvalidOperationException("Streaming methods must be invoked using HubConnection.Stream")); |
| 103 | + _subject.TryOnError(new OperationCanceledException("Connection terminated")); |
93 | 104 | }
|
94 | 105 | }
|
95 | 106 |
|
96 |
| - public void Dispose() |
| 107 | + private class NonStreaming : InvocationRequest |
97 | 108 | {
|
98 |
| - _logger.LogTrace("Invocation {invocationId} disposed", InvocationId); |
| 109 | + private readonly TaskCompletionSource<object> _completionSource = new TaskCompletionSource<object>(TaskCreationOptions.RunContinuationsAsynchronously); |
99 | 110 |
|
100 |
| - // Just in case it hasn't already been completed |
101 |
| - if (_streaming) |
| 111 | + public NonStreaming(CancellationToken cancellationToken, Type resultType, string invocationId, ILoggerFactory loggerFactory) |
| 112 | + : base(cancellationToken, resultType, invocationId, loggerFactory.CreateLogger<NonStreaming>()) |
102 | 113 | {
|
103 |
| - _subject.TryOnError(new OperationCanceledException("Connection terminated")); |
104 | 114 | }
|
105 |
| - else |
| 115 | + |
| 116 | + public Task<object> Result => _completionSource.Task; |
| 117 | + |
| 118 | + public override void Complete(object result) |
| 119 | + { |
| 120 | + Logger.LogTrace("Invocation {invocationId} marked as completed.", InvocationId); |
| 121 | + _completionSource.TrySetResult(result); |
| 122 | + } |
| 123 | + |
| 124 | + public override void Fail(Exception exception) |
106 | 125 | {
|
107 |
| - _completionSource.TrySetCanceled(); |
| 126 | + Logger.LogTrace("Invocation {invocationId} marked as failed.", InvocationId); |
| 127 | + _completionSource.TrySetException(exception); |
108 | 128 | }
|
109 | 129 |
|
110 |
| - _cancellationTokenRegistration.Dispose(); |
| 130 | + public override void StreamItem(object item) |
| 131 | + { |
| 132 | + Logger.LogError("Invocation {invocationId} received stream item but was invoked as a non-streamed invocation.", InvocationId); |
| 133 | + _completionSource.TrySetException(new InvalidOperationException("Streaming methods must be invoked using HubConnection.Stream")); |
| 134 | + } |
| 135 | + |
| 136 | + protected override void Cancel() |
| 137 | + { |
| 138 | + _completionSource.TrySetCanceled(); |
| 139 | + } |
111 | 140 | }
|
112 | 141 | }
|
113 | 142 | }
|
0 commit comments