// // Authors: // Ben Motmans // Lucas Ontivero lucasontivero@gmail.com // // Copyright (C) 2007 Ben Motmans // Copyright (C) 2014 Lucas Ontivero // // Permission is hereby granted, free of charge, to any person obtaining // a copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to // permit persons to whom the Software is furnished to do so, subject to // the following conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // using System; using System.Collections.Generic; using System.Linq; using System.Net; using System.Net.Sockets; using System.Threading; namespace Open.Nat { internal class PmpSearcher : Searcher { private readonly IIPAddressesProvider _ipprovider; private Dictionary> _gatewayLists; private int _timeout; internal PmpSearcher(IIPAddressesProvider ipprovider) { _ipprovider = ipprovider; _timeout = 250; CreateSocketsAndAddGateways(); } private void CreateSocketsAndAddGateways() { Sockets = new List(); _gatewayLists = new Dictionary>(); try { List gatewayList = _ipprovider.GatewayAddresses() .Select(ip => new IPEndPoint(ip, PmpConstants.ServerPort)) .ToList(); if (!gatewayList.Any()) { gatewayList.AddRange( _ipprovider.DnsAddresses() .Select(ip => new IPEndPoint(ip, PmpConstants.ServerPort))); } if (!gatewayList.Any()) return; foreach (IPAddress address in _ipprovider.UnicastAddresses()) { UdpClient client; try { client = new UdpClient(new IPEndPoint(address, 0)); } catch (SocketException) { continue; // Move on to the next address. } _gatewayLists.Add(client, gatewayList); Sockets.Add(client); } } catch (Exception e) { NatDiscoverer.TraceSource.LogError("There was a problem finding gateways: " + e); // NAT-PMP does not use multicast, so there isn't really a good fallback. } } protected override void Discover(UdpClient client, CancellationToken cancelationToken) { // Sort out the time for the next search first. The spec says the // timeout should double after each attempt. Once it reaches 64 seconds // (and that attempt fails), assume no devices available NextSearch = DateTime.UtcNow.AddMilliseconds(_timeout); _timeout *= 2; if (_timeout >= 3000) { _timeout = 250; NextSearch = DateTime.UtcNow.AddSeconds(10); return; } // The nat-pmp search message. Must be sent to GatewayIP:53531 var buffer = new[] {PmpConstants.Version, PmpConstants.OperationExternalAddressRequest}; foreach (IPEndPoint gatewayEndpoint in _gatewayLists[client]) { if (cancelationToken.IsCancellationRequested) return; client.Send(buffer, buffer.Length, gatewayEndpoint); } } private bool IsSearchAddress(IPAddress address) { return _gatewayLists.Values.SelectMany(x => x) .Any(x => x.Address.Equals(address)); } public override NatDevice AnalyseReceivedResponse(IPAddress localAddress, byte[] response, IPEndPoint endpoint) { if (!IsSearchAddress(endpoint.Address) || response.Length != 12 || response[0] != PmpConstants.Version || response[1] != PmpConstants.ServerNoop) return null; int errorcode = IPAddress.NetworkToHostOrder(BitConverter.ToInt16(response, 2)); if (errorcode != 0) NatDiscoverer.TraceSource.LogError("Non zero error: {0}", errorcode); var publicIp = new IPAddress(new[] {response[8], response[9], response[10], response[11]}); //NextSearch = DateTime.Now.AddMinutes(5); _timeout = 250; return new PmpNatDevice(endpoint.Address, publicIp); } } }