| | 1 | | #nullable disable |
| | 2 | |
|
| | 3 | | #pragma warning disable CS1591 |
| | 4 | |
|
| | 5 | | using System; |
| | 6 | | using System.Collections.Generic; |
| | 7 | | using System.Globalization; |
| | 8 | | using System.IO; |
| | 9 | | using System.Linq; |
| | 10 | | using System.Net; |
| | 11 | | using System.Net.Http; |
| | 12 | | using System.Threading; |
| | 13 | | using System.Threading.Tasks; |
| | 14 | | using Jellyfin.Extensions; |
| | 15 | | using MediaBrowser.Common.Extensions; |
| | 16 | | using MediaBrowser.Common.Net; |
| | 17 | | using MediaBrowser.Controller; |
| | 18 | | using MediaBrowser.Controller.Configuration; |
| | 19 | | using MediaBrowser.Controller.Library; |
| | 20 | | using MediaBrowser.Controller.LiveTv; |
| | 21 | | using MediaBrowser.Model.Dto; |
| | 22 | | using MediaBrowser.Model.Entities; |
| | 23 | | using MediaBrowser.Model.IO; |
| | 24 | | using MediaBrowser.Model.LiveTv; |
| | 25 | | using MediaBrowser.Model.MediaInfo; |
| | 26 | | using Microsoft.Extensions.Logging; |
| | 27 | | using Microsoft.Net.Http.Headers; |
| | 28 | |
|
| | 29 | | namespace Jellyfin.LiveTv.TunerHosts |
| | 30 | | { |
| | 31 | | public class M3UTunerHost : BaseTunerHost, ITunerHost, IConfigurableTunerHost |
| | 32 | | { |
| 0 | 33 | | private static readonly string[] _mimeTypesCanShareHttpStream = ["video/MP2T"]; |
| 0 | 34 | | private static readonly string[] _extensionsCanShareHttpStream = [".ts", ".tsv", ".m2t"]; |
| | 35 | |
|
| | 36 | | private readonly IHttpClientFactory _httpClientFactory; |
| | 37 | | private readonly IServerApplicationHost _appHost; |
| | 38 | | private readonly INetworkManager _networkManager; |
| | 39 | | private readonly IMediaSourceManager _mediaSourceManager; |
| | 40 | | private readonly IStreamHelper _streamHelper; |
| | 41 | |
|
| | 42 | | public M3UTunerHost( |
| | 43 | | IServerConfigurationManager config, |
| | 44 | | IMediaSourceManager mediaSourceManager, |
| | 45 | | ILogger<M3UTunerHost> logger, |
| | 46 | | IFileSystem fileSystem, |
| | 47 | | IHttpClientFactory httpClientFactory, |
| | 48 | | IServerApplicationHost appHost, |
| | 49 | | INetworkManager networkManager, |
| | 50 | | IStreamHelper streamHelper) |
| 21 | 51 | | : base(config, logger, fileSystem) |
| | 52 | | { |
| 21 | 53 | | _httpClientFactory = httpClientFactory; |
| 21 | 54 | | _appHost = appHost; |
| 21 | 55 | | _networkManager = networkManager; |
| 21 | 56 | | _mediaSourceManager = mediaSourceManager; |
| 21 | 57 | | _streamHelper = streamHelper; |
| 21 | 58 | | } |
| | 59 | |
|
| 4 | 60 | | public override string Type => "m3u"; |
| | 61 | |
|
| 0 | 62 | | public virtual string Name => "M3U Tuner"; |
| | 63 | |
|
| | 64 | | private string GetFullChannelIdPrefix(TunerHostInfo info) |
| | 65 | | { |
| 0 | 66 | | return ChannelIdPrefix + info.Url.GetMD5().ToString("N", CultureInfo.InvariantCulture); |
| | 67 | | } |
| | 68 | |
|
| | 69 | | protected override async Task<List<ChannelInfo>> GetChannelsInternal(TunerHostInfo tuner, CancellationToken canc |
| | 70 | | { |
| | 71 | | var channelIdPrefix = GetFullChannelIdPrefix(tuner); |
| | 72 | |
|
| | 73 | | return await new M3uParser(Logger, _httpClientFactory) |
| | 74 | | .Parse(tuner, channelIdPrefix, cancellationToken) |
| | 75 | | .ConfigureAwait(false); |
| | 76 | | } |
| | 77 | |
|
| | 78 | | protected override async Task<ILiveStream> GetChannelStream(TunerHostInfo tunerHost, ChannelInfo channel, string |
| | 79 | | { |
| | 80 | | var tunerCount = tunerHost.TunerCount; |
| | 81 | |
|
| | 82 | | if (tunerCount > 0) |
| | 83 | | { |
| | 84 | | var tunerHostId = tunerHost.Id; |
| | 85 | | var liveStreams = currentLiveStreams.Where(i => string.Equals(i.TunerHostId, tunerHostId, StringComparis |
| | 86 | |
|
| | 87 | | if (liveStreams.Count() >= tunerCount) |
| | 88 | | { |
| | 89 | | throw new LiveTvConflictException("M3U simultaneous stream limit has been reached."); |
| | 90 | | } |
| | 91 | | } |
| | 92 | |
|
| | 93 | | var sources = await GetChannelStreamMediaSources(tunerHost, channel, cancellationToken).ConfigureAwait(false |
| | 94 | |
|
| | 95 | | var mediaSource = sources[0]; |
| | 96 | |
|
| | 97 | | if (tunerHost.AllowStreamSharing && mediaSource.Protocol == MediaProtocol.Http && !mediaSource.RequiresLoopi |
| | 98 | | { |
| | 99 | | var extension = Path.GetExtension(new UriBuilder(mediaSource.Path).Path); |
| | 100 | |
|
| | 101 | | if (string.IsNullOrEmpty(extension)) |
| | 102 | | { |
| | 103 | | try |
| | 104 | | { |
| | 105 | | using var message = new HttpRequestMessage(HttpMethod.Head, mediaSource.Path); |
| | 106 | | using var response = await _httpClientFactory.CreateClient(NamedClient.Default) |
| | 107 | | .SendAsync(message, cancellationToken) |
| | 108 | | .ConfigureAwait(false); |
| | 109 | |
|
| | 110 | | if (response.IsSuccessStatusCode) |
| | 111 | | { |
| | 112 | | if (_mimeTypesCanShareHttpStream.Contains(response.Content.Headers.ContentType?.MediaType, S |
| | 113 | | { |
| | 114 | | return new SharedHttpStream(mediaSource, tunerHost, streamId, FileSystem, _httpClientFac |
| | 115 | | } |
| | 116 | | } |
| | 117 | | } |
| | 118 | | catch (Exception) |
| | 119 | | { |
| | 120 | | Logger.LogWarning("HEAD request to check MIME type failed, shared stream disabled"); |
| | 121 | | } |
| | 122 | | } |
| | 123 | | else if (_extensionsCanShareHttpStream.Contains(extension, StringComparison.OrdinalIgnoreCase)) |
| | 124 | | { |
| | 125 | | return new SharedHttpStream(mediaSource, tunerHost, streamId, FileSystem, _httpClientFactory, Logger |
| | 126 | | } |
| | 127 | | } |
| | 128 | |
|
| | 129 | | return new LiveStream(mediaSource, tunerHost, FileSystem, Logger, Config, _streamHelper); |
| | 130 | | } |
| | 131 | |
|
| | 132 | | public async Task Validate(TunerHostInfo info) |
| | 133 | | { |
| | 134 | | using (await new M3uParser(Logger, _httpClientFactory).GetListingsStream(info, CancellationToken.None).Confi |
| | 135 | | { |
| | 136 | | } |
| | 137 | | } |
| | 138 | |
|
| | 139 | | protected override Task<List<MediaSourceInfo>> GetChannelStreamMediaSources(TunerHostInfo tuner, ChannelInfo cha |
| | 140 | | { |
| 0 | 141 | | return Task.FromResult(new List<MediaSourceInfo> { CreateMediaSourceInfo(tuner, channel) }); |
| | 142 | | } |
| | 143 | |
|
| | 144 | | protected virtual MediaSourceInfo CreateMediaSourceInfo(TunerHostInfo info, ChannelInfo channel) |
| | 145 | | { |
| 0 | 146 | | var path = channel.Path; |
| | 147 | |
|
| 0 | 148 | | var supportsDirectPlay = !info.EnableStreamLooping && info.TunerCount == 0; |
| 0 | 149 | | var supportsDirectStream = !info.EnableStreamLooping; |
| | 150 | |
|
| 0 | 151 | | var protocol = _mediaSourceManager.GetPathProtocol(path); |
| | 152 | |
|
| 0 | 153 | | var isRemote = true; |
| 0 | 154 | | if (Uri.TryCreate(path, UriKind.Absolute, out var uri)) |
| | 155 | | { |
| 0 | 156 | | isRemote = !_networkManager.IsInLocalNetwork(uri.Host); |
| | 157 | | } |
| | 158 | |
|
| 0 | 159 | | var httpHeaders = new Dictionary<string, string>(); |
| | 160 | |
|
| 0 | 161 | | if (protocol == MediaProtocol.Http) |
| | 162 | | { |
| | 163 | | // Use user-defined user-agent. If there isn't one, make it look like a browser. |
| 0 | 164 | | httpHeaders[HeaderNames.UserAgent] = string.IsNullOrWhiteSpace(info.UserAgent) ? |
| 0 | 165 | | "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 S |
| 0 | 166 | | info.UserAgent; |
| | 167 | | } |
| | 168 | |
|
| 0 | 169 | | var mediaSource = new MediaSourceInfo |
| 0 | 170 | | { |
| 0 | 171 | | Path = path, |
| 0 | 172 | | Protocol = protocol, |
| 0 | 173 | | MediaStreams = new MediaStream[] |
| 0 | 174 | | { |
| 0 | 175 | | new MediaStream |
| 0 | 176 | | { |
| 0 | 177 | | Type = MediaStreamType.Video, |
| 0 | 178 | | // Set the index to -1 because we don't know the exact index of the video stream within the cont |
| 0 | 179 | | Index = -1, |
| 0 | 180 | | IsInterlaced = true |
| 0 | 181 | | }, |
| 0 | 182 | | new MediaStream |
| 0 | 183 | | { |
| 0 | 184 | | Type = MediaStreamType.Audio, |
| 0 | 185 | | // Set the index to -1 because we don't know the exact index of the audio stream within the cont |
| 0 | 186 | | Index = -1 |
| 0 | 187 | | } |
| 0 | 188 | | }, |
| 0 | 189 | | RequiresOpening = true, |
| 0 | 190 | | RequiresClosing = true, |
| 0 | 191 | | RequiresLooping = info.EnableStreamLooping, |
| 0 | 192 | |
|
| 0 | 193 | | ReadAtNativeFramerate = false, |
| 0 | 194 | |
|
| 0 | 195 | | Id = channel.Path.GetMD5().ToString("N", CultureInfo.InvariantCulture), |
| 0 | 196 | | IsInfiniteStream = true, |
| 0 | 197 | | IsRemote = isRemote, |
| 0 | 198 | |
|
| 0 | 199 | | IgnoreDts = info.IgnoreDts, |
| 0 | 200 | | SupportsDirectPlay = supportsDirectPlay, |
| 0 | 201 | | SupportsDirectStream = supportsDirectStream, |
| 0 | 202 | |
|
| 0 | 203 | | RequiredHttpHeaders = httpHeaders, |
| 0 | 204 | | UseMostCompatibleTranscodingProfile = !info.AllowFmp4TranscodingContainer, |
| 0 | 205 | | FallbackMaxStreamingBitrate = info.FallbackMaxStreamingBitrate |
| 0 | 206 | | }; |
| | 207 | |
|
| 0 | 208 | | mediaSource.InferTotalBitrate(); |
| | 209 | |
|
| 0 | 210 | | return mediaSource; |
| | 211 | | } |
| | 212 | |
|
| | 213 | | public Task<List<TunerHostInfo>> DiscoverDevices(int discoveryDurationMs, CancellationToken cancellationToken) |
| | 214 | | { |
| 1 | 215 | | return Task.FromResult(new List<TunerHostInfo>()); |
| | 216 | | } |
| | 217 | | } |
| | 218 | | } |