SyncPlayManager.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. using System;
  2. using System.Collections.Concurrent;
  3. using System.Collections.Generic;
  4. using System.Threading;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.Session;
  7. using MediaBrowser.Controller.SyncPlay;
  8. using MediaBrowser.Controller.SyncPlay.Requests;
  9. using MediaBrowser.Model.SyncPlay;
  10. using Microsoft.Extensions.Logging;
  11. namespace Emby.Server.Implementations.SyncPlay
  12. {
  13. /// <summary>
  14. /// Class SyncPlayManager.
  15. /// </summary>
  16. public class SyncPlayManager : ISyncPlayManager, IDisposable
  17. {
  18. /// <summary>
  19. /// The logger.
  20. /// </summary>
  21. private readonly ILogger<SyncPlayManager> _logger;
  22. /// <summary>
  23. /// The logger factory.
  24. /// </summary>
  25. private readonly ILoggerFactory _loggerFactory;
  26. /// <summary>
  27. /// The user manager.
  28. /// </summary>
  29. private readonly IUserManager _userManager;
  30. /// <summary>
  31. /// The session manager.
  32. /// </summary>
  33. private readonly ISessionManager _sessionManager;
  34. /// <summary>
  35. /// The library manager.
  36. /// </summary>
  37. private readonly ILibraryManager _libraryManager;
  38. /// <summary>
  39. /// The map between users and counter of active sessions.
  40. /// </summary>
  41. private readonly ConcurrentDictionary<Guid, int> _activeUsers =
  42. new ConcurrentDictionary<Guid, int>();
  43. /// <summary>
  44. /// The map between sessions and groups.
  45. /// </summary>
  46. private readonly ConcurrentDictionary<string, Group> _sessionToGroupMap =
  47. new ConcurrentDictionary<string, Group>(StringComparer.OrdinalIgnoreCase);
  48. /// <summary>
  49. /// The groups.
  50. /// </summary>
  51. private readonly ConcurrentDictionary<Guid, Group> _groups =
  52. new ConcurrentDictionary<Guid, Group>();
  53. /// <summary>
  54. /// Lock used for accessing multiple groups at once.
  55. /// </summary>
  56. /// <remarks>
  57. /// This lock has priority on locks made on <see cref="Group"/>.
  58. /// </remarks>
  59. private readonly object _groupsLock = new object();
  60. private bool _disposed = false;
  61. /// <summary>
  62. /// Initializes a new instance of the <see cref="SyncPlayManager" /> class.
  63. /// </summary>
  64. /// <param name="loggerFactory">The logger factory.</param>
  65. /// <param name="userManager">The user manager.</param>
  66. /// <param name="sessionManager">The session manager.</param>
  67. /// <param name="libraryManager">The library manager.</param>
  68. public SyncPlayManager(
  69. ILoggerFactory loggerFactory,
  70. IUserManager userManager,
  71. ISessionManager sessionManager,
  72. ILibraryManager libraryManager)
  73. {
  74. _loggerFactory = loggerFactory;
  75. _userManager = userManager;
  76. _sessionManager = sessionManager;
  77. _libraryManager = libraryManager;
  78. _logger = loggerFactory.CreateLogger<SyncPlayManager>();
  79. _sessionManager.SessionEnded += OnSessionEnded;
  80. }
  81. /// <inheritdoc />
  82. public void Dispose()
  83. {
  84. Dispose(true);
  85. GC.SuppressFinalize(this);
  86. }
  87. /// <inheritdoc />
  88. public void NewGroup(SessionInfo session, NewGroupRequest request, CancellationToken cancellationToken)
  89. {
  90. if (session == null)
  91. {
  92. throw new InvalidOperationException("Session is null!");
  93. }
  94. if (request == null)
  95. {
  96. throw new InvalidOperationException("Request is null!");
  97. }
  98. // Locking required to access list of groups.
  99. lock (_groupsLock)
  100. {
  101. // Make sure that session has not joined another group.
  102. if (_sessionToGroupMap.ContainsKey(session.Id))
  103. {
  104. var leaveGroupRequest = new LeaveGroupRequest();
  105. LeaveGroup(session, leaveGroupRequest, cancellationToken);
  106. }
  107. var group = new Group(_loggerFactory, _userManager, _sessionManager, _libraryManager);
  108. _groups[group.GroupId] = group;
  109. if (!_sessionToGroupMap.TryAdd(session.Id, group))
  110. {
  111. throw new InvalidOperationException("Could not add session to group!");
  112. }
  113. UpdateSessionsCounter(session.UserId, 1);
  114. group.CreateGroup(session, request, cancellationToken);
  115. }
  116. }
  117. /// <inheritdoc />
  118. public void JoinGroup(SessionInfo session, JoinGroupRequest request, CancellationToken cancellationToken)
  119. {
  120. if (session == null)
  121. {
  122. throw new InvalidOperationException("Session is null!");
  123. }
  124. if (request == null)
  125. {
  126. throw new InvalidOperationException("Request is null!");
  127. }
  128. var user = _userManager.GetUserById(session.UserId);
  129. // Locking required to access list of groups.
  130. lock (_groupsLock)
  131. {
  132. _groups.TryGetValue(request.GroupId, out Group group);
  133. if (group == null)
  134. {
  135. _logger.LogWarning("Session {SessionId} tried to join group {GroupId} that does not exist.", session.Id, request.GroupId);
  136. var error = new GroupUpdate<string>(Guid.Empty, GroupUpdateType.GroupDoesNotExist, string.Empty);
  137. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  138. return;
  139. }
  140. // Group lock required to let other requests end first.
  141. lock (group)
  142. {
  143. if (!group.HasAccessToPlayQueue(user))
  144. {
  145. _logger.LogWarning("Session {SessionId} tried to join group {GroupId} but does not have access to some content of the playing queue.", session.Id, group.GroupId.ToString());
  146. var error = new GroupUpdate<string>(group.GroupId, GroupUpdateType.LibraryAccessDenied, string.Empty);
  147. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  148. return;
  149. }
  150. if (_sessionToGroupMap.TryGetValue(session.Id, out var existingGroup))
  151. {
  152. if (existingGroup.GroupId.Equals(request.GroupId))
  153. {
  154. // Restore session.
  155. UpdateSessionsCounter(session.UserId, 1);
  156. group.SessionJoin(session, request, cancellationToken);
  157. return;
  158. }
  159. var leaveGroupRequest = new LeaveGroupRequest();
  160. LeaveGroup(session, leaveGroupRequest, cancellationToken);
  161. }
  162. if (!_sessionToGroupMap.TryAdd(session.Id, group))
  163. {
  164. throw new InvalidOperationException("Could not add session to group!");
  165. }
  166. UpdateSessionsCounter(session.UserId, 1);
  167. group.SessionJoin(session, request, cancellationToken);
  168. }
  169. }
  170. }
  171. /// <inheritdoc />
  172. public void LeaveGroup(SessionInfo session, LeaveGroupRequest request, CancellationToken cancellationToken)
  173. {
  174. if (session == null)
  175. {
  176. throw new InvalidOperationException("Session is null!");
  177. }
  178. if (request == null)
  179. {
  180. throw new InvalidOperationException("Request is null!");
  181. }
  182. // Locking required to access list of groups.
  183. lock (_groupsLock)
  184. {
  185. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  186. {
  187. // Group lock required to let other requests end first.
  188. lock (group)
  189. {
  190. if (_sessionToGroupMap.TryRemove(session.Id, out var tempGroup))
  191. {
  192. if (!tempGroup.GroupId.Equals(group.GroupId))
  193. {
  194. throw new InvalidOperationException("Session was in wrong group!");
  195. }
  196. }
  197. else
  198. {
  199. throw new InvalidOperationException("Could not remove session from group!");
  200. }
  201. UpdateSessionsCounter(session.UserId, -1);
  202. group.SessionLeave(session, request, cancellationToken);
  203. if (group.IsGroupEmpty())
  204. {
  205. _logger.LogInformation("Group {GroupId} is empty, removing it.", group.GroupId);
  206. _groups.Remove(group.GroupId, out _);
  207. }
  208. }
  209. }
  210. else
  211. {
  212. _logger.LogWarning("Session {SessionId} does not belong to any group.", session.Id);
  213. var error = new GroupUpdate<string>(Guid.Empty, GroupUpdateType.NotInGroup, string.Empty);
  214. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  215. return;
  216. }
  217. }
  218. }
  219. /// <inheritdoc />
  220. public List<GroupInfoDto> ListGroups(SessionInfo session, ListGroupsRequest request)
  221. {
  222. if (session == null)
  223. {
  224. throw new InvalidOperationException("Session is null!");
  225. }
  226. if (request == null)
  227. {
  228. throw new InvalidOperationException("Request is null!");
  229. }
  230. var user = _userManager.GetUserById(session.UserId);
  231. List<GroupInfoDto> list = new List<GroupInfoDto>();
  232. foreach (var group in _groups.Values)
  233. {
  234. // Locking required as group is not thread-safe.
  235. lock (group)
  236. {
  237. if (group.HasAccessToPlayQueue(user))
  238. {
  239. list.Add(group.GetInfo());
  240. }
  241. }
  242. }
  243. return list;
  244. }
  245. /// <inheritdoc />
  246. public void HandleRequest(SessionInfo session, IGroupPlaybackRequest request, CancellationToken cancellationToken)
  247. {
  248. if (session == null)
  249. {
  250. throw new InvalidOperationException("Session is null!");
  251. }
  252. if (request == null)
  253. {
  254. throw new InvalidOperationException("Request is null!");
  255. }
  256. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  257. {
  258. // Group lock required as Group is not thread-safe.
  259. lock (group)
  260. {
  261. // Make sure that session still belongs to this group.
  262. if (_sessionToGroupMap.TryGetValue(session.Id, out var checkGroup) && !checkGroup.GroupId.Equals(group.GroupId))
  263. {
  264. // Drop request.
  265. return;
  266. }
  267. // Drop request if group is empty.
  268. if (group.IsGroupEmpty())
  269. {
  270. return;
  271. }
  272. // Apply requested changes to group.
  273. group.HandleRequest(session, request, cancellationToken);
  274. }
  275. }
  276. else
  277. {
  278. _logger.LogWarning("Session {SessionId} does not belong to any group.", session.Id);
  279. var error = new GroupUpdate<string>(Guid.Empty, GroupUpdateType.NotInGroup, string.Empty);
  280. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  281. }
  282. }
  283. /// <inheritdoc />
  284. public bool IsUserActive(Guid userId)
  285. {
  286. if (_activeUsers.TryGetValue(userId, out var sessionsCounter))
  287. {
  288. return sessionsCounter > 0;
  289. }
  290. else
  291. {
  292. return false;
  293. }
  294. }
  295. /// <summary>
  296. /// Releases unmanaged and optionally managed resources.
  297. /// </summary>
  298. /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  299. protected virtual void Dispose(bool disposing)
  300. {
  301. if (_disposed)
  302. {
  303. return;
  304. }
  305. _sessionManager.SessionEnded -= OnSessionEnded;
  306. _disposed = true;
  307. }
  308. private void OnSessionEnded(object sender, SessionEventArgs e)
  309. {
  310. var session = e.SessionInfo;
  311. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  312. {
  313. var leaveGroupRequest = new LeaveGroupRequest();
  314. LeaveGroup(session, leaveGroupRequest, CancellationToken.None);
  315. }
  316. }
  317. private void UpdateSessionsCounter(Guid userId, int toAdd)
  318. {
  319. // Update sessions counter.
  320. var newSessionsCounter = _activeUsers.AddOrUpdate(
  321. userId,
  322. 1,
  323. (key, sessionsCounter) => sessionsCounter + toAdd);
  324. // Should never happen.
  325. if (newSessionsCounter < 0)
  326. {
  327. throw new InvalidOperationException("Sessions counter is negative!");
  328. }
  329. // Clean record if user has no more active sessions.
  330. if (newSessionsCounter == 0)
  331. {
  332. _activeUsers.TryRemove(new KeyValuePair<Guid, int>(userId, newSessionsCounter));
  333. }
  334. }
  335. }
  336. }