WebSocketController.cs 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. using MediaBrowser.Controller.Net;
  2. using MediaBrowser.Controller.Session;
  3. using MediaBrowser.Model.Entities;
  4. using MediaBrowser.Model.Logging;
  5. using MediaBrowser.Model.Net;
  6. using MediaBrowser.Model.Session;
  7. using MediaBrowser.Model.System;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Linq;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. namespace MediaBrowser.Server.Implementations.Session
  14. {
  15. public class WebSocketController : ISessionController, IDisposable
  16. {
  17. public SessionInfo Session { get; private set; }
  18. public IReadOnlyList<IWebSocketConnection> Sockets { get; private set; }
  19. private readonly ILogger _logger;
  20. private readonly ISessionManager _sessionManager;
  21. public WebSocketController(SessionInfo session, ILogger logger, ISessionManager sessionManager)
  22. {
  23. Session = session;
  24. _logger = logger;
  25. _sessionManager = sessionManager;
  26. Sockets = new List<IWebSocketConnection>();
  27. }
  28. private bool HasOpenSockets
  29. {
  30. get { return GetActiveSockets().Any(); }
  31. }
  32. public bool SupportsMediaControl
  33. {
  34. get { return HasOpenSockets; }
  35. }
  36. private bool _isActive;
  37. public bool IsSessionActive
  38. {
  39. get
  40. {
  41. return HasOpenSockets;
  42. }
  43. }
  44. public void OnActivity()
  45. {
  46. _isActive = true;
  47. }
  48. private IEnumerable<IWebSocketConnection> GetActiveSockets()
  49. {
  50. return Sockets
  51. .OrderByDescending(i => i.LastActivityDate)
  52. .Where(i => i.State == WebSocketState.Open);
  53. }
  54. public void AddWebSocket(IWebSocketConnection connection)
  55. {
  56. var sockets = Sockets.ToList();
  57. sockets.Add(connection);
  58. Sockets = sockets;
  59. connection.Closed += connection_Closed;
  60. }
  61. void connection_Closed(object sender, EventArgs e)
  62. {
  63. if (!GetActiveSockets().Any())
  64. {
  65. _isActive = false;
  66. try
  67. {
  68. _sessionManager.ReportSessionEnded(Session.Id);
  69. }
  70. catch (Exception ex)
  71. {
  72. _logger.ErrorException("Error reporting session ended.", ex);
  73. }
  74. }
  75. }
  76. private IWebSocketConnection GetActiveSocket()
  77. {
  78. var socket = GetActiveSockets()
  79. .FirstOrDefault();
  80. if (socket == null)
  81. {
  82. throw new InvalidOperationException("The requested session does not have an open web socket.");
  83. }
  84. return socket;
  85. }
  86. public Task SendPlayCommand(PlayRequest command, CancellationToken cancellationToken)
  87. {
  88. return SendMessageInternal(new WebSocketMessage<PlayRequest>
  89. {
  90. MessageType = "Play",
  91. Data = command
  92. }, cancellationToken);
  93. }
  94. public Task SendPlaystateCommand(PlaystateRequest command, CancellationToken cancellationToken)
  95. {
  96. return SendMessageInternal(new WebSocketMessage<PlaystateRequest>
  97. {
  98. MessageType = "Playstate",
  99. Data = command
  100. }, cancellationToken);
  101. }
  102. public Task SendLibraryUpdateInfo(LibraryUpdateInfo info, CancellationToken cancellationToken)
  103. {
  104. return SendMessagesInternal(new WebSocketMessage<LibraryUpdateInfo>
  105. {
  106. MessageType = "LibraryChanged",
  107. Data = info
  108. }, cancellationToken);
  109. }
  110. /// <summary>
  111. /// Sends the restart required message.
  112. /// </summary>
  113. /// <param name="info">The information.</param>
  114. /// <param name="cancellationToken">The cancellation token.</param>
  115. /// <returns>Task.</returns>
  116. public Task SendRestartRequiredNotification(SystemInfo info, CancellationToken cancellationToken)
  117. {
  118. return SendMessagesInternal(new WebSocketMessage<SystemInfo>
  119. {
  120. MessageType = "RestartRequired",
  121. Data = info
  122. }, cancellationToken);
  123. }
  124. /// <summary>
  125. /// Sends the user data change info.
  126. /// </summary>
  127. /// <param name="info">The info.</param>
  128. /// <param name="cancellationToken">The cancellation token.</param>
  129. /// <returns>Task.</returns>
  130. public Task SendUserDataChangeInfo(UserDataChangeInfo info, CancellationToken cancellationToken)
  131. {
  132. return SendMessagesInternal(new WebSocketMessage<UserDataChangeInfo>
  133. {
  134. MessageType = "UserDataChanged",
  135. Data = info
  136. }, cancellationToken);
  137. }
  138. /// <summary>
  139. /// Sends the server shutdown notification.
  140. /// </summary>
  141. /// <param name="cancellationToken">The cancellation token.</param>
  142. /// <returns>Task.</returns>
  143. public Task SendServerShutdownNotification(CancellationToken cancellationToken)
  144. {
  145. return SendMessagesInternal(new WebSocketMessage<string>
  146. {
  147. MessageType = "ServerShuttingDown",
  148. Data = string.Empty
  149. }, cancellationToken);
  150. }
  151. /// <summary>
  152. /// Sends the server restart notification.
  153. /// </summary>
  154. /// <param name="cancellationToken">The cancellation token.</param>
  155. /// <returns>Task.</returns>
  156. public Task SendServerRestartNotification(CancellationToken cancellationToken)
  157. {
  158. return SendMessagesInternal(new WebSocketMessage<string>
  159. {
  160. MessageType = "ServerRestarting",
  161. Data = string.Empty
  162. }, cancellationToken);
  163. }
  164. public Task SendGeneralCommand(GeneralCommand command, CancellationToken cancellationToken)
  165. {
  166. return SendMessageInternal(new WebSocketMessage<GeneralCommand>
  167. {
  168. MessageType = "GeneralCommand",
  169. Data = command
  170. }, cancellationToken);
  171. }
  172. public Task SendSessionEndedNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
  173. {
  174. return SendMessagesInternal(new WebSocketMessage<SessionInfoDto>
  175. {
  176. MessageType = "SessionEnded",
  177. Data = sessionInfo
  178. }, cancellationToken);
  179. }
  180. public Task SendPlaybackStartNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
  181. {
  182. return SendMessagesInternal(new WebSocketMessage<SessionInfoDto>
  183. {
  184. MessageType = "PlaybackStart",
  185. Data = sessionInfo
  186. }, cancellationToken);
  187. }
  188. public Task SendPlaybackStoppedNotification(SessionInfoDto sessionInfo, CancellationToken cancellationToken)
  189. {
  190. return SendMessagesInternal(new WebSocketMessage<SessionInfoDto>
  191. {
  192. MessageType = "PlaybackStopped",
  193. Data = sessionInfo
  194. }, cancellationToken);
  195. }
  196. public Task SendMessage<T>(string name, T data, CancellationToken cancellationToken)
  197. {
  198. return SendMessagesInternal(new WebSocketMessage<T>
  199. {
  200. Data = data,
  201. MessageType = name
  202. }, cancellationToken);
  203. }
  204. private Task SendMessageInternal<T>(WebSocketMessage<T> message, CancellationToken cancellationToken)
  205. {
  206. var socket = GetActiveSocket();
  207. return socket.SendAsync(message, cancellationToken);
  208. }
  209. private Task SendMessagesInternal<T>(WebSocketMessage<T> message, CancellationToken cancellationToken)
  210. {
  211. var tasks = GetActiveSockets().Select(i => Task.Run(async () =>
  212. {
  213. try
  214. {
  215. await i.SendAsync(message, cancellationToken).ConfigureAwait(false);
  216. }
  217. catch (Exception ex)
  218. {
  219. _logger.ErrorException("Error sending web socket message", ex);
  220. }
  221. }, cancellationToken));
  222. return Task.WhenAll(tasks);
  223. }
  224. public void Dispose()
  225. {
  226. foreach (var socket in Sockets.ToList())
  227. {
  228. socket.Closed -= connection_Closed;
  229. }
  230. }
  231. }
  232. }