| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 | #pragma warning disable CS1591using System;using System.Collections.Generic;using System.Globalization;using System.Linq;using System.Threading;using System.Threading.Tasks;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.Plugins;using MediaBrowser.Controller.Session;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Session;using Microsoft.Extensions.Logging;namespace Emby.Server.Implementations.EntryPoints{    public class UserDataChangeNotifier : IServerEntryPoint    {        private readonly ISessionManager _sessionManager;        private readonly ILogger _logger;        private readonly IUserDataManager _userDataManager;        private readonly IUserManager _userManager;        private readonly object _syncLock = new object();        private Timer UpdateTimer { get; set; }        private const int UpdateDuration = 500;        private readonly Dictionary<Guid, List<BaseItem>> _changedItems = new Dictionary<Guid, List<BaseItem>>();        public UserDataChangeNotifier(IUserDataManager userDataManager, ISessionManager sessionManager, ILogger logger, IUserManager userManager)        {            _userDataManager = userDataManager;            _sessionManager = sessionManager;            _logger = logger;            _userManager = userManager;        }        public Task RunAsync()        {            _userDataManager.UserDataSaved += _userDataManager_UserDataSaved;            return Task.CompletedTask;        }        void _userDataManager_UserDataSaved(object sender, UserDataSaveEventArgs e)        {            if (e.SaveReason == UserDataSaveReason.PlaybackProgress)            {                return;            }            lock (_syncLock)            {                if (UpdateTimer == null)                {                    UpdateTimer = new Timer(UpdateTimerCallback, null, UpdateDuration,                                                   Timeout.Infinite);                }                else                {                    UpdateTimer.Change(UpdateDuration, Timeout.Infinite);                }                if (!_changedItems.TryGetValue(e.UserId, out List<BaseItem> keys))                {                    keys = new List<BaseItem>();                    _changedItems[e.UserId] = keys;                }                keys.Add(e.Item);                var baseItem = e.Item;                // Go up one level for indicators                if (baseItem != null)                {                    var parent = baseItem.GetOwner() ?? baseItem.GetParent();                    if (parent != null)                    {                        keys.Add(parent);                    }                }            }        }        private void UpdateTimerCallback(object state)        {            lock (_syncLock)            {                // Remove dupes in case some were saved multiple times                var changes = _changedItems.ToList();                _changedItems.Clear();                var task = SendNotifications(changes, CancellationToken.None);                if (UpdateTimer != null)                {                    UpdateTimer.Dispose();                    UpdateTimer = null;                }            }        }        private async Task SendNotifications(List<KeyValuePair<Guid, List<BaseItem>>> changes, CancellationToken cancellationToken)        {            foreach (var pair in changes)            {                await SendNotifications(pair.Key, pair.Value, cancellationToken).ConfigureAwait(false);            }        }        private Task SendNotifications(Guid userId, List<BaseItem> changedItems, CancellationToken cancellationToken)        {            return _sessionManager.SendMessageToUserSessions(new List<Guid> { userId }, "UserDataChanged", () => GetUserDataChangeInfo(userId, changedItems), cancellationToken);        }        private UserDataChangeInfo GetUserDataChangeInfo(Guid userId, List<BaseItem> changedItems)        {            var user = _userManager.GetUserById(userId);            var dtoList = changedItems                .GroupBy(x => x.Id)                .Select(x => x.First())                .Select(i =>                {                    var dto = _userDataManager.GetUserDataDto(i, user);                    dto.ItemId = i.Id.ToString("N", CultureInfo.InvariantCulture);                    return dto;                })                .ToArray();            var userIdString = userId.ToString("N", CultureInfo.InvariantCulture);            return new UserDataChangeInfo            {                UserId = userIdString,                UserDataList = dtoList            };        }        public void Dispose()        {            if (UpdateTimer != null)            {                UpdateTimer.Dispose();                UpdateTimer = null;            }            _userDataManager.UserDataSaved -= _userDataManager_UserDataSaved;        }    }}
 |