530 lines
19 KiB
C#
Raw Normal View History

//------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//------------------------------------------------------------
namespace System.ServiceModel.Channels
{
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Net;
using System.Net.NetworkInformation;
using System.Net.Sockets;
using System.Runtime;
using System.ServiceModel;
using System.Threading;
// IP address helper class for multi-homing support
class PeerIPHelper
{
public event EventHandler AddressChanged;
bool isOpen;
readonly IPAddress listenAddress; // To listen on a single IP address.
IPAddress[] localAddresses;
AddressChangeHelper addressChangeHelper;
Socket ipv6Socket;
object thisLock;
const uint Six2FourPrefix = 0x220;
const uint TeredoPrefix = 0x00000120;
const uint IsatapIdentifier = 0xfe5e0000;
enum AddressType
{
Unknown,
Teredo,
Isatap,
Six2Four
}
public PeerIPHelper()
{
Initialize();
}
public PeerIPHelper(IPAddress listenAddress)
{
if (!(listenAddress != null))
{
throw Fx.AssertAndThrow("listenAddress expected to be non-null");
}
this.listenAddress = listenAddress;
Initialize();
}
void Initialize()
{
this.localAddresses = new IPAddress[0];
this.thisLock = new object();
}
// NOTE: This is just for suites -- to skip timeout usage
internal int AddressChangeWaitTimeout
{
set { this.addressChangeHelper.Timeout = value; }
}
object ThisLock
{
get { return this.thisLock; }
}
// Compares if the specified collection matches our local cache. Return true on mismatch.
public bool AddressesChanged(ReadOnlyCollection<IPAddress> addresses)
{
bool changed = false;
lock (ThisLock)
{
if (addresses.Count != this.localAddresses.Length)
{
changed = true;
}
else
{
// If every specified addresses exist in the cache, addresses haven't changed
foreach (IPAddress address in this.localAddresses)
{
if (!addresses.Contains(address))
{
changed = true;
break;
}
}
}
}
return changed;
}
// Since scope ID of IPAddress is mutable, you want to be able to clone an IP address
public static IPAddress CloneAddress(IPAddress source, bool maskScopeId)
{
IPAddress clone = null;
if (maskScopeId || V4Address(source))
clone = new IPAddress(source.GetAddressBytes());
else
clone = new IPAddress(source.GetAddressBytes(), source.ScopeId);
return clone;
}
// Since scope ID of IPAddress is mutable, you want to be able to clone IP addresses in an array or collection
static ReadOnlyCollection<IPAddress> CloneAddresses(IPAddress[] sourceArray)
{
IPAddress[] cloneArray = new IPAddress[sourceArray.Length];
for (int i = 0; i < sourceArray.Length; i++)
{
cloneArray[i] = CloneAddress(sourceArray[i], false);
}
return new ReadOnlyCollection<IPAddress>(cloneArray);
}
public static ReadOnlyCollection<IPAddress> CloneAddresses(ReadOnlyCollection<IPAddress> sourceCollection, bool maskScopeId)
{
IPAddress[] cloneArray = new IPAddress[sourceCollection.Count];
for (int i = 0; i < sourceCollection.Count; i++)
{
cloneArray[i] = CloneAddress(sourceCollection[i], maskScopeId);
}
return new ReadOnlyCollection<IPAddress>(cloneArray);
}
// When listening on a specific IP address, creates an array containing just that address
static IPAddress[] CreateAddressArray(IPAddress address)
{
IPAddress[] addressArray = new IPAddress[1];
addressArray[0] = CloneAddress(address, false);
return addressArray;
}
public void Close()
{
if (this.isOpen)
{
lock (ThisLock)
{
if (this.isOpen)
{
this.addressChangeHelper.Unregister();
if (this.ipv6Socket != null)
{
this.ipv6Socket.Close();
}
this.isOpen = false;
this.addressChangeHelper = null;
}
}
}
}
// Retrieve the IP addresses configured on the machine.
IPAddress[] GetAddresses()
{
List<IPAddress> addresses = new List<IPAddress>();
List<IPAddress> temporaryAddresses = new List<IPAddress>();
if (this.listenAddress != null) // single local address scenario?
{
// Check if the specified address is configured
if (ValidAddress(this.listenAddress))
{
return (CreateAddressArray(this.listenAddress));
}
}
// Walk the interfaces
NetworkInterface[] networkIfs = NetworkInterface.GetAllNetworkInterfaces();
foreach (NetworkInterface networkIf in networkIfs)
{
if (!ValidInterface(networkIf))
{
continue;
}
// Process each unicast address for the interface. Pick at most one IPv4 and one IPv6 address.
// Add remaining eligible addresses on the list to surplus list. We can add these back to the list
// being returned, if there is room.
IPInterfaceProperties properties = networkIf.GetIPProperties();
if (properties != null)
{
foreach (UnicastIPAddressInformation unicastAddress in properties.UnicastAddresses)
{
if (NonTransientAddress(unicastAddress))
{
if (unicastAddress.SuffixOrigin == SuffixOrigin.Random)
temporaryAddresses.Add(unicastAddress.Address);
else
addresses.Add(unicastAddress.Address);
}
}
}
}
if (addresses.Count > 0)
return ReorderAddresses(addresses);
else
return temporaryAddresses.ToArray();
}
internal static IPAddress[] ReorderAddresses(IEnumerable<IPAddress> sourceAddresses)
{
List<IPAddress> result = new List<IPAddress>();
List<IPAddress> notAdded = new List<IPAddress>();
AddressType addressType = AddressType.Unknown;
IPAddress v4Address = null, v6Address = null, isatapAddress = null, teredoAddress = null, six2FourAddress = null;
foreach (IPAddress address in sourceAddresses)
{
if (address.AddressFamily == AddressFamily.InterNetwork)
{
if (v4Address != null)
notAdded.Add(address);
else
{
v4Address = address;
}
continue;
}
if (address.AddressFamily != AddressFamily.InterNetworkV6)
{
notAdded.Add(address);
continue;
}
if (address.IsIPv6LinkLocal || address.IsIPv6SiteLocal)
{
notAdded.Add(address);
continue;
}
addressType = GetAddressType(address);
switch (addressType)
{
case AddressType.Teredo:
{
if (teredoAddress == null)
{
teredoAddress = address;
}
else
{
notAdded.Add(address);
}
continue;
}
case AddressType.Six2Four:
{
if (six2FourAddress == null)
{
six2FourAddress = address;
}
else
{
notAdded.Add(address);
}
continue;
}
case AddressType.Isatap:
{
if (isatapAddress == null)
{
isatapAddress = address;
}
else
{
notAdded.Add(address);
}
continue;
}
default:
{
if (v6Address != null)
notAdded.Add(address);
else
{
v6Address = address;
}
continue;
}
}
}
if (six2FourAddress != null)
result.Add(six2FourAddress);
if (teredoAddress != null)
result.Add(teredoAddress);
if (isatapAddress != null)
result.Add(isatapAddress);
if (v6Address != null)
result.Add(v6Address);
if (v4Address != null)
result.Add(v4Address);
result.AddRange(notAdded);
return result.ToArray();
}
static AddressType GetAddressType(IPAddress address)
{
AddressType result = AddressType.Unknown;
byte[] bytes = address.GetAddressBytes();
if (BitConverter.ToUInt16(bytes, 0) == Six2FourPrefix)
result = AddressType.Six2Four;
else if (BitConverter.ToUInt32(bytes, 0) == TeredoPrefix)
result = AddressType.Teredo;
else if (BitConverter.ToUInt32(bytes, 8) == IsatapIdentifier)
result = AddressType.Isatap;
return result;
}
// Given an EPR, replaces its URI with the specified IP address
public static EndpointAddress GetIPEndpointAddress(EndpointAddress epr, IPAddress address)
{
EndpointAddressBuilder eprBuilder = new EndpointAddressBuilder(epr);
eprBuilder.Uri = GetIPUri(epr.Uri, address);
return eprBuilder.ToEndpointAddress();
}
// Given a hostName based URI, replaces hostName with the IP address
public static Uri GetIPUri(Uri uri, IPAddress ipAddress)
{
UriBuilder uriBuilder = new UriBuilder(uri);
if (V6Address(ipAddress) && (ipAddress.IsIPv6LinkLocal || ipAddress.IsIPv6SiteLocal))
{
// We make a copy of the IP address because scopeID will not be part of ToString() if set after IP address was created
uriBuilder.Host = new IPAddress(ipAddress.GetAddressBytes(), ipAddress.ScopeId).ToString();
}
else
{
uriBuilder.Host = ipAddress.ToString();
}
return uriBuilder.Uri;
}
// Retrieve the currently configured addresses (cached)
public ReadOnlyCollection<IPAddress> GetLocalAddresses()
{
lock (ThisLock)
{
// Return a clone of the address cache
return CloneAddresses(this.localAddresses);
}
}
// An address is valid if it is a non-transient addresss (if global, must be DNS-eligible as well)
static bool NonTransientAddress(UnicastIPAddressInformation address)
{
return (!address.IsTransient);
}
public static bool V4Address(IPAddress address)
{
return address.AddressFamily == AddressFamily.InterNetwork;
}
public static bool V6Address(IPAddress address)
{
return address.AddressFamily == AddressFamily.InterNetworkV6;
}
// Returns true if the specified address is configured on the machine
public static bool ValidAddress(IPAddress address)
{
// Walk the interfaces
NetworkInterface[] networkIfs = NetworkInterface.GetAllNetworkInterfaces();
foreach (NetworkInterface networkIf in networkIfs)
{
if (ValidInterface(networkIf))
{
IPInterfaceProperties properties = networkIf.GetIPProperties();
if (properties != null)
{
foreach (UnicastIPAddressInformation unicastAddress in properties.UnicastAddresses)
{
if (address.Equals(unicastAddress.Address))
{
return true;
}
}
}
}
}
return false;
}
static bool ValidInterface(NetworkInterface networkIf)
{
return (networkIf.NetworkInterfaceType != NetworkInterfaceType.Loopback &&
networkIf.OperationalStatus == OperationalStatus.Up);
}
// Process the address change notification from the system and check if the addresses
// have really changed. If so, update the local cache and notify interested parties.
void OnAddressChanged()
{
bool changed = false;
IPAddress[] newAddresses = GetAddresses();
lock (ThisLock)
{
if (AddressesChanged(Array.AsReadOnly<IPAddress>(newAddresses)))
{
this.localAddresses = newAddresses;
changed = true;
}
}
if (changed)
{
EventHandler handler = AddressChanged;
if (handler != null && this.isOpen)
{
handler(this, EventArgs.Empty);
}
}
}
public void Open()
{
lock (ThisLock)
{
Fx.Assert(!this.isOpen, "Helper not expected to be open");
// Register for addr changed event and retrieve addresses from the system
this.addressChangeHelper = new AddressChangeHelper(OnAddressChanged);
this.localAddresses = GetAddresses();
if (Socket.OSSupportsIPv6)
{
this.ipv6Socket = new Socket(AddressFamily.InterNetworkV6, SocketType.Stream, ProtocolType.IP);
}
this.isOpen = true;
}
}
// Sorts the collection of addresses using sort ioctl
public ReadOnlyCollection<IPAddress> SortAddresses(ReadOnlyCollection<IPAddress> addresses)
{
ReadOnlyCollection<IPAddress> sortedAddresses = SocketAddressList.SortAddresses(this.ipv6Socket, listenAddress, addresses);
// If listening on specific address, copy the scope ID that we're listing on for the
// link and site local addresses in the sorted list (so that the connect will work)
if (this.listenAddress != null)
{
if (this.listenAddress.IsIPv6LinkLocal)
{
foreach (IPAddress address in sortedAddresses)
{
if (address.IsIPv6LinkLocal)
{
address.ScopeId = this.listenAddress.ScopeId;
}
}
}
else if (this.listenAddress.IsIPv6SiteLocal)
{
foreach (IPAddress address in sortedAddresses)
{
if (address.IsIPv6SiteLocal)
{
address.ScopeId = this.listenAddress.ScopeId;
}
}
}
}
return sortedAddresses;
}
//
// Helper class to handle system address change events. Because multiple events can be fired as a result of
// a single significant change (such as interface being enabled/disabled), we try to handle the event just
// once using the below mechanism:
// Start a timer that goes off after Timeout seconds. If get another address change event from the system
// within this timespan, reset the timer to go off after another Timeout secs. When the timer finally fires,
// the registered handlers are notified. This should minimize (but not completely eliminate -- think wireless
// DHCP interface being enabled, for instance -- this could take longer) reacting multiple times for
// a single change.
//
class AddressChangeHelper
{
public delegate void AddedChangedCallback();
// To avoid processing multiple addr change events within this time span (5 seconds)
public int Timeout = 5000;
IOThreadTimer timer;
AddedChangedCallback addressChanged;
public AddressChangeHelper(AddedChangedCallback addressChanged)
{
Fx.Assert(addressChanged != null, "addressChanged expected to be non-null");
this.addressChanged = addressChanged;
this.timer = new IOThreadTimer(new Action<object>(FireAddressChange), null, true);
NetworkChange.NetworkAddressChanged += OnAddressChange;
}
public void Unregister()
{
NetworkChange.NetworkAddressChanged -= OnAddressChange;
}
void OnAddressChange(object sender, EventArgs args)
{
this.timer.Set(Timeout);
}
// Now fire address change event to the interested parties
void FireAddressChange(object asyncState)
{
this.timer.Cancel();
this.addressChanged();
}
}
}
}