SyncPlayManager.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading;
  5. using Jellyfin.Data.Enums;
  6. using MediaBrowser.Controller.Library;
  7. using MediaBrowser.Controller.Session;
  8. using MediaBrowser.Controller.SyncPlay;
  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 user manager.
  24. /// </summary>
  25. private readonly IUserManager _userManager;
  26. /// <summary>
  27. /// The session manager.
  28. /// </summary>
  29. private readonly ISessionManager _sessionManager;
  30. /// <summary>
  31. /// The library manager.
  32. /// </summary>
  33. private readonly ILibraryManager _libraryManager;
  34. /// <summary>
  35. /// The map between sessions and groups.
  36. /// </summary>
  37. private readonly Dictionary<string, IGroupController> _sessionToGroupMap =
  38. new Dictionary<string, IGroupController>(StringComparer.OrdinalIgnoreCase);
  39. /// <summary>
  40. /// The groups.
  41. /// </summary>
  42. private readonly Dictionary<Guid, IGroupController> _groups =
  43. new Dictionary<Guid, IGroupController>();
  44. /// <summary>
  45. /// Lock used for accesing any group.
  46. /// </summary>
  47. private readonly object _groupsLock = new object();
  48. private bool _disposed = false;
  49. /// <summary>
  50. /// Initializes a new instance of the <see cref="SyncPlayManager" /> class.
  51. /// </summary>
  52. /// <param name="logger">The logger.</param>
  53. /// <param name="userManager">The user manager.</param>
  54. /// <param name="sessionManager">The session manager.</param>
  55. /// <param name="libraryManager">The library manager.</param>
  56. public SyncPlayManager(
  57. ILogger<SyncPlayManager> logger,
  58. IUserManager userManager,
  59. ISessionManager sessionManager,
  60. ILibraryManager libraryManager)
  61. {
  62. _logger = logger;
  63. _userManager = userManager;
  64. _sessionManager = sessionManager;
  65. _libraryManager = libraryManager;
  66. _sessionManager.SessionStarted += OnSessionManagerSessionStarted;
  67. _sessionManager.SessionEnded += OnSessionManagerSessionEnded;
  68. _sessionManager.PlaybackStart += OnSessionManagerPlaybackStart;
  69. _sessionManager.PlaybackStopped += OnSessionManagerPlaybackStopped;
  70. }
  71. /// <summary>
  72. /// Gets all groups.
  73. /// </summary>
  74. /// <value>All groups.</value>
  75. public IEnumerable<IGroupController> Groups => _groups.Values;
  76. /// <inheritdoc />
  77. public void Dispose()
  78. {
  79. Dispose(true);
  80. GC.SuppressFinalize(this);
  81. }
  82. /// <summary>
  83. /// Releases unmanaged and optionally managed resources.
  84. /// </summary>
  85. /// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  86. protected virtual void Dispose(bool disposing)
  87. {
  88. if (_disposed)
  89. {
  90. return;
  91. }
  92. _sessionManager.SessionStarted -= OnSessionManagerSessionStarted;
  93. _sessionManager.SessionEnded -= OnSessionManagerSessionEnded;
  94. _sessionManager.PlaybackStart -= OnSessionManagerPlaybackStart;
  95. _sessionManager.PlaybackStopped -= OnSessionManagerPlaybackStopped;
  96. _disposed = true;
  97. }
  98. private void OnSessionManagerSessionStarted(object sender, SessionEventArgs e)
  99. {
  100. var session = e.SessionInfo;
  101. if (!IsSessionInGroup(session))
  102. {
  103. return;
  104. }
  105. var groupId = GetSessionGroup(session) ?? Guid.Empty;
  106. var request = new JoinGroupRequest()
  107. {
  108. GroupId = groupId
  109. };
  110. JoinGroup(session, groupId, request, CancellationToken.None);
  111. }
  112. private void OnSessionManagerSessionEnded(object sender, SessionEventArgs e)
  113. {
  114. var session = e.SessionInfo;
  115. if (!IsSessionInGroup(session))
  116. {
  117. return;
  118. }
  119. // TODO: probably remove this event, not used at the moment.
  120. }
  121. private void OnSessionManagerPlaybackStart(object sender, PlaybackProgressEventArgs e)
  122. {
  123. var session = e.Session;
  124. if (!IsSessionInGroup(session))
  125. {
  126. return;
  127. }
  128. // TODO: probably remove this event, not used at the moment.
  129. }
  130. private void OnSessionManagerPlaybackStopped(object sender, PlaybackStopEventArgs e)
  131. {
  132. var session = e.Session;
  133. if (!IsSessionInGroup(session))
  134. {
  135. return;
  136. }
  137. // TODO: probably remove this event, not used at the moment.
  138. }
  139. private bool IsRequestValid<T>(SessionInfo session, GroupRequestType requestType, T request, bool checkRequest = true)
  140. {
  141. if (session == null || (request == null && checkRequest))
  142. {
  143. return false;
  144. }
  145. var user = _userManager.GetUserById(session.UserId);
  146. if (user.SyncPlayAccess == SyncPlayAccess.None)
  147. {
  148. _logger.LogWarning("IsRequestValid: {0} does not have access to SyncPlay. Requested {1}.", session.Id, requestType);
  149. var error = new GroupUpdate<string>()
  150. {
  151. // TODO: rename to a more generic error. Next PR will fix this.
  152. Type = GroupUpdateType.JoinGroupDenied
  153. };
  154. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  155. return false;
  156. }
  157. if (requestType.Equals(GroupRequestType.NewGroup) && user.SyncPlayAccess != SyncPlayAccess.CreateAndJoinGroups)
  158. {
  159. _logger.LogWarning("IsRequestValid: {0} does not have permission to create groups.", session.Id);
  160. var error = new GroupUpdate<string>
  161. {
  162. Type = GroupUpdateType.CreateGroupDenied
  163. };
  164. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  165. return false;
  166. }
  167. return true;
  168. }
  169. private bool IsRequestValid(SessionInfo session, GroupRequestType requestType)
  170. {
  171. return IsRequestValid(session, requestType, session, false);
  172. }
  173. private bool IsSessionInGroup(SessionInfo session)
  174. {
  175. return _sessionToGroupMap.ContainsKey(session.Id);
  176. }
  177. private Guid? GetSessionGroup(SessionInfo session)
  178. {
  179. _sessionToGroupMap.TryGetValue(session.Id, out var group);
  180. return group?.GroupId;
  181. }
  182. /// <inheritdoc />
  183. public void NewGroup(SessionInfo session, NewGroupRequest request, CancellationToken cancellationToken)
  184. {
  185. // TODO: create abstract class for GroupRequests to avoid explicit request type here.
  186. if (!IsRequestValid(session, GroupRequestType.NewGroup, request))
  187. {
  188. return;
  189. }
  190. lock (_groupsLock)
  191. {
  192. if (IsSessionInGroup(session))
  193. {
  194. LeaveGroup(session, cancellationToken);
  195. }
  196. var group = new GroupController(_logger, _userManager, _sessionManager, _libraryManager, this);
  197. _groups[group.GroupId] = group;
  198. group.CreateGroup(session, request, cancellationToken);
  199. }
  200. }
  201. /// <inheritdoc />
  202. public void JoinGroup(SessionInfo session, Guid groupId, JoinGroupRequest request, CancellationToken cancellationToken)
  203. {
  204. // TODO: create abstract class for GroupRequests to avoid explicit request type here.
  205. if (!IsRequestValid(session, GroupRequestType.JoinGroup, request))
  206. {
  207. return;
  208. }
  209. var user = _userManager.GetUserById(session.UserId);
  210. lock (_groupsLock)
  211. {
  212. _groups.TryGetValue(groupId, out IGroupController group);
  213. if (group == null)
  214. {
  215. _logger.LogWarning("JoinGroup: {0} tried to join group {0} that does not exist.", session.Id, groupId);
  216. var error = new GroupUpdate<string>()
  217. {
  218. Type = GroupUpdateType.GroupDoesNotExist
  219. };
  220. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  221. return;
  222. }
  223. if (!group.HasAccessToPlayQueue(user))
  224. {
  225. _logger.LogWarning("JoinGroup: {0} does not have access to some content from the playing queue of group {1}.", session.Id, group.GroupId.ToString());
  226. var error = new GroupUpdate<string>()
  227. {
  228. GroupId = group.GroupId.ToString(),
  229. Type = GroupUpdateType.LibraryAccessDenied
  230. };
  231. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  232. return;
  233. }
  234. if (IsSessionInGroup(session))
  235. {
  236. if (GetSessionGroup(session).Equals(groupId))
  237. {
  238. group.SessionRestore(session, request, cancellationToken);
  239. return;
  240. }
  241. LeaveGroup(session, cancellationToken);
  242. }
  243. group.SessionJoin(session, request, cancellationToken);
  244. }
  245. }
  246. /// <inheritdoc />
  247. public void LeaveGroup(SessionInfo session, CancellationToken cancellationToken)
  248. {
  249. // TODO: create abstract class for GroupRequests to avoid explicit request type here.
  250. if (!IsRequestValid(session, GroupRequestType.LeaveGroup))
  251. {
  252. return;
  253. }
  254. // TODO: determine what happens to users that are in a group and get their permissions revoked.
  255. lock (_groupsLock)
  256. {
  257. _sessionToGroupMap.TryGetValue(session.Id, out var group);
  258. if (group == null)
  259. {
  260. _logger.LogWarning("LeaveGroup: {0} does not belong to any group.", session.Id);
  261. var error = new GroupUpdate<string>()
  262. {
  263. Type = GroupUpdateType.NotInGroup
  264. };
  265. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  266. return;
  267. }
  268. group.SessionLeave(session, cancellationToken);
  269. if (group.IsGroupEmpty())
  270. {
  271. _logger.LogInformation("LeaveGroup: removing empty group {0}.", group.GroupId);
  272. _groups.Remove(group.GroupId, out _);
  273. }
  274. }
  275. }
  276. /// <inheritdoc />
  277. public List<GroupInfoDto> ListGroups(SessionInfo session)
  278. {
  279. // TODO: create abstract class for GroupRequests to avoid explicit request type here.
  280. if (!IsRequestValid(session, GroupRequestType.ListGroups))
  281. {
  282. return new List<GroupInfoDto>();
  283. }
  284. var user = _userManager.GetUserById(session.UserId);
  285. return _groups
  286. .Values
  287. .Where(group => group.HasAccessToPlayQueue(user))
  288. .Select(group => group.GetInfo())
  289. .ToList();
  290. }
  291. /// <inheritdoc />
  292. public void HandleRequest(SessionInfo session, IGroupPlaybackRequest request, CancellationToken cancellationToken)
  293. {
  294. // TODO: create abstract class for GroupRequests to avoid explicit request type here.
  295. if (!IsRequestValid(session, GroupRequestType.Playback, request))
  296. {
  297. return;
  298. }
  299. lock (_groupsLock)
  300. {
  301. _sessionToGroupMap.TryGetValue(session.Id, out var group);
  302. if (group == null)
  303. {
  304. _logger.LogWarning("HandleRequest: {0} does not belong to any group.", session.Id);
  305. var error = new GroupUpdate<string>()
  306. {
  307. Type = GroupUpdateType.NotInGroup
  308. };
  309. _sessionManager.SendSyncPlayGroupUpdate(session, error, CancellationToken.None);
  310. return;
  311. }
  312. group.HandleRequest(session, request, cancellationToken);
  313. }
  314. }
  315. /// <inheritdoc />
  316. public void AddSessionToGroup(SessionInfo session, IGroupController group)
  317. {
  318. if (session == null)
  319. {
  320. throw new InvalidOperationException("Session is null!");
  321. }
  322. if (IsSessionInGroup(session))
  323. {
  324. throw new InvalidOperationException("Session in other group already!");
  325. }
  326. _sessionToGroupMap[session.Id] = group ?? throw new InvalidOperationException("Group is null!");
  327. }
  328. /// <inheritdoc />
  329. public void RemoveSessionFromGroup(SessionInfo session, IGroupController group)
  330. {
  331. if (session == null)
  332. {
  333. throw new InvalidOperationException("Session is null!");
  334. }
  335. if (group == null)
  336. {
  337. throw new InvalidOperationException("Group is null!");
  338. }
  339. if (!IsSessionInGroup(session))
  340. {
  341. throw new InvalidOperationException("Session not in any group!");
  342. }
  343. _sessionToGroupMap.Remove(session.Id, out var tempGroup);
  344. if (!tempGroup.GroupId.Equals(group.GroupId))
  345. {
  346. throw new InvalidOperationException("Session was in wrong group!");
  347. }
  348. }
  349. }
  350. }