| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218 | 
							- #pragma warning disable CS1591
 
- using System;
 
- using System.Collections.Concurrent;
 
- using System.Collections.Generic;
 
- using System.Net;
 
- using System.Text;
 
- using System.Threading;
 
- using System.Threading.Tasks;
 
- using Jellyfin.Data.Events;
 
- using MediaBrowser.Controller;
 
- using MediaBrowser.Controller.Configuration;
 
- using MediaBrowser.Controller.Plugins;
 
- using MediaBrowser.Model.Dlna;
 
- using Microsoft.Extensions.Logging;
 
- using Mono.Nat;
 
- namespace Emby.Server.Implementations.EntryPoints
 
- {
 
-     /// <summary>
 
-     /// Server entrypoint handling external port forwarding.
 
-     /// </summary>
 
-     public class ExternalPortForwarding : IServerEntryPoint
 
-     {
 
-         private readonly IServerApplicationHost _appHost;
 
-         private readonly ILogger<ExternalPortForwarding> _logger;
 
-         private readonly IServerConfigurationManager _config;
 
-         private readonly IDeviceDiscovery _deviceDiscovery;
 
-         private readonly ConcurrentDictionary<IPEndPoint, byte> _createdRules = new ConcurrentDictionary<IPEndPoint, byte>();
 
-         private Timer _timer;
 
-         private string _configIdentifier;
 
-         private bool _disposed = false;
 
-         /// <summary>
 
-         /// Initializes a new instance of the <see cref="ExternalPortForwarding"/> class.
 
-         /// </summary>
 
-         /// <param name="logger">The logger.</param>
 
-         /// <param name="appHost">The application host.</param>
 
-         /// <param name="config">The configuration manager.</param>
 
-         /// <param name="deviceDiscovery">The device discovery.</param>
 
-         public ExternalPortForwarding(
 
-             ILogger<ExternalPortForwarding> logger,
 
-             IServerApplicationHost appHost,
 
-             IServerConfigurationManager config,
 
-             IDeviceDiscovery deviceDiscovery)
 
-         {
 
-             _logger = logger;
 
-             _appHost = appHost;
 
-             _config = config;
 
-             _deviceDiscovery = deviceDiscovery;
 
-         }
 
-         private string GetConfigIdentifier()
 
-         {
 
-             const char Separator = '|';
 
-             var config = _config.Configuration;
 
-             return new StringBuilder(32)
 
-                 .Append(config.EnableUPnP).Append(Separator)
 
-                 .Append(config.PublicPort).Append(Separator)
 
-                 .Append(config.PublicHttpsPort).Append(Separator)
 
-                 .Append(_appHost.HttpPort).Append(Separator)
 
-                 .Append(_appHost.HttpsPort).Append(Separator)
 
-                 .Append(_appHost.ListenWithHttps).Append(Separator)
 
-                 .Append(config.EnableRemoteAccess).Append(Separator)
 
-                 .ToString();
 
-         }
 
-         private void OnConfigurationUpdated(object sender, EventArgs e)
 
-         {
 
-             var oldConfigIdentifier = _configIdentifier;
 
-             _configIdentifier = GetConfigIdentifier();
 
-             if (!string.Equals(_configIdentifier, oldConfigIdentifier, StringComparison.OrdinalIgnoreCase))
 
-             {
 
-                 Stop();
 
-                 Start();
 
-             }
 
-         }
 
-         /// <inheritdoc />
 
-         public Task RunAsync()
 
-         {
 
-             Start();
 
-             _config.ConfigurationUpdated += OnConfigurationUpdated;
 
-             return Task.CompletedTask;
 
-         }
 
-         private void Start()
 
-         {
 
-             if (!_config.Configuration.EnableUPnP || !_config.Configuration.EnableRemoteAccess)
 
-             {
 
-                 return;
 
-             }
 
-             _logger.LogInformation("Starting NAT discovery");
 
-             NatUtility.DeviceFound += OnNatUtilityDeviceFound;
 
-             NatUtility.StartDiscovery();
 
-             _timer = new Timer((_) => _createdRules.Clear(), null, TimeSpan.FromMinutes(10), TimeSpan.FromMinutes(10));
 
-             _deviceDiscovery.DeviceDiscovered += OnDeviceDiscoveryDeviceDiscovered;
 
-         }
 
-         private void Stop()
 
-         {
 
-             _logger.LogInformation("Stopping NAT discovery");
 
-             NatUtility.StopDiscovery();
 
-             NatUtility.DeviceFound -= OnNatUtilityDeviceFound;
 
-             _timer?.Dispose();
 
-             _deviceDiscovery.DeviceDiscovered -= OnDeviceDiscoveryDeviceDiscovered;
 
-         }
 
-         private void OnDeviceDiscoveryDeviceDiscovered(object sender, GenericEventArgs<UpnpDeviceInfo> e)
 
-         {
 
-             NatUtility.Search(e.Argument.LocalIpAddress, NatProtocol.Upnp);
 
-         }
 
-         private async void OnNatUtilityDeviceFound(object sender, DeviceEventArgs e)
 
-         {
 
-             try
 
-             {
 
-                 await CreateRules(e.Device).ConfigureAwait(false);
 
-             }
 
-             catch (Exception ex)
 
-             {
 
-                 _logger.LogError(ex, "Error creating port forwarding rules");
 
-             }
 
-         }
 
-         private Task CreateRules(INatDevice device)
 
-         {
 
-             if (_disposed)
 
-             {
 
-                 throw new ObjectDisposedException(GetType().Name);
 
-             }
 
-             // On some systems the device discovered event seems to fire repeatedly
 
-             // This check will help ensure we're not trying to port map the same device over and over
 
-             if (!_createdRules.TryAdd(device.DeviceEndpoint, 0))
 
-             {
 
-                 return Task.CompletedTask;
 
-             }
 
-             return Task.WhenAll(CreatePortMaps(device));
 
-         }
 
-         private IEnumerable<Task> CreatePortMaps(INatDevice device)
 
-         {
 
-             yield return CreatePortMap(device, _appHost.HttpPort, _config.Configuration.PublicPort);
 
-             if (_appHost.ListenWithHttps)
 
-             {
 
-                 yield return CreatePortMap(device, _appHost.HttpsPort, _config.Configuration.PublicHttpsPort);
 
-             }
 
-         }
 
-         private async Task CreatePortMap(INatDevice device, int privatePort, int publicPort)
 
-         {
 
-             _logger.LogDebug(
 
-                 "Creating port map on local port {LocalPort} to public port {PublicPort} with device {DeviceEndpoint}",
 
-                 privatePort,
 
-                 publicPort,
 
-                 device.DeviceEndpoint);
 
-             try
 
-             {
 
-                 var mapping = new Mapping(Protocol.Tcp, privatePort, publicPort, 0, _appHost.Name);
 
-                 await device.CreatePortMapAsync(mapping).ConfigureAwait(false);
 
-             }
 
-             catch (Exception ex)
 
-             {
 
-                 _logger.LogError(
 
-                     ex,
 
-                     "Error creating port map on local port {LocalPort} to public port {PublicPort} with device {DeviceEndpoint}.",
 
-                     privatePort,
 
-                     publicPort,
 
-                     device.DeviceEndpoint);
 
-             }
 
-         }
 
-         /// <inheritdoc />
 
-         public void Dispose()
 
-         {
 
-             Dispose(true);
 
-             GC.SuppressFinalize(this);
 
-         }
 
-         /// <summary>
 
-         /// Releases unmanaged and - optionally - managed resources.
 
-         /// </summary>
 
-         /// <param name="dispose"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
 
-         protected virtual void Dispose(bool dispose)
 
-         {
 
-             if (_disposed)
 
-             {
 
-                 return;
 
-             }
 
-             _config.ConfigurationUpdated -= OnConfigurationUpdated;
 
-             Stop();
 
-             _timer = null;
 
-             _disposed = true;
 
-         }
 
-     }
 
- }
 
 
  |