2020-06-22 16:02:57 +00:00
using System ;
2020-06-24 12:40:37 +00:00
using System.Collections.Generic ;
2020-08-06 14:17:45 +00:00
using System.ComponentModel.DataAnnotations ;
2020-06-22 16:02:57 +00:00
using System.Linq ;
using Jellyfin.Api.Constants ;
using Jellyfin.Api.Extensions ;
2020-11-09 21:59:04 +00:00
using Jellyfin.Api.ModelBinders ;
2020-08-03 18:01:24 +00:00
using Jellyfin.Data.Enums ;
2021-06-19 16:02:33 +00:00
using Jellyfin.Extensions ;
2020-06-22 16:02:57 +00:00
using MediaBrowser.Controller.Dto ;
using MediaBrowser.Controller.Entities ;
using MediaBrowser.Controller.Entities.TV ;
using MediaBrowser.Controller.Library ;
using MediaBrowser.Controller.TV ;
using MediaBrowser.Model.Dto ;
2020-10-09 23:52:39 +00:00
using MediaBrowser.Model.Entities ;
2020-06-22 16:02:57 +00:00
using MediaBrowser.Model.Querying ;
using Microsoft.AspNetCore.Authorization ;
2020-06-24 12:40:37 +00:00
using Microsoft.AspNetCore.Http ;
2020-06-22 16:02:57 +00:00
using Microsoft.AspNetCore.Mvc ;
namespace Jellyfin.Api.Controllers
{
/// <summary>
/// The tv shows controller.
/// </summary>
2020-08-04 14:27:54 +00:00
[Route("Shows")]
2020-06-22 16:02:57 +00:00
[Authorize(Policy = Policies.DefaultAuthorization)]
public class TvShowsController : BaseJellyfinApiController
{
private readonly IUserManager _userManager ;
private readonly ILibraryManager _libraryManager ;
private readonly IDtoService _dtoService ;
private readonly ITVSeriesManager _tvSeriesManager ;
/// <summary>
/// Initializes a new instance of the <see cref="TvShowsController"/> class.
/// </summary>
/// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
/// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
/// <param name="dtoService">Instance of the <see cref="IDtoService"/> interface.</param>
/// <param name="tvSeriesManager">Instance of the <see cref="ITVSeriesManager"/> interface.</param>
public TvShowsController (
IUserManager userManager ,
ILibraryManager libraryManager ,
IDtoService dtoService ,
2020-06-24 12:40:37 +00:00
ITVSeriesManager tvSeriesManager )
2020-06-22 16:02:57 +00:00
{
_userManager = userManager ;
_libraryManager = libraryManager ;
_dtoService = dtoService ;
_tvSeriesManager = tvSeriesManager ;
}
/// <summary>
/// Gets a list of next up episodes.
/// </summary>
/// <param name="userId">The user id of the user to get the next up episodes for.</param>
/// <param name="startIndex">Optional. The record index to start at. All items with a lower index will be dropped from the results.</param>
/// <param name="limit">Optional. The maximum number of records to return.</param>
2020-10-09 23:35:08 +00:00
/// <param name="fields">Optional. Specify additional fields of information to return in the output.</param>
2020-06-22 16:02:57 +00:00
/// <param name="seriesId">Optional. Filter by series id.</param>
/// <param name="parentId">Optional. Specify this to localize the search to a specific item or folder. Omit to use the root.</param>
2021-12-19 00:17:03 +00:00
/// <param name="enableImages">Optional. Include image information in output.</param>
2020-06-22 16:02:57 +00:00
/// <param name="imageTypeLimit">Optional. The max number of images to return, per image type.</param>
/// <param name="enableImageTypes">Optional. The image types to include in the output.</param>
/// <param name="enableUserData">Optional. Include user data.</param>
2021-05-27 02:49:53 +00:00
/// <param name="nextUpDateCutoff">Optional. Starting date of shows to show in Next Up section.</param>
2020-06-22 16:02:57 +00:00
/// <param name="enableTotalRecordCount">Whether to enable the total records count. Defaults to true.</param>
2021-01-15 22:08:48 +00:00
/// <param name="disableFirstEpisode">Whether to disable sending the first episode in a series as next up.</param>
2022-03-01 05:49:29 +00:00
/// <param name="enableRewatching">Whether to include watched episode in next up results.</param>
2020-06-22 16:02:57 +00:00
/// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the next up episodes.</returns>
[HttpGet("NextUp")]
2020-06-24 12:40:37 +00:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-06-22 16:02:57 +00:00
public ActionResult < QueryResult < BaseItemDto > > GetNextUp (
2020-09-09 20:28:30 +00:00
[FromQuery] Guid ? userId ,
2020-06-22 16:02:57 +00:00
[FromQuery] int? startIndex ,
[FromQuery] int? limit ,
2020-11-09 21:59:04 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields [ ] fields ,
2022-08-14 11:03:48 +00:00
[FromQuery] Guid ? seriesId ,
2020-12-01 18:07:41 +00:00
[FromQuery] Guid ? parentId ,
2021-12-19 00:17:03 +00:00
[FromQuery] bool? enableImages ,
2020-06-22 16:02:57 +00:00
[FromQuery] int? imageTypeLimit ,
2020-11-09 21:53:23 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType [ ] enableImageTypes ,
2020-06-22 16:02:57 +00:00
[FromQuery] bool? enableUserData ,
2021-04-16 17:57:22 +00:00
[FromQuery] DateTime ? nextUpDateCutoff ,
2021-01-15 22:08:48 +00:00
[FromQuery] bool enableTotalRecordCount = true ,
2022-02-20 17:05:57 +00:00
[FromQuery] bool disableFirstEpisode = false ,
2022-03-01 05:49:29 +00:00
[FromQuery] bool enableRewatching = false )
2020-06-22 16:02:57 +00:00
{
2020-10-29 17:36:45 +00:00
var options = new DtoOptions { Fields = fields }
2020-06-22 16:02:57 +00:00
. AddClientFields ( Request )
2021-12-19 00:17:03 +00:00
. AddAdditionalDtoOptions ( enableImages , enableUserData , imageTypeLimit , enableImageTypes ) ;
2020-06-22 16:02:57 +00:00
var result = _tvSeriesManager . GetNextUp (
new NextUpQuery
{
Limit = limit ,
ParentId = parentId ,
SeriesId = seriesId ,
StartIndex = startIndex ,
2020-07-07 15:10:51 +00:00
UserId = userId ? ? Guid . Empty ,
2021-01-15 22:08:48 +00:00
EnableTotalRecordCount = enableTotalRecordCount ,
2021-04-15 18:44:21 +00:00
DisableFirstEpisode = disableFirstEpisode ,
2022-02-20 17:05:57 +00:00
NextUpDateCutoff = nextUpDateCutoff ? ? DateTime . MinValue ,
2022-03-01 05:49:29 +00:00
EnableRewatching = enableRewatching
2020-06-22 16:02:57 +00:00
} ,
options ) ;
2022-02-21 13:15:09 +00:00
var user = userId is null | | userId . Value . Equals ( default )
? null
: _userManager . GetUserById ( userId . Value ) ;
2020-06-22 16:02:57 +00:00
var returnItems = _dtoService . GetBaseItemDtos ( result . Items , options , user ) ;
2022-01-20 15:46:17 +00:00
return new QueryResult < BaseItemDto > (
startIndex ,
result . TotalRecordCount ,
returnItems ) ;
2020-06-22 16:02:57 +00:00
}
/// <summary>
/// Gets a list of upcoming episodes.
/// </summary>
/// <param name="userId">The user id of the user to get the upcoming episodes for.</param>
/// <param name="startIndex">Optional. The record index to start at. All items with a lower index will be dropped from the results.</param>
/// <param name="limit">Optional. The maximum number of records to return.</param>
2020-10-09 23:35:08 +00:00
/// <param name="fields">Optional. Specify additional fields of information to return in the output.</param>
2020-06-22 16:02:57 +00:00
/// <param name="parentId">Optional. Specify this to localize the search to a specific item or folder. Omit to use the root.</param>
2021-12-19 00:17:03 +00:00
/// <param name="enableImages">Optional. Include image information in output.</param>
2020-06-22 16:02:57 +00:00
/// <param name="imageTypeLimit">Optional. The max number of images to return, per image type.</param>
/// <param name="enableImageTypes">Optional. The image types to include in the output.</param>
/// <param name="enableUserData">Optional. Include user data.</param>
/// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the next up episodes.</returns>
[HttpGet("Upcoming")]
2020-06-24 12:40:37 +00:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-06-22 16:02:57 +00:00
public ActionResult < QueryResult < BaseItemDto > > GetUpcomingEpisodes (
2020-09-09 20:28:30 +00:00
[FromQuery] Guid ? userId ,
2020-06-22 16:02:57 +00:00
[FromQuery] int? startIndex ,
[FromQuery] int? limit ,
2020-11-09 21:59:04 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields [ ] fields ,
2020-12-01 18:07:41 +00:00
[FromQuery] Guid ? parentId ,
2021-12-19 00:17:03 +00:00
[FromQuery] bool? enableImages ,
2020-06-22 16:02:57 +00:00
[FromQuery] int? imageTypeLimit ,
2020-11-09 21:53:23 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType [ ] enableImageTypes ,
2020-06-24 12:40:37 +00:00
[FromQuery] bool? enableUserData )
2020-06-22 16:02:57 +00:00
{
2022-02-21 13:15:09 +00:00
var user = userId is null | | userId . Value . Equals ( default )
? null
: _userManager . GetUserById ( userId . Value ) ;
2020-06-22 16:02:57 +00:00
2021-09-20 23:21:45 +00:00
var minPremiereDate = DateTime . UtcNow . Date . AddDays ( - 1 ) ;
2020-06-22 16:02:57 +00:00
2020-12-01 18:07:41 +00:00
var parentIdGuid = parentId ? ? Guid . Empty ;
2020-06-22 16:02:57 +00:00
2020-10-29 17:36:45 +00:00
var options = new DtoOptions { Fields = fields }
2020-06-22 16:02:57 +00:00
. AddClientFields ( Request )
2021-12-19 00:17:03 +00:00
. AddAdditionalDtoOptions ( enableImages , enableUserData , imageTypeLimit , enableImageTypes ) ;
2020-06-22 16:02:57 +00:00
var itemsResult = _libraryManager . GetItemList ( new InternalItemsQuery ( user )
{
2021-12-12 02:31:30 +00:00
IncludeItemTypes = new [ ] { BaseItemKind . Episode } ,
2021-04-21 20:25:08 +00:00
OrderBy = new [ ] { ( ItemSortBy . PremiereDate , SortOrder . Ascending ) , ( ItemSortBy . SortName , SortOrder . Ascending ) } ,
2020-06-22 16:02:57 +00:00
MinPremiereDate = minPremiereDate ,
StartIndex = startIndex ,
Limit = limit ,
ParentId = parentIdGuid ,
Recursive = true ,
DtoOptions = options
} ) ;
var returnItems = _dtoService . GetBaseItemDtos ( itemsResult , options , user ) ;
2022-01-20 15:46:17 +00:00
return new QueryResult < BaseItemDto > (
startIndex ,
itemsResult . Count ,
returnItems ) ;
2020-06-22 16:02:57 +00:00
}
2020-06-24 12:40:37 +00:00
/// <summary>
/// Gets episodes for a tv season.
/// </summary>
/// <param name="seriesId">The series id.</param>
/// <param name="userId">The user id.</param>
2020-11-18 13:23:45 +00:00
/// <param name="fields">Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimited. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines, TrailerUrls.</param>
2020-06-24 12:40:37 +00:00
/// <param name="season">Optional filter by season number.</param>
/// <param name="seasonId">Optional. Filter by season id.</param>
/// <param name="isMissing">Optional. Filter by items that are missing episodes or not.</param>
/// <param name="adjacentTo">Optional. Return items that are siblings of a supplied item.</param>
/// <param name="startItemId">Optional. Skip through the list until a given item is found.</param>
/// <param name="startIndex">Optional. The record index to start at. All items with a lower index will be dropped from the results.</param>
/// <param name="limit">Optional. The maximum number of records to return.</param>
/// <param name="enableImages">Optional, include image information in output.</param>
/// <param name="imageTypeLimit">Optional, the max number of images to return, per image type.</param>
/// <param name="enableImageTypes">Optional. The image types to include in the output.</param>
/// <param name="enableUserData">Optional. Include user data.</param>
2020-11-18 13:23:45 +00:00
/// <param name="sortBy">Optional. Specify one or more sort orders, comma delimited. Options: Album, AlbumArtist, Artist, Budget, CommunityRating, CriticRating, DateCreated, DatePlayed, PlayCount, PremiereDate, ProductionYear, SortName, Random, Revenue, Runtime.</param>
2020-06-24 12:40:37 +00:00
/// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the episodes on success or a <see cref="NotFoundResult"/> if the series was not found.</returns>
[HttpGet("{seriesId}/Episodes")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public ActionResult < QueryResult < BaseItemDto > > GetEpisodes (
2020-12-01 18:07:41 +00:00
[FromRoute, Required] Guid seriesId ,
2020-09-08 00:45:06 +00:00
[FromQuery] Guid ? userId ,
2020-11-09 21:59:04 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields [ ] fields ,
2020-06-24 12:40:37 +00:00
[FromQuery] int? season ,
2020-12-01 18:07:41 +00:00
[FromQuery] Guid ? seasonId ,
2020-06-24 12:40:37 +00:00
[FromQuery] bool? isMissing ,
2022-08-14 10:47:25 +00:00
[FromQuery] Guid ? adjacentTo ,
2020-12-01 18:07:41 +00:00
[FromQuery] Guid ? startItemId ,
2020-06-24 12:40:37 +00:00
[FromQuery] int? startIndex ,
[FromQuery] int? limit ,
[FromQuery] bool? enableImages ,
[FromQuery] int? imageTypeLimit ,
2020-11-09 21:53:23 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType [ ] enableImageTypes ,
2020-06-24 12:40:37 +00:00
[FromQuery] bool? enableUserData ,
2020-06-24 20:43:28 +00:00
[FromQuery] string? sortBy )
2020-06-24 12:40:37 +00:00
{
2022-02-21 13:15:09 +00:00
var user = userId is null | | userId . Value . Equals ( default )
? null
: _userManager . GetUserById ( userId . Value ) ;
2020-06-24 12:40:37 +00:00
List < BaseItem > episodes ;
2020-10-29 17:36:45 +00:00
var dtoOptions = new DtoOptions { Fields = fields }
2020-06-24 12:40:37 +00:00
. AddClientFields ( Request )
2021-10-03 03:43:05 +00:00
. AddAdditionalDtoOptions ( enableImages , enableUserData , imageTypeLimit , enableImageTypes ) ;
2020-06-24 12:40:37 +00:00
2020-12-01 18:07:41 +00:00
if ( seasonId . HasValue ) // Season id was supplied. Get episodes by season id.
2020-06-24 12:40:37 +00:00
{
2020-12-01 18:07:41 +00:00
var item = _libraryManager . GetItemById ( seasonId . Value ) ;
2021-08-28 22:32:50 +00:00
if ( item is not Season seasonItem )
2020-06-24 12:40:37 +00:00
{
return NotFound ( "No season exists with Id " + seasonId ) ;
}
episodes = seasonItem . GetEpisodes ( user , dtoOptions ) ;
}
else if ( season . HasValue ) // Season number was supplied. Get episodes by season number
{
2021-08-28 22:32:50 +00:00
if ( _libraryManager . GetItemById ( seriesId ) is not Series series )
2020-06-24 12:40:37 +00:00
{
return NotFound ( "Series not found" ) ;
}
var seasonItem = series
. GetSeasons ( user , dtoOptions )
. FirstOrDefault ( i = > i . IndexNumber = = season . Value ) ;
episodes = seasonItem = = null ?
new List < BaseItem > ( )
: ( ( Season ) seasonItem ) . GetEpisodes ( user , dtoOptions ) ;
}
else // No season number or season id was supplied. Returning all episodes.
{
2021-08-28 22:32:50 +00:00
if ( _libraryManager . GetItemById ( seriesId ) is not Series series )
2020-06-24 12:40:37 +00:00
{
return NotFound ( "Series not found" ) ;
}
episodes = series . GetEpisodes ( user , dtoOptions ) . ToList ( ) ;
}
// Filter after the fact in case the ui doesn't want them
if ( isMissing . HasValue )
{
var val = isMissing . Value ;
episodes = episodes
. Where ( i = > ( ( Episode ) i ) . IsMissingEpisode = = val )
. ToList ( ) ;
}
2020-12-01 18:07:41 +00:00
if ( startItemId . HasValue )
2020-06-24 12:40:37 +00:00
{
episodes = episodes
2021-01-03 02:23:54 +00:00
. SkipWhile ( i = > ! startItemId . Value . Equals ( i . Id ) )
2020-06-24 12:40:37 +00:00
. ToList ( ) ;
}
// This must be the last filter
2022-08-14 10:58:38 +00:00
if ( adjacentTo . HasValue & & ! adjacentTo . Value . Equals ( default ) )
2020-06-24 12:40:37 +00:00
{
2022-08-14 10:47:25 +00:00
episodes = UserViewBuilder . FilterForAdjacency ( episodes , adjacentTo . Value ) . ToList ( ) ;
2020-06-24 12:40:37 +00:00
}
if ( string . Equals ( sortBy , ItemSortBy . Random , StringComparison . OrdinalIgnoreCase ) )
{
episodes . Shuffle ( ) ;
}
var returnItems = episodes ;
if ( startIndex . HasValue | | limit . HasValue )
{
returnItems = ApplyPaging ( episodes , startIndex , limit ) . ToList ( ) ;
}
var dtos = _dtoService . GetBaseItemDtos ( returnItems , dtoOptions , user ) ;
2022-01-20 15:46:17 +00:00
return new QueryResult < BaseItemDto > (
startIndex ,
episodes . Count ,
dtos ) ;
2020-06-24 12:40:37 +00:00
}
/// <summary>
/// Gets seasons for a tv series.
/// </summary>
/// <param name="seriesId">The series id.</param>
/// <param name="userId">The user id.</param>
2020-11-18 13:23:45 +00:00
/// <param name="fields">Optional. Specify additional fields of information to return in the output. This allows multiple, comma delimited. Options: Budget, Chapters, DateCreated, Genres, HomePageUrl, IndexOptions, MediaStreams, Overview, ParentId, Path, People, ProviderIds, PrimaryImageAspectRatio, Revenue, SortName, Studios, Taglines, TrailerUrls.</param>
2020-06-24 12:40:37 +00:00
/// <param name="isSpecialSeason">Optional. Filter by special season.</param>
/// <param name="isMissing">Optional. Filter by items that are missing episodes or not.</param>
/// <param name="adjacentTo">Optional. Return items that are siblings of a supplied item.</param>
/// <param name="enableImages">Optional. Include image information in output.</param>
/// <param name="imageTypeLimit">Optional. The max number of images to return, per image type.</param>
/// <param name="enableImageTypes">Optional. The image types to include in the output.</param>
/// <param name="enableUserData">Optional. Include user data.</param>
/// <returns>A <see cref="QueryResult{BaseItemDto}"/> on success or a <see cref="NotFoundResult"/> if the series was not found.</returns>
[HttpGet("{seriesId}/Seasons")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
public ActionResult < QueryResult < BaseItemDto > > GetSeasons (
2020-12-01 18:07:41 +00:00
[FromRoute, Required] Guid seriesId ,
2020-09-08 00:45:06 +00:00
[FromQuery] Guid ? userId ,
2020-11-09 21:59:04 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ItemFields [ ] fields ,
2020-06-24 12:40:37 +00:00
[FromQuery] bool? isSpecialSeason ,
[FromQuery] bool? isMissing ,
2022-08-14 10:47:25 +00:00
[FromQuery] Guid ? adjacentTo ,
2020-06-24 12:40:37 +00:00
[FromQuery] bool? enableImages ,
[FromQuery] int? imageTypeLimit ,
2020-11-09 21:53:23 +00:00
[FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] ImageType [ ] enableImageTypes ,
2020-06-24 12:40:37 +00:00
[FromQuery] bool? enableUserData )
{
2022-02-21 13:15:09 +00:00
var user = userId is null | | userId . Value . Equals ( default )
? null
: _userManager . GetUserById ( userId . Value ) ;
2020-06-24 12:40:37 +00:00
2021-08-28 22:32:50 +00:00
if ( _libraryManager . GetItemById ( seriesId ) is not Series series )
2020-06-24 12:40:37 +00:00
{
return NotFound ( "Series not found" ) ;
}
var seasons = series . GetItemList ( new InternalItemsQuery ( user )
{
IsMissing = isMissing ,
IsSpecialSeason = isSpecialSeason ,
AdjacentTo = adjacentTo
} ) ;
2020-10-29 17:36:45 +00:00
var dtoOptions = new DtoOptions { Fields = fields }
2020-06-24 12:40:37 +00:00
. AddClientFields ( Request )
2021-10-03 03:43:05 +00:00
. AddAdditionalDtoOptions ( enableImages , enableUserData , imageTypeLimit , enableImageTypes ) ;
2020-06-24 12:40:37 +00:00
var returnItems = _dtoService . GetBaseItemDtos ( seasons , dtoOptions , user ) ;
2022-01-20 15:46:17 +00:00
return new QueryResult < BaseItemDto > ( returnItems ) ;
2020-06-24 12:40:37 +00:00
}
/// <summary>
/// Applies the paging.
/// </summary>
/// <param name="items">The items.</param>
/// <param name="startIndex">The start index.</param>
/// <param name="limit">The limit.</param>
/// <returns>IEnumerable{BaseItem}.</returns>
private IEnumerable < BaseItem > ApplyPaging ( IEnumerable < BaseItem > items , int? startIndex , int? limit )
{
// Start at
if ( startIndex . HasValue )
{
items = items . Skip ( startIndex . Value ) ;
}
// Return limit
if ( limit . HasValue )
{
items = items . Take ( limit . Value ) ;
}
return items ;
}
2020-06-22 16:02:57 +00:00
}
}