SyncPlayManager.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  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. lock (_groupsLock)
  233. {
  234. foreach (var (_, group) in _groups)
  235. {
  236. // Locking required as group is not thread-safe.
  237. lock (group)
  238. {
  239. if (group.HasAccessToPlayQueue(user))
  240. {
  241. list.Add(group.GetInfo());
  242. }
  243. }
  244. }
  245. }
  246. return list;
  247. }
  248. /// <inheritdoc />
  249. public void HandleRequest(SessionInfo session, IGroupPlaybackRequest request, CancellationToken cancellationToken)
  250. {
  251. if (session == null)
  252. {
  253. throw new InvalidOperationException("Session is null!");
  254. }
  255. if (request == null)
  256. {
  257. throw new InvalidOperationException("Request is null!");
  258. }
  259. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  260. {
  261. // Group lock required as Group is not thread-safe.
  262. lock (group)
  263. {
  264. // Make sure that session still belongs to this group.
  265. if (_sessionToGroupMap.TryGetValue(session.Id, out var checkGroup) && !checkGroup.GroupId.Equals(group.GroupId))
  266. {
  267. // Drop request.
  268. return;
  269. }
  270. // Drop request if group is empty.
  271. if (group.IsGroupEmpty())
  272. {
  273. return;
  274. }
  275. // Apply requested changes to group.
  276. group.HandleRequest(session, request, cancellationToken);
  277. }
  278. }
  279. else
  280. {
  281. _logger.LogWarning("Session {SessionId} does not belong to any group.", session.Id);
  282. var error = new GroupUpdate<string>(Guid.Empty, GroupUpdateType.NotInGroup, string.Empty);
  283. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  284. }
  285. }
  286. /// <inheritdoc />
  287. public bool IsUserActive(Guid userId)
  288. {
  289. if (_activeUsers.TryGetValue(userId, out var sessionsCounter))
  290. {
  291. return sessionsCounter > 0;
  292. }
  293. else
  294. {
  295. return false;
  296. }
  297. }
  298. /// <summary>
  299. /// Releases unmanaged and optionally managed resources.
  300. /// </summary>
  301. /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  302. protected virtual void Dispose(bool disposing)
  303. {
  304. if (_disposed)
  305. {
  306. return;
  307. }
  308. _sessionManager.SessionEnded -= OnSessionEnded;
  309. _disposed = true;
  310. }
  311. private void OnSessionEnded(object sender, SessionEventArgs e)
  312. {
  313. var session = e.SessionInfo;
  314. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  315. {
  316. var leaveGroupRequest = new LeaveGroupRequest();
  317. LeaveGroup(session, leaveGroupRequest, CancellationToken.None);
  318. }
  319. }
  320. private void UpdateSessionsCounter(Guid userId, int toAdd)
  321. {
  322. // Update sessions counter.
  323. var newSessionsCounter = _activeUsers.AddOrUpdate(
  324. userId,
  325. 1,
  326. (key, sessionsCounter) => sessionsCounter + toAdd);
  327. // Should never happen.
  328. if (newSessionsCounter < 0)
  329. {
  330. throw new InvalidOperationException("Sessions counter is negative!");
  331. }
  332. // Clean record if user has no more active sessions.
  333. if (newSessionsCounter == 0)
  334. {
  335. _activeUsers.TryRemove(new KeyValuePair<Guid, int>(userId, newSessionsCounter));
  336. }
  337. }
  338. }
  339. }