UserDataChangeNotifier.cs 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. using MediaBrowser.Controller.Dto;
  2. using MediaBrowser.Controller.Library;
  3. using MediaBrowser.Controller.Plugins;
  4. using MediaBrowser.Controller.Session;
  5. using MediaBrowser.Model.Entities;
  6. using MediaBrowser.Model.Logging;
  7. using MediaBrowser.Model.Session;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Linq;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. namespace MediaBrowser.Server.Implementations.EntryPoints
  14. {
  15. class UserDataChangeNotifier : IServerEntryPoint
  16. {
  17. private readonly ISessionManager _sessionManager;
  18. private readonly ILogger _logger;
  19. private readonly IDtoService _dtoService;
  20. private readonly IUserDataManager _userDataManager;
  21. private readonly object _syncLock = new object();
  22. private Timer UpdateTimer { get; set; }
  23. private const int UpdateDuration = 500;
  24. private readonly Dictionary<Guid, List<string>> _changedKeys = new Dictionary<Guid, List<string>>();
  25. public UserDataChangeNotifier(IUserDataManager userDataManager, ISessionManager sessionManager, IDtoService dtoService, ILogger logger)
  26. {
  27. _userDataManager = userDataManager;
  28. _sessionManager = sessionManager;
  29. _dtoService = dtoService;
  30. _logger = logger;
  31. }
  32. public void Run()
  33. {
  34. _userDataManager.UserDataSaved += _userDataManager_UserDataSaved;
  35. }
  36. void _userDataManager_UserDataSaved(object sender, UserDataSaveEventArgs e)
  37. {
  38. if (e.SaveReason == UserDataSaveReason.PlaybackProgress)
  39. {
  40. return;
  41. }
  42. lock (_syncLock)
  43. {
  44. if (UpdateTimer == null)
  45. {
  46. UpdateTimer = new Timer(UpdateTimerCallback, null, UpdateDuration,
  47. Timeout.Infinite);
  48. }
  49. else
  50. {
  51. UpdateTimer.Change(UpdateDuration, Timeout.Infinite);
  52. }
  53. List<string> keys;
  54. if (!_changedKeys.TryGetValue(e.UserId, out keys))
  55. {
  56. keys = new List<string>();
  57. _changedKeys[e.UserId] = keys;
  58. }
  59. keys.Add(e.Key);
  60. }
  61. }
  62. private void UpdateTimerCallback(object state)
  63. {
  64. lock (_syncLock)
  65. {
  66. // Remove dupes in case some were saved multiple times
  67. var changes = _changedKeys.ToList();
  68. _changedKeys.Clear();
  69. SendNotifications(changes, CancellationToken.None);
  70. if (UpdateTimer != null)
  71. {
  72. UpdateTimer.Dispose();
  73. UpdateTimer = null;
  74. }
  75. }
  76. }
  77. private async Task SendNotifications(List<KeyValuePair<Guid, List<string>>> changes, CancellationToken cancellationToken)
  78. {
  79. foreach (var pair in changes)
  80. {
  81. var userId = pair.Key;
  82. var userSessions = _sessionManager.Sessions
  83. .Where(u => u.User != null && u.User.Id == userId && u.SessionController != null && u.IsActive)
  84. .ToList();
  85. if (userSessions.Count > 0)
  86. {
  87. var dtoList = pair.Value
  88. .Select(i => _dtoService.GetUserItemDataDto(_userDataManager.GetUserData(userId, i)))
  89. .ToList();
  90. var info = new UserDataChangeInfo
  91. {
  92. UserId = userId.ToString("N"),
  93. UserDataList = dtoList
  94. };
  95. foreach (var userSession in userSessions)
  96. {
  97. try
  98. {
  99. await userSession.SessionController.SendUserDataChangeInfo(info, cancellationToken).ConfigureAwait(false);
  100. }
  101. catch (Exception ex)
  102. {
  103. _logger.ErrorException("Error sending UserDataChanged message", ex);
  104. }
  105. }
  106. }
  107. }
  108. }
  109. public void Dispose()
  110. {
  111. if (UpdateTimer != null)
  112. {
  113. UpdateTimer.Dispose();
  114. UpdateTimer = null;
  115. }
  116. _userDataManager.UserDataSaved -= _userDataManager_UserDataSaved;
  117. }
  118. }
  119. }