SyncPlayManager.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  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 Lock _groupsLock = new();
  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 is null)
  92. {
  93. throw new InvalidOperationException("Session is null!");
  94. }
  95. if (request is 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 is null)
  122. {
  123. throw new InvalidOperationException("Session is null!");
  124. }
  125. if (request is 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 is 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.Id, 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.Id, 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 is null)
  176. {
  177. throw new InvalidOperationException("Session is null!");
  178. }
  179. if (request is 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.Id, error, CancellationToken.None);
  216. }
  217. }
  218. }
  219. /// <inheritdoc />
  220. public List<GroupInfoDto> ListGroups(SessionInfo session, ListGroupsRequest request)
  221. {
  222. if (session is null)
  223. {
  224. throw new InvalidOperationException("Session is null!");
  225. }
  226. if (request is 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 is null)
  252. {
  253. throw new InvalidOperationException("Session is null!");
  254. }
  255. if (request is 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.Id, 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. return false;
  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 _))
  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. (_, 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. }