SyncPlayManager.cs 14 KB

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