@ -1,5 +1,8 @@
#pragma warning disable CS1591
using System ;
using System ;
using System.Collections.Generic ;
using System.Collections.Generic ;
using System.Diagnostics ;
using System.Globalization ;
using System.Globalization ;
using System.IO ;
using System.IO ;
using System.Linq ;
using System.Linq ;
@ -19,9 +22,8 @@ using MediaBrowser.Model.Globalization;
using MediaBrowser.Model.IO ;
using MediaBrowser.Model.IO ;
using MediaBrowser.Model.MediaInfo ;
using MediaBrowser.Model.MediaInfo ;
using MediaBrowser.Model.System ;
using MediaBrowser.Model.System ;
using Microsoft.Extensions.Logging ;
using System.Diagnostics ;
using Microsoft.Extensions.Configuration ;
using Microsoft.Extensions.Configuration ;
using Microsoft.Extensions.Logging ;
namespace MediaBrowser.MediaEncoding.Encoder
namespace MediaBrowser.MediaEncoding.Encoder
{
{
@ -35,6 +37,11 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// </summary>
/// </summary>
internal const int DefaultImageExtractionTimeout = 5000 ;
internal const int DefaultImageExtractionTimeout = 5000 ;
/// <summary>
/// The us culture.
/// </summary>
private readonly CultureInfo _usCulture = new CultureInfo ( "en-US" ) ;
private readonly ILogger < MediaEncoder > _logger ;
private readonly ILogger < MediaEncoder > _logger ;
private readonly IServerConfigurationManager _configurationManager ;
private readonly IServerConfigurationManager _configurationManager ;
private readonly IFileSystem _fileSystem ;
private readonly IFileSystem _fileSystem ;
@ -47,6 +54,10 @@ namespace MediaBrowser.MediaEncoding.Encoder
private readonly object _runningProcessesLock = new object ( ) ;
private readonly object _runningProcessesLock = new object ( ) ;
private readonly List < ProcessWrapper > _runningProcesses = new List < ProcessWrapper > ( ) ;
private readonly List < ProcessWrapper > _runningProcesses = new List < ProcessWrapper > ( ) ;
private List < string > _encoders = new List < string > ( ) ;
private List < string > _decoders = new List < string > ( ) ;
private List < string > _hwaccels = new List < string > ( ) ;
private string _ffmpegPath = string . Empty ;
private string _ffmpegPath = string . Empty ;
private string _ffprobePath ;
private string _ffprobePath ;
@ -77,7 +88,7 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// <summary>
/// <summary>
/// Run at startup or if the user removes a Custom path from transcode page.
/// Run at startup or if the user removes a Custom path from transcode page.
/// Sets global variables FFmpegPath.
/// Sets global variables FFmpegPath.
/// Precedence is: Config > CLI > $PATH
/// Precedence is: Config > CLI > $PATH .
/// </summary>
/// </summary>
public void SetFFmpegPath ( )
public void SetFFmpegPath ( )
{
{
@ -122,8 +133,8 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// Triggered from the Settings > Transcoding UI page when users submits Custom FFmpeg path to use.
/// Triggered from the Settings > Transcoding UI page when users submits Custom FFmpeg path to use.
/// Only write the new path to xml if it exists. Do not perform validation checks on ffmpeg here.
/// Only write the new path to xml if it exists. Do not perform validation checks on ffmpeg here.
/// </summary>
/// </summary>
/// <param name="path"> </param>
/// <param name="path"> The path. </param>
/// <param name="pathType"> </param>
/// <param name="pathType"> The path type. </param>
public void UpdateEncoderPath ( string path , string pathType )
public void UpdateEncoderPath ( string path , string pathType )
{
{
string newPath ;
string newPath ;
@ -168,8 +179,8 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// If checks pass, global variable FFmpegPath and EncoderLocation are updated.
/// If checks pass, global variable FFmpegPath and EncoderLocation are updated.
/// </summary>
/// </summary>
/// <param name="path">FQPN to test.</param>
/// <param name="path">FQPN to test.</param>
/// <param name="location">Location (External, Custom, System) of tool </param>
/// <param name="location">Location (External, Custom, System) of tool . </param>
/// <returns>< /returns>
/// <returns>< c>true</c> if the version validation succeeded; otherwise, <c>false</c>.< /returns>
private bool ValidatePath ( string path , FFmpegLocation location )
private bool ValidatePath ( string path , FFmpegLocation location )
{
{
bool rc = false ;
bool rc = false ;
@ -221,8 +232,8 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// <summary>
/// <summary>
/// Search the system $PATH environment variable looking for given filename.
/// Search the system $PATH environment variable looking for given filename.
/// </summary>
/// </summary>
/// <param name="fileName"> </param>
/// <param name="fileName"> The filename. </param>
/// <returns> </returns>
/// <returns> The full path to the file. </returns>
private string ExistsOnSystemPath ( string fileName )
private string ExistsOnSystemPath ( string fileName )
{
{
var inJellyfinPath = GetEncoderPathFromDirectory ( AppContext . BaseDirectory , fileName , recursive : true ) ;
var inJellyfinPath = GetEncoderPathFromDirectory ( AppContext . BaseDirectory , fileName , recursive : true ) ;
@ -246,25 +257,19 @@ namespace MediaBrowser.MediaEncoding.Encoder
return null ;
return null ;
}
}
private List < string > _encoders = new List < string > ( ) ;
public void SetAvailableEncoders ( IEnumerable < string > list )
public void SetAvailableEncoders ( IEnumerable < string > list )
{
{
_encoders = list . ToList ( ) ;
_encoders = list . ToList ( ) ;
// _logger.Info("Supported encoders: {0}", string.Join(",", list.ToArray()));
}
}
private List < string > _decoders = new List < string > ( ) ;
public void SetAvailableDecoders ( IEnumerable < string > list )
public void SetAvailableDecoders ( IEnumerable < string > list )
{
{
_decoders = list . ToList ( ) ;
_decoders = list . ToList ( ) ;
// _logger.Info("Supported decoders: {0}", string.Join(",", list.ToArray()));
}
}
private List < string > _hwaccels = new List < string > ( ) ;
public void SetAvailableHwaccels ( IEnumerable < string > list )
public void SetAvailableHwaccels ( IEnumerable < string > list )
{
{
_hwaccels = list . ToList ( ) ;
_hwaccels = list . ToList ( ) ;
//_logger.Info("Supported hwaccels: {0}", string.Join(",", list.ToArray()));
}
}
public bool SupportsEncoder ( string encoder )
public bool SupportsEncoder ( string encoder )
@ -332,8 +337,16 @@ namespace MediaBrowser.MediaEncoding.Encoder
var forceEnableLogging = request . MediaSource . Protocol ! = MediaProtocol . File ;
var forceEnableLogging = request . MediaSource . Protocol ! = MediaProtocol . File ;
return GetMediaInfoInternal ( GetInputArgument ( inputFiles , request . MediaSource . Protocol ) , request . MediaSource . Path , request . MediaSource . Protocol , extractChapters ,
return GetMediaInfoInternal (
probeSize , request . MediaType = = DlnaProfileType . Audio , request . MediaSource . VideoType , forceEnableLogging , cancellationToken ) ;
GetInputArgument ( inputFiles , request . MediaSource . Protocol ) ,
request . MediaSource . Path ,
request . MediaSource . Protocol ,
extractChapters ,
probeSize ,
request . MediaType = = DlnaProfileType . Audio ,
request . MediaSource . VideoType ,
forceEnableLogging ,
cancellationToken ) ;
}
}
/// <summary>
/// <summary>
@ -342,7 +355,7 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// <param name="inputFiles">The input files.</param>
/// <param name="inputFiles">The input files.</param>
/// <param name="protocol">The protocol.</param>
/// <param name="protocol">The protocol.</param>
/// <returns>System.String.</returns>
/// <returns>System.String.</returns>
/// <exception cref="ArgumentException">Unrecognized InputType </exception>
/// <exception cref="ArgumentException">Unrecognized InputType . </exception>
public string GetInputArgument ( IReadOnlyList < string > inputFiles , MediaProtocol protocol )
public string GetInputArgument ( IReadOnlyList < string > inputFiles , MediaProtocol protocol )
= > EncodingUtils . GetInputArgument ( inputFiles , protocol ) ;
= > EncodingUtils . GetInputArgument ( inputFiles , protocol ) ;
@ -350,7 +363,8 @@ namespace MediaBrowser.MediaEncoding.Encoder
/// Gets the media info internal.
/// Gets the media info internal.
/// </summary>
/// </summary>
/// <returns>Task{MediaInfoResult}.</returns>
/// <returns>Task{MediaInfoResult}.</returns>
private async Task < MediaInfo > GetMediaInfoInternal ( string inputPath ,
private async Task < MediaInfo > GetMediaInfoInternal (
string inputPath ,
string primaryPath ,
string primaryPath ,
MediaProtocol protocol ,
MediaProtocol protocol ,
bool extractChapters ,
bool extractChapters ,
@ -378,7 +392,6 @@ namespace MediaBrowser.MediaEncoding.Encoder
FileName = _ffprobePath ,
FileName = _ffprobePath ,
Arguments = args ,
Arguments = args ,
WindowStyle = ProcessWindowStyle . Hidden ,
WindowStyle = ProcessWindowStyle . Hidden ,
ErrorDialog = false ,
ErrorDialog = false ,
} ,
} ,
@ -439,11 +452,6 @@ namespace MediaBrowser.MediaEncoding.Encoder
}
}
}
}
/// <summary>
/// The us culture.
/// </summary>
protected readonly CultureInfo UsCulture = new CultureInfo ( "en-US" ) ;
public Task < string > ExtractAudioImage ( string path , int? imageStreamIndex , CancellationToken cancellationToken )
public Task < string > ExtractAudioImage ( string path , int? imageStreamIndex , CancellationToken cancellationToken )
{
{
return ExtractImage ( new [ ] { path } , null , null , imageStreamIndex , MediaProtocol . File , true , null , null , cancellationToken ) ;
return ExtractImage ( new [ ] { path } , null , null , imageStreamIndex , MediaProtocol . File , true , null , null , cancellationToken ) ;
@ -459,8 +467,16 @@ namespace MediaBrowser.MediaEncoding.Encoder
return ExtractImage ( inputFiles , container , imageStream , imageStreamIndex , protocol , false , null , null , cancellationToken ) ;
return ExtractImage ( inputFiles , container , imageStream , imageStreamIndex , protocol , false , null , null , cancellationToken ) ;
}
}
private async Task < string > ExtractImage ( string [ ] inputFiles , string container , MediaStream videoStream , int? imageStreamIndex , MediaProtocol protocol , bool isAudio ,
private async Task < string > ExtractImage (
Video3DFormat ? threedFormat , TimeSpan ? offset , CancellationToken cancellationToken )
string [ ] inputFiles ,
string container ,
MediaStream videoStream ,
int? imageStreamIndex ,
MediaProtocol protocol ,
bool isAudio ,
Video3DFormat ? threedFormat ,
TimeSpan ? offset ,
CancellationToken cancellationToken )
{
{
var inputArgument = GetInputArgument ( inputFiles , protocol ) ;
var inputArgument = GetInputArgument ( inputFiles , protocol ) ;
@ -645,7 +661,7 @@ namespace MediaBrowser.MediaEncoding.Encoder
public string GetTimeParameter ( TimeSpan time )
public string GetTimeParameter ( TimeSpan time )
{
{
return time . ToString ( @"hh\:mm\:ss\.fff" , U sCulture) ;
return time . ToString ( @"hh\:mm\:ss\.fff" , _u sCulture) ;
}
}
public async Task ExtractVideoImagesOnInterval (
public async Task ExtractVideoImagesOnInterval (
@ -662,11 +678,11 @@ namespace MediaBrowser.MediaEncoding.Encoder
{
{
var inputArgument = GetInputArgument ( inputFiles , protocol ) ;
var inputArgument = GetInputArgument ( inputFiles , protocol ) ;
var vf = "fps=fps=1/" + interval . TotalSeconds . ToString ( U sCulture) ;
var vf = "fps=fps=1/" + interval . TotalSeconds . ToString ( _u sCulture) ;
if ( maxWidth . HasValue )
if ( maxWidth . HasValue )
{
{
var maxWidthParam = maxWidth . Value . ToString ( U sCulture) ;
var maxWidthParam = maxWidth . Value . ToString ( _u sCulture) ;
vf + = string . Format ( ",scale=min(iw\\,{0}):trunc(ow/dar/2)*2" , maxWidthParam ) ;
vf + = string . Format ( ",scale=min(iw\\,{0}):trunc(ow/dar/2)*2" , maxWidthParam ) ;
}
}
@ -859,6 +875,7 @@ namespace MediaBrowser.MediaEncoding.Encoder
if ( dispose )
if ( dispose )
{
{
StopProcesses ( ) ;
StopProcesses ( ) ;
_thumbnailResourcePool . Dispose ( ) ;
}
}
}
}