using System; using System.Diagnostics; using System.Diagnostics.CodeAnalysis; using System.Threading; using MediaBrowser.Controller.MediaEncoding; using MediaBrowser.Model.Dto; using Microsoft.Extensions.Logging; namespace Jellyfin.Api.Models.PlaybackDtos; /// /// Class TranscodingJob. /// public class TranscodingJobDto : IDisposable { /// /// The process lock. /// [SuppressMessage("Microsoft.Performance", "CA1051:NoVisibleInstanceFields", MessageId = "ProcessLock", Justification = "Imported from ServiceStack")] [SuppressMessage("Microsoft.Performance", "SA1401:PrivateField", MessageId = "ProcessLock", Justification = "Imported from ServiceStack")] public readonly object ProcessLock = new object(); /// /// Timer lock. /// private readonly object _timerLock = new object(); /// /// Initializes a new instance of the class. /// /// Instance of the interface. public TranscodingJobDto(ILogger logger) { Logger = logger; } /// /// Gets or sets the play session identifier. /// /// The play session identifier. public string? PlaySessionId { get; set; } /// /// Gets or sets the live stream identifier. /// /// The live stream identifier. public string? LiveStreamId { get; set; } /// /// Gets or sets a value indicating whether is live output. /// public bool IsLiveOutput { get; set; } /// /// Gets or sets the path. /// /// The path. public MediaSourceInfo? MediaSource { get; set; } /// /// Gets or sets path. /// public string? Path { get; set; } /// /// Gets or sets the type. /// /// The type. public TranscodingJobType Type { get; set; } /// /// Gets or sets the process. /// /// The process. public Process? Process { get; set; } /// /// Gets logger. /// public ILogger Logger { get; private set; } /// /// Gets or sets the active request count. /// /// The active request count. public int ActiveRequestCount { get; set; } /// /// Gets or sets the kill timer. /// /// The kill timer. private Timer? KillTimer { get; set; } /// /// Gets or sets device id. /// public string? DeviceId { get; set; } /// /// Gets or sets cancellation token source. /// public CancellationTokenSource? CancellationTokenSource { get; set; } /// /// Gets or sets a value indicating whether has exited. /// public bool HasExited { get; set; } /// /// Gets or sets exit code. /// public int ExitCode { get; set; } /// /// Gets or sets a value indicating whether is user paused. /// public bool IsUserPaused { get; set; } /// /// Gets or sets id. /// public string? Id { get; set; } /// /// Gets or sets framerate. /// public float? Framerate { get; set; } /// /// Gets or sets completion percentage. /// public double? CompletionPercentage { get; set; } /// /// Gets or sets bytes downloaded. /// public long BytesDownloaded { get; set; } /// /// Gets or sets bytes transcoded. /// public long? BytesTranscoded { get; set; } /// /// Gets or sets bit rate. /// public int? BitRate { get; set; } /// /// Gets or sets transcoding position ticks. /// public long? TranscodingPositionTicks { get; set; } /// /// Gets or sets download position ticks. /// public long? DownloadPositionTicks { get; set; } /// /// Gets or sets transcoding throttler. /// public TranscodingThrottler? TranscodingThrottler { get; set; } /// /// Gets or sets last ping date. /// public DateTime LastPingDate { get; set; } /// /// Gets or sets ping timeout. /// public int PingTimeout { get; set; } /// /// Stop kill timer. /// public void StopKillTimer() { lock (_timerLock) { KillTimer?.Change(Timeout.Infinite, Timeout.Infinite); } } /// /// Dispose kill timer. /// public void DisposeKillTimer() { lock (_timerLock) { if (KillTimer is not null) { KillTimer.Dispose(); KillTimer = null; } } } /// /// Start kill timer. /// /// Callback action. public void StartKillTimer(Action callback) { StartKillTimer(callback, PingTimeout); } /// /// Start kill timer. /// /// Callback action. /// Callback interval. public void StartKillTimer(Action callback, int intervalMs) { if (HasExited) { return; } lock (_timerLock) { if (KillTimer is null) { Logger.LogDebug("Starting kill timer at {0}ms. JobId {1} PlaySessionId {2}", intervalMs, Id, PlaySessionId); KillTimer = new Timer(new TimerCallback(callback), this, intervalMs, Timeout.Infinite); } else { Logger.LogDebug("Changing kill timer to {0}ms. JobId {1} PlaySessionId {2}", intervalMs, Id, PlaySessionId); KillTimer.Change(intervalMs, Timeout.Infinite); } } } /// /// Change kill timer if started. /// public void ChangeKillTimerIfStarted() { if (HasExited) { return; } lock (_timerLock) { if (KillTimer is not null) { var intervalMs = PingTimeout; Logger.LogDebug("Changing kill timer to {0}ms. JobId {1} PlaySessionId {2}", intervalMs, Id, PlaySessionId); KillTimer.Change(intervalMs, Timeout.Infinite); } } } /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } /// /// Dispose all resources. /// /// Whether to dispose all resources. protected virtual void Dispose(bool disposing) { if (disposing) { Process?.Dispose(); Process = null; KillTimer?.Dispose(); KillTimer = null; CancellationTokenSource?.Dispose(); CancellationTokenSource = null; TranscodingThrottler?.Dispose(); TranscodingThrottler = null; } } }