jellyfin-server/MediaBrowser.Api/HttpHandlers/VideoHandler.cs

238 lines
7.6 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using MediaBrowser.Model.Entities;
namespace MediaBrowser.Api.HttpHandlers
{
/// <summary>
/// Supported output formats: mkv,m4v,mp4,asf,wmv,mov,webm,ogv,3gp,avi,ts,flv
/// </summary>
class VideoHandler : BaseMediaHandler<Video>
{
/// <summary>
/// We can output these files directly, but we can't encode them
/// </summary>
protected override IEnumerable<string> UnsupportedOutputEncodingFormats
{
get
{
2012-08-14 02:24:35 +00:00
// mp4, 3gp, mov - muxer does not support non-seekable output
// avi, mov, mkv, m4v - can't stream these when encoding. the player will try to download them completely before starting playback.
// wmv - can't seem to figure out the output format name
return new string[] { "mp4", "3gp", "m4v", "mkv", "avi", "mov", "wmv" };
}
}
protected override bool RequiresConversion()
{
2012-08-13 02:14:39 +00:00
string currentFormat = Path.GetExtension(LibraryItem.Path).Replace(".", string.Empty);
// For now we won't allow these to pass through.
// Later we'll add some intelligence to allow it when possible
2012-08-13 02:22:34 +00:00
if (currentFormat.Equals("mp4", StringComparison.OrdinalIgnoreCase) || currentFormat.Equals("mkv", StringComparison.OrdinalIgnoreCase) || currentFormat.Equals("m4v", StringComparison.OrdinalIgnoreCase))
2012-08-13 02:14:39 +00:00
{
return true;
}
if (base.RequiresConversion())
{
return true;
}
AudioStream audio = LibraryItem.AudioStreams.FirstOrDefault();
if (audio != null)
{
// If the number of channels is greater than our desired channels, we need to transcode
if (AudioChannels.HasValue && AudioChannels.Value < audio.Channels)
{
return true;
}
}
// Yay
return false;
}
2012-08-13 02:14:39 +00:00
/// <summary>
/// Translates the file extension to the format param that follows "-f" on the ffmpeg command line
/// </summary>
private string GetFFMpegOutputFormat(string outputFormat)
{
if (outputFormat.Equals("mkv", StringComparison.OrdinalIgnoreCase))
{
return "matroska";
}
else if (outputFormat.Equals("ts", StringComparison.OrdinalIgnoreCase))
{
return "mpegts";
}
else if (outputFormat.Equals("ogv", StringComparison.OrdinalIgnoreCase))
{
return "ogg";
}
return outputFormat;
}
/// <summary>
/// Creates arguments to pass to ffmpeg
/// </summary>
protected override string GetCommandLineArguments()
{
List<string> audioTranscodeParams = new List<string>();
string outputFormat = GetConversionOutputFormat();
2012-08-13 02:14:39 +00:00
return string.Format("-i \"{0}\" -threads 0 {1} {2} -f {3} -",
LibraryItem.Path,
2012-08-13 02:14:39 +00:00
GetVideoArguments(outputFormat),
GetAudioArguments(outputFormat),
GetFFMpegOutputFormat(outputFormat)
);
}
2012-08-13 02:14:39 +00:00
private string GetVideoArguments(string outputFormat)
{
string codec = GetVideoCodec(outputFormat);
string args = "-vcodec " + codec;
return args;
}
private string GetAudioArguments(string outputFormat)
{
AudioStream audioStream = LibraryItem.AudioStreams.FirstOrDefault();
if (audioStream == null)
{
return string.Empty;
}
string codec = GetAudioCodec(audioStream, outputFormat);
2012-08-13 02:14:39 +00:00
string args = "-acodec " + codec;
if (!codec.Equals("copy", StringComparison.OrdinalIgnoreCase))
{
int? channels = GetNumAudioChannelsParam(codec, audioStream.Channels);
2012-08-13 02:14:39 +00:00
if (channels.HasValue)
{
args += " -ac " + channels.Value;
}
int? sampleRate = GetSampleRateParam(audioStream.SampleRate);
if (sampleRate.HasValue)
2012-08-13 02:14:39 +00:00
{
args += " -ar " + sampleRate.Value;
2012-08-13 02:14:39 +00:00
}
}
return args;
}
private string GetVideoCodec(string outputFormat)
{
if (outputFormat.Equals("webm"))
{
// Per webm specification, it must be vpx
return "libvpx";
}
else if (outputFormat.Equals("asf"))
{
return "wmv2";
}
2012-08-14 02:24:35 +00:00
else if (outputFormat.Equals("wmv"))
{
return "wmv2";
}
else if (outputFormat.Equals("ogv"))
{
return "libtheora";
}
2012-08-13 02:14:39 +00:00
return "libx264";
2012-08-13 02:14:39 +00:00
}
private string GetAudioCodec(AudioStream audioStream, string outputFormat)
{
2012-08-13 02:14:39 +00:00
if (outputFormat.Equals("webm"))
{
// Per webm specification, it must be vorbis
return "libvorbis";
}
2012-08-14 02:24:35 +00:00
else if (outputFormat.Equals("asf"))
{
return "wmav2";
}
else if (outputFormat.Equals("wmv"))
{
return "wmav2";
}
else if (outputFormat.Equals("ogv"))
{
return "libvorbis";
}
// See if we can just copy the stream
if (HasBasicAudio(audioStream))
2012-08-13 02:14:39 +00:00
{
return "copy";
2012-08-13 02:14:39 +00:00
}
return "libvo_aacenc";
}
private int? GetNumAudioChannelsParam(string audioCodec, int libraryItemChannels)
{
2012-08-14 02:24:35 +00:00
if (libraryItemChannels > 2)
2012-08-13 02:14:39 +00:00
{
2012-08-14 02:24:35 +00:00
if (audioCodec.Equals("libvo_aacenc"))
{
// libvo_aacenc currently only supports two channel output
return 2;
}
else if (audioCodec.Equals("wmav2"))
{
// wmav2 currently only supports two channel output
return 2;
}
2012-08-13 02:14:39 +00:00
}
return GetNumAudioChannelsParam(libraryItemChannels);
}
private bool HasBasicAudio(AudioStream audio)
{
2012-08-14 02:24:35 +00:00
if (AudioChannels.HasValue)
{
2012-08-14 02:24:35 +00:00
if (audio.Channels > AudioChannels.Value)
{
return false;
}
}
if (audio.AudioFormat.IndexOf("aac", StringComparison.OrdinalIgnoreCase) != -1)
{
return true;
}
if (audio.AudioFormat.IndexOf("ac-3", StringComparison.OrdinalIgnoreCase) != -1 || audio.AudioFormat.IndexOf("ac3", StringComparison.OrdinalIgnoreCase) != -1)
{
return true;
}
if (audio.AudioFormat.IndexOf("mpeg", StringComparison.OrdinalIgnoreCase) != -1 || audio.AudioFormat.IndexOf("mp3", StringComparison.OrdinalIgnoreCase) != -1)
{
return true;
}
return false;
}
}
}