jellyfin/Emby.Server.Implementations/EntryPoints/UdpServerEntryPoint.cs

146 lines
5.3 KiB
C#
Raw Normal View History

2021-02-23 16:45:10 +01:00
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
2020-07-13 17:33:39 +02:00
using System.Net.Sockets;
2020-01-12 18:59:10 +01:00
using System.Threading;
2019-01-27 15:40:37 +01:00
using System.Threading.Tasks;
using Emby.Server.Implementations.Udp;
2022-02-14 14:39:33 +01:00
using Jellyfin.Networking.Configuration;
2023-07-03 21:51:36 +02:00
using Jellyfin.Networking.Extensions;
2022-02-14 14:39:33 +01:00
using MediaBrowser.Common.Configuration;
using MediaBrowser.Common.Net;
2014-07-28 00:01:29 +02:00
using MediaBrowser.Controller;
2013-05-19 00:07:59 +02:00
using MediaBrowser.Controller.Plugins;
2020-05-02 18:56:09 +02:00
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;
2013-05-19 00:07:59 +02:00
2016-11-04 19:56:47 +01:00
namespace Emby.Server.Implementations.EntryPoints
2013-05-19 00:07:59 +02:00
{
/// <summary>
2019-11-01 18:38:54 +01:00
/// Class UdpServerEntryPoint.
/// </summary>
2020-01-12 18:59:10 +01:00
public sealed class UdpServerEntryPoint : IServerEntryPoint
2013-05-19 00:07:59 +02:00
{
/// <summary>
2019-11-01 18:38:54 +01:00
/// The port of the UDP server.
2013-05-19 00:07:59 +02:00
/// </summary>
2019-11-01 18:38:54 +01:00
public const int PortNumber = 7359;
2013-05-19 00:07:59 +02:00
/// <summary>
/// The logger.
/// </summary>
2020-06-06 02:15:56 +02:00
private readonly ILogger<UdpServerEntryPoint> _logger;
2014-07-28 00:01:29 +02:00
private readonly IServerApplicationHost _appHost;
2020-05-02 18:56:09 +02:00
private readonly IConfiguration _config;
2022-02-14 14:39:33 +01:00
private readonly IConfigurationManager _configurationManager;
private readonly INetworkManager _networkManager;
private readonly bool _enableMultiSocketBinding;
2013-05-19 00:07:59 +02:00
2019-11-01 18:38:54 +01:00
/// <summary>
/// The UDP server.
/// </summary>
private List<UdpServer> _udpServers;
2020-01-12 18:59:10 +01:00
private CancellationTokenSource _cancellationTokenSource = new CancellationTokenSource();
private bool _disposed = false;
2013-09-25 02:54:51 +02:00
/// <summary>
2014-08-20 00:28:35 +02:00
/// Initializes a new instance of the <see cref="UdpServerEntryPoint" /> class.
/// </summary>
2021-08-29 00:32:50 +02:00
/// <param name="logger">Instance of the <see cref="ILogger{UdpServerEntryPoint}"/> interface.</param>
/// <param name="appHost">Instance of the <see cref="IServerApplicationHost"/> interface.</param>
/// <param name="configuration">Instance of the <see cref="IConfiguration"/> interface.</param>
2022-02-14 14:39:33 +01:00
/// <param name="configurationManager">Instance of the <see cref="IConfigurationManager"/> interface.</param>
/// <param name="networkManager">Instance of the <see cref="INetworkManager"/> interface.</param>
2019-11-01 18:38:54 +01:00
public UdpServerEntryPoint(
2020-01-12 18:59:10 +01:00
ILogger<UdpServerEntryPoint> logger,
2020-05-02 18:56:09 +02:00
IServerApplicationHost appHost,
2022-02-14 14:39:33 +01:00
IConfiguration configuration,
IConfigurationManager configurationManager,
INetworkManager networkManager)
2013-05-19 00:07:59 +02:00
{
_logger = logger;
2014-07-28 00:01:29 +02:00
_appHost = appHost;
2020-05-02 18:56:09 +02:00
_config = configuration;
2022-02-14 14:39:33 +01:00
_configurationManager = configurationManager;
_networkManager = networkManager;
_udpServers = new List<UdpServer>();
_enableMultiSocketBinding = OperatingSystem.IsWindows() || OperatingSystem.IsLinux();
2013-05-19 00:07:59 +02:00
}
2019-11-01 18:38:54 +01:00
/// <inheritdoc />
public Task RunAsync()
2013-05-19 00:07:59 +02:00
{
2021-02-23 16:45:10 +01:00
CheckDisposed();
2022-03-08 22:11:06 +01:00
if (!_configurationManager.GetNetworkConfiguration().AutoDiscovery)
2022-02-14 14:39:33 +01:00
{
return Task.CompletedTask;
}
try
{
if (_enableMultiSocketBinding)
{
2022-10-15 17:27:37 +02:00
// Add global broadcast socket
2023-02-23 13:55:27 +01:00
var server = new UdpServer(_logger, _appHost, _config, IPAddress.Broadcast, PortNumber);
server.Start(_cancellationTokenSource.Token);
_udpServers.Add(server);
2022-10-15 17:27:37 +02:00
// Add bind address specific broadcast sockets
// IPv6 is currently unsupported
var validInterfaces = _networkManager.GetInternalBindAddresses().Where(i => i.AddressFamily == AddressFamily.InterNetwork);
foreach (var intf in validInterfaces)
{
var broadcastAddress = NetworkExtensions.GetBroadcastAddress(intf.Subnet);
2023-02-23 13:55:27 +01:00
_logger.LogDebug("Binding UDP server to {Address} on port {PortNumber}", broadcastAddress, PortNumber);
2022-10-15 17:27:37 +02:00
2023-02-23 13:55:27 +01:00
server = new UdpServer(_logger, _appHost, _config, broadcastAddress, PortNumber);
server.Start(_cancellationTokenSource.Token);
_udpServers.Add(server);
}
}
else
{
var server = new UdpServer(_logger, _appHost, _config, IPAddress.Any, PortNumber);
2023-02-15 22:34:44 +01:00
server.Start(_cancellationTokenSource.Token);
_udpServers.Add(server);
2023-02-15 22:34:44 +01:00
}
}
2020-07-13 17:33:39 +02:00
catch (SocketException ex)
{
_logger.LogWarning(ex, "Unable to start AutoDiscovery listener on UDP port {PortNumber}", PortNumber);
}
return Task.CompletedTask;
2013-05-19 00:07:59 +02:00
}
2021-02-23 16:45:10 +01:00
private void CheckDisposed()
{
if (_disposed)
{
throw new ObjectDisposedException(this.GetType().Name);
}
}
2020-01-12 18:59:10 +01:00
/// <inheritdoc />
public void Dispose()
2013-05-19 00:07:59 +02:00
{
2020-01-12 18:59:10 +01:00
if (_disposed)
2013-05-19 00:07:59 +02:00
{
2020-01-12 18:59:10 +01:00
return;
2013-05-19 00:07:59 +02:00
}
2020-01-12 18:59:10 +01:00
_cancellationTokenSource.Cancel();
_cancellationTokenSource.Dispose();
foreach (var server in _udpServers)
{
server.Dispose();
}
2020-01-12 18:59:10 +01:00
_udpServers.Clear();
2020-01-12 18:59:10 +01:00
_disposed = true;
2013-05-19 00:07:59 +02:00
}
}
}