1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510 |
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Common.IO;
- using MediaBrowser.Common.Progress;
- using MediaBrowser.Common.ScheduledTasks;
- using MediaBrowser.Controller.Configuration;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Entities.Audio;
- using MediaBrowser.Controller.Entities.TV;
- using MediaBrowser.Controller.IO;
- using MediaBrowser.Controller.Library;
- using MediaBrowser.Controller.Persistence;
- using MediaBrowser.Controller.Providers;
- using MediaBrowser.Controller.Resolvers;
- using MediaBrowser.Controller.Sorting;
- using MediaBrowser.Model.Configuration;
- using MediaBrowser.Model.Entities;
- using MediaBrowser.Model.Logging;
- using MediaBrowser.Server.Implementations.Library.Validators;
- using MediaBrowser.Server.Implementations.ScheduledTasks;
- using MoreLinq;
- using System;
- using System.Collections.Concurrent;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- using System.Linq;
- using System.Threading;
- using System.Threading.Tasks;
- using SortOrder = MediaBrowser.Model.Entities.SortOrder;
- namespace MediaBrowser.Server.Implementations.Library
- {
- /// <summary>
- /// Class LibraryManager
- /// </summary>
- public class LibraryManager : ILibraryManager
- {
- /// <summary>
- /// Gets or sets the postscan tasks.
- /// </summary>
- /// <value>The postscan tasks.</value>
- private ILibraryPostScanTask[] PostscanTasks { get; set; }
- /// <summary>
- /// Gets the intro providers.
- /// </summary>
- /// <value>The intro providers.</value>
- private IIntroProvider[] IntroProviders { get; set; }
- /// <summary>
- /// Gets the list of entity resolution ignore rules
- /// </summary>
- /// <value>The entity resolution ignore rules.</value>
- private IResolverIgnoreRule[] EntityResolutionIgnoreRules { get; set; }
- /// <summary>
- /// Gets the list of BasePluginFolders added by plugins
- /// </summary>
- /// <value>The plugin folders.</value>
- private IVirtualFolderCreator[] PluginFolderCreators { get; set; }
- /// <summary>
- /// Gets the list of currently registered entity resolvers
- /// </summary>
- /// <value>The entity resolvers enumerable.</value>
- private IItemResolver[] EntityResolvers { get; set; }
- /// <summary>
- /// Gets or sets the comparers.
- /// </summary>
- /// <value>The comparers.</value>
- private IBaseItemComparer[] Comparers { get; set; }
- /// <summary>
- /// Gets the active item repository
- /// </summary>
- /// <value>The item repository.</value>
- public IItemRepository ItemRepository { get; set; }
- /// <summary>
- /// Occurs when [item added].
- /// </summary>
- public event EventHandler<ItemChangeEventArgs> ItemAdded;
- /// <summary>
- /// Occurs when [item updated].
- /// </summary>
- public event EventHandler<ItemChangeEventArgs> ItemUpdated;
- /// <summary>
- /// Occurs when [item removed].
- /// </summary>
- public event EventHandler<ItemChangeEventArgs> ItemRemoved;
- /// <summary>
- /// The _logger
- /// </summary>
- private readonly ILogger _logger;
- /// <summary>
- /// The _task manager
- /// </summary>
- private readonly ITaskManager _taskManager;
- /// <summary>
- /// The _user manager
- /// </summary>
- private readonly IUserManager _userManager;
- /// <summary>
- /// The _user data repository
- /// </summary>
- private readonly IUserDataManager _userDataRepository;
- /// <summary>
- /// Gets or sets the configuration manager.
- /// </summary>
- /// <value>The configuration manager.</value>
- private IServerConfigurationManager ConfigurationManager { get; set; }
- /// <summary>
- /// A collection of items that may be referenced from multiple physical places in the library
- /// (typically, multiple user roots). We store them here and be sure they all reference a
- /// single instance.
- /// </summary>
- /// <value>The by reference items.</value>
- private ConcurrentDictionary<Guid, BaseItem> ByReferenceItems { get; set; }
- private readonly Func<ILibraryMonitor> _libraryMonitorFactory;
- private readonly Func<IProviderManager> _providerManagerFactory;
- /// <summary>
- /// The _library items cache
- /// </summary>
- private readonly ConcurrentDictionary<Guid, BaseItem> _libraryItemsCache;
- /// <summary>
- /// Gets the library items cache.
- /// </summary>
- /// <value>The library items cache.</value>
- private ConcurrentDictionary<Guid, BaseItem> LibraryItemsCache
- {
- get
- {
- return _libraryItemsCache;
- }
- }
- private readonly IFileSystem _fileSystem;
- /// <summary>
- /// Initializes a new instance of the <see cref="LibraryManager" /> class.
- /// </summary>
- /// <param name="logger">The logger.</param>
- /// <param name="taskManager">The task manager.</param>
- /// <param name="userManager">The user manager.</param>
- /// <param name="configurationManager">The configuration manager.</param>
- /// <param name="userDataRepository">The user data repository.</param>
- public LibraryManager(ILogger logger, ITaskManager taskManager, IUserManager userManager, IServerConfigurationManager configurationManager, IUserDataManager userDataRepository, Func<ILibraryMonitor> libraryMonitorFactory, IFileSystem fileSystem, Func<IProviderManager> providerManagerFactory)
- {
- _logger = logger;
- _taskManager = taskManager;
- _userManager = userManager;
- ConfigurationManager = configurationManager;
- _userDataRepository = userDataRepository;
- _libraryMonitorFactory = libraryMonitorFactory;
- _fileSystem = fileSystem;
- _providerManagerFactory = providerManagerFactory;
- ByReferenceItems = new ConcurrentDictionary<Guid, BaseItem>();
- _libraryItemsCache = new ConcurrentDictionary<Guid, BaseItem>();
- ConfigurationManager.ConfigurationUpdated += ConfigurationUpdated;
- RecordConfigurationValues(configurationManager.Configuration);
- }
- /// <summary>
- /// Adds the parts.
- /// </summary>
- /// <param name="rules">The rules.</param>
- /// <param name="pluginFolders">The plugin folders.</param>
- /// <param name="resolvers">The resolvers.</param>
- /// <param name="introProviders">The intro providers.</param>
- /// <param name="itemComparers">The item comparers.</param>
- /// <param name="postscanTasks">The postscan tasks.</param>
- public void AddParts(IEnumerable<IResolverIgnoreRule> rules,
- IEnumerable<IVirtualFolderCreator> pluginFolders,
- IEnumerable<IItemResolver> resolvers,
- IEnumerable<IIntroProvider> introProviders,
- IEnumerable<IBaseItemComparer> itemComparers,
- IEnumerable<ILibraryPostScanTask> postscanTasks)
- {
- EntityResolutionIgnoreRules = rules.ToArray();
- PluginFolderCreators = pluginFolders.ToArray();
- EntityResolvers = resolvers.OrderBy(i => i.Priority).ToArray();
- IntroProviders = introProviders.ToArray();
- Comparers = itemComparers.ToArray();
- PostscanTasks = postscanTasks.OrderBy(i =>
- {
- var hasOrder = i as IHasOrder;
- return hasOrder == null ? 0 : hasOrder.Order;
- }).ToArray();
- }
- /// <summary>
- /// The _root folder
- /// </summary>
- private AggregateFolder _rootFolder;
- /// <summary>
- /// The _root folder sync lock
- /// </summary>
- private object _rootFolderSyncLock = new object();
- /// <summary>
- /// The _root folder initialized
- /// </summary>
- private bool _rootFolderInitialized;
- /// <summary>
- /// Gets the root folder.
- /// </summary>
- /// <value>The root folder.</value>
- public AggregateFolder RootFolder
- {
- get
- {
- LazyInitializer.EnsureInitialized(ref _rootFolder, ref _rootFolderInitialized, ref _rootFolderSyncLock, CreateRootFolder);
- return _rootFolder;
- }
- private set
- {
- _rootFolder = value;
- if (value == null)
- {
- _rootFolderInitialized = false;
- }
- }
- }
- /// <summary>
- /// The _items by name path
- /// </summary>
- private string _itemsByNamePath;
- /// <summary>
- /// The _season zero display name
- /// </summary>
- private string _seasonZeroDisplayName;
- private bool _wizardCompleted;
- /// <summary>
- /// Records the configuration values.
- /// </summary>
- /// <param name="configuration">The configuration.</param>
- private void RecordConfigurationValues(ServerConfiguration configuration)
- {
- _seasonZeroDisplayName = configuration.SeasonZeroDisplayName;
- _itemsByNamePath = ConfigurationManager.ApplicationPaths.ItemsByNamePath;
- _wizardCompleted = configuration.IsStartupWizardCompleted;
- }
- /// <summary>
- /// Configurations the updated.
- /// </summary>
- /// <param name="sender">The sender.</param>
- /// <param name="e">The <see cref="EventArgs" /> instance containing the event data.</param>
- void ConfigurationUpdated(object sender, EventArgs e)
- {
- var config = ConfigurationManager.Configuration;
- var ibnPathChanged = !string.Equals(_itemsByNamePath, ConfigurationManager.ApplicationPaths.ItemsByNamePath, StringComparison.CurrentCulture);
- if (ibnPathChanged)
- {
- RemoveItemsByNameFromCache();
- }
- var newSeasonZeroName = ConfigurationManager.Configuration.SeasonZeroDisplayName;
- var seasonZeroNameChanged = !string.Equals(_seasonZeroDisplayName, newSeasonZeroName, StringComparison.CurrentCulture);
- var wizardChanged = config.IsStartupWizardCompleted != _wizardCompleted;
- RecordConfigurationValues(config);
- Task.Run(async () =>
- {
- if (seasonZeroNameChanged)
- {
- await UpdateSeasonZeroNames(newSeasonZeroName, CancellationToken.None).ConfigureAwait(false);
- }
- if (seasonZeroNameChanged || ibnPathChanged || wizardChanged)
- {
- _taskManager.CancelIfRunningAndQueue<RefreshMediaLibraryTask>();
- }
- });
- }
- private void RemoveItemsByNameFromCache()
- {
- RemoveItemsFromCache(i => i is Person);
- RemoveItemsFromCache(i => i is Year);
- RemoveItemsFromCache(i => i is Genre);
- RemoveItemsFromCache(i => i is MusicGenre);
- RemoveItemsFromCache(i => i is GameGenre);
- RemoveItemsFromCache(i => i is Studio);
- RemoveItemsFromCache(i =>
- {
- var artist = i as MusicArtist;
- return artist != null && artist.IsAccessedByName;
- });
- }
- private void RemoveItemsFromCache(Func<BaseItem, bool> remove)
- {
- var items = _libraryItemsCache.ToList().Where(i => remove(i.Value)).ToList();
- foreach (var item in items)
- {
- BaseItem value;
- _libraryItemsCache.TryRemove(item.Key, out value);
- }
- }
- /// <summary>
- /// Updates the season zero names.
- /// </summary>
- /// <param name="newName">The new name.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- private async Task UpdateSeasonZeroNames(string newName, CancellationToken cancellationToken)
- {
- var seasons = RootFolder.RecursiveChildren
- .OfType<Season>()
- .Where(i => i.IndexNumber.HasValue && i.IndexNumber.Value == 0 && !string.Equals(i.Name, newName, StringComparison.CurrentCulture))
- .ToList();
- foreach (var season in seasons)
- {
- season.Name = newName;
- try
- {
- await UpdateItem(season, ItemUpdateType.MetadataDownload, cancellationToken).ConfigureAwait(false);
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error saving {0}", ex, season.Path);
- }
- }
- }
- /// <summary>
- /// Updates the item in library cache.
- /// </summary>
- /// <param name="item">The item.</param>
- private void UpdateItemInLibraryCache(BaseItem item)
- {
- RegisterItem(item);
- }
- public void RegisterItem(BaseItem item)
- {
- if (item == null)
- {
- throw new ArgumentNullException("item");
- }
- RegisterItem(item.Id, item);
- }
- private void RegisterItem(Guid id, BaseItem item)
- {
- LibraryItemsCache.AddOrUpdate(id, item, delegate { return item; });
- }
- public async Task DeleteItem(BaseItem item, DeleteOptions options)
- {
- _logger.Debug("Deleting item, Type: {0}, Name: {1}, Path: {2}, Id: {3}",
- item.GetType().Name,
- item.Name,
- item.Path ?? string.Empty,
- item.Id);
- var parent = item.Parent;
- var locationType = item.LocationType;
- var children = item.IsFolder
- ? ((Folder)item).RecursiveChildren.ToList()
- : new List<BaseItem>();
- foreach (var metadataPath in GetMetadataPaths(item, children))
- {
- _logger.Debug("Deleting path {0}", metadataPath);
- try
- {
- Directory.Delete(metadataPath, true);
- }
- catch (DirectoryNotFoundException)
- {
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error deleting {0}", ex, metadataPath);
- }
- }
- if (options.DeleteFileLocation && locationType != LocationType.Remote && locationType != LocationType.Virtual)
- {
- foreach (var path in item.GetDeletePaths().ToList())
- {
- if (Directory.Exists(path))
- {
- _logger.Debug("Deleting path {0}", path);
- Directory.Delete(path, true);
- }
- else if (File.Exists(path))
- {
- _logger.Debug("Deleting path {0}", path);
- File.Delete(path);
- }
- }
- if (parent != null)
- {
- await parent.ValidateChildren(new Progress<double>(), CancellationToken.None)
- .ConfigureAwait(false);
- }
- }
- else if (parent != null)
- {
- await parent.RemoveChild(item, CancellationToken.None).ConfigureAwait(false);
- }
- await ItemRepository.DeleteItem(item.Id, CancellationToken.None).ConfigureAwait(false);
- foreach (var child in children)
- {
- await ItemRepository.DeleteItem(child.Id, CancellationToken.None).ConfigureAwait(false);
- }
- BaseItem removed;
- _libraryItemsCache.TryRemove(item.Id, out removed);
- ReportItemRemoved(item);
- }
- private IEnumerable<string> GetMetadataPaths(BaseItem item, IEnumerable<BaseItem> children)
- {
- var list = new List<string>
- {
- ConfigurationManager.ApplicationPaths.GetInternalMetadataPath(item.Id)
- };
- list.AddRange(children.Select(i => ConfigurationManager.ApplicationPaths.GetInternalMetadataPath(i.Id)));
- return list;
- }
- /// <summary>
- /// Resolves the item.
- /// </summary>
- /// <param name="args">The args.</param>
- /// <returns>BaseItem.</returns>
- public BaseItem ResolveItem(ItemResolveArgs args)
- {
- var item = EntityResolvers.Select(r =>
- {
- try
- {
- return r.ResolvePath(args);
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error in {0} resolving {1}", ex, r.GetType().Name, args.Path);
- return null;
- }
- }).FirstOrDefault(i => i != null);
- if (item != null)
- {
- ResolverHelper.SetInitialItemValues(item, args, _fileSystem);
- }
- return item;
- }
- public IEnumerable<BaseItem> ReplaceVideosWithPrimaryVersions(IEnumerable<BaseItem> items)
- {
- return items.Select(i =>
- {
- var video = i as Video;
- if (video != null)
- {
- if (video.PrimaryVersionId.HasValue)
- {
- var primary = GetItemById(video.PrimaryVersionId.Value) as Video;
- if (primary != null)
- {
- return primary;
- }
- }
- }
- return i;
- }).DistinctBy(i => i.Id);
- }
- /// <summary>
- /// Ensure supplied item has only one instance throughout
- /// </summary>
- /// <param name="item">The item.</param>
- /// <returns>The proper instance to the item</returns>
- public BaseItem GetOrAddByReferenceItem(BaseItem item)
- {
- // Add this item to our list if not there already
- if (!ByReferenceItems.TryAdd(item.Id, item))
- {
- // Already there - return the existing reference
- item = ByReferenceItems[item.Id];
- }
- return item;
- }
- public BaseItem ResolvePath(FileSystemInfo fileInfo, Folder parent = null)
- {
- return ResolvePath(fileInfo, new DirectoryService(_logger), parent);
- }
- /// <summary>
- /// Resolves a path into a BaseItem
- /// </summary>
- /// <param name="fileInfo">The file info.</param>
- /// <param name="directoryService">The directory service.</param>
- /// <param name="parent">The parent.</param>
- /// <returns>BaseItem.</returns>
- /// <exception cref="System.ArgumentNullException">fileInfo</exception>
- public BaseItem ResolvePath(FileSystemInfo fileInfo, IDirectoryService directoryService, Folder parent = null)
- {
- if (fileInfo == null)
- {
- throw new ArgumentNullException("fileInfo");
- }
- var args = new ItemResolveArgs(ConfigurationManager.ApplicationPaths, this, directoryService)
- {
- Parent = parent,
- Path = fileInfo.FullName,
- FileInfo = fileInfo
- };
- // Return null if ignore rules deem that we should do so
- if (EntityResolutionIgnoreRules.Any(r => r.ShouldIgnore(args)))
- {
- return null;
- }
- // Gather child folder and files
- if (args.IsDirectory)
- {
- var isPhysicalRoot = args.IsPhysicalRoot;
- // When resolving the root, we need it's grandchildren (children of user views)
- var flattenFolderDepth = isPhysicalRoot ? 2 : 0;
- var fileSystemDictionary = FileData.GetFilteredFileSystemEntries(directoryService, args.Path, _fileSystem, _logger, args, flattenFolderDepth: flattenFolderDepth, resolveShortcuts: isPhysicalRoot || args.IsVf);
- // Need to remove subpaths that may have been resolved from shortcuts
- // Example: if \\server\movies exists, then strip out \\server\movies\action
- if (isPhysicalRoot)
- {
- var paths = NormalizeRootPathList(fileSystemDictionary.Keys);
- fileSystemDictionary = paths.Select(i => (FileSystemInfo)new DirectoryInfo(i)).ToDictionary(i => i.FullName);
- }
- args.FileSystemDictionary = fileSystemDictionary;
- }
- // Check to see if we should resolve based on our contents
- if (args.IsDirectory && !ShouldResolvePathContents(args))
- {
- return null;
- }
- return ResolveItem(args);
- }
- public IEnumerable<string> NormalizeRootPathList(IEnumerable<string> paths)
- {
- var list = paths.Select(_fileSystem.NormalizePath)
- .Distinct(StringComparer.OrdinalIgnoreCase)
- .ToList();
- var dupes = list.Where(subPath => !subPath.EndsWith(":\\", StringComparison.OrdinalIgnoreCase) && list.Any(i => _fileSystem.ContainsSubPath(i, subPath)))
- .ToList();
- foreach (var dupe in dupes)
- {
- _logger.Info("Found duplicate path: {0}", dupe);
- }
- return list.Except(dupes, StringComparer.OrdinalIgnoreCase);
- }
- /// <summary>
- /// Determines whether a path should be ignored based on its contents - called after the contents have been read
- /// </summary>
- /// <param name="args">The args.</param>
- /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
- private static bool ShouldResolvePathContents(ItemResolveArgs args)
- {
- // Ignore any folders containing a file called .ignore
- return !args.ContainsFileSystemEntryByName(".ignore");
- }
- /// <summary>
- /// Resolves a set of files into a list of BaseItem
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="files">The files.</param>
- /// <param name="directoryService">The directory service.</param>
- /// <param name="parent">The parent.</param>
- /// <returns>List{``0}.</returns>
- public List<T> ResolvePaths<T>(IEnumerable<FileSystemInfo> files, IDirectoryService directoryService, Folder parent)
- where T : BaseItem
- {
- var list = new List<T>();
- Parallel.ForEach(files, f =>
- {
- try
- {
- var item = ResolvePath(f, directoryService, parent) as T;
- if (item != null)
- {
- lock (list)
- {
- list.Add(item);
- }
- }
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error resolving path {0}", ex, f.FullName);
- }
- });
- return list;
- }
- /// <summary>
- /// Creates the root media folder
- /// </summary>
- /// <returns>AggregateFolder.</returns>
- /// <exception cref="System.InvalidOperationException">Cannot create the root folder until plugins have loaded</exception>
- public AggregateFolder CreateRootFolder()
- {
- var rootFolderPath = ConfigurationManager.ApplicationPaths.RootFolderPath;
- Directory.CreateDirectory(rootFolderPath);
- var rootFolder = GetItemById(rootFolderPath.GetMBId(typeof(AggregateFolder))) as AggregateFolder ?? (AggregateFolder)ResolvePath(new DirectoryInfo(rootFolderPath));
- // Add in the plug-in folders
- foreach (var child in PluginFolderCreators)
- {
- var folder = child.GetFolder();
- if (folder != null)
- {
- if (folder.Id == Guid.Empty)
- {
- folder.Id = (folder.Path ?? folder.GetType().Name).GetMBId(folder.GetType());
- }
- folder = GetItemById(folder.Id) as BasePluginFolder ?? folder;
- rootFolder.AddVirtualChild(folder);
- RegisterItem(folder);
- }
- }
- return rootFolder;
- }
- private UserRootFolder _userRootFolder;
- public Folder GetUserRootFolder()
- {
- if (_userRootFolder == null)
- {
- var userRootPath = ConfigurationManager.ApplicationPaths.DefaultUserViewsPath;
- Directory.CreateDirectory(userRootPath);
- _userRootFolder = GetItemById(userRootPath.GetMBId(typeof(UserRootFolder))) as UserRootFolder ??
- (UserRootFolder)ResolvePath(new DirectoryInfo(userRootPath));
- }
- return _userRootFolder;
- }
- /// <summary>
- /// Gets a Person
- /// </summary>
- /// <param name="name">The name.</param>
- /// <returns>Task{Person}.</returns>
- public Person GetPerson(string name)
- {
- return GetItemByName<Person>(ConfigurationManager.ApplicationPaths.PeoplePath, name);
- }
- /// <summary>
- /// Gets a Studio
- /// </summary>
- /// <param name="name">The name.</param>
- /// <returns>Task{Studio}.</returns>
- public Studio GetStudio(string name)
- {
- return GetItemByName<Studio>(ConfigurationManager.ApplicationPaths.StudioPath, name);
- }
- /// <summary>
- /// Gets a Genre
- /// </summary>
- /// <param name="name">The name.</param>
- /// <returns>Task{Genre}.</returns>
- public Genre GetGenre(string name)
- {
- return GetItemByName<Genre>(ConfigurationManager.ApplicationPaths.GenrePath, name);
- }
- /// <summary>
- /// Gets the genre.
- /// </summary>
- /// <param name="name">The name.</param>
- /// <returns>Task{MusicGenre}.</returns>
- public MusicGenre GetMusicGenre(string name)
- {
- return GetItemByName<MusicGenre>(ConfigurationManager.ApplicationPaths.MusicGenrePath, name);
- }
- /// <summary>
- /// Gets the game genre.
- /// </summary>
- /// <param name="name">The name.</param>
- /// <returns>Task{GameGenre}.</returns>
- public GameGenre GetGameGenre(string name)
- {
- return GetItemByName<GameGenre>(ConfigurationManager.ApplicationPaths.GameGenrePath, name);
- }
- /// <summary>
- /// The us culture
- /// </summary>
- private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
- /// <summary>
- /// Gets a Year
- /// </summary>
- /// <param name="value">The value.</param>
- /// <returns>Task{Year}.</returns>
- /// <exception cref="System.ArgumentOutOfRangeException"></exception>
- public Year GetYear(int value)
- {
- if (value <= 0)
- {
- throw new ArgumentOutOfRangeException("Years less than or equal to 0 are invalid.");
- }
- return GetItemByName<Year>(ConfigurationManager.ApplicationPaths.YearPath, value.ToString(UsCulture));
- }
- /// <summary>
- /// Gets a Genre
- /// </summary>
- /// <param name="name">The name.</param>
- /// <returns>Task{Genre}.</returns>
- public MusicArtist GetArtist(string name)
- {
- return GetItemByName<MusicArtist>(ConfigurationManager.ApplicationPaths.ArtistsPath, name);
- }
- private T GetItemByName<T>(string path, string name)
- where T : BaseItem, new()
- {
- if (string.IsNullOrWhiteSpace(path))
- {
- throw new ArgumentNullException("path");
- }
- if (string.IsNullOrWhiteSpace(name))
- {
- throw new ArgumentNullException("name");
- }
- var validFilename = _fileSystem.GetValidFilename(name).Trim();
- string subFolderPrefix = null;
- var type = typeof(T);
- if (type == typeof(Person) && ConfigurationManager.Configuration.EnablePeoplePrefixSubFolders)
- {
- subFolderPrefix = validFilename.Substring(0, 1);
- }
- var fullPath = string.IsNullOrEmpty(subFolderPrefix) ?
- Path.Combine(path, validFilename) :
- Path.Combine(path, subFolderPrefix, validFilename);
- var id = fullPath.GetMBId(type);
- BaseItem obj;
- if (!_libraryItemsCache.TryGetValue(id, out obj))
- {
- obj = CreateItemByName<T>(fullPath, name, id);
- RegisterItem(id, obj);
- }
- return obj as T;
- }
- /// <summary>
- /// Creates an IBN item based on a given path
- /// </summary>
- /// <typeparam name="T"></typeparam>
- /// <param name="path">The path.</param>
- /// <param name="name">The name.</param>
- /// <returns>Task{``0}.</returns>
- /// <exception cref="System.IO.IOException">Path not created: + path</exception>
- private T CreateItemByName<T>(string path, string name, Guid id)
- where T : BaseItem, new()
- {
- var isArtist = typeof(T) == typeof(MusicArtist);
- if (isArtist)
- {
- var existing = RootFolder.RecursiveChildren
- .OfType<T>()
- .FirstOrDefault(i => string.Equals(i.Name, name, StringComparison.OrdinalIgnoreCase));
- if (existing != null)
- {
- return existing;
- }
- }
- var fileInfo = new DirectoryInfo(path);
- var isNew = false;
- if (!fileInfo.Exists)
- {
- fileInfo = Directory.CreateDirectory(path);
- isNew = true;
- }
- var item = isNew ? null : GetItemById(id) as T;
- if (item == null)
- {
- item = new T
- {
- Name = name,
- Id = id,
- DateCreated = _fileSystem.GetCreationTimeUtc(fileInfo),
- DateModified = _fileSystem.GetLastWriteTimeUtc(fileInfo),
- Path = path
- };
- }
- if (isArtist)
- {
- (item as MusicArtist).IsAccessedByName = true;
- }
- return item;
- }
- /// <summary>
- /// Validate and refresh the People sub-set of the IBN.
- /// The items are stored in the db but not loaded into memory until actually requested by an operation.
- /// </summary>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="progress">The progress.</param>
- /// <returns>Task.</returns>
- public Task ValidatePeople(CancellationToken cancellationToken, IProgress<double> progress)
- {
- // Ensure the location is available.
- Directory.CreateDirectory(ConfigurationManager.ApplicationPaths.PeoplePath);
- return new PeopleValidator(this, _logger, ConfigurationManager).ValidatePeople(cancellationToken, progress);
- }
- /// <summary>
- /// Validates the artists.
- /// </summary>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="progress">The progress.</param>
- /// <returns>Task.</returns>
- public Task ValidateArtists(CancellationToken cancellationToken, IProgress<double> progress)
- {
- // Ensure the location is unavailable.
- Directory.CreateDirectory(ConfigurationManager.ApplicationPaths.ArtistsPath);
- return new ArtistsValidator(this, _userManager, _logger).Run(progress, cancellationToken);
- }
- /// <summary>
- /// Validates the music genres.
- /// </summary>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="progress">The progress.</param>
- /// <returns>Task.</returns>
- public Task ValidateMusicGenres(CancellationToken cancellationToken, IProgress<double> progress)
- {
- // Ensure the location is unavailable.
- Directory.CreateDirectory(ConfigurationManager.ApplicationPaths.MusicGenrePath);
- return new MusicGenresValidator(this, _logger).Run(progress, cancellationToken);
- }
- /// <summary>
- /// Validates the game genres.
- /// </summary>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="progress">The progress.</param>
- /// <returns>Task.</returns>
- public Task ValidateGameGenres(CancellationToken cancellationToken, IProgress<double> progress)
- {
- // Ensure the location is unavailable.
- Directory.CreateDirectory(ConfigurationManager.ApplicationPaths.GameGenrePath);
- return new GameGenresValidator(this, _userManager, _logger).Run(progress, cancellationToken);
- }
- /// <summary>
- /// Validates the studios.
- /// </summary>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="progress">The progress.</param>
- /// <returns>Task.</returns>
- public Task ValidateStudios(CancellationToken cancellationToken, IProgress<double> progress)
- {
- // Ensure the location is unavailable.
- Directory.CreateDirectory(ConfigurationManager.ApplicationPaths.StudioPath);
- return new StudiosValidator(this, _userManager, _logger).Run(progress, cancellationToken);
- }
- /// <summary>
- /// Validates the genres.
- /// </summary>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="progress">The progress.</param>
- /// <returns>Task.</returns>
- public Task ValidateGenres(CancellationToken cancellationToken, IProgress<double> progress)
- {
- // Ensure the location is unavailable.
- Directory.CreateDirectory(ConfigurationManager.ApplicationPaths.GenrePath);
- return new GenresValidator(this, _userManager, _logger).Run(progress, cancellationToken);
- }
- /// <summary>
- /// Reloads the root media folder
- /// </summary>
- /// <param name="progress">The progress.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- public Task ValidateMediaLibrary(IProgress<double> progress, CancellationToken cancellationToken)
- {
- // Just run the scheduled task so that the user can see it
- _taskManager.CancelIfRunningAndQueue<RefreshMediaLibraryTask>();
- return Task.FromResult(true);
- }
- /// <summary>
- /// Queues the library scan.
- /// </summary>
- public void QueueLibraryScan()
- {
- // Just run the scheduled task so that the user can see it
- _taskManager.QueueScheduledTask<RefreshMediaLibraryTask>();
- }
- /// <summary>
- /// Validates the media library internal.
- /// </summary>
- /// <param name="progress">The progress.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- public async Task ValidateMediaLibraryInternal(IProgress<double> progress, CancellationToken cancellationToken)
- {
- _libraryMonitorFactory().Stop();
- try
- {
- await PerformLibraryValidation(progress, cancellationToken).ConfigureAwait(false);
- }
- finally
- {
- _libraryMonitorFactory().Start();
- }
- }
- private async Task PerformLibraryValidation(IProgress<double> progress, CancellationToken cancellationToken)
- {
- _logger.Info("Validating media library");
- await RootFolder.RefreshMetadata(cancellationToken).ConfigureAwait(false);
- progress.Report(.5);
- // Start by just validating the children of the root, but go no further
- await RootFolder.ValidateChildren(new Progress<double>(), cancellationToken, new MetadataRefreshOptions(), recursive: false);
- progress.Report(1);
- var userRoot = GetUserRootFolder();
- await userRoot.RefreshMetadata(cancellationToken).ConfigureAwait(false);
- await userRoot.ValidateChildren(new Progress<double>(), cancellationToken, new MetadataRefreshOptions(), recursive: false).ConfigureAwait(false);
- progress.Report(2);
- var innerProgress = new ActionableProgress<double>();
- innerProgress.RegisterAction(pct => progress.Report(2 + pct * .73));
- // Now validate the entire media library
- await RootFolder.ValidateChildren(innerProgress, cancellationToken, new MetadataRefreshOptions(), recursive: true).ConfigureAwait(false);
- progress.Report(75);
- innerProgress = new ActionableProgress<double>();
- innerProgress.RegisterAction(pct => progress.Report(75 + pct * .25));
- // Run post-scan tasks
- await RunPostScanTasks(innerProgress, cancellationToken).ConfigureAwait(false);
- progress.Report(100);
- // Bad practice, i know. But we keep a lot in memory, unfortunately.
- GC.Collect(2, GCCollectionMode.Forced, true);
- GC.Collect(2, GCCollectionMode.Forced, true);
- }
- /// <summary>
- /// Runs the post scan tasks.
- /// </summary>
- /// <param name="progress">The progress.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- private async Task RunPostScanTasks(IProgress<double> progress, CancellationToken cancellationToken)
- {
- var tasks = PostscanTasks.ToList();
- var numComplete = 0;
- var numTasks = tasks.Count;
- foreach (var task in tasks)
- {
- var innerProgress = new ActionableProgress<double>();
- // Prevent access to modified closure
- var currentNumComplete = numComplete;
- innerProgress.RegisterAction(pct =>
- {
- double innerPercent = (currentNumComplete * 100) + pct;
- innerPercent /= numTasks;
- progress.Report(innerPercent);
- });
- try
- {
- await task.Run(innerProgress, cancellationToken);
- }
- catch (OperationCanceledException)
- {
- _logger.Info("Post-scan task cancelled: {0}", task.GetType().Name);
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error running postscan task", ex);
- }
- numComplete++;
- double percent = numComplete;
- percent /= numTasks;
- progress.Report(percent * 100);
- }
- progress.Report(100);
- }
- /// <summary>
- /// Gets the default view.
- /// </summary>
- /// <returns>IEnumerable{VirtualFolderInfo}.</returns>
- public IEnumerable<VirtualFolderInfo> GetDefaultVirtualFolders()
- {
- return GetView(ConfigurationManager.ApplicationPaths.DefaultUserViewsPath);
- }
- /// <summary>
- /// Gets the view.
- /// </summary>
- /// <param name="user">The user.</param>
- /// <returns>IEnumerable{VirtualFolderInfo}.</returns>
- public IEnumerable<VirtualFolderInfo> GetVirtualFolders(User user)
- {
- return GetDefaultVirtualFolders();
- }
- /// <summary>
- /// Gets the view.
- /// </summary>
- /// <param name="path">The path.</param>
- /// <returns>IEnumerable{VirtualFolderInfo}.</returns>
- private IEnumerable<VirtualFolderInfo> GetView(string path)
- {
- return Directory.EnumerateDirectories(path, "*", SearchOption.TopDirectoryOnly)
- .Select(dir => new VirtualFolderInfo
- {
- Name = Path.GetFileName(dir),
- Locations = Directory.EnumerateFiles(dir, "*.mblink", SearchOption.TopDirectoryOnly)
- .Select(_fileSystem.ResolveShortcut)
- .OrderBy(i => i)
- .ToList(),
- CollectionType = GetCollectionType(dir)
- });
- }
- private string GetCollectionType(string path)
- {
- return new DirectoryInfo(path).EnumerateFiles("*.collection", SearchOption.TopDirectoryOnly)
- .Select(i => _fileSystem.GetFileNameWithoutExtension(i))
- .FirstOrDefault();
- }
- /// <summary>
- /// Gets the item by id.
- /// </summary>
- /// <param name="id">The id.</param>
- /// <returns>BaseItem.</returns>
- /// <exception cref="System.ArgumentNullException">id</exception>
- public BaseItem GetItemById(Guid id)
- {
- if (id == Guid.Empty)
- {
- throw new ArgumentNullException("id");
- }
- BaseItem item;
- if (LibraryItemsCache.TryGetValue(id, out item))
- {
- return item;
- }
- item = RetrieveItem(id);
- if (item != null)
- {
- RegisterItem(item);
- }
- return item;
- }
- /// <summary>
- /// Gets the intros.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="user">The user.</param>
- /// <returns>IEnumerable{System.String}.</returns>
- public IEnumerable<Video> GetIntros(BaseItem item, User user)
- {
- return IntroProviders.SelectMany(i => i.GetIntros(item, user))
- .Select(ResolveIntro)
- .Where(i => i != null);
- }
- /// <summary>
- /// Gets all intro files.
- /// </summary>
- /// <returns>IEnumerable{System.String}.</returns>
- public IEnumerable<string> GetAllIntroFiles()
- {
- return IntroProviders.SelectMany(i => i.GetAllIntroFiles());
- }
- /// <summary>
- /// Resolves the intro.
- /// </summary>
- /// <param name="info">The info.</param>
- /// <returns>Video.</returns>
- private Video ResolveIntro(IntroInfo info)
- {
- Video video = null;
- if (info.ItemId.HasValue)
- {
- // Get an existing item by Id
- video = GetItemById(info.ItemId.Value) as Video;
- if (video == null)
- {
- _logger.Error("Unable to locate item with Id {0}.", info.ItemId.Value);
- }
- }
- else if (!string.IsNullOrEmpty(info.Path))
- {
- try
- {
- // Try to resolve the path into a video
- video = ResolvePath(_fileSystem.GetFileSystemInfo(info.Path)) as Video;
- if (video == null)
- {
- _logger.Error("Intro resolver returned null for {0}.", info.Path);
- }
- else
- {
- // Pull the saved db item that will include metadata
- var dbItem = GetItemById(video.Id) as Video;
- if (dbItem != null)
- {
- video = dbItem;
- }
- }
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error resolving path {0}.", ex, info.Path);
- }
- }
- else
- {
- _logger.Error("IntroProvider returned an IntroInfo with null Path and ItemId.");
- }
- return video;
- }
- /// <summary>
- /// Sorts the specified sort by.
- /// </summary>
- /// <param name="items">The items.</param>
- /// <param name="user">The user.</param>
- /// <param name="sortBy">The sort by.</param>
- /// <param name="sortOrder">The sort order.</param>
- /// <returns>IEnumerable{BaseItem}.</returns>
- public IEnumerable<BaseItem> Sort(IEnumerable<BaseItem> items, User user, IEnumerable<string> sortBy, SortOrder sortOrder)
- {
- var isFirst = true;
- IOrderedEnumerable<BaseItem> orderedItems = null;
- foreach (var orderBy in sortBy.Select(o => GetComparer(o, user)).Where(c => c != null))
- {
- if (isFirst)
- {
- orderedItems = sortOrder == SortOrder.Descending ? items.OrderByDescending(i => i, orderBy) : items.OrderBy(i => i, orderBy);
- }
- else
- {
- orderedItems = sortOrder == SortOrder.Descending ? orderedItems.ThenByDescending(i => i, orderBy) : orderedItems.ThenBy(i => i, orderBy);
- }
- isFirst = false;
- }
- return orderedItems ?? items;
- }
- /// <summary>
- /// Gets the comparer.
- /// </summary>
- /// <param name="name">The name.</param>
- /// <param name="user">The user.</param>
- /// <returns>IBaseItemComparer.</returns>
- private IBaseItemComparer GetComparer(string name, User user)
- {
- var comparer = Comparers.FirstOrDefault(c => string.Equals(name, c.Name, StringComparison.OrdinalIgnoreCase));
- if (comparer != null)
- {
- // If it requires a user, create a new one, and assign the user
- if (comparer is IUserBaseItemComparer)
- {
- var userComparer = (IUserBaseItemComparer)Activator.CreateInstance(comparer.GetType());
- userComparer.User = user;
- userComparer.UserManager = _userManager;
- userComparer.UserDataRepository = _userDataRepository;
- return userComparer;
- }
- }
- return comparer;
- }
- /// <summary>
- /// Creates the item.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- public Task CreateItem(BaseItem item, CancellationToken cancellationToken)
- {
- return CreateItems(new[] { item }, cancellationToken);
- }
- /// <summary>
- /// Creates the items.
- /// </summary>
- /// <param name="items">The items.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- public async Task CreateItems(IEnumerable<BaseItem> items, CancellationToken cancellationToken)
- {
- var list = items.ToList();
- await ItemRepository.SaveItems(list, cancellationToken).ConfigureAwait(false);
- foreach (var item in list)
- {
- UpdateItemInLibraryCache(item);
- }
- if (ItemAdded != null)
- {
- foreach (var item in list)
- {
- try
- {
- ItemAdded(this, new ItemChangeEventArgs { Item = item });
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error in ItemAdded event handler", ex);
- }
- }
- }
- }
- /// <summary>
- /// Updates the item.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="updateReason">The update reason.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- public async Task UpdateItem(BaseItem item, ItemUpdateType updateReason, CancellationToken cancellationToken)
- {
- var locationType = item.LocationType;
- if (locationType != LocationType.Remote && locationType != LocationType.Virtual)
- {
- await _providerManagerFactory().SaveMetadata(item, updateReason).ConfigureAwait(false);
- }
- item.DateLastSaved = DateTime.UtcNow;
- _logger.Debug("Saving {0} to database.", item.Path ?? item.Name);
- await ItemRepository.SaveItem(item, cancellationToken).ConfigureAwait(false);
- UpdateItemInLibraryCache(item);
- if (ItemUpdated != null)
- {
- try
- {
- ItemUpdated(this, new ItemChangeEventArgs
- {
- Item = item,
- UpdateReason = updateReason
- });
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error in ItemUpdated event handler", ex);
- }
- }
- }
- /// <summary>
- /// Reports the item removed.
- /// </summary>
- /// <param name="item">The item.</param>
- public void ReportItemRemoved(BaseItem item)
- {
- if (ItemRemoved != null)
- {
- try
- {
- ItemRemoved(this, new ItemChangeEventArgs { Item = item });
- }
- catch (Exception ex)
- {
- _logger.ErrorException("Error in ItemRemoved event handler", ex);
- }
- }
- }
- /// <summary>
- /// Retrieves the item.
- /// </summary>
- /// <param name="id">The id.</param>
- /// <returns>BaseItem.</returns>
- public BaseItem RetrieveItem(Guid id)
- {
- return ItemRepository.RetrieveItem(id);
- }
- /// <summary>
- /// Finds the type of the collection.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <returns>System.String.</returns>
- public string FindCollectionType(BaseItem item)
- {
- while (!(item.Parent is AggregateFolder) && item.Parent != null)
- {
- item = item.Parent;
- }
- if (item == null)
- {
- return null;
- }
- var collectionTypes = _userManager.Users
- .Select(i => i.RootFolder)
- .Distinct()
- .SelectMany(i => i.Children)
- .OfType<CollectionFolder>()
- .Where(i => string.Equals(i.Path, item.Path, StringComparison.OrdinalIgnoreCase) || i.PhysicalLocations.Contains(item.Path))
- .Select(i => i.CollectionType)
- .Where(i => !string.IsNullOrEmpty(i))
- .Distinct()
- .ToList();
- return collectionTypes.Count == 1 ? collectionTypes[0] : null;
- }
- public Task<UserView> GetNamedView(string name, string type, string sortName, CancellationToken cancellationToken)
- {
- return GetNamedView(name, null, type, sortName, cancellationToken);
- }
- public async Task<UserView> GetNamedView(string name, string category, string type, string sortName, CancellationToken cancellationToken)
- {
- var path = Path.Combine(ConfigurationManager.ApplicationPaths.ItemsByNamePath,
- "views");
- if (!string.IsNullOrWhiteSpace(category))
- {
- path = Path.Combine(path, _fileSystem.GetValidFilename(category));
- }
- path = Path.Combine(path, _fileSystem.GetValidFilename(type));
- var id = (path + "_namedview_" + name).GetMBId(typeof(UserView));
- var item = GetItemById(id) as UserView;
- if (item == null || !string.Equals(item.Path, path, StringComparison.OrdinalIgnoreCase))
- {
- Directory.CreateDirectory(Path.GetDirectoryName(path));
- item = new UserView
- {
- Path = path,
- Id = id,
- DateCreated = DateTime.UtcNow,
- Name = name,
- ViewType = type,
- ForcedSortName = sortName
- };
- await CreateItem(item, cancellationToken).ConfigureAwait(false);
- await item.RefreshMetadata(cancellationToken).ConfigureAwait(false);
- }
- return item;
- }
- }
- }
|