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

134 lines
5.2 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using MediaBrowser.Common.Configuration;
using MediaBrowser.Controller;
2015-05-18 15:23:03 -07:00
using MediaBrowser.Controller.Entities;
2015-03-28 13:22:27 -07:00
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
2015-04-04 12:35:29 -07:00
using MediaBrowser.Controller.MediaEncoding;
2015-03-28 13:22:27 -07:00
using MediaBrowser.Model.Dto;
2015-05-18 15:23:03 -07:00
using MediaBrowser.Model.MediaInfo;
2015-03-28 21:56:39 -07:00
using MediaBrowser.Model.Serialization;
using Microsoft.Extensions.Logging;
2015-03-28 13:22:27 -07:00
2016-11-03 16:35:19 -07:00
namespace Emby.Server.Implementations.LiveTv
2015-03-28 13:22:27 -07:00
{
public class LiveTvMediaSourceProvider : IMediaSourceProvider
{
private readonly ILiveTvManager _liveTvManager;
2015-03-28 21:56:39 -07:00
private readonly IJsonSerializer _jsonSerializer;
private readonly ILogger _logger;
2015-03-31 09:24:16 -07:00
private readonly IMediaSourceManager _mediaSourceManager;
2015-04-04 12:35:29 -07:00
private readonly IMediaEncoder _mediaEncoder;
2015-05-18 15:23:03 -07:00
private readonly IServerApplicationHost _appHost;
2018-09-12 10:26:21 -07:00
private IApplicationPaths _appPaths;
2015-03-28 13:22:27 -07:00
public LiveTvMediaSourceProvider(ILiveTvManager liveTvManager, IApplicationPaths appPaths, IJsonSerializer jsonSerializer, ILoggerFactory loggerFactory, IMediaSourceManager mediaSourceManager, IMediaEncoder mediaEncoder, IServerApplicationHost appHost)
2015-03-28 13:22:27 -07:00
{
_liveTvManager = liveTvManager;
2015-03-28 21:56:39 -07:00
_jsonSerializer = jsonSerializer;
2015-03-31 09:24:16 -07:00
_mediaSourceManager = mediaSourceManager;
2015-04-04 12:35:29 -07:00
_mediaEncoder = mediaEncoder;
2015-05-18 15:23:03 -07:00
_appHost = appHost;
_logger = loggerFactory.CreateLogger(GetType().Name);
2018-09-12 10:26:21 -07:00
_appPaths = appPaths;
2015-03-28 13:22:27 -07:00
}
2018-09-12 10:26:21 -07:00
public Task<IEnumerable<MediaSourceInfo>> GetMediaSources(BaseItem item, CancellationToken cancellationToken)
2015-03-28 13:22:27 -07:00
{
2016-03-19 14:17:08 -07:00
var baseItem = (BaseItem)item;
2015-03-28 13:22:27 -07:00
2016-03-19 14:17:08 -07:00
if (baseItem.SourceType == SourceType.LiveTV)
2015-03-28 13:22:27 -07:00
{
2017-08-23 12:45:52 -07:00
var activeRecordingInfo = _liveTvManager.GetActiveRecordingInfo(item.Path);
2018-09-12 10:26:21 -07:00
if (string.IsNullOrEmpty(baseItem.Path) || activeRecordingInfo != null)
2015-03-28 13:22:27 -07:00
{
2017-08-23 12:45:52 -07:00
return GetMediaSourcesInternal(item, activeRecordingInfo, cancellationToken);
2015-03-28 13:22:27 -07:00
}
}
2018-09-12 10:26:21 -07:00
return Task.FromResult<IEnumerable<MediaSourceInfo>>(Array.Empty<MediaSourceInfo>());
2015-03-28 13:22:27 -07:00
}
2015-07-18 11:07:03 -07:00
// Do not use a pipe here because Roku http requests to the server will fail, without any explicit error message.
private const char StreamIdDelimeter = '_';
2015-07-23 16:40:54 -07:00
private const string StreamIdDelimeterString = "_";
2016-03-19 14:17:08 -07:00
2018-09-12 10:26:21 -07:00
private async Task<IEnumerable<MediaSourceInfo>> GetMediaSourcesInternal(BaseItem item, ActiveRecordingInfo activeRecordingInfo, CancellationToken cancellationToken)
2015-03-28 13:22:27 -07:00
{
2015-03-28 21:56:39 -07:00
IEnumerable<MediaSourceInfo> sources;
2015-03-28 13:22:27 -07:00
2015-08-02 12:08:55 -07:00
var forceRequireOpening = false;
2015-03-28 21:56:39 -07:00
try
{
2018-09-12 10:26:21 -07:00
if (activeRecordingInfo != null)
2015-03-28 21:56:39 -07:00
{
2018-09-12 10:26:21 -07:00
sources = await EmbyTV.EmbyTV.Current.GetRecordingStreamMediaSources(activeRecordingInfo, cancellationToken)
2017-08-23 12:45:52 -07:00
.ConfigureAwait(false);
2015-03-28 21:56:39 -07:00
}
else
{
2018-09-12 10:26:21 -07:00
sources = await _liveTvManager.GetChannelMediaSources(item, cancellationToken)
.ConfigureAwait(false);
2015-03-28 21:56:39 -07:00
}
}
catch (NotImplementedException)
{
2018-09-12 10:26:21 -07:00
sources = _mediaSourceManager.GetStaticMediaSources(item, false);
2015-08-02 12:08:55 -07:00
forceRequireOpening = true;
2015-03-28 21:56:39 -07:00
}
2015-03-28 13:22:27 -07:00
2015-03-28 21:56:39 -07:00
var list = sources.ToList();
2017-11-23 08:46:16 -07:00
var serverUrl = await _appHost.GetLocalApiUrl(cancellationToken).ConfigureAwait(false);
2015-03-28 21:56:39 -07:00
foreach (var source in list)
2015-03-28 13:22:27 -07:00
{
source.Type = MediaSourceType.Default;
2015-08-24 05:54:10 -07:00
source.BufferMs = source.BufferMs ?? 1500;
2015-03-28 13:22:27 -07:00
2015-08-02 12:08:55 -07:00
if (source.RequiresOpening || forceRequireOpening)
2015-07-23 16:40:54 -07:00
{
source.RequiresOpening = true;
}
2015-08-02 12:08:55 -07:00
if (source.RequiresOpening)
2015-07-23 16:40:54 -07:00
{
var openKeys = new List<string>();
openKeys.Add(item.GetType().Name);
openKeys.Add(item.Id.ToString("N", CultureInfo.InvariantCulture));
2015-07-23 16:40:54 -07:00
openKeys.Add(source.Id ?? string.Empty);
2018-12-28 08:48:26 -07:00
source.OpenToken = string.Join(StreamIdDelimeterString, openKeys.ToArray());
}
2015-05-18 15:23:03 -07:00
// Dummy this up so that direct play checks can still run
if (string.IsNullOrEmpty(source.Path) && source.Protocol == MediaProtocol.Http)
{
source.Path = serverUrl;
}
2015-03-28 13:22:27 -07:00
}
_logger.LogDebug("MediaSources: {0}", _jsonSerializer.SerializeToString(list));
2015-03-28 21:56:39 -07:00
return list;
2015-03-28 13:22:27 -07:00
}
2018-09-12 10:26:21 -07:00
public async Task<ILiveStream> OpenMediaSource(string openToken, List<ILiveStream> currentLiveStreams, CancellationToken cancellationToken)
2015-03-28 13:22:27 -07:00
{
2015-07-18 11:07:03 -07:00
var keys = openToken.Split(new[] { StreamIdDelimeter }, 3);
2015-08-24 05:54:10 -07:00
var mediaSourceId = keys.Length >= 3 ? keys[2] : null;
2015-04-04 12:35:29 -07:00
2018-09-12 10:26:21 -07:00
var info = await _liveTvManager.GetChannelStream(keys[1], mediaSourceId, currentLiveStreams, cancellationToken).ConfigureAwait(false);
var liveStream = info.Item2;
2015-04-19 12:17:17 -07:00
2018-09-12 10:26:21 -07:00
return liveStream;
2015-03-28 13:22:27 -07:00
}
}
}