| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525 | 
							- using MediaBrowser.Common;
 
- using MediaBrowser.Common.Net;
 
- using MediaBrowser.Controller;
 
- using MediaBrowser.Controller.Configuration;
 
- using MediaBrowser.Model.Logging;
 
- using MediaBrowser.Model.Serialization;
 
- using System;
 
- using System.Collections.Generic;
 
- using System.Diagnostics;
 
- using System.IO;
 
- using System.Linq;
 
- using System.Net;
 
- using System.Net.Sockets;
 
- using System.Reflection;
 
- using System.Text;
 
- using System.Threading;
 
- using System.Threading.Tasks;
 
- namespace MediaBrowser.Server.Implementations.ServerManager
 
- {
 
-     /// <summary>
 
-     /// Manages the Http Server, Udp Server and WebSocket connections
 
-     /// </summary>
 
-     public class ServerManager : IServerManager, IDisposable
 
-     {
 
-         /// <summary>
 
-         /// This is the udp server used for server discovery by clients
 
-         /// </summary>
 
-         /// <value>The UDP server.</value>
 
-         private IUdpServer UdpServer { get; set; }
 
-         /// <summary>
 
-         /// Both the Ui and server will have a built-in HttpServer.
 
-         /// People will inevitably want remote control apps so it's needed in the Ui too.
 
-         /// </summary>
 
-         /// <value>The HTTP server.</value>
 
-         private IHttpServer HttpServer { get; set; }
 
-         /// <summary>
 
-         /// Gets or sets the json serializer.
 
-         /// </summary>
 
-         /// <value>The json serializer.</value>
 
-         private readonly IJsonSerializer _jsonSerializer;
 
-         /// <summary>
 
-         /// This subscribes to HttpListener requests and finds the appropriate BaseHandler to process it
 
-         /// </summary>
 
-         /// <value>The HTTP listener.</value>
 
-         private IDisposable HttpListener { get; set; }
 
-         /// <summary>
 
-         /// The web socket connections
 
-         /// </summary>
 
-         private readonly List<IWebSocketConnection> _webSocketConnections = new List<IWebSocketConnection>();
 
-         /// <summary>
 
-         /// Gets or sets the external web socket server.
 
-         /// </summary>
 
-         /// <value>The external web socket server.</value>
 
-         private IWebSocketServer ExternalWebSocketServer { get; set; }
 
-         /// <summary>
 
-         /// The _logger
 
-         /// </summary>
 
-         private readonly ILogger _logger;
 
-         /// <summary>
 
-         /// The _network manager
 
-         /// </summary>
 
-         private readonly INetworkManager _networkManager;
 
-         /// <summary>
 
-         /// The _application host
 
-         /// </summary>
 
-         private readonly IApplicationHost _applicationHost;
 
-         /// <summary>
 
-         /// Gets or sets the configuration manager.
 
-         /// </summary>
 
-         /// <value>The configuration manager.</value>
 
-         private IServerConfigurationManager ConfigurationManager { get; set; }
 
-         /// <summary>
 
-         /// Gets a value indicating whether [supports web socket].
 
-         /// </summary>
 
-         /// <value><c>true</c> if [supports web socket]; otherwise, <c>false</c>.</value>
 
-         public bool SupportsNativeWebSocket
 
-         {
 
-             get { return HttpServer != null && HttpServer.SupportsWebSockets; }
 
-         }
 
-         /// <summary>
 
-         /// Gets the web socket port number.
 
-         /// </summary>
 
-         /// <value>The web socket port number.</value>
 
-         public int WebSocketPortNumber
 
-         {
 
-             get { return SupportsNativeWebSocket ? ConfigurationManager.Configuration.HttpServerPortNumber : ConfigurationManager.Configuration.LegacyWebSocketPortNumber; }
 
-         }
 
-         /// <summary>
 
-         /// Gets the web socket listeners.
 
-         /// </summary>
 
-         /// <value>The web socket listeners.</value>
 
-         private readonly List<IWebSocketListener> _webSocketListeners = new List<IWebSocketListener>();
 
-         private Kernel _kernel;
 
-         /// <summary>
 
-         /// Initializes a new instance of the <see cref="ServerManager" /> class.
 
-         /// </summary>
 
-         /// <param name="applicationHost">The application host.</param>
 
-         /// <param name="networkManager">The network manager.</param>
 
-         /// <param name="jsonSerializer">The json serializer.</param>
 
-         /// <param name="logger">The logger.</param>
 
-         /// <param name="configurationManager">The configuration manager.</param>
 
-         /// <exception cref="System.ArgumentNullException">applicationHost</exception>
 
-         public ServerManager(IApplicationHost applicationHost, INetworkManager networkManager, IJsonSerializer jsonSerializer, ILogger logger, IServerConfigurationManager configurationManager, Kernel kernel)
 
-         {
 
-             if (applicationHost == null)
 
-             {
 
-                 throw new ArgumentNullException("applicationHost");
 
-             }
 
-             if (networkManager == null)
 
-             {
 
-                 throw new ArgumentNullException("networkManager");
 
-             }
 
-             if (jsonSerializer == null)
 
-             {
 
-                 throw new ArgumentNullException("jsonSerializer");
 
-             }
 
-             if (logger == null)
 
-             {
 
-                 throw new ArgumentNullException("logger");
 
-             }
 
-             _logger = logger;
 
-             _jsonSerializer = jsonSerializer;
 
-             _applicationHost = applicationHost;
 
-             _networkManager = networkManager;
 
-             ConfigurationManager = configurationManager;
 
-             _kernel = kernel;
 
-         }
 
-         /// <summary>
 
-         /// Starts this instance.
 
-         /// </summary>
 
-         public void Start()
 
-         {
 
-             if (_applicationHost.IsFirstRun)
 
-             {
 
-                 RegisterServerWithAdministratorAccess();
 
-             }
 
-             ReloadUdpServer();
 
-             ReloadHttpServer();
 
-             if (!SupportsNativeWebSocket)
 
-             {
 
-                 ReloadExternalWebSocketServer();
 
-             }
 
-             ConfigurationManager.ConfigurationUpdated += _kernel_ConfigurationUpdated;
 
-         }
 
-         /// <summary>
 
-         /// Starts the external web socket server.
 
-         /// </summary>
 
-         private void ReloadExternalWebSocketServer()
 
-         {
 
-             DisposeExternalWebSocketServer();
 
-             ExternalWebSocketServer = _applicationHost.Resolve<IWebSocketServer>();
 
-             ExternalWebSocketServer.Start(ConfigurationManager.Configuration.LegacyWebSocketPortNumber);
 
-             ExternalWebSocketServer.WebSocketConnected += HttpServer_WebSocketConnected;
 
-         }
 
-         /// <summary>
 
-         /// Restarts the Http Server, or starts it if not currently running
 
-         /// </summary>
 
-         /// <param name="registerServerOnFailure">if set to <c>true</c> [register server on failure].</param>
 
-         private void ReloadHttpServer(bool registerServerOnFailure = true)
 
-         {
 
-             // Only reload if the port has changed, so that we don't disconnect any active users
 
-             if (HttpServer != null && HttpServer.UrlPrefix.Equals(_kernel.HttpServerUrlPrefix, StringComparison.OrdinalIgnoreCase))
 
-             {
 
-                 return;
 
-             }
 
-             DisposeHttpServer();
 
-             _logger.Info("Loading Http Server");
 
-             try
 
-             {
 
-                 HttpServer = _applicationHost.Resolve<IHttpServer>();
 
-                 HttpServer.EnableHttpRequestLogging = ConfigurationManager.Configuration.EnableHttpLevelLogging;
 
-                 HttpServer.Start(_kernel.HttpServerUrlPrefix);
 
-             }
 
-             catch (HttpListenerException ex)
 
-             {
 
-                 _logger.ErrorException("Error starting Http Server", ex);
 
-                 if (registerServerOnFailure)
 
-                 {
 
-                     RegisterServerWithAdministratorAccess();
 
-                     // Don't get stuck in a loop
 
-                     ReloadHttpServer(false);
 
-                     return;
 
-                 }
 
-                 throw;
 
-             }
 
-             HttpServer.WebSocketConnected += HttpServer_WebSocketConnected;
 
-         }
 
-         /// <summary>
 
-         /// Handles the WebSocketConnected event of the HttpServer control.
 
-         /// </summary>
 
-         /// <param name="sender">The source of the event.</param>
 
-         /// <param name="e">The <see cref="WebSocketConnectEventArgs" /> instance containing the event data.</param>
 
-         void HttpServer_WebSocketConnected(object sender, WebSocketConnectEventArgs e)
 
-         {
 
-             var connection = new WebSocketConnection(e.WebSocket, e.Endpoint, _jsonSerializer, _logger) { OnReceive = ProcessWebSocketMessageReceived };
 
-             _webSocketConnections.Add(connection);
 
-         }
 
-         /// <summary>
 
-         /// Processes the web socket message received.
 
-         /// </summary>
 
-         /// <param name="result">The result.</param>
 
-         private async void ProcessWebSocketMessageReceived(WebSocketMessageInfo result)
 
-         {
 
-             var tasks = _webSocketListeners.Select(i => Task.Run(async () =>
 
-             {
 
-                 try
 
-                 {
 
-                     await i.ProcessMessage(result).ConfigureAwait(false);
 
-                 }
 
-                 catch (Exception ex)
 
-                 {
 
-                     _logger.ErrorException("{0} failed processing WebSocket message {1}", ex, i.GetType().Name, result.MessageType);
 
-                 }
 
-             }));
 
-             await Task.WhenAll(tasks).ConfigureAwait(false);
 
-         }
 
-         /// <summary>
 
-         /// Starts or re-starts the udp server
 
-         /// </summary>
 
-         private void ReloadUdpServer()
 
-         {
 
-             // For now, there's no reason to keep reloading this over and over
 
-             if (UdpServer != null)
 
-             {
 
-                 return;
 
-             }
 
-             DisposeUdpServer();
 
-             try
 
-             {
 
-                 // The port number can't be in configuration because we don't want it to ever change
 
-                 UdpServer = _applicationHost.Resolve<IUdpServer>();
 
-                 UdpServer.Start(_kernel.UdpServerPortNumber);
 
-             }
 
-             catch (SocketException ex)
 
-             {
 
-                 _logger.ErrorException("Failed to start UDP Server", ex);
 
-                 return;
 
-             }
 
-             UdpServer.MessageReceived += UdpServer_MessageReceived;
 
-         }
 
-         /// <summary>
 
-         /// Handles the MessageReceived event of the UdpServer control.
 
-         /// </summary>
 
-         /// <param name="sender">The source of the event.</param>
 
-         /// <param name="e">The <see cref="UdpMessageReceivedEventArgs" /> instance containing the event data.</param>
 
-         async void UdpServer_MessageReceived(object sender, UdpMessageReceivedEventArgs e)
 
-         {
 
-             var context = "Server";
 
-             var expectedMessage = String.Format("who is MediaBrowser{0}?", context);
 
-             var expectedMessageBytes = Encoding.UTF8.GetBytes(expectedMessage);
 
-             if (expectedMessageBytes.SequenceEqual(e.Bytes))
 
-             {
 
-                 _logger.Info("Received UDP server request from " + e.RemoteEndPoint);
 
-                 // Send a response back with our ip address and port
 
-                 var response = String.Format("MediaBrowser{0}|{1}:{2}", context, _networkManager.GetLocalIpAddress(), _kernel.UdpServerPortNumber);
 
-                 await UdpServer.SendAsync(Encoding.UTF8.GetBytes(response), e.RemoteEndPoint);
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Sends a message to all clients currently connected via a web socket
 
-         /// </summary>
 
-         /// <typeparam name="T"></typeparam>
 
-         /// <param name="messageType">Type of the message.</param>
 
-         /// <param name="data">The data.</param>
 
-         /// <returns>Task.</returns>
 
-         public void SendWebSocketMessage<T>(string messageType, T data)
 
-         {
 
-             SendWebSocketMessage(messageType, () => data);
 
-         }
 
-         /// <summary>
 
-         /// Sends a message to all clients currently connected via a web socket
 
-         /// </summary>
 
-         /// <typeparam name="T"></typeparam>
 
-         /// <param name="messageType">Type of the message.</param>
 
-         /// <param name="dataFunction">The function that generates the data to send, if there are any connected clients</param>
 
-         public void SendWebSocketMessage<T>(string messageType, Func<T> dataFunction)
 
-         {
 
-             Task.Run(async () => await SendWebSocketMessageAsync(messageType, dataFunction, CancellationToken.None).ConfigureAwait(false));
 
-         }
 
-         /// <summary>
 
-         /// Sends a message to all clients currently connected via a web socket
 
-         /// </summary>
 
-         /// <typeparam name="T"></typeparam>
 
-         /// <param name="messageType">Type of the message.</param>
 
-         /// <param name="dataFunction">The function that generates the data to send, if there are any connected clients</param>
 
-         /// <param name="cancellationToken">The cancellation token.</param>
 
-         /// <returns>Task.</returns>
 
-         /// <exception cref="System.ArgumentNullException">messageType</exception>
 
-         public async Task SendWebSocketMessageAsync<T>(string messageType, Func<T> dataFunction, CancellationToken cancellationToken)
 
-         {
 
-             if (string.IsNullOrEmpty(messageType))
 
-             {
 
-                 throw new ArgumentNullException("messageType");
 
-             }
 
-             if (dataFunction == null)
 
-             {
 
-                 throw new ArgumentNullException("dataFunction");
 
-             }
 
-             if (cancellationToken == null)
 
-             {
 
-                 throw new ArgumentNullException("cancellationToken");
 
-             }
 
-             cancellationToken.ThrowIfCancellationRequested();
 
-             var connections = _webSocketConnections.Where(s => s.State == WebSocketState.Open).ToList();
 
-             if (connections.Count > 0)
 
-             {
 
-                 _logger.Info("Sending web socket message {0}", messageType);
 
-                 var message = new WebSocketMessage<T> { MessageType = messageType, Data = dataFunction() };
 
-                 var bytes = _jsonSerializer.SerializeToBytes(message);
 
-                 var tasks = connections.Select(s => Task.Run(() =>
 
-                 {
 
-                     try
 
-                     {
 
-                         s.SendAsync(bytes, cancellationToken);
 
-                     }
 
-                     catch (OperationCanceledException)
 
-                     {
 
-                         throw;
 
-                     }
 
-                     catch (Exception ex)
 
-                     {
 
-                         _logger.ErrorException("Error sending web socket message {0} to {1}", ex, messageType, s.RemoteEndPoint);
 
-                     }
 
-                 }));
 
-                 await Task.WhenAll(tasks).ConfigureAwait(false);
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Disposes the udp server
 
-         /// </summary>
 
-         private void DisposeUdpServer()
 
-         {
 
-             if (UdpServer != null)
 
-             {
 
-                 UdpServer.MessageReceived -= UdpServer_MessageReceived;
 
-                 UdpServer.Dispose();
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Disposes the current HttpServer
 
-         /// </summary>
 
-         private void DisposeHttpServer()
 
-         {
 
-             foreach (var socket in _webSocketConnections)
 
-             {
 
-                 // Dispose the connection
 
-                 socket.Dispose();
 
-             }
 
-             _webSocketConnections.Clear();
 
-             if (HttpServer != null)
 
-             {
 
-                 HttpServer.WebSocketConnected -= HttpServer_WebSocketConnected;
 
-                 HttpServer.Dispose();
 
-             }
 
-             if (HttpListener != null)
 
-             {
 
-                 HttpListener.Dispose();
 
-             }
 
-             DisposeExternalWebSocketServer();
 
-         }
 
-         /// <summary>
 
-         /// Registers the server with administrator access.
 
-         /// </summary>
 
-         private void RegisterServerWithAdministratorAccess()
 
-         {
 
-             // Create a temp file path to extract the bat file to
 
-             var tmpFile = Path.Combine(ConfigurationManager.CommonApplicationPaths.TempDirectory, Guid.NewGuid() + ".bat");
 
-             // Extract the bat file
 
-             using (var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream("MediaBrowser.Server.Implementations.ServerManager.RegisterServer.bat"))
 
-             {
 
-                 using (var fileStream = File.Create(tmpFile))
 
-                 {
 
-                     stream.CopyTo(fileStream);
 
-                 }
 
-             }
 
-             var startInfo = new ProcessStartInfo
 
-             {
 
-                 FileName = tmpFile,
 
-                 Arguments = string.Format("{0} {1} {2} {3}", ConfigurationManager.Configuration.HttpServerPortNumber,
 
-                 _kernel.HttpServerUrlPrefix,
 
-                 _kernel.UdpServerPortNumber,
 
-                 ConfigurationManager.Configuration.LegacyWebSocketPortNumber),
 
-                 CreateNoWindow = true,
 
-                 WindowStyle = ProcessWindowStyle.Hidden,
 
-                 Verb = "runas",
 
-                 ErrorDialog = false
 
-             };
 
-             using (var process = Process.Start(startInfo))
 
-             {
 
-                 process.WaitForExit();
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.
 
-         /// </summary>
 
-         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 (dispose)
 
-             {
 
-                 DisposeUdpServer();
 
-                 DisposeHttpServer();
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Disposes the external web socket server.
 
-         /// </summary>
 
-         private void DisposeExternalWebSocketServer()
 
-         {
 
-             if (ExternalWebSocketServer != null)
 
-             {
 
-                 ExternalWebSocketServer.Dispose();
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Handles the ConfigurationUpdated event of the _kernel control.
 
-         /// </summary>
 
-         /// <param name="sender">The source of the event.</param>
 
-         /// <param name="e">The <see cref="EventArgs" /> instance containing the event data.</param>
 
-         /// <exception cref="System.NotImplementedException"></exception>
 
-         void _kernel_ConfigurationUpdated(object sender, EventArgs e)
 
-         {
 
-             HttpServer.EnableHttpRequestLogging = ConfigurationManager.Configuration.EnableHttpLevelLogging;
 
-             if (!string.Equals(HttpServer.UrlPrefix, _kernel.HttpServerUrlPrefix, StringComparison.OrdinalIgnoreCase))
 
-             {
 
-                 ReloadHttpServer();
 
-             }
 
-             if (!SupportsNativeWebSocket && ExternalWebSocketServer != null && ExternalWebSocketServer.Port != ConfigurationManager.Configuration.LegacyWebSocketPortNumber)
 
-             {
 
-                 ReloadExternalWebSocketServer();
 
-             }
 
-         }
 
-         /// <summary>
 
-         /// Adds the web socket listeners.
 
-         /// </summary>
 
-         /// <param name="listeners">The listeners.</param>
 
-         public void AddWebSocketListeners(IEnumerable<IWebSocketListener> listeners)
 
-         {
 
-             _webSocketListeners.AddRange(listeners);
 
-         }
 
-     }
 
- }
 
 
  |