| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243 | using MediaBrowser.Common.Configuration;using MediaBrowser.Common.Extensions;using MediaBrowser.Common.IO;using MediaBrowser.Controller.Sync;using MediaBrowser.Model.Logging;using MediaBrowser.Model.Serialization;using MediaBrowser.Model.Sync;using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Threading;using System.Threading.Tasks;namespace MediaBrowser.Server.Implementations.Sync{    public class TargetDataProvider : ISyncDataProvider    {        private readonly SyncTarget _target;        private readonly IServerSyncProvider _provider;        private readonly SemaphoreSlim _dataLock = new SemaphoreSlim(1, 1);        private List<LocalItem> _items;        private readonly ILogger _logger;        private readonly IJsonSerializer _json;        private readonly IFileSystem _fileSystem;        private readonly IApplicationPaths _appPaths;        private readonly string _serverId;        private readonly SemaphoreSlim _cacheFileLock = new SemaphoreSlim(1, 1);        public TargetDataProvider(IServerSyncProvider provider, SyncTarget target, string serverId, ILogger logger, IJsonSerializer json, IFileSystem fileSystem, IApplicationPaths appPaths)        {            _logger = logger;            _json = json;            _provider = provider;            _target = target;            _fileSystem = fileSystem;            _appPaths = appPaths;            _serverId = serverId;        }        private string GetCachePath()        {            return Path.Combine(_appPaths.DataPath, "sync", _target.Id.GetMD5().ToString("N") + ".json");        }        private string GetRemotePath()        {            var parts = new List<string>            {                _serverId,                "data.json"            };            return _provider.GetFullPath(parts, _target);        }        private async Task CacheData(Stream stream)        {            var cachePath = GetCachePath();            await _cacheFileLock.WaitAsync().ConfigureAwait(false);            try            {                Directory.CreateDirectory(Path.GetDirectoryName(cachePath));                using (var fileStream = _fileSystem.GetFileStream(cachePath, FileMode.Create, FileAccess.Write, FileShare.Read, true))                {                    await stream.CopyToAsync(fileStream).ConfigureAwait(false);                }            }            catch (Exception ex)            {                _logger.ErrorException("Error saving sync data to {0}", ex, cachePath);            }            finally            {                _cacheFileLock.Release();            }        }        private async Task EnsureData(CancellationToken cancellationToken)        {            if (_items == null)            {                try                {                    using (var stream = await _provider.GetFile(GetRemotePath(), _target, new Progress<double>(), cancellationToken))                    {                        _items = _json.DeserializeFromStream<List<LocalItem>>(stream);                    }                }                catch (FileNotFoundException)                {                    _items = new List<LocalItem>();                }                catch (DirectoryNotFoundException)                {                    _items = new List<LocalItem>();                }                using (var memoryStream = new MemoryStream())                {                    _json.SerializeToStream(_items, memoryStream);                                        // Now cache it                    memoryStream.Position = 0;                    await CacheData(memoryStream).ConfigureAwait(false);                }            }        }        private async Task SaveData(CancellationToken cancellationToken)        {            using (var stream = new MemoryStream())            {                _json.SerializeToStream(_items, stream);                // Save to sync provider                stream.Position = 0;                await _provider.SendFile(stream, GetRemotePath(), _target, new Progress<double>(), cancellationToken).ConfigureAwait(false);                // Now cache it                stream.Position = 0;                await CacheData(stream).ConfigureAwait(false);            }        }        private async Task<T> GetData<T>(Func<List<LocalItem>, T> dataFactory)        {            await _dataLock.WaitAsync().ConfigureAwait(false);            try            {                await EnsureData(CancellationToken.None).ConfigureAwait(false);                return dataFactory(_items);            }            finally            {                _dataLock.Release();            }        }        private async Task UpdateData(Func<List<LocalItem>, List<LocalItem>> action)        {            await _dataLock.WaitAsync().ConfigureAwait(false);            try            {                await EnsureData(CancellationToken.None).ConfigureAwait(false);                _items = action(_items);                await SaveData(CancellationToken.None).ConfigureAwait(false);            }            finally            {                _dataLock.Release();            }        }        public Task<List<string>> GetServerItemIds(SyncTarget target, string serverId)        {            return GetData(items => items.Where(i => string.Equals(i.ServerId, serverId, StringComparison.OrdinalIgnoreCase)).Select(i => i.ItemId).ToList());        }        public Task AddOrUpdate(SyncTarget target, LocalItem item)        {            return UpdateData(items =>            {                var list = items.Where(i => !string.Equals(i.Id, item.Id, StringComparison.OrdinalIgnoreCase))                    .ToList();                list.Add(item);                return list;            });        }        public Task Delete(SyncTarget target, string id)        {            return UpdateData(items => items.Where(i => !string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase)).ToList());        }        public Task<LocalItem> Get(SyncTarget target, string id)        {            return GetData(items => items.FirstOrDefault(i => string.Equals(i.Id, id, StringComparison.OrdinalIgnoreCase)));        }        private async Task<List<LocalItem>> GetCachedData()        {            if (_items == null)            {                await _cacheFileLock.WaitAsync().ConfigureAwait(false);                try                {                    if (_items == null)                    {                        try                        {                            _items = _json.DeserializeFromFile<List<LocalItem>>(GetCachePath());                        }                        catch (FileNotFoundException)                        {                            _items = new List<LocalItem>();                        }                        catch (DirectoryNotFoundException)                        {                            _items = new List<LocalItem>();                        }                    }                }                finally                {                    _cacheFileLock.Release();                }            }            return _items.ToList();        }        public async Task<List<string>> GetCachedServerItemIds(SyncTarget target, string serverId)        {            var items = await GetCachedData().ConfigureAwait(false);            return items.Where(i => string.Equals(i.ServerId, serverId, StringComparison.OrdinalIgnoreCase))                    .Select(i => i.ItemId)                    .ToList();        }        public async Task<List<LocalItem>> GetCachedItems(SyncTarget target, string serverId, string itemId)        {            var items = await GetCachedData().ConfigureAwait(false);            return items.Where(i => string.Equals(i.ServerId, serverId, StringComparison.OrdinalIgnoreCase) && string.Equals(i.ItemId, itemId, StringComparison.OrdinalIgnoreCase))                    .ToList();        }    }}
 |