2019-01-13 20:01:16 +00:00
using System ;
2019-01-13 19:24:58 +00:00
using System.Globalization ;
using System.Threading.Tasks ;
2020-05-20 17:07:53 +00:00
using Jellyfin.Data.Entities ;
2019-11-17 22:05:39 +00:00
using MediaBrowser.Controller.Configuration ;
2018-12-15 05:17:07 +00:00
using MediaBrowser.Controller.Library ;
using MediaBrowser.Controller.Net ;
using MediaBrowser.Controller.Session ;
using MediaBrowser.Model.Dto ;
using MediaBrowser.Model.Services ;
2019-01-13 19:24:58 +00:00
using MediaBrowser.Model.Session ;
2018-12-20 10:38:42 +00:00
using Microsoft.Extensions.Logging ;
2018-12-15 05:17:07 +00:00
namespace MediaBrowser.Api.UserLibrary
{
/// <summary>
2020-06-15 22:37:52 +00:00
/// Class MarkPlayedItem.
2018-12-15 05:17:07 +00:00
/// </summary>
[Route("/Users/{UserId}/PlayedItems/{Id}", "POST", Summary = "Marks an item as played")]
public class MarkPlayedItem : IReturn < UserItemDataDto >
{
/// <summary>
/// Gets or sets the user id.
/// </summary>
/// <value>The user id.</value>
[ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
public string UserId { get ; set ; }
[ApiMember(Name = "DatePlayed", Description = "The date the item was played (if any). Format = yyyyMMddHHmmss", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string DatePlayed { get ; set ; }
/// <summary>
/// Gets or sets the id.
/// </summary>
/// <value>The id.</value>
[ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
public string Id { get ; set ; }
}
/// <summary>
2020-06-15 22:37:52 +00:00
/// Class MarkUnplayedItem.
2018-12-15 05:17:07 +00:00
/// </summary>
[Route("/Users/{UserId}/PlayedItems/{Id}", "DELETE", Summary = "Marks an item as unplayed")]
public class MarkUnplayedItem : IReturn < UserItemDataDto >
{
/// <summary>
/// Gets or sets the user id.
/// </summary>
/// <value>The user id.</value>
[ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
public string UserId { get ; set ; }
/// <summary>
/// Gets or sets the id.
/// </summary>
/// <value>The id.</value>
[ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
public string Id { get ; set ; }
}
[Route("/Sessions/Playing", "POST", Summary = "Reports playback has started within a session")]
public class ReportPlaybackStart : PlaybackStartInfo , IReturnVoid
{
}
[Route("/Sessions/Playing/Progress", "POST", Summary = "Reports playback progress within a session")]
public class ReportPlaybackProgress : PlaybackProgressInfo , IReturnVoid
{
}
[Route("/Sessions/Playing/Ping", "POST", Summary = "Pings a playback session")]
public class PingPlaybackSession : IReturnVoid
{
[ApiMember(Name = "PlaySessionId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string PlaySessionId { get ; set ; }
}
[Route("/Sessions/Playing/Stopped", "POST", Summary = "Reports playback has stopped within a session")]
public class ReportPlaybackStopped : PlaybackStopInfo , IReturnVoid
{
}
/// <summary>
2020-06-15 22:37:52 +00:00
/// Class OnPlaybackStart.
2018-12-15 05:17:07 +00:00
/// </summary>
[Route("/Users/{UserId}/PlayingItems/{Id}", "POST", Summary = "Reports that a user has begun playing an item")]
public class OnPlaybackStart : IReturnVoid
{
/// <summary>
/// Gets or sets the user id.
/// </summary>
/// <value>The user id.</value>
[ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
public string UserId { get ; set ; }
/// <summary>
/// Gets or sets the id.
/// </summary>
/// <value>The id.</value>
[ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
public string Id { get ; set ; }
[ApiMember(Name = "MediaSourceId", Description = "The id of the MediaSource", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "POST")]
public string MediaSourceId { get ; set ; }
[ApiMember(Name = "CanSeek", Description = "Indicates if the client can seek", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "POST")]
public bool CanSeek { get ; set ; }
[ApiMember(Name = "AudioStreamIndex", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
public int? AudioStreamIndex { get ; set ; }
[ApiMember(Name = "SubtitleStreamIndex", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
public int? SubtitleStreamIndex { get ; set ; }
[ApiMember(Name = "PlayMethod", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public PlayMethod PlayMethod { get ; set ; }
[ApiMember(Name = "LiveStreamId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string LiveStreamId { get ; set ; }
[ApiMember(Name = "PlaySessionId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string PlaySessionId { get ; set ; }
}
/// <summary>
2020-06-15 22:37:52 +00:00
/// Class OnPlaybackProgress.
2018-12-15 05:17:07 +00:00
/// </summary>
[Route("/Users/{UserId}/PlayingItems/{Id}/Progress", "POST", Summary = "Reports a user's playback progress")]
public class OnPlaybackProgress : IReturnVoid
{
/// <summary>
/// Gets or sets the user id.
/// </summary>
/// <value>The user id.</value>
[ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
public string UserId { get ; set ; }
/// <summary>
/// Gets or sets the id.
/// </summary>
/// <value>The id.</value>
[ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
public string Id { get ; set ; }
[ApiMember(Name = "MediaSourceId", Description = "The id of the MediaSource", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "POST")]
public string MediaSourceId { get ; set ; }
/// <summary>
/// Gets or sets the position ticks.
/// </summary>
/// <value>The position ticks.</value>
[ApiMember(Name = "PositionTicks", Description = "Optional. The current position, in ticks. 1 tick = 10000 ms", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
public long? PositionTicks { get ; set ; }
[ApiMember(Name = "IsPaused", Description = "Indicates if the player is paused.", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "POST")]
public bool IsPaused { get ; set ; }
[ApiMember(Name = "IsMuted", Description = "Indicates if the player is muted.", IsRequired = false, DataType = "boolean", ParameterType = "query", Verb = "POST")]
public bool IsMuted { get ; set ; }
[ApiMember(Name = "AudioStreamIndex", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
public int? AudioStreamIndex { get ; set ; }
[ApiMember(Name = "SubtitleStreamIndex", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
public int? SubtitleStreamIndex { get ; set ; }
[ApiMember(Name = "VolumeLevel", Description = "Scale of 0-100", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
public int? VolumeLevel { get ; set ; }
[ApiMember(Name = "PlayMethod", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public PlayMethod PlayMethod { get ; set ; }
[ApiMember(Name = "LiveStreamId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string LiveStreamId { get ; set ; }
[ApiMember(Name = "PlaySessionId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string PlaySessionId { get ; set ; }
[ApiMember(Name = "RepeatMode", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public RepeatMode RepeatMode { get ; set ; }
}
/// <summary>
2020-06-15 22:37:52 +00:00
/// Class OnPlaybackStopped.
2018-12-15 05:17:07 +00:00
/// </summary>
[Route("/Users/{UserId}/PlayingItems/{Id}", "DELETE", Summary = "Reports that a user has stopped playing an item")]
public class OnPlaybackStopped : IReturnVoid
{
/// <summary>
/// Gets or sets the user id.
/// </summary>
/// <value>The user id.</value>
[ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
public string UserId { get ; set ; }
/// <summary>
/// Gets or sets the id.
/// </summary>
/// <value>The id.</value>
[ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
public string Id { get ; set ; }
[ApiMember(Name = "MediaSourceId", Description = "The id of the MediaSource", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "DELETE")]
public string MediaSourceId { get ; set ; }
[ApiMember(Name = "NextMediaType", Description = "The next media type that will play", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "DELETE")]
public string NextMediaType { get ; set ; }
/// <summary>
/// Gets or sets the position ticks.
/// </summary>
/// <value>The position ticks.</value>
[ApiMember(Name = "PositionTicks", Description = "Optional. The position, in ticks, where playback stopped. 1 tick = 10000 ms", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "DELETE")]
public long? PositionTicks { get ; set ; }
[ApiMember(Name = "LiveStreamId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string LiveStreamId { get ; set ; }
[ApiMember(Name = "PlaySessionId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
public string PlaySessionId { get ; set ; }
}
[Authenticated]
public class PlaystateService : BaseApiService
{
private readonly IUserManager _userManager ;
private readonly IUserDataManager _userDataRepository ;
private readonly ILibraryManager _libraryManager ;
private readonly ISessionManager _sessionManager ;
private readonly ISessionContext _sessionContext ;
private readonly IAuthorizationContext _authContext ;
2019-11-17 22:05:39 +00:00
public PlaystateService (
ILogger < PlaystateService > logger ,
IServerConfigurationManager serverConfigurationManager ,
IHttpResultFactory httpResultFactory ,
IUserManager userManager ,
IUserDataManager userDataRepository ,
ILibraryManager libraryManager ,
ISessionManager sessionManager ,
ISessionContext sessionContext ,
IAuthorizationContext authContext )
: base ( logger , serverConfigurationManager , httpResultFactory )
2018-12-15 05:17:07 +00:00
{
_userManager = userManager ;
_userDataRepository = userDataRepository ;
_libraryManager = libraryManager ;
_sessionManager = sessionManager ;
_sessionContext = sessionContext ;
_authContext = authContext ;
}
/// <summary>
/// Posts the specified request.
/// </summary>
/// <param name="request">The request.</param>
2019-02-01 20:56:50 +00:00
public object Post ( MarkPlayedItem request )
2018-12-15 05:17:07 +00:00
{
2019-02-01 20:56:50 +00:00
var result = MarkPlayed ( request ) ;
2018-12-15 05:17:07 +00:00
return ToOptimizedResult ( result ) ;
}
2019-02-01 20:56:50 +00:00
private UserItemDataDto MarkPlayed ( MarkPlayedItem request )
2018-12-15 05:17:07 +00:00
{
2019-11-17 22:05:39 +00:00
var user = _userManager . GetUserById ( Guid . Parse ( request . UserId ) ) ;
2018-12-15 05:17:07 +00:00
DateTime ? datePlayed = null ;
if ( ! string . IsNullOrEmpty ( request . DatePlayed ) )
{
datePlayed = DateTime . ParseExact ( request . DatePlayed , "yyyyMMddHHmmss" , CultureInfo . InvariantCulture , DateTimeStyles . AssumeUniversal ) ;
}
2018-12-15 05:17:53 +00:00
var session = GetSession ( _sessionContext ) ;
2018-12-15 05:17:07 +00:00
2019-01-25 22:05:01 +00:00
var dto = UpdatePlayedStatus ( user , request . Id , true , datePlayed ) ;
2018-12-15 05:17:07 +00:00
foreach ( var additionalUserInfo in session . AdditionalUsers )
{
var additionalUser = _userManager . GetUserById ( additionalUserInfo . UserId ) ;
2019-01-25 22:05:01 +00:00
UpdatePlayedStatus ( additionalUser , request . Id , true , datePlayed ) ;
2018-12-15 05:17:07 +00:00
}
return dto ;
}
private PlayMethod ValidatePlayMethod ( PlayMethod method , string playSessionId )
{
if ( method = = PlayMethod . Transcode )
{
var job = string . IsNullOrWhiteSpace ( playSessionId ) ? null : ApiEntryPoint . Instance . GetTranscodingJob ( playSessionId ) ;
if ( job = = null )
{
return PlayMethod . DirectPlay ;
}
}
return method ;
}
/// <summary>
/// Posts the specified request.
/// </summary>
/// <param name="request">The request.</param>
public void Post ( OnPlaybackStart request )
{
Post ( new ReportPlaybackStart
{
CanSeek = request . CanSeek ,
2018-12-15 05:17:53 +00:00
ItemId = new Guid ( request . Id ) ,
2018-12-15 05:17:07 +00:00
MediaSourceId = request . MediaSourceId ,
AudioStreamIndex = request . AudioStreamIndex ,
SubtitleStreamIndex = request . SubtitleStreamIndex ,
PlayMethod = request . PlayMethod ,
PlaySessionId = request . PlaySessionId ,
LiveStreamId = request . LiveStreamId
} ) ;
}
public void Post ( ReportPlaybackStart request )
{
request . PlayMethod = ValidatePlayMethod ( request . PlayMethod , request . PlaySessionId ) ;
2018-12-15 05:17:53 +00:00
request . SessionId = GetSession ( _sessionContext ) . Id ;
2018-12-15 05:17:07 +00:00
var task = _sessionManager . OnPlaybackStart ( request ) ;
Task . WaitAll ( task ) ;
}
/// <summary>
/// Posts the specified request.
/// </summary>
/// <param name="request">The request.</param>
public void Post ( OnPlaybackProgress request )
{
Post ( new ReportPlaybackProgress
{
2018-12-15 05:17:53 +00:00
ItemId = new Guid ( request . Id ) ,
2018-12-15 05:17:07 +00:00
PositionTicks = request . PositionTicks ,
IsMuted = request . IsMuted ,
IsPaused = request . IsPaused ,
MediaSourceId = request . MediaSourceId ,
AudioStreamIndex = request . AudioStreamIndex ,
SubtitleStreamIndex = request . SubtitleStreamIndex ,
VolumeLevel = request . VolumeLevel ,
PlayMethod = request . PlayMethod ,
PlaySessionId = request . PlaySessionId ,
LiveStreamId = request . LiveStreamId ,
RepeatMode = request . RepeatMode
} ) ;
}
public void Post ( ReportPlaybackProgress request )
{
request . PlayMethod = ValidatePlayMethod ( request . PlayMethod , request . PlaySessionId ) ;
2018-12-15 05:17:53 +00:00
request . SessionId = GetSession ( _sessionContext ) . Id ;
2018-12-15 05:17:07 +00:00
var task = _sessionManager . OnPlaybackProgress ( request ) ;
Task . WaitAll ( task ) ;
}
public void Post ( PingPlaybackSession request )
{
ApiEntryPoint . Instance . PingTranscodingJob ( request . PlaySessionId , null ) ;
}
/// <summary>
/// Posts the specified request.
/// </summary>
/// <param name="request">The request.</param>
2019-01-29 20:39:12 +00:00
public Task Delete ( OnPlaybackStopped request )
2018-12-15 05:17:07 +00:00
{
2019-01-29 20:39:12 +00:00
return Post ( new ReportPlaybackStopped
2018-12-15 05:17:07 +00:00
{
2018-12-15 05:17:53 +00:00
ItemId = new Guid ( request . Id ) ,
2018-12-15 05:17:07 +00:00
PositionTicks = request . PositionTicks ,
MediaSourceId = request . MediaSourceId ,
PlaySessionId = request . PlaySessionId ,
LiveStreamId = request . LiveStreamId ,
NextMediaType = request . NextMediaType
} ) ;
}
2019-01-29 20:39:12 +00:00
public async Task Post ( ReportPlaybackStopped request )
2018-12-15 05:17:07 +00:00
{
2018-12-20 10:38:42 +00:00
Logger . LogDebug ( "ReportPlaybackStopped PlaySessionId: {0}" , request . PlaySessionId ? ? string . Empty ) ;
2018-12-15 05:17:07 +00:00
if ( ! string . IsNullOrWhiteSpace ( request . PlaySessionId ) )
{
2019-01-29 20:39:12 +00:00
await ApiEntryPoint . Instance . KillTranscodingJobs ( _authContext . GetAuthorizationInfo ( Request ) . DeviceId , request . PlaySessionId , s = > true ) ;
2018-12-15 05:17:07 +00:00
}
2018-12-15 05:17:53 +00:00
request . SessionId = GetSession ( _sessionContext ) . Id ;
2018-12-15 05:17:07 +00:00
2019-01-29 20:39:12 +00:00
await _sessionManager . OnPlaybackStopped ( request ) ;
2018-12-15 05:17:07 +00:00
}
/// <summary>
/// Deletes the specified request.
/// </summary>
/// <param name="request">The request.</param>
public object Delete ( MarkUnplayedItem request )
{
var task = MarkUnplayed ( request ) ;
2019-02-01 20:56:50 +00:00
return ToOptimizedResult ( task ) ;
2018-12-15 05:17:07 +00:00
}
2019-02-01 20:56:50 +00:00
private UserItemDataDto MarkUnplayed ( MarkUnplayedItem request )
2018-12-15 05:17:07 +00:00
{
2019-11-17 22:05:39 +00:00
var user = _userManager . GetUserById ( Guid . Parse ( request . UserId ) ) ;
2018-12-15 05:17:07 +00:00
2018-12-15 05:17:53 +00:00
var session = GetSession ( _sessionContext ) ;
2018-12-15 05:17:07 +00:00
2019-01-25 22:05:01 +00:00
var dto = UpdatePlayedStatus ( user , request . Id , false , null ) ;
2018-12-15 05:17:07 +00:00
foreach ( var additionalUserInfo in session . AdditionalUsers )
{
var additionalUser = _userManager . GetUserById ( additionalUserInfo . UserId ) ;
2019-01-25 22:05:01 +00:00
UpdatePlayedStatus ( additionalUser , request . Id , false , null ) ;
2018-12-15 05:17:07 +00:00
}
return dto ;
}
/// <summary>
/// Updates the played status.
/// </summary>
/// <param name="user">The user.</param>
/// <param name="itemId">The item id.</param>
/// <param name="wasPlayed">if set to <c>true</c> [was played].</param>
/// <param name="datePlayed">The date played.</param>
/// <returns>Task.</returns>
2020-05-20 17:07:53 +00:00
private UserItemDataDto UpdatePlayedStatus ( User user , string itemId , bool wasPlayed , DateTime ? datePlayed )
2018-12-15 05:17:07 +00:00
{
var item = _libraryManager . GetItemById ( itemId ) ;
if ( wasPlayed )
{
2018-12-15 05:17:53 +00:00
item . MarkPlayed ( user , datePlayed , true ) ;
2018-12-15 05:17:07 +00:00
}
else
{
2018-12-15 05:17:53 +00:00
item . MarkUnplayed ( user ) ;
2018-12-15 05:17:07 +00:00
}
return _userDataRepository . GetUserDataDto ( item , user ) ;
}
}
2018-12-15 05:17:53 +00:00
}