jellyfin/MediaBrowser.Server.Implementations/LiveTv/LiveTvMediaSourceProvider.cs

209 lines
7.4 KiB
C#
Raw Normal View History

2015-03-28 21:22:27 +01:00
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
2015-04-04 21:35:29 +02:00
using MediaBrowser.Controller.MediaEncoding;
2015-04-05 17:01:57 +02:00
using MediaBrowser.Model.Dlna;
2015-03-28 21:22:27 +01:00
using MediaBrowser.Model.Dto;
2015-03-29 06:56:39 +02:00
using MediaBrowser.Model.Logging;
using MediaBrowser.Model.Serialization;
2015-03-28 21:22:27 +01:00
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
namespace MediaBrowser.Server.Implementations.LiveTv
{
public class LiveTvMediaSourceProvider : IMediaSourceProvider
{
private readonly ILiveTvManager _liveTvManager;
2015-03-29 06:56:39 +02:00
private readonly IJsonSerializer _jsonSerializer;
private readonly ILogger _logger;
2015-03-31 18:24:16 +02:00
private readonly IMediaSourceManager _mediaSourceManager;
2015-04-04 21:35:29 +02:00
private readonly IMediaEncoder _mediaEncoder;
2015-03-28 21:22:27 +01:00
2015-04-04 21:35:29 +02:00
public LiveTvMediaSourceProvider(ILiveTvManager liveTvManager, IJsonSerializer jsonSerializer, ILogManager logManager, IMediaSourceManager mediaSourceManager, IMediaEncoder mediaEncoder)
2015-03-28 21:22:27 +01:00
{
_liveTvManager = liveTvManager;
2015-03-29 06:56:39 +02:00
_jsonSerializer = jsonSerializer;
2015-03-31 18:24:16 +02:00
_mediaSourceManager = mediaSourceManager;
2015-04-04 21:35:29 +02:00
_mediaEncoder = mediaEncoder;
2015-03-29 06:56:39 +02:00
_logger = logManager.GetLogger(GetType().Name);
2015-03-28 21:22:27 +01:00
}
public Task<IEnumerable<MediaSourceInfo>> GetMediaSources(IHasMediaSources item, CancellationToken cancellationToken)
{
var channelItem = item as ILiveTvItem;
if (channelItem != null)
{
var hasMetadata = (IHasMetadata)channelItem;
if (string.IsNullOrWhiteSpace(hasMetadata.Path))
{
return GetMediaSourcesInternal(channelItem, cancellationToken);
}
}
return Task.FromResult<IEnumerable<MediaSourceInfo>>(new List<MediaSourceInfo>());
}
private async Task<IEnumerable<MediaSourceInfo>> GetMediaSourcesInternal(ILiveTvItem item, CancellationToken cancellationToken)
{
2015-03-29 06:56:39 +02:00
IEnumerable<MediaSourceInfo> sources;
2015-03-28 21:22:27 +01:00
2015-03-29 06:56:39 +02:00
try
{
if (item is ILiveTvRecording)
{
sources = await _liveTvManager.GetRecordingMediaSources(item.Id.ToString("N"), cancellationToken)
.ConfigureAwait(false);
}
else
{
sources = await _liveTvManager.GetChannelMediaSources(item.Id.ToString("N"), cancellationToken)
.ConfigureAwait(false);
}
}
catch (NotImplementedException)
{
var hasMediaSources = (IHasMediaSources)item;
2015-03-31 18:24:16 +02:00
sources = _mediaSourceManager.GetStaticMediaSources(hasMediaSources, false)
2015-03-29 06:56:39 +02:00
.ToList();
}
2015-03-28 21:22:27 +01:00
2015-03-29 06:56:39 +02:00
var list = sources.ToList();
foreach (var source in list)
2015-03-28 21:22:27 +01:00
{
source.Type = MediaSourceType.Default;
source.RequiresOpening = true;
2015-03-29 06:56:39 +02:00
source.BufferMs = source.BufferMs ?? 1500;
2015-03-28 21:22:27 +01:00
var openKeys = new List<string>();
openKeys.Add(item.GetType().Name);
openKeys.Add(item.Id.ToString("N"));
2015-05-16 04:36:47 +02:00
openKeys.Add(source.Id ?? string.Empty);
2015-03-29 06:56:39 +02:00
source.OpenToken = string.Join("|", openKeys.ToArray());
2015-03-28 21:22:27 +01:00
}
2015-03-29 06:56:39 +02:00
_logger.Debug("MediaSources: {0}", _jsonSerializer.SerializeToString(list));
return list;
2015-03-28 21:22:27 +01:00
}
2015-03-29 06:56:39 +02:00
public async Task<MediaSourceInfo> OpenMediaSource(string openToken, CancellationToken cancellationToken)
2015-03-28 21:22:27 +01:00
{
2015-04-04 21:35:29 +02:00
MediaSourceInfo stream;
2015-05-16 04:36:47 +02:00
const bool isAudio = false;
2015-04-04 21:35:29 +02:00
2015-05-16 04:36:47 +02:00
var keys = openToken.Split(new[] { '|' }, 3);
var mediaSourceId = keys.Length >= 3 ? keys[2] : null;
2015-03-28 21:22:27 +01:00
if (string.Equals(keys[0], typeof(LiveTvChannel).Name, StringComparison.OrdinalIgnoreCase))
{
2015-05-16 04:36:47 +02:00
stream = await _liveTvManager.GetChannelStream(keys[1], mediaSourceId, cancellationToken).ConfigureAwait(false);
2015-04-04 21:35:29 +02:00
}
else
{
stream = await _liveTvManager.GetRecordingStream(keys[1], cancellationToken).ConfigureAwait(false);
}
try
{
await AddMediaInfo(stream, isAudio, cancellationToken).ConfigureAwait(false);
}
catch (Exception ex)
{
_logger.ErrorException("Error probing live tv stream", ex);
}
return stream;
}
private async Task AddMediaInfo(MediaSourceInfo mediaSource, bool isAudio, CancellationToken cancellationToken)
{
2015-04-19 21:17:17 +02:00
var originalRuntime = mediaSource.RunTimeTicks;
2015-04-04 21:35:29 +02:00
2015-04-05 17:01:57 +02:00
var info = await _mediaEncoder.GetMediaInfo(new MediaInfoRequest
{
InputPath = mediaSource.Path,
Protocol = mediaSource.Protocol,
MediaType = isAudio ? DlnaProfileType.Audio : DlnaProfileType.Video,
ExtractChapters = false
2015-04-19 21:17:17 +02:00
}, cancellationToken).ConfigureAwait(false);
2015-04-04 21:35:29 +02:00
mediaSource.Bitrate = info.Bitrate;
mediaSource.Container = info.Container;
mediaSource.Formats = info.Formats;
mediaSource.MediaStreams = info.MediaStreams;
mediaSource.RunTimeTicks = info.RunTimeTicks;
mediaSource.Size = info.Size;
mediaSource.Timestamp = info.Timestamp;
mediaSource.Video3DFormat = info.Video3DFormat;
mediaSource.VideoType = info.VideoType;
mediaSource.DefaultSubtitleStreamIndex = null;
2015-04-19 21:17:17 +02:00
// Null this out so that it will be treated like a live stream
if (!originalRuntime.HasValue)
{
mediaSource.RunTimeTicks = null;
}
2015-04-04 21:35:29 +02:00
var audioStream = mediaSource.MediaStreams.FirstOrDefault(i => i.Type == Model.Entities.MediaStreamType.Audio);
if (audioStream == null || audioStream.Index == -1)
{
mediaSource.DefaultAudioStreamIndex = null;
}
else
{
mediaSource.DefaultAudioStreamIndex = audioStream.Index;
2015-03-28 21:22:27 +01:00
}
2015-05-17 05:17:23 +02:00
var videoStream = mediaSource.MediaStreams.FirstOrDefault(i => i.Type == Model.Entities.MediaStreamType.Video);
if (videoStream != null)
2015-04-04 21:35:29 +02:00
{
2015-05-17 05:17:23 +02:00
if (!videoStream.BitRate.HasValue)
2015-04-04 21:35:29 +02:00
{
var width = videoStream.Width ?? 1920;
if (width >= 1900)
{
2015-05-17 05:17:23 +02:00
videoStream.BitRate = 8000000;
2015-04-04 21:35:29 +02:00
}
else if (width >= 1260)
{
2015-05-17 05:17:23 +02:00
videoStream.BitRate = 3000000;
2015-04-04 21:35:29 +02:00
}
else if (width >= 700)
{
2015-05-17 05:17:23 +02:00
videoStream.BitRate = 1000000;
2015-04-04 21:35:29 +02:00
}
}
}
2015-05-17 05:17:23 +02:00
// Try to estimate this
if (!mediaSource.Bitrate.HasValue)
{
var total = mediaSource.MediaStreams.Select(i => i.BitRate ?? 0).Sum();
if (total > 0)
{
mediaSource.Bitrate = total;
}
}
2015-03-28 21:22:27 +01:00
}
2015-03-29 06:56:39 +02:00
public Task CloseMediaSource(string liveStreamId, CancellationToken cancellationToken)
2015-03-28 21:22:27 +01:00
{
2015-03-29 06:56:39 +02:00
return _liveTvManager.CloseLiveStream(liveStreamId, cancellationToken);
2015-03-28 21:22:27 +01:00
}
}
}