Skip to content

Commit

Permalink
Add docs to help with disconnect investigations
Browse files Browse the repository at this point in the history
  • Loading branch information
AArnott committed Sep 12, 2024
1 parent 529012a commit bf4b520
Show file tree
Hide file tree
Showing 3 changed files with 41 additions and 1 deletion.
32 changes: 31 additions & 1 deletion doc/disconnecting.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
The JSON-RPC connection is disconnected for any of these scenarios:

1. The `JsonRpc.Dispose` method is invoked.
1. The transport unexpectedly closes.
1. The transport unexpectedly closes (e.g. lost network connection, remote process crashed, remote side closed the connection intentionally).
1. A message is received that does not parse to a valid JSON-RPC message, or some other protocol-level violation is detected.

When any of the above conditions are met, `JsonRpc` will (in no particular order):
Expand All @@ -14,3 +14,33 @@ When any of the above conditions are met, `JsonRpc` will (in no particular order
1. Remove event handlers that may have been added to any target objects.
1. Completes the `Task` returned from the `JsonRpc.Completion` property, either succesfully or with the exception that led to the connection termination.
1. Raise the `JsonRpc.Disconnected` event, with an argument specifying the reason for the disconnection.

## Understanding the cause for an unexpected `ConnectionLostException`

An unexpected disconnection may occur for any of the reasons given above.
Understanding *which* possible condition led to termination may require access to the logs and/or code running on both ends.

Given the `JsonRpc.Disconnected` event includes the reason for disconnect, an event handler attached to this which writes any unexpected disconnect reasons to a file can be invaluable during an investigation.
It can be important to add the event handler to *both* sides of the connection since the root cause will typically be logged by only one side, while the other side may report a less helpful cause (e.g. the other side disconnected).

Following is a table of reasons that may be given by each party, and a plausible explanation for each one.
Party 1 is likely the first to notice and report a disconnect.

Party 1 | Party 2 | Explanation
--|--|--
LocallyDisposed | RemotePartyTerminated | Party 1 explicitly disconnected, and party 2 noticed.
RemotePartyTerminated | RemotePartyTerminated | The transport failed. May be common across unreliable networks.
StreamError | RemotePartyTerminated so StreamError | Party 1 experienced some I/O error that made reading from the stream impossible. Party 2 may experience the same, or notice when Party 1 disconnects as a result of the first error.
LocalContractViolation | RemotePartyTerminated | Party 1 knew it would be unable to maintain its protocol contract (e.g. a critical serialization failure or misbehaving extension) and killed its own connection.
RemoteProtocolViolation, ParseError | * | Party 1 recognized that Party 2 sent a message that violated the protocol, so party 1 disconnected.

In addition to the `JsonRpcDisconnectedEventArgs.Reason` given to a handler of the `JsonRpc.Disconnected` event, the `JsonRpcDisconnectedEventArgs` class includes several other properties that can be very helpful in understanding details of a failure.
Logging the whole object can be a good idea.

### Using TraceSource for logging

Most of the properties available on this object are also traced out via `JsonRpc.TraceSource` on disconnect.
Adding a `TraceListener` to that `TraceSource` is usually adequate for logging and captures many other warning and error conditions that may occur without a disconnect.

`LocallyDisposed` and `RemotePartyTerminated` disconnection reasons are considered normal and are traced with `TraceEventType.Information` severity level.
All other reasons are traced with `TraceEventType.Critical` severity level.
4 changes: 4 additions & 0 deletions doc/troubleshooting.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,10 @@ The above is just a sample. The full list of events is available on the `JsonRpc

## Other issues

### Requests failing with ConnectionLostException

Please see our [disconnecting](disconnecting.md) documentation.

### Hangs after connecting over IPC pipes

When connecting two processes using Windows (named) pipes, be sure to use `PipeOptions.Asynchronous`
Expand Down
6 changes: 6 additions & 0 deletions src/StreamJsonRpc/EventArgs/DisconnectedReason.cs
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,12 @@ public enum DisconnectedReason
/// <summary>
/// A fatal exception was thrown in a local method that was requested by the remote party.
/// </summary>
/// <remarks>
/// Exceptions thrown by locally dispatched method calls are not considered fatal by default.
/// The <see langword="virtual" /> method <see cref="JsonRpc.IsFatalException" /> may be overridden
/// by an application in order to enable a JSON-RPC server to throw an exception that can terminate
/// the entire connection.
/// </remarks>
FatalException,

/// <summary>
Expand Down

0 comments on commit bf4b520

Please sign in to comment.