SyncPlayManager.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  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 GroupInfoDto 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. return group.GetInfo();
  117. }
  118. }
  119. /// <inheritdoc />
  120. public void JoinGroup(SessionInfo session, JoinGroupRequest request, CancellationToken cancellationToken)
  121. {
  122. if (session is null)
  123. {
  124. throw new InvalidOperationException("Session is null!");
  125. }
  126. if (request is null)
  127. {
  128. throw new InvalidOperationException("Request is null!");
  129. }
  130. var user = _userManager.GetUserById(session.UserId);
  131. // Locking required to access list of groups.
  132. lock (_groupsLock)
  133. {
  134. _groups.TryGetValue(request.GroupId, out Group group);
  135. if (group is null)
  136. {
  137. _logger.LogWarning("Session {SessionId} tried to join group {GroupId} that does not exist.", session.Id, request.GroupId);
  138. var error = new SyncPlayGroupDoesNotExistUpdate(Guid.Empty, string.Empty);
  139. _sessionManager.SendSyncPlayGroupUpdate(session.Id, error, CancellationToken.None);
  140. return;
  141. }
  142. // Group lock required to let other requests end first.
  143. lock (group)
  144. {
  145. if (!group.HasAccessToPlayQueue(user))
  146. {
  147. _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());
  148. var error = new SyncPlayLibraryAccessDeniedUpdate(group.GroupId, string.Empty);
  149. _sessionManager.SendSyncPlayGroupUpdate(session.Id, error, CancellationToken.None);
  150. return;
  151. }
  152. if (_sessionToGroupMap.TryGetValue(session.Id, out var existingGroup))
  153. {
  154. if (existingGroup.GroupId.Equals(request.GroupId))
  155. {
  156. // Restore session.
  157. UpdateSessionsCounter(session.UserId, 1);
  158. group.SessionJoin(session, request, cancellationToken);
  159. return;
  160. }
  161. var leaveGroupRequest = new LeaveGroupRequest();
  162. LeaveGroup(session, leaveGroupRequest, cancellationToken);
  163. }
  164. if (!_sessionToGroupMap.TryAdd(session.Id, group))
  165. {
  166. throw new InvalidOperationException("Could not add session to group!");
  167. }
  168. UpdateSessionsCounter(session.UserId, 1);
  169. group.SessionJoin(session, request, cancellationToken);
  170. }
  171. }
  172. }
  173. /// <inheritdoc />
  174. public void LeaveGroup(SessionInfo session, LeaveGroupRequest request, CancellationToken cancellationToken)
  175. {
  176. if (session is null)
  177. {
  178. throw new InvalidOperationException("Session is null!");
  179. }
  180. if (request is null)
  181. {
  182. throw new InvalidOperationException("Request is null!");
  183. }
  184. // Locking required to access list of groups.
  185. lock (_groupsLock)
  186. {
  187. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  188. {
  189. // Group lock required to let other requests end first.
  190. lock (group)
  191. {
  192. if (_sessionToGroupMap.TryRemove(session.Id, out var tempGroup))
  193. {
  194. if (!tempGroup.GroupId.Equals(group.GroupId))
  195. {
  196. throw new InvalidOperationException("Session was in wrong group!");
  197. }
  198. }
  199. else
  200. {
  201. throw new InvalidOperationException("Could not remove session from group!");
  202. }
  203. UpdateSessionsCounter(session.UserId, -1);
  204. group.SessionLeave(session, request, cancellationToken);
  205. if (group.IsGroupEmpty())
  206. {
  207. _logger.LogInformation("Group {GroupId} is empty, removing it.", group.GroupId);
  208. _groups.Remove(group.GroupId, out _);
  209. }
  210. }
  211. }
  212. else
  213. {
  214. _logger.LogWarning("Session {SessionId} does not belong to any group.", session.Id);
  215. var error = new SyncPlayNotInGroupUpdate(Guid.Empty, string.Empty);
  216. _sessionManager.SendSyncPlayGroupUpdate(session.Id, error, CancellationToken.None);
  217. }
  218. }
  219. }
  220. /// <inheritdoc />
  221. public List<GroupInfoDto> ListGroups(SessionInfo session, ListGroupsRequest request)
  222. {
  223. if (session is null)
  224. {
  225. throw new InvalidOperationException("Session is null!");
  226. }
  227. if (request is 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 GroupInfoDto GetGroup(SessionInfo session, Guid groupId)
  251. {
  252. ArgumentNullException.ThrowIfNull(session);
  253. var user = _userManager.GetUserById(session.UserId);
  254. lock (_groupsLock)
  255. {
  256. foreach (var (_, group) in _groups)
  257. {
  258. // Locking required as group is not thread-safe.
  259. lock (group)
  260. {
  261. if (group.GroupId.Equals(groupId) && group.HasAccessToPlayQueue(user))
  262. {
  263. return group.GetInfo();
  264. }
  265. }
  266. }
  267. }
  268. return null;
  269. }
  270. /// <inheritdoc />
  271. public void HandleRequest(SessionInfo session, IGroupPlaybackRequest request, CancellationToken cancellationToken)
  272. {
  273. if (session is null)
  274. {
  275. throw new InvalidOperationException("Session is null!");
  276. }
  277. if (request is null)
  278. {
  279. throw new InvalidOperationException("Request is null!");
  280. }
  281. if (_sessionToGroupMap.TryGetValue(session.Id, out var group))
  282. {
  283. // Group lock required as Group is not thread-safe.
  284. lock (group)
  285. {
  286. // Make sure that session still belongs to this group.
  287. if (_sessionToGroupMap.TryGetValue(session.Id, out var checkGroup) && !checkGroup.GroupId.Equals(group.GroupId))
  288. {
  289. // Drop request.
  290. return;
  291. }
  292. // Drop request if group is empty.
  293. if (group.IsGroupEmpty())
  294. {
  295. return;
  296. }
  297. // Apply requested changes to group.
  298. group.HandleRequest(session, request, cancellationToken);
  299. }
  300. }
  301. else
  302. {
  303. _logger.LogWarning("Session {SessionId} does not belong to any group.", session.Id);
  304. var error = new SyncPlayNotInGroupUpdate(Guid.Empty, string.Empty);
  305. _sessionManager.SendSyncPlayGroupUpdate(session.Id, error, CancellationToken.None);
  306. }
  307. }
  308. /// <inheritdoc />
  309. public bool IsUserActive(Guid userId)
  310. {
  311. if (_activeUsers.TryGetValue(userId, out var sessionsCounter))
  312. {
  313. return sessionsCounter > 0;
  314. }
  315. return false;
  316. }
  317. /// <summary>
  318. /// Releases unmanaged and optionally managed resources.
  319. /// </summary>
  320. /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  321. protected virtual void Dispose(bool disposing)
  322. {
  323. if (_disposed)
  324. {
  325. return;
  326. }
  327. _sessionManager.SessionEnded -= OnSessionEnded;
  328. _disposed = true;
  329. }
  330. private void OnSessionEnded(object sender, SessionEventArgs e)
  331. {
  332. var session = e.SessionInfo;
  333. if (_sessionToGroupMap.TryGetValue(session.Id, out _))
  334. {
  335. var leaveGroupRequest = new LeaveGroupRequest();
  336. LeaveGroup(session, leaveGroupRequest, CancellationToken.None);
  337. }
  338. }
  339. private void UpdateSessionsCounter(Guid userId, int toAdd)
  340. {
  341. // Update sessions counter.
  342. var newSessionsCounter = _activeUsers.AddOrUpdate(
  343. userId,
  344. 1,
  345. (_, sessionsCounter) => sessionsCounter + toAdd);
  346. // Should never happen.
  347. if (newSessionsCounter < 0)
  348. {
  349. throw new InvalidOperationException("Sessions counter is negative!");
  350. }
  351. // Clean record if user has no more active sessions.
  352. if (newSessionsCounter == 0)
  353. {
  354. _activeUsers.TryRemove(new KeyValuePair<Guid, int>(userId, newSessionsCounter));
  355. }
  356. }
  357. }
  358. }