| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 | #pragma warning disable CS1591using 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 Jellyfin.Networking.Configuration;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.GetNetworkConfiguration();            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()        {            var config = _config.GetNetworkConfiguration();            if (!config.EnableUPnP || !config.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)        {            var config = _config.GetNetworkConfiguration();            yield return CreatePortMap(device, _appHost.HttpPort, config.PublicPort);            if (_appHost.ListenWithHttps)            {                yield return CreatePortMap(device, _appHost.HttpsPort, config.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;        }    }}
 |