e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
1726 lines
67 KiB
C#
1726 lines
67 KiB
C#
//------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//------------------------------------------------------------
|
|
namespace System.ServiceModel.Channels
|
|
{
|
|
using System.Collections.Generic;
|
|
using System.Collections.ObjectModel;
|
|
using System.Diagnostics;
|
|
using System.Net;
|
|
using System.Runtime;
|
|
using System.ServiceModel;
|
|
using System.ServiceModel.Description;
|
|
using System.ServiceModel.Diagnostics;
|
|
using System.ServiceModel.Dispatcher;
|
|
using System.Threading;
|
|
|
|
// Neighbor Manager is responsible for managing a set of neighbors for a node.
|
|
class PeerNeighborManager
|
|
{
|
|
public event EventHandler<PeerNeighborCloseEventArgs> NeighborClosed;
|
|
public event EventHandler<PeerNeighborCloseEventArgs> NeighborClosing;
|
|
public event EventHandler NeighborConnected;
|
|
public event EventHandler NeighborOpened;
|
|
public event EventHandler Offline;
|
|
public event EventHandler Online;
|
|
|
|
// Delegate to determine if neighbor manager is closing or closed
|
|
delegate bool ClosedCallback();
|
|
|
|
enum State
|
|
{
|
|
Created,
|
|
Opened,
|
|
ShuttingDown,
|
|
Shutdown,
|
|
Closed,
|
|
}
|
|
|
|
PeerNodeConfig config;
|
|
|
|
//
|
|
// Contains the neighbors in connected state
|
|
// We maintain a connectedNeighborList in addition to neighborList for two reasons:
|
|
// (a) Several operations are on connected neighbors
|
|
// (b) To correctly handle online/offline conditions
|
|
//
|
|
List<IPeerNeighbor> connectedNeighborList;
|
|
|
|
bool isOnline;
|
|
PeerIPHelper ipHelper;
|
|
List<PeerNeighbor> neighborList; // contains all the neighbors known to neighbor manager
|
|
ManualResetEvent shutdownEvent;
|
|
State state;
|
|
object thisLock;
|
|
PeerNodeTraceRecord traceRecord;
|
|
PeerService service;
|
|
Binding serviceBinding;
|
|
IPeerNodeMessageHandling messageHandler;
|
|
|
|
public PeerNeighborManager(PeerIPHelper ipHelper, PeerNodeConfig config)
|
|
:
|
|
this(ipHelper, config, null) { }
|
|
public PeerNeighborManager(PeerIPHelper ipHelper, PeerNodeConfig config, IPeerNodeMessageHandling messageHandler)
|
|
{
|
|
Fx.Assert(ipHelper != null, "Non-null ipHelper is expected");
|
|
Fx.Assert(config != null, "Non-null Config is expected");
|
|
|
|
this.neighborList = new List<PeerNeighbor>();
|
|
this.connectedNeighborList = new List<IPeerNeighbor>();
|
|
this.ipHelper = ipHelper;
|
|
this.messageHandler = messageHandler;
|
|
this.config = config;
|
|
this.thisLock = new object();
|
|
this.traceRecord = new PeerNodeTraceRecord(config.NodeId);
|
|
this.state = State.Created;
|
|
}
|
|
|
|
// Returns the count of connected neighbors
|
|
public int ConnectedNeighborCount
|
|
{
|
|
get
|
|
{
|
|
return this.connectedNeighborList.Count;
|
|
}
|
|
}
|
|
|
|
public int NonClosingNeighborCount
|
|
{
|
|
get
|
|
{
|
|
int count = 0;
|
|
foreach (PeerNeighbor neighbor in this.connectedNeighborList)
|
|
{
|
|
if (!neighbor.IsClosing) count++;
|
|
}
|
|
return count;
|
|
}
|
|
}
|
|
|
|
// Returns true if Neighbor Manager is online
|
|
// (i.e., has one or more connected neighbors)
|
|
public bool IsOnline
|
|
{
|
|
get
|
|
{
|
|
return this.isOnline;
|
|
}
|
|
}
|
|
|
|
// Returns the count of connected neighbors
|
|
public int NeighborCount
|
|
{
|
|
get
|
|
{
|
|
return this.neighborList.Count;
|
|
}
|
|
}
|
|
|
|
object ThisLock
|
|
{
|
|
get
|
|
{
|
|
return this.thisLock;
|
|
}
|
|
}
|
|
|
|
// Ungracefully shutdown the neighbor manager
|
|
void Abort(PeerNeighbor[] neighbors)
|
|
{
|
|
foreach (PeerNeighbor neighbor in neighbors)
|
|
neighbor.Abort(PeerCloseReason.LeavingMesh, PeerCloseInitiator.LocalNode);
|
|
}
|
|
|
|
public IAsyncResult BeginOpenNeighbor(PeerNodeAddress remoteAddress, TimeSpan timeout, AsyncCallback callback, object asyncState)
|
|
{
|
|
ThrowIfNotOpen();
|
|
|
|
// It's okay if neighbor manager is shutdown and closed after the above check
|
|
// because the new neighbor is only added to neighborList in NeighborOpened
|
|
// handler if the neighbor manager is still open.
|
|
|
|
// Sort the IP addresses
|
|
ReadOnlyCollection<IPAddress> sortedAddresses = this.ipHelper.SortAddresses(remoteAddress.IPAddresses);
|
|
PeerNodeAddress address = new PeerNodeAddress(remoteAddress.EndpointAddress, sortedAddresses);
|
|
return BeginOpenNeighborInternal(address, timeout, callback, asyncState);
|
|
}
|
|
|
|
internal IAsyncResult BeginOpenNeighborInternal(PeerNodeAddress remoteAddress, TimeSpan timeout, AsyncCallback callback, object asyncState)
|
|
{
|
|
PeerNeighbor neighbor = new PeerNeighbor(this.config, this.messageHandler);
|
|
RegisterForNeighborEvents(neighbor);
|
|
|
|
return new NeighborOpenAsyncResult(neighbor, remoteAddress, this.serviceBinding, this.service,
|
|
new ClosedCallback(Closed), timeout, callback, asyncState);
|
|
}
|
|
|
|
// Cleanup after shutdown
|
|
void Cleanup(bool graceful)
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
// In case of g----ful shutdown, we wait for neighbor list to become empty. connectedNeighborList should become
|
|
// empty as well.
|
|
if (graceful)
|
|
{
|
|
Fx.Assert(this.neighborList.Count == 0, "neighbor count should be 0");
|
|
Fx.Assert(this.connectedNeighborList.Count == 0, "Connected neighbor count should be 0");
|
|
|
|
// shutdownEvent is only relevant for a g----ful close. And should be closed by the thread
|
|
// performing g----ful close
|
|
if (this.shutdownEvent != null)
|
|
this.shutdownEvent.Close();
|
|
}
|
|
this.state = State.Shutdown;
|
|
}
|
|
}
|
|
|
|
// To clear the neighbor lists in case of unexpected exceptions during shutdown
|
|
void ClearNeighborList()
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
this.neighborList.Clear();
|
|
this.connectedNeighborList.Clear();
|
|
}
|
|
}
|
|
|
|
// Close the neighbor manager. It should be called after Shutdown().
|
|
// Can also be called before Open.
|
|
public void Close()
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
this.state = State.Closed;
|
|
}
|
|
}
|
|
|
|
// Returns true if neighbor manager is closing or closed
|
|
bool Closed()
|
|
{
|
|
return this.state != State.Opened;
|
|
}
|
|
|
|
//
|
|
// Close the specified neighbor. Ok to call multiple times, but NeighborClosing
|
|
// and NeighborClosed events are fired just once.
|
|
// If the closeReason specified is InvalidNeighbor, it will be closed ungracefully
|
|
//
|
|
public void CloseNeighbor(IPeerNeighbor neighbor, PeerCloseReason closeReason,
|
|
PeerCloseInitiator closeInitiator)
|
|
{
|
|
CloseNeighbor(neighbor, closeReason, closeInitiator, null);
|
|
}
|
|
|
|
public void CloseNeighbor(IPeerNeighbor neighbor, PeerCloseReason closeReason,
|
|
PeerCloseInitiator closeInitiator, Exception closeException)
|
|
{
|
|
PeerNeighbor nbr = (PeerNeighbor)neighbor;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
if (!(this.state != State.Created))
|
|
{
|
|
throw Fx.AssertAndThrow("Neighbor Manager is not expected to be in Created state");
|
|
}
|
|
|
|
// Check that the neighbor is known to neighbor manager
|
|
if (!this.neighborList.Contains(nbr))
|
|
return;
|
|
}
|
|
|
|
// initiate closing of the neighbor
|
|
if (closeReason != PeerCloseReason.InvalidNeighbor)
|
|
{
|
|
if (!nbr.IsClosing)
|
|
InvokeAsyncNeighborClose(nbr, closeReason, closeInitiator, closeException, null);
|
|
}
|
|
else // Call abort even if neighbor is already closing
|
|
{
|
|
nbr.Abort(closeReason, closeInitiator);
|
|
}
|
|
}
|
|
|
|
public IPeerNeighbor EndOpenNeighbor(IAsyncResult result)
|
|
{
|
|
return NeighborOpenAsyncResult.End(result);
|
|
}
|
|
|
|
static void FireEvent(EventHandler handler, PeerNeighborManager manager)
|
|
{
|
|
if (handler != null)
|
|
handler(manager, EventArgs.Empty);
|
|
}
|
|
|
|
static void FireEvent(EventHandler handler, PeerNeighbor neighbor)
|
|
{
|
|
if (handler != null)
|
|
handler(neighbor, EventArgs.Empty);
|
|
}
|
|
|
|
static void FireEvent(EventHandler<PeerNeighborCloseEventArgs> handler,
|
|
PeerNeighbor neighbor, PeerCloseReason closeReason,
|
|
PeerCloseInitiator closeInitiator, Exception closeException)
|
|
{
|
|
if (handler != null)
|
|
{
|
|
PeerNeighborCloseEventArgs args = new PeerNeighborCloseEventArgs(
|
|
closeReason, closeInitiator, closeException);
|
|
handler(neighbor, args);
|
|
}
|
|
}
|
|
|
|
// Find a duplicate neighbor matching the nodeId
|
|
public IPeerNeighbor FindDuplicateNeighbor(ulong nodeId)
|
|
{
|
|
return FindDuplicateNeighbor(nodeId, null);
|
|
}
|
|
|
|
// Find a duplicate neighbor (excluding skipNeighbor) matching the nodeID.
|
|
public IPeerNeighbor FindDuplicateNeighbor(ulong nodeId, IPeerNeighbor skipNeighbor)
|
|
{
|
|
PeerNeighbor duplicateNeighbor = null;
|
|
|
|
if (nodeId != PeerTransportConstants.InvalidNodeId)
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
foreach (PeerNeighbor neighbor in this.neighborList)
|
|
{
|
|
// We restrict search to neighbors that are not yet closing.
|
|
if (neighbor != (PeerNeighbor)skipNeighbor && neighbor.NodeId == nodeId &&
|
|
!neighbor.IsClosing &&
|
|
neighbor.State < PeerNeighborState.Disconnecting)
|
|
{
|
|
duplicateNeighbor = neighbor;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return duplicateNeighbor;
|
|
}
|
|
|
|
public bool PingNeighbor(IPeerNeighbor peer)
|
|
{
|
|
bool result = true;
|
|
Message message = Message.CreateMessage(MessageVersion.Soap12WSAddressing10, PeerStrings.PingAction);
|
|
try
|
|
{
|
|
peer.Ping(message);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
DiagnosticUtility.TraceHandledException(e, TraceEventType.Information);
|
|
peer.Abort(PeerCloseReason.InternalFailure, PeerCloseInitiator.LocalNode);
|
|
result = false;
|
|
}
|
|
return result;
|
|
|
|
}
|
|
|
|
public void PingNeighbors()
|
|
{
|
|
List<IPeerNeighbor> neighbors = GetConnectedNeighbors();
|
|
foreach (IPeerNeighbor neighbor in neighbors)
|
|
{
|
|
PingNeighbor(neighbor);
|
|
}
|
|
}
|
|
|
|
// Find a duplicate neighbor matching the address
|
|
public IPeerNeighbor FindDuplicateNeighbor(PeerNodeAddress address)
|
|
{
|
|
return FindDuplicateNeighbor(address, null);
|
|
}
|
|
|
|
// Find a duplicate neighbor (excluding skipNeighbor) matching the address.
|
|
public IPeerNeighbor FindDuplicateNeighbor(PeerNodeAddress address, IPeerNeighbor skipNeighbor)
|
|
{
|
|
PeerNeighbor duplicateNeighbor = null;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
foreach (PeerNeighbor neighbor in this.neighborList)
|
|
{
|
|
// We restrict search to neighbors that are not yet closing.
|
|
if (neighbor != (PeerNeighbor)skipNeighbor &&
|
|
neighbor.ListenAddress != null &&
|
|
neighbor.ListenAddress.ServicePath == address.ServicePath &&
|
|
!neighbor.IsClosing &&
|
|
neighbor.State < PeerNeighborState.Disconnecting)
|
|
{
|
|
duplicateNeighbor = neighbor;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return duplicateNeighbor;
|
|
}
|
|
|
|
// Returns a copy of the list of connected neighbors.
|
|
public List<IPeerNeighbor> GetConnectedNeighbors()
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
return new List<IPeerNeighbor>(this.connectedNeighborList);
|
|
}
|
|
}
|
|
|
|
// Used to retrieve a neighbor given the proxy.
|
|
// Maps the proxy from the incoming message's service context to a neighbor instance.
|
|
public IPeerNeighbor GetNeighborFromProxy(IPeerProxy proxy)
|
|
{
|
|
PeerNeighbor neighbor = null;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
if (state == State.Opened)
|
|
{
|
|
// Find the neighbor containing the specified proxy.
|
|
foreach (PeerNeighbor nbr in this.neighborList)
|
|
{
|
|
if (nbr.Proxy == proxy)
|
|
{
|
|
neighbor = nbr;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return neighbor;
|
|
}
|
|
|
|
// Calls neighbor.BeginClose or EndClose and catches appropriate exceptions for any cleanup.
|
|
// We use a single method for both BeginClose and EndClose processing since exception handling
|
|
// is very similar in both cases.
|
|
void InvokeAsyncNeighborClose(PeerNeighbor neighbor, PeerCloseReason closeReason,
|
|
PeerCloseInitiator closeInitiator, Exception closeException, IAsyncResult endResult)
|
|
{
|
|
// initiate invoking BeginClose or EndClose
|
|
try
|
|
{
|
|
if (endResult == null)
|
|
{
|
|
IAsyncResult beginResult = neighbor.BeginClose(closeReason, closeInitiator,
|
|
closeException, Fx.ThunkCallback(new AsyncCallback(OnNeighborClosedCallback)), neighbor);
|
|
if (beginResult.CompletedSynchronously)
|
|
neighbor.EndClose(beginResult);
|
|
}
|
|
else
|
|
{
|
|
neighbor.EndClose(endResult);
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
DiagnosticUtility.TraceHandledException(e, TraceEventType.Information);
|
|
|
|
neighbor.TraceEventHelper(TraceEventType.Warning, TraceCode.PeerNeighborCloseFailed, SR.GetString(SR.TraceCodePeerNeighborCloseFailed), e);
|
|
// May get InvalidOperationException or ObjectDisposedException due to simultaneous close from both sides (and autoclose is enabled)
|
|
if (e is InvalidOperationException || e is CommunicationException || e is TimeoutException)
|
|
{
|
|
neighbor.Abort();
|
|
}
|
|
else
|
|
{
|
|
throw;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Handler for processing neighbor closed event.
|
|
//
|
|
// We should allow this event to be processed even if the neighbor manager is shutting
|
|
// down because neighbor manager will be waiting on the shutdown event which is set in
|
|
// this handler once the last neighbor's close event is processed.
|
|
//
|
|
void OnNeighborClosed(object source, EventArgs args)
|
|
{
|
|
RemoveNeighbor((PeerNeighbor)source);
|
|
}
|
|
|
|
// Callback that is invoked when BeginClose completes.
|
|
void OnNeighborClosedCallback(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
// Call neighbor.EndClose -- PeerCloseReason and PeerCloseInitiator are dummy values
|
|
InvokeAsyncNeighborClose((PeerNeighbor)result.AsyncState, PeerCloseReason.None,
|
|
PeerCloseInitiator.LocalNode, null, result);
|
|
}
|
|
}
|
|
|
|
// Handles neighbor disconnecting or disconnected events
|
|
void OnNeighborClosing(object source, EventArgs args)
|
|
{
|
|
//
|
|
// Remove the neighbor from connected list. But closed and offline events are
|
|
// fired upon processing closed event. If, due to thread scheduling issues,
|
|
// closed handler executes before this handler, it will have already done the
|
|
// work and Remove() below is a NOP.
|
|
//
|
|
lock (ThisLock)
|
|
{
|
|
this.connectedNeighborList.Remove((IPeerNeighbor)source);
|
|
}
|
|
}
|
|
|
|
// handler to process neighbor connected event
|
|
void OnNeighborConnected(object source, EventArgs args)
|
|
{
|
|
PeerNeighbor neighbor = (PeerNeighbor)source;
|
|
bool fireConnected = false;
|
|
bool fireOnline = false;
|
|
|
|
// we may get this event after the neighbor has been closed. So, we check to see if
|
|
// the neighbor exists in the neighbor list before processing the event.
|
|
lock (ThisLock)
|
|
{
|
|
if (this.neighborList.Contains(neighbor))
|
|
{
|
|
fireConnected = true;
|
|
|
|
// Add the neighbor to connected list and determine if online should be fired
|
|
this.connectedNeighborList.Add(neighbor);
|
|
if (!this.isOnline)
|
|
{
|
|
// Fire online event
|
|
this.isOnline = true;
|
|
fireOnline = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fireConnected)
|
|
{
|
|
FireEvent(NeighborConnected, neighbor);
|
|
}
|
|
else
|
|
neighbor.TraceEventHelper(TraceEventType.Warning, TraceCode.PeerNeighborNotFound, SR.GetString(SR.TraceCodePeerNeighborNotFound));
|
|
|
|
if (fireOnline)
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceInformation)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Information, TraceCode.PeerNeighborManagerOnline,
|
|
SR.GetString(SR.TraceCodePeerNeighborManagerOnline), this.traceRecord, this, null);
|
|
}
|
|
FireEvent(Online, this);
|
|
}
|
|
}
|
|
|
|
// handler to process neighbor opened event
|
|
void OnNeighborOpened(object source, EventArgs args)
|
|
{
|
|
PeerNeighbor neighbor = (PeerNeighbor)source;
|
|
bool accept = false;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
// Add the neighbor to neighborList if neighbor manager is still open
|
|
if (this.state == State.Opened)
|
|
{
|
|
// StateManager assures that neighbor Opened and Closed events are
|
|
// serialized. So, we should never have to process a closed event handler
|
|
// before opened is complete.
|
|
if (!(neighbor.State == PeerNeighborState.Opened))
|
|
{
|
|
throw Fx.AssertAndThrow("Neighbor expected to be in Opened state");
|
|
}
|
|
this.neighborList.Add(neighbor);
|
|
accept = true;
|
|
}
|
|
}
|
|
if (accept)
|
|
{
|
|
FireEvent(NeighborOpened, neighbor);
|
|
}
|
|
else // close the neighbor ungracefully
|
|
{
|
|
neighbor.Abort();
|
|
neighbor.TraceEventHelper(TraceEventType.Warning, TraceCode.PeerNeighborNotAccepted, SR.GetString(SR.TraceCodePeerNeighborNotAccepted));
|
|
}
|
|
}
|
|
|
|
// Opens the neighbor manager. When this method returns the neighbor manager is ready
|
|
// to accept incoming neighbor requests and to establish outgoing neighbors.
|
|
public void Open(Binding serviceBinding, PeerService service)
|
|
{
|
|
Fx.Assert(serviceBinding != null, "serviceBinding must not be null");
|
|
Fx.Assert(service != null, "service must not be null");
|
|
|
|
lock (ThisLock)
|
|
{
|
|
this.service = service;
|
|
this.serviceBinding = serviceBinding;
|
|
if (!(this.state == State.Created))
|
|
{
|
|
throw Fx.AssertAndThrow("Neighbor Manager is expected to be in Created state");
|
|
}
|
|
this.state = State.Opened;
|
|
}
|
|
}
|
|
|
|
// Process an inbound channel
|
|
public bool ProcessIncomingChannel(IClientChannel channel)
|
|
{
|
|
bool accepted = false;
|
|
IPeerProxy proxy = (IPeerProxy)channel;
|
|
|
|
Fx.Assert(GetNeighborFromProxy(proxy) == null, "Channel should not map to an existing neighbor");
|
|
if (this.state == State.Opened)
|
|
{
|
|
// It is okay if neighbor manager is closed after the above check because the
|
|
// new neighbor is only added to neighborList in neighbor Opened handler if the
|
|
// neighbor manager is still open.
|
|
PeerNeighbor neighbor = new PeerNeighbor(this.config, this.messageHandler);
|
|
RegisterForNeighborEvents(neighbor);
|
|
neighbor.Open(proxy);
|
|
accepted = true;
|
|
}
|
|
else
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceWarning)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Warning, TraceCode.PeerNeighborNotAccepted,
|
|
SR.GetString(SR.TraceCodePeerNeighborNotAccepted), this.traceRecord, this, null);
|
|
}
|
|
}
|
|
|
|
return accepted;
|
|
}
|
|
|
|
void RegisterForNeighborEvents(PeerNeighbor neighbor)
|
|
{
|
|
neighbor.Opened += OnNeighborOpened;
|
|
neighbor.Connected += OnNeighborConnected;
|
|
neighbor.Closed += OnNeighborClosed;
|
|
|
|
// We want the neighbor to call Closing handlers directly, so we delegate
|
|
neighbor.Closing += this.NeighborClosing;
|
|
|
|
// Disconnecting and Disconnected are treated the same way
|
|
neighbor.Disconnecting += OnNeighborClosing;
|
|
neighbor.Disconnected += OnNeighborClosing;
|
|
}
|
|
|
|
// Remove neighbor from the list and fire relevant events
|
|
void RemoveNeighbor(PeerNeighbor neighbor)
|
|
{
|
|
bool fireClosed = false;
|
|
bool fireOffline = false;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
if (this.neighborList.Contains(neighbor))
|
|
{
|
|
fireClosed = true;
|
|
|
|
// Remove neighbor from our lists and determine if offline should be fired.
|
|
this.neighborList.Remove(neighbor);
|
|
this.connectedNeighborList.Remove(neighbor);
|
|
if (this.isOnline && this.connectedNeighborList.Count == 0)
|
|
{
|
|
this.isOnline = false;
|
|
fireOffline = true;
|
|
}
|
|
// If in the process of shutting down neighbor manager, signal completion
|
|
// upon closing of the last remaining neighbor
|
|
if (this.neighborList.Count == 0 && this.shutdownEvent != null)
|
|
{
|
|
this.shutdownEvent.Set();
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fire events
|
|
if (fireClosed)
|
|
{
|
|
FireEvent(NeighborClosed, neighbor, neighbor.CloseReason,
|
|
neighbor.CloseInitiator, neighbor.CloseException);
|
|
}
|
|
else
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceWarning)
|
|
{
|
|
neighbor.TraceEventHelper(TraceEventType.Warning, TraceCode.PeerNeighborNotFound, SR.GetString(SR.TraceCodePeerNeighborNotFound));
|
|
}
|
|
}
|
|
if (fireOffline)
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceInformation)
|
|
{
|
|
TraceUtility.TraceEvent(TraceEventType.Information, TraceCode.PeerNeighborManagerOffline,
|
|
SR.GetString(SR.TraceCodePeerNeighborManagerOffline), this.traceRecord, this, null);
|
|
}
|
|
FireEvent(Offline, this);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Shutdown the neighbor manager. Shutdown should be called prior to Close(). It stops
|
|
// processing inbound neighbor sessions and closes all the neighbors. Outbound neighbor
|
|
// sessions are also disabled as a result of setting the state to ShuttingDown
|
|
// (and then Shutdown).
|
|
//
|
|
public void Shutdown(bool graceful, TimeSpan timeout)
|
|
{
|
|
PeerNeighbor[] neighbors = null;
|
|
TimeoutHelper timeoutHelper = new TimeoutHelper(timeout);
|
|
|
|
try
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
if (this.state == State.Shutdown || this.state == State.Closed)
|
|
return;
|
|
this.state = State.ShuttingDown;
|
|
|
|
// Create a copy of neighbor list in order to close neighbors outside lock.
|
|
neighbors = this.neighborList.ToArray();
|
|
|
|
// In case of g----ful shutdown, if there are any neighbors to close, create an event
|
|
// to wait until they are closed
|
|
if (graceful && neighbors.Length > 0)
|
|
this.shutdownEvent = new ManualResetEvent(false);
|
|
}
|
|
|
|
// Close each neighbor. Do this outside the lock due to Closing and Closed event handlers being invoked
|
|
if (graceful)
|
|
Shutdown(neighbors, timeoutHelper.RemainingTime());
|
|
else
|
|
Abort(neighbors);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
// Purge neighbor list in case of unexpected exceptions
|
|
if (Fx.IsFatal(e)) throw;
|
|
try
|
|
{
|
|
ClearNeighborList();
|
|
}
|
|
catch (Exception ee)
|
|
{
|
|
if (Fx.IsFatal(ee)) throw;
|
|
DiagnosticUtility.TraceHandledException(ee, TraceEventType.Information);
|
|
}
|
|
throw;
|
|
}
|
|
finally
|
|
{
|
|
Cleanup(graceful);
|
|
}
|
|
}
|
|
|
|
// G----ful shutdown
|
|
void Shutdown(PeerNeighbor[] neighbors, TimeSpan timeout)
|
|
{
|
|
TimeoutHelper timeoutHelper = new TimeoutHelper(timeout);
|
|
foreach (PeerNeighbor neighbor in neighbors)
|
|
CloseNeighbor(neighbor, PeerCloseReason.LeavingMesh, PeerCloseInitiator.LocalNode, null);
|
|
|
|
// Wait for all the neighbors to close (the event object is set when the last
|
|
// neighbor is closed). Specify a timeout for wait event handle in case event.Set
|
|
// fails for some reason (it doesn't throw exception). Bail out of the loop when
|
|
// the neighbor count reaches 0. This ensures that Shutdown() doesn't hang.
|
|
if (neighbors.Length > 0)
|
|
{
|
|
if (!TimeoutHelper.WaitOne(this.shutdownEvent, timeoutHelper.RemainingTime()))
|
|
{
|
|
Abort(neighbors); // abort neighbors that haven't been closed yet
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new TimeoutException());
|
|
}
|
|
}
|
|
}
|
|
|
|
void ThrowIfNotOpen()
|
|
{
|
|
if (!(this.state != State.Created))
|
|
{
|
|
throw Fx.AssertAndThrow("Neighbor manager not expected to be in Created state");
|
|
}
|
|
if (Closed())
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ObjectDisposedException(this.ToString()));
|
|
}
|
|
}
|
|
|
|
class PeerNeighbor : IPeerNeighbor, IInputSessionShutdown
|
|
{
|
|
public event EventHandler Closed;
|
|
public event EventHandler<PeerNeighborCloseEventArgs> Closing;
|
|
public event EventHandler Connected;
|
|
public event EventHandler Disconnected;
|
|
public event EventHandler Disconnecting;
|
|
public event EventHandler Opened;
|
|
|
|
ChannelFactory<IPeerProxy> channelFactory;
|
|
|
|
// Used after closing the neighbor to find details of the close reason and who
|
|
// initiated closing.
|
|
Exception closeException;
|
|
PeerCloseInitiator closeInitiator;
|
|
PeerCloseReason closeReason;
|
|
|
|
PeerNodeConfig config;
|
|
IPAddress connectIPAddress; // relevant for initiator neighbor. Indicates the IP address used for connection
|
|
ExtensionCollection<IPeerNeighbor> extensions;
|
|
bool initiator;
|
|
bool isClosing; // If true, the neighbor is being closed or already closed
|
|
PeerNodeAddress listenAddress; // The address that the remote endpoint is listening on
|
|
ulong nodeId; // The nodeID of the remote endpoint
|
|
IPeerProxy proxy; // Proxy channel to talk to the remote endpoint
|
|
IClientChannel proxyChannel; // To access inner Channel from proxy w/o casting
|
|
PeerNeighborState state; // Current state of the neighbor
|
|
object thisLock = new object();
|
|
IPeerNodeMessageHandling messageHandler;
|
|
UtilityExtension utility;
|
|
|
|
// Dictates if attempt to set neighbor state should throw exception on failure.
|
|
enum SetStateBehavior
|
|
{
|
|
ThrowException,
|
|
TrySet,
|
|
}
|
|
|
|
public PeerNeighbor(PeerNodeConfig config,
|
|
IPeerNodeMessageHandling messageHandler)
|
|
{
|
|
this.closeReason = PeerCloseReason.None;
|
|
this.closeInitiator = PeerCloseInitiator.LocalNode;
|
|
this.config = config;
|
|
this.state = PeerNeighborState.Created;
|
|
this.extensions = new ExtensionCollection<IPeerNeighbor>(this, thisLock);
|
|
this.messageHandler = messageHandler;
|
|
}
|
|
|
|
public IPAddress ConnectIPAddress
|
|
{
|
|
get
|
|
{
|
|
return this.connectIPAddress;
|
|
}
|
|
set
|
|
{
|
|
this.connectIPAddress = value;
|
|
}
|
|
}
|
|
|
|
// To retrieve reason for closing the neighbor
|
|
public PeerCloseReason CloseReason
|
|
{
|
|
get
|
|
{
|
|
return this.closeReason;
|
|
}
|
|
}
|
|
|
|
// Indicates if close was initiated by local or remote node
|
|
public PeerCloseInitiator CloseInitiator
|
|
{
|
|
get
|
|
{
|
|
return this.closeInitiator;
|
|
}
|
|
}
|
|
|
|
// If an exception during processing caused the neighbor to be closed
|
|
public Exception CloseException
|
|
{
|
|
get
|
|
{
|
|
return this.closeException;
|
|
}
|
|
}
|
|
|
|
public IExtensionCollection<IPeerNeighbor> Extensions
|
|
{
|
|
get
|
|
{
|
|
return extensions;
|
|
}
|
|
}
|
|
|
|
// Returns true if the neighbor is currently closing or already closed
|
|
public bool IsClosing
|
|
{
|
|
get
|
|
{
|
|
return isClosing;
|
|
}
|
|
}
|
|
|
|
// Returns true if neighbor is in connected, synchronizing, or synchronized states
|
|
public bool IsConnected
|
|
{
|
|
get
|
|
{
|
|
return PeerNeighborStateHelper.IsConnected(this.state);
|
|
}
|
|
}
|
|
|
|
// NOTE: If the property is accessed before the neighbor transitions to connected
|
|
// state, the returned listen address may be null for the accepting neighbor.
|
|
public PeerNodeAddress ListenAddress
|
|
{
|
|
get
|
|
{
|
|
// Return a copy since the scope ID is settable
|
|
PeerNodeAddress address = this.listenAddress;
|
|
if (address != null)
|
|
return new PeerNodeAddress(address.EndpointAddress, PeerIPHelper.CloneAddresses(address.IPAddresses, true));
|
|
else
|
|
return address;
|
|
}
|
|
set
|
|
{
|
|
|
|
lock (ThisLock)
|
|
{
|
|
if (!(!this.initiator))
|
|
{
|
|
throw Fx.AssertAndThrow("Cannot be set for initiator neighbors");
|
|
}
|
|
ThrowIfClosed();
|
|
|
|
if (value != null)
|
|
{
|
|
this.listenAddress = value;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Returns true if the neighbor is an initiator
|
|
public bool IsInitiator
|
|
{
|
|
get
|
|
{
|
|
return this.initiator;
|
|
}
|
|
}
|
|
|
|
// Returns the node ID of the neighbor. If this property is accessed before the
|
|
// neighbor transitions to connected state, the returned node ID may be 0.
|
|
public ulong NodeId
|
|
{
|
|
get
|
|
{
|
|
return this.nodeId;
|
|
}
|
|
|
|
set
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
ThrowIfClosed();
|
|
this.nodeId = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Returns the proxy for the neighbor (i.e., the channel that SFx maintains to the
|
|
// remote node associated with this neighbor instance).
|
|
public IPeerProxy Proxy
|
|
{
|
|
get
|
|
{
|
|
return this.proxy;
|
|
}
|
|
set
|
|
{
|
|
this.proxy = value;
|
|
this.proxyChannel = (IClientChannel)this.proxy;
|
|
RegisterForChannelEvents();
|
|
}
|
|
}
|
|
|
|
// The only states that are settable are connecting, connected, synchronizing,
|
|
// synchronized, disconnecting, and disconnected.
|
|
public PeerNeighborState State
|
|
{
|
|
get
|
|
{
|
|
return this.state;
|
|
}
|
|
|
|
set
|
|
{
|
|
if (!(PeerNeighborStateHelper.IsSettable(value)))
|
|
{
|
|
throw Fx.AssertAndThrow("A valid settable state is expected");
|
|
}
|
|
SetState(value, SetStateBehavior.ThrowException);
|
|
}
|
|
}
|
|
|
|
object ThisLock
|
|
{
|
|
get
|
|
{
|
|
return this.thisLock;
|
|
}
|
|
}
|
|
|
|
// NOTE: Closing handlers not invoked when a neighbor is aborted; but Closed handlers are.
|
|
public void Abort(PeerCloseReason reason, PeerCloseInitiator closeInit)
|
|
{
|
|
lock (ThisLock)
|
|
{
|
|
// Set close reason etc. if they are not already set.
|
|
if (!this.isClosing)
|
|
{
|
|
this.isClosing = true;
|
|
this.closeReason = reason;
|
|
this.closeInitiator = closeInit;
|
|
}
|
|
}
|
|
Abort();
|
|
}
|
|
|
|
public void Abort()
|
|
{
|
|
if (this.channelFactory != null)
|
|
this.channelFactory.Abort();
|
|
else
|
|
this.proxyChannel.Abort();
|
|
}
|
|
|
|
// Close a neighbor gracefully
|
|
public IAsyncResult BeginClose(PeerCloseReason reason,
|
|
PeerCloseInitiator closeInit, Exception exception,
|
|
AsyncCallback callback, object asyncState)
|
|
{
|
|
bool callClosing = false;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
// Set close reason etc. if they are not already set.
|
|
if (!this.isClosing)
|
|
{
|
|
callClosing = true;
|
|
this.isClosing = true;
|
|
this.closeReason = reason;
|
|
this.closeInitiator = closeInit;
|
|
this.closeException = exception;
|
|
}
|
|
}
|
|
|
|
// Initiate close, if another thread has not already done so....
|
|
// NOTE: NeighborClosing handlers should not throw any catchable exceptions.
|
|
if (callClosing)
|
|
{
|
|
EventHandler<PeerNeighborCloseEventArgs> handler = this.Closing;
|
|
if (handler != null)
|
|
{
|
|
try
|
|
{
|
|
PeerNeighborCloseEventArgs args = new PeerNeighborCloseEventArgs(
|
|
reason, closeInitiator, exception);
|
|
handler(this, args);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
Abort();
|
|
throw;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.channelFactory != null)
|
|
return this.channelFactory.BeginClose(callback, asyncState);
|
|
else
|
|
return this.proxyChannel.BeginClose(callback, asyncState);
|
|
}
|
|
|
|
// Begin opening of a neighbor channel to 'to'. instanceContext is where the remote
|
|
// endpoint should send messages to (it will be a reference to PeerNeighborManager).
|
|
public IAsyncResult BeginOpen(PeerNodeAddress remoteAddress, Binding binding,
|
|
PeerService service, ClosedCallback closedCallback, TimeSpan timeout,
|
|
AsyncCallback callback, object asyncState)
|
|
{
|
|
this.initiator = true;
|
|
this.listenAddress = remoteAddress;
|
|
OpenAsyncResult result = new OpenAsyncResult(this, remoteAddress, binding, service,
|
|
closedCallback, timeout, callback, state);
|
|
return result;
|
|
}
|
|
|
|
// Called by OpenAsyncResult
|
|
public IAsyncResult BeginOpenProxy(EndpointAddress remoteAddress, Binding binding,
|
|
InstanceContext instanceContext, TimeSpan timeout, AsyncCallback callback, object state)
|
|
{
|
|
TimeoutHelper timeoutHelper = new TimeoutHelper(timeout);
|
|
if (this.channelFactory != null)
|
|
Abort(); // to close previously created factory, if any
|
|
|
|
EndpointAddressBuilder meshEprBuilder = new EndpointAddressBuilder(remoteAddress);
|
|
meshEprBuilder.Uri = config.GetMeshUri();
|
|
this.channelFactory = new DuplexChannelFactory<IPeerProxy>(instanceContext, binding, meshEprBuilder.ToEndpointAddress());
|
|
this.channelFactory.Endpoint.Behaviors.Add(new ClientViaBehavior(remoteAddress.Uri));
|
|
this.channelFactory.Endpoint.Behaviors.Add(new PeerNeighborBehavior(this));
|
|
this.channelFactory.Endpoint.Contract.Behaviors.Add(new PeerOperationSelectorBehavior(this.messageHandler));
|
|
this.config.SecurityManager.ApplyClientSecurity(channelFactory);
|
|
this.channelFactory.Open(timeoutHelper.RemainingTime());
|
|
this.Proxy = this.channelFactory.CreateChannel();
|
|
|
|
IAsyncResult result = this.proxyChannel.BeginOpen(timeoutHelper.RemainingTime(), callback, state);
|
|
if (result.CompletedSynchronously)
|
|
this.proxyChannel.EndOpen(result);
|
|
|
|
return result;
|
|
}
|
|
|
|
public IAsyncResult BeginSend(Message message,
|
|
AsyncCallback callback, object asyncState)
|
|
{
|
|
return this.proxy.BeginSend(message, callback, asyncState);
|
|
}
|
|
|
|
public IAsyncResult BeginSend(Message message,
|
|
TimeSpan timeout, AsyncCallback callback, object asyncState)
|
|
{
|
|
return this.proxy.BeginSend(message, timeout, callback, asyncState);
|
|
}
|
|
|
|
public void Send(Message message)
|
|
{
|
|
this.proxy.Send(message);
|
|
}
|
|
|
|
// Called to Abort channelFactory in case BeginOpenProxy or EndOpenProxy throw
|
|
public void CleanupProxy()
|
|
{
|
|
this.channelFactory.Abort();
|
|
}
|
|
|
|
public void EndClose(IAsyncResult result)
|
|
{
|
|
if (this.channelFactory != null)
|
|
this.channelFactory.EndClose(result);
|
|
else
|
|
this.proxyChannel.EndClose(result);
|
|
}
|
|
|
|
public void EndOpen(IAsyncResult result)
|
|
{
|
|
OpenAsyncResult.End(result);
|
|
}
|
|
|
|
// Called by OpenAsyncResult
|
|
public void EndOpenProxy(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
this.proxyChannel.EndOpen(result);
|
|
}
|
|
|
|
public void EndSend(IAsyncResult result)
|
|
{
|
|
this.proxy.EndSend(result);
|
|
}
|
|
|
|
|
|
public Message RequestSecurityToken(Message request)
|
|
{
|
|
return this.proxy.ProcessRequestSecurityToken(request);
|
|
}
|
|
|
|
public void Ping(Message request)
|
|
{
|
|
this.proxy.Ping(request);
|
|
}
|
|
|
|
// Service channel closed event handler.
|
|
void OnChannelClosed(object source, EventArgs args)
|
|
{
|
|
if (this.state < PeerNeighborState.Closed)
|
|
OnChannelClosedOrFaulted(PeerCloseReason.Closed);
|
|
|
|
// If the other side closed the channel, abort the factory (if one exists)
|
|
if (this.closeInitiator != PeerCloseInitiator.LocalNode && this.channelFactory != null)
|
|
this.channelFactory.Abort();
|
|
}
|
|
|
|
// Does heavy-lifting of processing closed/faulted events
|
|
void OnChannelClosedOrFaulted(PeerCloseReason reason)
|
|
{
|
|
PeerNeighborState oldState;
|
|
|
|
lock (ThisLock)
|
|
{
|
|
// We don't call SetState here because it should not be called inside lock,
|
|
// and to avoid race conditions, we need to set the state before the lock
|
|
// can be released.
|
|
oldState = this.state;
|
|
this.state = PeerNeighborState.Closed;
|
|
|
|
// Set close reason etc. if they are not already set (as a result of local
|
|
// node initiating Close)
|
|
if (!this.isClosing)
|
|
{
|
|
this.isClosing = true;
|
|
this.closeReason = reason;
|
|
this.closeInitiator = PeerCloseInitiator.RemoteNode;
|
|
}
|
|
TraceClosedEvent(oldState);
|
|
}
|
|
|
|
// Update traces and counters and notify interested parties
|
|
OnStateChanged(PeerNeighborState.Closed);
|
|
}
|
|
|
|
// Service channel faulted event handler.
|
|
void OnChannelFaulted(object source, EventArgs args)
|
|
{
|
|
try
|
|
{
|
|
OnChannelClosedOrFaulted(PeerCloseReason.Faulted);
|
|
}
|
|
finally
|
|
{
|
|
Abort();
|
|
}
|
|
}
|
|
|
|
// Service channel opened event handler.
|
|
void OnChannelOpened(object source, EventArgs args)
|
|
{
|
|
// TrySetState is not used because it asserts for a settable state
|
|
// and is meant for use by upper layers. Only PeerNeighbor can set
|
|
// the state to Opened. So, it calls SetState directly.
|
|
SetState(PeerNeighborState.Opened, SetStateBehavior.TrySet);
|
|
}
|
|
|
|
//
|
|
// Invokes the appropriate state changed event handler.
|
|
// WARNING: This method should not be called within lock.
|
|
//
|
|
void OnStateChanged(PeerNeighborState newState)
|
|
{
|
|
EventHandler handler = null;
|
|
switch (newState)
|
|
{
|
|
case PeerNeighborState.Opened:
|
|
handler = this.Opened;
|
|
break;
|
|
case PeerNeighborState.Closed:
|
|
handler = this.Closed;
|
|
break;
|
|
case PeerNeighborState.Connected:
|
|
handler = this.Connected;
|
|
break;
|
|
case PeerNeighborState.Disconnecting:
|
|
handler = this.Disconnecting;
|
|
break;
|
|
case PeerNeighborState.Disconnected:
|
|
handler = this.Disconnected;
|
|
break;
|
|
}
|
|
if (handler != null)
|
|
handler(this, EventArgs.Empty);
|
|
}
|
|
|
|
// Open an accepting (incoming) neighbor. callbackInstance is where msgs meant for
|
|
// remote endpoint should be sent.
|
|
public void Open(IPeerProxy callbackInstance)
|
|
{
|
|
this.initiator = false;
|
|
this.Proxy = callbackInstance;
|
|
}
|
|
|
|
// Register for channel events
|
|
void RegisterForChannelEvents()
|
|
{
|
|
this.state = PeerNeighborState.Created; // reset state if the previous proxy failed
|
|
this.proxyChannel.Opened += OnChannelOpened;
|
|
this.proxyChannel.Closed += OnChannelClosed;
|
|
this.proxyChannel.Faulted += OnChannelFaulted;
|
|
}
|
|
|
|
// WARNING: This method should not be called within the lock -- it may invoke state
|
|
// changed event handlers
|
|
bool SetState(PeerNeighborState newState, SetStateBehavior behavior)
|
|
{
|
|
bool stateChanged = false;
|
|
PeerNeighborState oldState;
|
|
|
|
// Attempt to set the state
|
|
lock (ThisLock)
|
|
{
|
|
oldState = this.State;
|
|
if (behavior == SetStateBehavior.ThrowException)
|
|
ThrowIfInvalidState(newState);
|
|
if (newState > this.state)
|
|
{
|
|
this.state = newState;
|
|
stateChanged = true;
|
|
if (DiagnosticUtility.ShouldTraceInformation)
|
|
{
|
|
TraceEventHelper(TraceEventType.Information, TraceCode.PeerNeighborStateChanged, SR.GetString(SR.TraceCodePeerNeighborStateChanged), null, null, newState, oldState);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceInformation)
|
|
{
|
|
TraceEventHelper(TraceEventType.Information, TraceCode.PeerNeighborStateChangeFailed, SR.GetString(SR.TraceCodePeerNeighborStateChangeFailed), null, null, oldState, newState);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (stateChanged)
|
|
{
|
|
// Pass state change notification on to interested subscribers.
|
|
OnStateChanged(newState);
|
|
}
|
|
|
|
return stateChanged;
|
|
}
|
|
|
|
// Attempts to set to specified state.
|
|
// Returns true if succeed and false otherwise.
|
|
public bool TrySetState(PeerNeighborState newState)
|
|
{
|
|
if (!(PeerNeighborStateHelper.IsSettable(newState)))
|
|
{
|
|
throw Fx.AssertAndThrow("A valid settable state is expected");
|
|
}
|
|
return SetState(newState, SetStateBehavior.TrySet);
|
|
}
|
|
|
|
public void ThrowIfClosed()
|
|
{
|
|
if (this.state == PeerNeighborState.Closed)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ObjectDisposedException(
|
|
this.ToString()));
|
|
}
|
|
}
|
|
|
|
// Throws if the new state being set on the neighbor is invalid compared to the
|
|
// current state (such as setting state to connecting when it is already in
|
|
// disconnected state). Also throws if neighbor is already closed.
|
|
// NOTE: This method should be called within the lock.
|
|
void ThrowIfInvalidState(PeerNeighborState newState)
|
|
{
|
|
if (this.state == PeerNeighborState.Closed)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ObjectDisposedException(
|
|
this.ToString()));
|
|
}
|
|
if (this.state >= newState)
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(
|
|
SR.GetString(SR.PeerNeighborInvalidState, this.state.ToString(),
|
|
newState.ToString())));
|
|
}
|
|
}
|
|
|
|
public void TraceClosedEvent(PeerNeighborState previousState)
|
|
{
|
|
if (DiagnosticUtility.ShouldTraceInformation)
|
|
{
|
|
TraceEventType severity = TraceEventType.Information;
|
|
|
|
// Override tracing severity based on close reason
|
|
switch (this.closeReason)
|
|
{
|
|
case PeerCloseReason.InvalidNeighbor:
|
|
case PeerCloseReason.DuplicateNodeId:
|
|
severity = TraceEventType.Error;
|
|
break;
|
|
|
|
case PeerCloseReason.ConnectTimedOut:
|
|
case PeerCloseReason.InternalFailure:
|
|
case PeerCloseReason.Faulted:
|
|
severity = TraceEventType.Warning;
|
|
break;
|
|
}
|
|
|
|
PeerNeighborCloseTraceRecord record = new PeerNeighborCloseTraceRecord(
|
|
this.nodeId, this.config.NodeId, null, null,
|
|
this.GetHashCode(), this.initiator,
|
|
PeerNeighborState.Closed.ToString(), previousState.ToString(), null,
|
|
this.closeInitiator.ToString(), this.closeReason.ToString()
|
|
);
|
|
|
|
TraceUtility.TraceEvent(severity, TraceCode.PeerNeighborStateChanged,
|
|
SR.GetString(SR.TraceCodePeerNeighborStateChanged), record, this, this.closeException);
|
|
}
|
|
}
|
|
|
|
public void TraceEventHelper(TraceEventType severity, int traceCode, string traceDescription)
|
|
{
|
|
PeerNeighborState nbrState = this.state;
|
|
this.TraceEventHelper(severity, traceCode, traceDescription, null, null, nbrState, nbrState);
|
|
}
|
|
|
|
public void TraceEventHelper(TraceEventType severity, int traceCode, string traceDescription, Exception e)
|
|
{
|
|
PeerNeighborState nbrState = this.state;
|
|
this.TraceEventHelper(severity, traceCode, traceDescription, e, null, nbrState, nbrState);
|
|
}
|
|
|
|
public void TraceEventHelper(TraceEventType severity, int traceCode, string traceDescription, Exception e,
|
|
string action, PeerNeighborState nbrState, PeerNeighborState previousOrAttemptedState)
|
|
{
|
|
if (DiagnosticUtility.ShouldTrace(severity))
|
|
{
|
|
string attemptedState = null;
|
|
string previousState = null;
|
|
PeerNodeAddress listenAddr = null;
|
|
IPAddress connectIPAddr = null;
|
|
|
|
if (nbrState >= PeerNeighborState.Opened && nbrState <= PeerNeighborState.Connected)
|
|
{
|
|
listenAddr = this.ListenAddress;
|
|
connectIPAddr = this.ConnectIPAddress;
|
|
}
|
|
|
|
if (traceCode == TraceCode.PeerNeighborStateChangeFailed)
|
|
attemptedState = previousOrAttemptedState.ToString();
|
|
else if (traceCode == TraceCode.PeerNeighborStateChanged)
|
|
previousState = previousOrAttemptedState.ToString();
|
|
|
|
PeerNeighborTraceRecord record = new PeerNeighborTraceRecord(this.nodeId,
|
|
this.config.NodeId, listenAddr, connectIPAddr, this.GetHashCode(),
|
|
this.initiator, nbrState.ToString(), previousState, attemptedState, action);
|
|
|
|
if (severity == TraceEventType.Verbose && e != null)
|
|
severity = TraceEventType.Information; // need to be >= info for exceptions
|
|
|
|
TraceUtility.TraceEvent(severity, traceCode, traceDescription, record, this, e);
|
|
}
|
|
}
|
|
|
|
// Helper class to implement PeerNeighbor's AsyncOpen by iterating over the IPAddress array
|
|
class OpenAsyncResult : AsyncResult
|
|
{
|
|
bool completedSynchronously;
|
|
ClosedCallback closed;
|
|
int currentIndex; // index into the ipAddress array
|
|
PeerNeighbor neighbor;
|
|
PeerNodeAddress remoteAddress;
|
|
Binding binding;
|
|
PeerService service;
|
|
AsyncCallback onOpen;
|
|
Exception lastException;
|
|
TimeoutHelper timeoutHelper;
|
|
|
|
public OpenAsyncResult(PeerNeighbor neighbor, PeerNodeAddress remoteAddress, Binding binding,
|
|
PeerService service, ClosedCallback closedCallback, TimeSpan timeout,
|
|
AsyncCallback callback, object state)
|
|
: base(callback, state)
|
|
{
|
|
Fx.Assert(remoteAddress != null && remoteAddress.IPAddresses.Count > 0, "Non-empty IPAddress collection expected");
|
|
|
|
this.timeoutHelper = new TimeoutHelper(timeout);
|
|
this.neighbor = neighbor;
|
|
this.currentIndex = 0;
|
|
this.completedSynchronously = true; // initially
|
|
this.remoteAddress = remoteAddress;
|
|
this.service = service;
|
|
this.binding = binding;
|
|
this.onOpen = Fx.ThunkCallback(new AsyncCallback(OnOpen));
|
|
this.closed = closedCallback;
|
|
BeginOpen();
|
|
}
|
|
|
|
void BeginOpen()
|
|
{
|
|
bool success = false;
|
|
|
|
try
|
|
{
|
|
while (this.currentIndex < this.remoteAddress.IPAddresses.Count)
|
|
{
|
|
EndpointAddress remoteAddress = PeerIPHelper.GetIPEndpointAddress(
|
|
this.remoteAddress.EndpointAddress, this.remoteAddress.IPAddresses[this.currentIndex]);
|
|
if (this.closed())
|
|
{
|
|
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ObjectDisposedException(this.GetType().ToString()));
|
|
}
|
|
try
|
|
{
|
|
this.neighbor.ConnectIPAddress = this.remoteAddress.IPAddresses[this.currentIndex];
|
|
IAsyncResult result = this.neighbor.BeginOpenProxy(remoteAddress, binding, new InstanceContext(null, service, false), this.timeoutHelper.RemainingTime(), onOpen, null);
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
return;
|
|
}
|
|
|
|
this.neighbor.EndOpenProxy(result);
|
|
this.lastException = null;
|
|
success = true;
|
|
neighbor.isClosing = false;
|
|
break;
|
|
}
|
|
#pragma warning suppress 56500 // covered by FxCOP
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
try
|
|
{
|
|
this.neighbor.CleanupProxy();
|
|
}
|
|
catch (Exception ee)
|
|
{
|
|
if (Fx.IsFatal(ee)) throw;
|
|
DiagnosticUtility.TraceHandledException(ee, TraceEventType.Information);
|
|
}
|
|
DiagnosticUtility.TraceHandledException(e, TraceEventType.Information);
|
|
if (!ContinuableException(e)) throw;
|
|
}
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
DiagnosticUtility.TraceHandledException(e, TraceEventType.Information);
|
|
this.lastException = e;
|
|
}
|
|
|
|
// Indicate completion to the caller
|
|
if (success)
|
|
{
|
|
Fx.Assert(this.lastException == null, "lastException expected to be null");
|
|
}
|
|
else
|
|
{
|
|
Fx.Assert(this.lastException != null, "lastException expected to be non-null");
|
|
}
|
|
base.Complete(this.completedSynchronously, this.lastException);
|
|
}
|
|
|
|
public static void End(IAsyncResult result)
|
|
{
|
|
AsyncResult.End<OpenAsyncResult>(result);
|
|
}
|
|
|
|
// Checks if the exception can be handled
|
|
bool ContinuableException(Exception exception)
|
|
{
|
|
if (
|
|
(
|
|
exception is EndpointNotFoundException
|
|
|| exception is TimeoutException
|
|
)
|
|
&& timeoutHelper.RemainingTime() > TimeSpan.Zero
|
|
)
|
|
{
|
|
this.lastException = exception;
|
|
this.currentIndex++;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Open completion callback. If open failed, reattempts with the next IP address in the list
|
|
void OnOpen(IAsyncResult result)
|
|
{
|
|
Exception exception = null;
|
|
bool completed = false;
|
|
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
this.completedSynchronously = false;
|
|
try
|
|
{
|
|
this.neighbor.EndOpenProxy(result);
|
|
completed = true;
|
|
neighbor.isClosing = false;
|
|
}
|
|
#pragma warning suppress 56500 // covered by FxCOP
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
try
|
|
{
|
|
this.neighbor.CleanupProxy();
|
|
}
|
|
catch (Exception ee)
|
|
{
|
|
if (Fx.IsFatal(ee)) throw;
|
|
DiagnosticUtility.TraceHandledException(ee, TraceEventType.Information);
|
|
}
|
|
exception = e;
|
|
if (ContinuableException(exception))
|
|
{
|
|
// attempt connection with the next IP address
|
|
try
|
|
{
|
|
BeginOpen();
|
|
}
|
|
catch (Exception ee)
|
|
{
|
|
if (Fx.IsFatal(ee)) throw;
|
|
DiagnosticUtility.TraceHandledException(ee, TraceEventType.Information);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
completed = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (completed)
|
|
base.Complete(this.completedSynchronously, exception);
|
|
}
|
|
}
|
|
|
|
#region IInputSessionShutdown Members
|
|
|
|
void IInputSessionShutdown.ChannelFaulted(IDuplexContextChannel channel)
|
|
{
|
|
//Noop
|
|
}
|
|
|
|
void IInputSessionShutdown.DoneReceiving(IDuplexContextChannel channel)
|
|
{
|
|
//Close it if the neighbor it was connected to has disconnected
|
|
if (channel.State == CommunicationState.Opened)
|
|
{
|
|
channel.Close();
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
public UtilityExtension Utility
|
|
{
|
|
get
|
|
{
|
|
if (this.utility == null)
|
|
{
|
|
this.utility = this.Extensions.Find<UtilityExtension>();
|
|
}
|
|
return this.utility;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Helper class to implement PeerNeighborManager's async neighbor open
|
|
class NeighborOpenAsyncResult : AsyncResult
|
|
{
|
|
PeerNeighbor neighbor;
|
|
|
|
// ClosedCallback is a delegate to determine if caller has closed. If so, we bail out of open operation
|
|
public NeighborOpenAsyncResult(PeerNeighbor neighbor, PeerNodeAddress remoteAddress, Binding binding,
|
|
PeerService service, ClosedCallback closedCallback, TimeSpan timeout, AsyncCallback callback, object state)
|
|
: base(callback, state)
|
|
{
|
|
this.neighbor = neighbor;
|
|
|
|
IAsyncResult result = null;
|
|
try
|
|
{
|
|
result = neighbor.BeginOpen(remoteAddress, binding, service, closedCallback, timeout,
|
|
Fx.ThunkCallback(new AsyncCallback(OnOpen)), null);
|
|
if (result.CompletedSynchronously)
|
|
{
|
|
neighbor.EndOpen(result);
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
neighbor.TraceEventHelper(TraceEventType.Warning, TraceCode.PeerNeighborOpenFailed, SR.GetString(SR.TraceCodePeerNeighborOpenFailed));
|
|
throw;
|
|
}
|
|
|
|
// Indicate [....] completion to the caller
|
|
if (result.CompletedSynchronously)
|
|
base.Complete(true);
|
|
}
|
|
|
|
void OnOpen(IAsyncResult result)
|
|
{
|
|
if (!result.CompletedSynchronously)
|
|
{
|
|
Exception exception = null;
|
|
|
|
try
|
|
{
|
|
this.neighbor.EndOpen(result);
|
|
}
|
|
#pragma warning suppress 56500 // covered by FxCOP
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e)) throw;
|
|
DiagnosticUtility.TraceHandledException(e, TraceEventType.Information);
|
|
neighbor.TraceEventHelper(TraceEventType.Warning, TraceCode.PeerNeighborOpenFailed, SR.GetString(SR.TraceCodePeerNeighborOpenFailed));
|
|
exception = e;
|
|
}
|
|
|
|
base.Complete(result.CompletedSynchronously, exception);
|
|
}
|
|
}
|
|
|
|
public static IPeerNeighbor End(IAsyncResult result)
|
|
{
|
|
NeighborOpenAsyncResult asyncResult = AsyncResult.End<NeighborOpenAsyncResult>(result);
|
|
return asyncResult.neighbor;
|
|
}
|
|
}
|
|
class PeerNeighborBehavior : IEndpointBehavior
|
|
{
|
|
PeerNeighbor neighbor;
|
|
|
|
public PeerNeighborBehavior(PeerNeighbor neighbor)
|
|
{
|
|
this.neighbor = neighbor;
|
|
}
|
|
|
|
#region IEndpointBehavior Members
|
|
|
|
public void Validate(ServiceEndpoint serviceEndpoint)
|
|
{
|
|
}
|
|
|
|
public void AddBindingParameters(ServiceEndpoint serviceEndpoint, BindingParameterCollection bindingParameters)
|
|
{
|
|
}
|
|
|
|
public void ApplyDispatchBehavior(ServiceEndpoint serviceEndpoint, EndpointDispatcher endpointDispatcher)
|
|
{
|
|
}
|
|
|
|
public void ApplyClientBehavior(ServiceEndpoint serviceEndpoint, ClientRuntime behavior)
|
|
{
|
|
behavior.DispatchRuntime.InputSessionShutdownHandlers.Add(this.neighbor);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
|
|
public IPeerNeighbor SlowestNeighbor()
|
|
{
|
|
List<IPeerNeighbor> neighbors = this.GetConnectedNeighbors();
|
|
IPeerNeighbor slowNeighbor = null;
|
|
UtilityExtension utility = null;
|
|
//if the neighbor has below this number, we wont consider for pruning
|
|
int pending = PeerTransportConstants.MessageThreshold;
|
|
foreach (IPeerNeighbor peer in neighbors)
|
|
{
|
|
utility = peer.Utility;
|
|
if (utility == null || !peer.IsConnected)
|
|
continue;
|
|
if (utility.PendingMessages > pending)
|
|
{
|
|
slowNeighbor = peer;
|
|
pending = utility.PendingMessages;
|
|
}
|
|
}
|
|
return slowNeighbor;
|
|
}
|
|
|
|
}
|
|
}
|