< Summary - Jellyfin

Information
Class: MediaBrowser.Controller.Entities.TV.Series
Assembly: MediaBrowser.Controller
File(s): /srv/git/jellyfin/MediaBrowser.Controller/Entities/TV/Series.cs
Line coverage
1%
Covered lines: 3
Uncovered lines: 193
Coverable lines: 196
Total lines: 527
Line coverage: 1.5%
Branch coverage
0%
Covered branches: 0
Total branches: 56
Branch coverage: 0%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

File(s)

/srv/git/jellyfin/MediaBrowser.Controller/Entities/TV/Series.cs

#LineLine coverage
 1#nullable disable
 2
 3#pragma warning disable CS1591
 4
 5using System;
 6using System.Collections.Generic;
 7using System.Globalization;
 8using System.Linq;
 9using System.Text.Json.Serialization;
 10using System.Threading;
 11using System.Threading.Tasks;
 12using Jellyfin.Data;
 13using Jellyfin.Data.Enums;
 14using Jellyfin.Database.Implementations.Entities;
 15using Jellyfin.Database.Implementations.Enums;
 16using MediaBrowser.Controller.Dto;
 17using MediaBrowser.Controller.Providers;
 18using MediaBrowser.Model.Entities;
 19using MediaBrowser.Model.Querying;
 20using MetadataProvider = MediaBrowser.Model.Entities.MetadataProvider;
 21
 22namespace MediaBrowser.Controller.Entities.TV
 23{
 24    /// <summary>
 25    /// Class Series.
 26    /// </summary>
 27    public class Series : Folder, IHasTrailers, IHasDisplayOrder, IHasLookupInfo<SeriesInfo>, IMetadataContainer, ISuppo
 28    {
 10529        public Series()
 30        {
 10531            AirDays = Array.Empty<DayOfWeek>();
 10532        }
 33
 34        public DayOfWeek[] AirDays { get; set; }
 35
 36        public string AirTime { get; set; }
 37
 38        [JsonIgnore]
 039        public override bool SupportsAddingToPlaylist => true;
 40
 41        [JsonIgnore]
 042        public override bool IsPreSorted => true;
 43
 44        [JsonIgnore]
 045        public override bool SupportsDateLastMediaAdded => true;
 46
 47        [JsonIgnore]
 048        public override bool SupportsInheritedParentImages => false;
 49
 50        [JsonIgnore]
 051        public override bool SupportsPeople => true;
 52
 53        /// <inheritdoc />
 54        [JsonIgnore]
 055        public IReadOnlyList<BaseItem> LocalTrailers => GetExtras()
 056            .Where(extra => extra.ExtraType == Model.Entities.ExtraType.Trailer)
 057            .ToArray();
 58
 59        /// <summary>
 60        /// Gets or sets the display order.
 61        /// </summary>
 62        /// <remarks>
 63        /// Valid options are airdate, dvd or absolute.
 64        /// </remarks>
 65        public string DisplayOrder { get; set; }
 66
 67        /// <summary>
 68        /// Gets or sets the status.
 69        /// </summary>
 70        /// <value>The status.</value>
 71        public SeriesStatus? Status { get; set; }
 72
 73        public override double GetDefaultPrimaryImageAspectRatio()
 74        {
 075            double value = 2;
 076            value /= 3;
 77
 078            return value;
 79        }
 80
 81        public override string CreatePresentationUniqueKey()
 82        {
 083            if (LibraryManager.GetLibraryOptions(this).EnableAutomaticSeriesGrouping)
 84            {
 085                var userdatakeys = GetUserDataKeys();
 86
 087                if (userdatakeys.Count > 1)
 88                {
 089                    return AddLibrariesToPresentationUniqueKey(userdatakeys[0]);
 90                }
 91            }
 92
 093            return base.CreatePresentationUniqueKey();
 94        }
 95
 96        private string AddLibrariesToPresentationUniqueKey(string key)
 97        {
 098            var lang = GetPreferredMetadataLanguage();
 099            if (!string.IsNullOrEmpty(lang))
 100            {
 0101                key += "-" + lang;
 102            }
 103
 0104            var folders = LibraryManager.GetCollectionFolders(this)
 0105                .Select(i => i.Id.ToString("N", CultureInfo.InvariantCulture))
 0106                .ToArray();
 107
 0108            if (folders.Length == 0)
 109            {
 0110                return key;
 111            }
 112
 0113            return key + "-" + string.Join('-', folders);
 114        }
 115
 116        private static string GetUniqueSeriesKey(BaseItem series)
 117        {
 0118            return series.GetPresentationUniqueKey();
 119        }
 120
 121        public override int GetChildCount(User user)
 122        {
 0123            var seriesKey = GetUniqueSeriesKey(this);
 124
 0125            var result = LibraryManager.GetCount(new InternalItemsQuery(user)
 0126            {
 0127                AncestorWithPresentationUniqueKey = null,
 0128                SeriesPresentationUniqueKey = seriesKey,
 0129                IncludeItemTypes = new[] { BaseItemKind.Season },
 0130                IsVirtualItem = false,
 0131                Limit = 0,
 0132                DtoOptions = new DtoOptions(false)
 0133                {
 0134                    EnableImages = false
 0135                }
 0136            });
 137
 0138            return result;
 139        }
 140
 141        public override int GetRecursiveChildCount(User user)
 142        {
 0143            var seriesKey = GetUniqueSeriesKey(this);
 144
 0145            var query = new InternalItemsQuery(user)
 0146            {
 0147                AncestorWithPresentationUniqueKey = null,
 0148                SeriesPresentationUniqueKey = seriesKey,
 0149                DtoOptions = new DtoOptions(false)
 0150                {
 0151                    EnableImages = false
 0152                }
 0153            };
 154
 0155            if (query.IncludeItemTypes.Length == 0)
 156            {
 0157                query.IncludeItemTypes = new[] { BaseItemKind.Episode };
 158            }
 159
 0160            query.IsVirtualItem = false;
 0161            query.Limit = 0;
 0162            var totalRecordCount = LibraryManager.GetCount(query);
 163
 0164            return totalRecordCount;
 165        }
 166
 167        /// <summary>
 168        /// Gets the user data key.
 169        /// </summary>
 170        /// <returns>System.String.</returns>
 171        public override List<string> GetUserDataKeys()
 172        {
 0173            var list = base.GetUserDataKeys();
 174
 0175            if (this.TryGetProviderId(MetadataProvider.Imdb, out var key))
 176            {
 0177                list.Insert(0, key);
 178            }
 179
 0180            if (this.TryGetProviderId(MetadataProvider.Tvdb, out key))
 181            {
 0182                list.Insert(0, key);
 183            }
 184
 0185            if (this.TryGetProviderId(MetadataProvider.Custom, out key))
 186            {
 0187                list.Insert(0, key);
 188            }
 189
 0190            return list;
 191        }
 192
 193        public override IReadOnlyList<BaseItem> GetChildren(User user, bool includeLinkedChildren, InternalItemsQuery qu
 194        {
 0195            return GetSeasons(user, new DtoOptions(true));
 196        }
 197
 198        public IReadOnlyList<BaseItem> GetSeasons(User user, DtoOptions options)
 199        {
 0200            var query = new InternalItemsQuery(user)
 0201            {
 0202                DtoOptions = options
 0203            };
 204
 0205            SetSeasonQueryOptions(query, user);
 206
 0207            return LibraryManager.GetItemList(query);
 208        }
 209
 210        private void SetSeasonQueryOptions(InternalItemsQuery query, User user)
 211        {
 0212            var seriesKey = GetUniqueSeriesKey(this);
 213
 0214            query.AncestorWithPresentationUniqueKey = null;
 0215            query.SeriesPresentationUniqueKey = seriesKey;
 0216            query.IncludeItemTypes = new[] { BaseItemKind.Season };
 0217            query.OrderBy = new[] { (ItemSortBy.SortName, SortOrder.Ascending) };
 218
 0219            if (user is not null && !user.DisplayMissingEpisodes)
 220            {
 0221                query.IsMissing = false;
 222            }
 0223        }
 224
 225        protected override QueryResult<BaseItem> GetItemsInternal(InternalItemsQuery query)
 226        {
 0227            var user = query.User;
 228
 0229            if (SourceType == SourceType.Channel)
 230            {
 231                try
 232                {
 0233                    query.Parent = this;
 0234                    query.ChannelIds = [ChannelId];
 0235                    return ChannelManager.GetChannelItemsInternal(query, new Progress<double>(), CancellationToken.None)
 236                }
 0237                catch
 238                {
 239                    // Already logged at lower levels
 0240                    return new QueryResult<BaseItem>();
 241                }
 242            }
 243
 0244            if (query.Recursive)
 245            {
 0246                var seriesKey = GetUniqueSeriesKey(this);
 247
 0248                query.AncestorWithPresentationUniqueKey = null;
 0249                query.SeriesPresentationUniqueKey = seriesKey;
 0250                if (query.OrderBy.Count == 0)
 251                {
 0252                    query.OrderBy = new[] { (ItemSortBy.SortName, SortOrder.Ascending) };
 253                }
 254
 0255                if (query.IncludeItemTypes.Length == 0)
 256                {
 0257                    query.IncludeItemTypes = new[] { BaseItemKind.Episode, BaseItemKind.Season };
 258                }
 259
 0260                query.IsVirtualItem = false;
 0261                return LibraryManager.GetItemsResult(query);
 262            }
 263
 0264            SetSeasonQueryOptions(query, user);
 265
 0266            return LibraryManager.GetItemsResult(query);
 0267        }
 268
 269        public IEnumerable<BaseItem> GetEpisodes(User user, DtoOptions options, bool shouldIncludeMissingEpisodes)
 270        {
 0271            var seriesKey = GetUniqueSeriesKey(this);
 272
 0273            var query = new InternalItemsQuery(user)
 0274            {
 0275                AncestorWithPresentationUniqueKey = null,
 0276                SeriesPresentationUniqueKey = seriesKey,
 0277                IncludeItemTypes = new[] { BaseItemKind.Episode, BaseItemKind.Season },
 0278                OrderBy = new[] { (ItemSortBy.SortName, SortOrder.Ascending) },
 0279                DtoOptions = options,
 0280            };
 281
 0282            if (!shouldIncludeMissingEpisodes)
 283            {
 0284                query.IsMissing = false;
 285            }
 286
 0287            var allItems = LibraryManager.GetItemList(query);
 288
 0289            var allSeriesEpisodes = allItems.OfType<Episode>().ToList();
 290
 0291            var allEpisodes = allItems.OfType<Season>()
 0292                .SelectMany(i => i.GetEpisodes(this, user, allSeriesEpisodes, options, shouldIncludeMissingEpisodes))
 0293                .Reverse();
 294
 295            // Specials could appear twice based on above - once in season 0, once in the aired season
 296            // This depends on settings for that series
 297            // When this happens, remove the duplicate from season 0
 298
 0299            return allEpisodes.DistinctBy(i => i.Id).Reverse();
 300        }
 301
 302        public async Task RefreshAllMetadata(MetadataRefreshOptions refreshOptions, IProgress<double> progress, Cancella
 303        {
 304            // Refresh bottom up, seasons and episodes first, then the series
 305            var items = GetRecursiveChildren();
 306
 307            var totalItems = items.Count;
 308            var numComplete = 0;
 309
 310            // Refresh seasons
 311            foreach (var item in items)
 312            {
 313                if (item is not Season)
 314                {
 315                    continue;
 316                }
 317
 318                cancellationToken.ThrowIfCancellationRequested();
 319
 320                if (refreshOptions.RefreshItem(item))
 321                {
 322                    await item.RefreshMetadata(refreshOptions, cancellationToken).ConfigureAwait(false);
 323                }
 324
 325                numComplete++;
 326                double percent = numComplete;
 327                percent /= totalItems;
 328                progress.Report(percent * 100);
 329            }
 330
 331            // Refresh episodes and other children
 332            foreach (var item in items)
 333            {
 334                if (item is Season)
 335                {
 336                    continue;
 337                }
 338
 339                cancellationToken.ThrowIfCancellationRequested();
 340
 341                bool skipItem = item is Episode episode
 342                    && refreshOptions.MetadataRefreshMode != MetadataRefreshMode.FullRefresh
 343                    && !refreshOptions.ReplaceAllMetadata
 344                    && episode.IsMissingEpisode
 345                    && episode.LocationType == LocationType.Virtual
 346                    && episode.PremiereDate.HasValue
 347                    && (DateTime.UtcNow - episode.PremiereDate.Value).TotalDays > 30;
 348
 349                if (!skipItem)
 350                {
 351                    if (refreshOptions.RefreshItem(item))
 352                    {
 353                        await item.RefreshMetadata(refreshOptions, cancellationToken).ConfigureAwait(false);
 354                    }
 355                }
 356
 357                numComplete++;
 358                double percent = numComplete;
 359                percent /= totalItems;
 360                progress.Report(percent * 100);
 361            }
 362
 363            refreshOptions = new MetadataRefreshOptions(refreshOptions);
 364            await ProviderManager.RefreshSingleItem(this, refreshOptions, cancellationToken).ConfigureAwait(false);
 365        }
 366
 367        public List<BaseItem> GetSeasonEpisodes(Season parentSeason, User user, DtoOptions options, bool shouldIncludeMi
 368        {
 0369            var queryFromSeries = ConfigurationManager.Configuration.DisplaySpecialsWithinSeasons;
 370
 371            // add optimization when this setting is not enabled
 0372            var seriesKey = queryFromSeries ?
 0373                GetUniqueSeriesKey(this) :
 0374                GetUniqueSeriesKey(parentSeason);
 375
 0376            var query = new InternalItemsQuery(user)
 0377            {
 0378                AncestorWithPresentationUniqueKey = queryFromSeries ? null : seriesKey,
 0379                SeriesPresentationUniqueKey = queryFromSeries ? seriesKey : null,
 0380                IncludeItemTypes = new[] { BaseItemKind.Episode },
 0381                OrderBy = new[] { (ItemSortBy.SortName, SortOrder.Ascending) },
 0382                DtoOptions = options
 0383            };
 384
 0385            if (!shouldIncludeMissingEpisodes)
 386            {
 0387                query.IsMissing = false;
 388            }
 389
 390            IReadOnlyList<BaseItem> allItems;
 0391            if (SourceType == SourceType.Channel)
 392            {
 393                try
 394                {
 0395                    query.Parent = parentSeason;
 0396                    query.ChannelIds = [ChannelId];
 0397                    allItems = [.. ChannelManager.GetChannelItemsInternal(query, new Progress<double>(), CancellationTok
 0398                }
 0399                catch
 400                {
 401                    // Already logged at lower levels
 0402                    return [];
 403                }
 404            }
 405            else
 406            {
 0407                allItems = LibraryManager.GetItemList(query);
 408            }
 409
 0410            return GetSeasonEpisodes(parentSeason, user, allItems, options, shouldIncludeMissingEpisodes);
 0411        }
 412
 413        public List<BaseItem> GetSeasonEpisodes(Season parentSeason, User user, IEnumerable<BaseItem> allSeriesEpisodes,
 414        {
 0415            if (allSeriesEpisodes is null)
 416            {
 0417                return GetSeasonEpisodes(parentSeason, user, options, shouldIncludeMissingEpisodes);
 418            }
 419
 0420            var episodes = FilterEpisodesBySeason(allSeriesEpisodes, parentSeason, ConfigurationManager.Configuration.Di
 421
 0422            var sortBy = (parentSeason.IndexNumber ?? -1) == 0 ? ItemSortBy.SortName : ItemSortBy.AiredEpisodeOrder;
 423
 0424            return LibraryManager.Sort(episodes, user, new[] { sortBy }, SortOrder.Ascending).ToList();
 425        }
 426
 427        /// <summary>
 428        /// Filters the episodes by season.
 429        /// </summary>
 430        /// <param name="episodes">The episodes.</param>
 431        /// <param name="parentSeason">The season.</param>
 432        /// <param name="includeSpecials"><c>true</c> to include special, <c>false</c> to not.</param>
 433        /// <returns>The set of episodes.</returns>
 434        public static IEnumerable<BaseItem> FilterEpisodesBySeason(IEnumerable<BaseItem> episodes, Season parentSeason, 
 435        {
 0436            var seasonNumber = parentSeason.IndexNumber;
 0437            var seasonPresentationKey = GetUniqueSeriesKey(parentSeason);
 438
 0439            var supportSpecialsInSeason = includeSpecials && seasonNumber.HasValue && seasonNumber.Value != 0;
 440
 0441            return episodes.Where(episode =>
 0442            {
 0443                var episodeItem = (Episode)episode;
 0444
 0445                var currentSeasonNumber = supportSpecialsInSeason ? episodeItem.AiredSeasonNumber : episode.ParentIndexN
 0446                if (currentSeasonNumber.HasValue && seasonNumber.HasValue && currentSeasonNumber.Value == seasonNumber.V
 0447                {
 0448                    return true;
 0449                }
 0450
 0451                if (!currentSeasonNumber.HasValue && !seasonNumber.HasValue && parentSeason.LocationType == LocationType
 0452                {
 0453                    return true;
 0454                }
 0455
 0456                var season = episodeItem.Season;
 0457                return season is not null && string.Equals(GetUniqueSeriesKey(season), seasonPresentationKey, StringComp
 0458            });
 459        }
 460
 461        /// <summary>
 462        /// Filters the episodes by season.
 463        /// </summary>
 464        /// <param name="episodes">The episodes.</param>
 465        /// <param name="seasonNumber">The season.</param>
 466        /// <param name="includeSpecials"><c>true</c> to include special, <c>false</c> to not.</param>
 467        /// <returns>The set of episodes.</returns>
 468        public static IEnumerable<Episode> FilterEpisodesBySeason(IEnumerable<Episode> episodes, int seasonNumber, bool 
 469        {
 0470            if (!includeSpecials || seasonNumber < 1)
 471            {
 0472                return episodes.Where(i => (i.ParentIndexNumber ?? -1) == seasonNumber);
 473            }
 474
 0475            return episodes.Where(i =>
 0476            {
 0477                var episode = i;
 0478
 0479                if (episode is not null)
 0480                {
 0481                    var currentSeasonNumber = episode.AiredSeasonNumber;
 0482
 0483                    return currentSeasonNumber.HasValue && currentSeasonNumber.Value == seasonNumber;
 0484                }
 0485
 0486                return false;
 0487            });
 488        }
 489
 490        protected override bool GetBlockUnratedValue(User user)
 491        {
 0492            return user.GetPreferenceValues<UnratedItem>(PreferenceKind.BlockUnratedItems).Contains(UnratedItem.Series);
 493        }
 494
 495        public override UnratedItem GetBlockUnratedType()
 496        {
 0497            return UnratedItem.Series;
 498        }
 499
 500        public SeriesInfo GetLookupInfo()
 501        {
 0502            var info = GetItemLookupInfo<SeriesInfo>();
 503
 0504            return info;
 505        }
 506
 507        public override bool BeforeMetadataRefresh(bool replaceAllMetadata)
 508        {
 0509            var hasChanges = base.BeforeMetadataRefresh(replaceAllMetadata);
 510
 0511            if (!ProductionYear.HasValue)
 512            {
 0513                var info = LibraryManager.ParseName(Name);
 514
 0515                var yearInName = info.Year;
 516
 0517                if (yearInName.HasValue)
 518                {
 0519                    ProductionYear = yearInName;
 0520                    hasChanges = true;
 521                }
 522            }
 523
 0524            return hasChanges;
 525        }
 526    }
 527}

Methods/Properties

.ctor()
get_SupportsAddingToPlaylist()
get_IsPreSorted()
get_SupportsDateLastMediaAdded()
get_SupportsInheritedParentImages()
get_SupportsPeople()
get_LocalTrailers()
GetDefaultPrimaryImageAspectRatio()
CreatePresentationUniqueKey()
AddLibrariesToPresentationUniqueKey(System.String)
GetUniqueSeriesKey(MediaBrowser.Controller.Entities.BaseItem)
GetChildCount(Jellyfin.Database.Implementations.Entities.User)
GetRecursiveChildCount(Jellyfin.Database.Implementations.Entities.User)
GetUserDataKeys()
GetChildren(Jellyfin.Database.Implementations.Entities.User,System.Boolean,MediaBrowser.Controller.Entities.InternalItemsQuery)
GetSeasons(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Dto.DtoOptions)
SetSeasonQueryOptions(MediaBrowser.Controller.Entities.InternalItemsQuery,Jellyfin.Database.Implementations.Entities.User)
GetItemsInternal(MediaBrowser.Controller.Entities.InternalItemsQuery)
GetEpisodes(Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Dto.DtoOptions,System.Boolean)
GetSeasonEpisodes(MediaBrowser.Controller.Entities.TV.Season,Jellyfin.Database.Implementations.Entities.User,MediaBrowser.Controller.Dto.DtoOptions,System.Boolean)
GetSeasonEpisodes(MediaBrowser.Controller.Entities.TV.Season,Jellyfin.Database.Implementations.Entities.User,System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Entities.BaseItem>,MediaBrowser.Controller.Dto.DtoOptions,System.Boolean)
FilterEpisodesBySeason(System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Entities.BaseItem>,MediaBrowser.Controller.Entities.TV.Season,System.Boolean)
FilterEpisodesBySeason(System.Collections.Generic.IEnumerable`1<MediaBrowser.Controller.Entities.TV.Episode>,System.Int32,System.Boolean)
GetBlockUnratedValue(Jellyfin.Database.Implementations.Entities.User)
GetBlockUnratedType()
GetLookupInfo()
BeforeMetadataRefresh(System.Boolean)