jellyfin/src/Jellyfin.LiveTv/TunerHosts/M3uParser.cs

327 lines
12 KiB
C#
Raw Normal View History

#nullable disable
#pragma warning disable CS1591
using System;
2016-02-19 07:20:18 +01:00
using System.Collections.Generic;
2016-12-07 21:03:00 +01:00
using System.Globalization;
2016-02-19 07:20:18 +01:00
using System.IO;
using System.Net.Http;
2016-02-21 18:22:13 +01:00
using System.Text.RegularExpressions;
2016-02-19 07:20:18 +01:00
using System.Threading;
using System.Threading.Tasks;
using Jellyfin.Extensions;
2016-02-19 07:20:18 +01:00
using MediaBrowser.Common.Extensions;
using MediaBrowser.Common.Net;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Model.IO;
using MediaBrowser.Model.LiveTv;
using Microsoft.Extensions.Logging;
2016-02-19 07:20:18 +01:00
2023-12-28 21:15:03 +01:00
namespace Jellyfin.LiveTv.TunerHosts
2016-02-19 07:20:18 +01:00
{
2023-05-22 22:48:09 +02:00
public partial class M3uParser
2016-02-19 07:20:18 +01:00
{
2021-05-28 14:33:54 +02:00
private const string ExtInfPrefix = "#EXTINF:";
2016-02-19 07:20:18 +01:00
private readonly ILogger _logger;
private readonly IHttpClientFactory _httpClientFactory;
2016-02-19 07:20:18 +01:00
2021-05-28 14:33:54 +02:00
public M3uParser(ILogger logger, IHttpClientFactory httpClientFactory)
2016-02-19 07:20:18 +01:00
{
_logger = logger;
_httpClientFactory = httpClientFactory;
2016-02-19 07:20:18 +01:00
}
2023-05-22 22:48:09 +02:00
[GeneratedRegex(@"([a-z0-9\-_]+)=\""([^""]+)\""", RegexOptions.IgnoreCase, "en-US")]
private static partial Regex KeyValueRegex();
public async Task<List<ChannelInfo>> Parse(TunerHostInfo info, string channelIdPrefix, CancellationToken cancellationToken)
2016-02-19 07:20:18 +01:00
{
// Read the file and display it line by line.
using (var reader = new StreamReader(await GetListingsStream(info, cancellationToken).ConfigureAwait(false)))
2016-02-19 07:20:18 +01:00
{
2021-01-08 23:57:27 +01:00
return await GetChannelsAsync(reader, channelIdPrefix, info.Id).ConfigureAwait(false);
2017-01-14 04:46:02 +01:00
}
}
public async Task<Stream> GetListingsStream(TunerHostInfo info, CancellationToken cancellationToken)
2016-02-19 07:20:18 +01:00
{
ArgumentNullException.ThrowIfNull(info);
2021-08-05 02:19:03 +02:00
if (!info.Url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
{
return AsyncFile.OpenRead(info.Url);
2021-08-05 02:19:03 +02:00
}
2021-08-05 02:19:03 +02:00
using var requestMessage = new HttpRequestMessage(HttpMethod.Get, info.Url);
if (!string.IsNullOrEmpty(info.UserAgent))
{
requestMessage.Headers.UserAgent.TryParseAdd(info.UserAgent);
2016-02-19 07:20:18 +01:00
}
2019-08-09 23:16:24 +02:00
2021-08-06 17:07:50 +02:00
// Set HttpCompletionOption.ResponseHeadersRead to prevent timeouts on larger files
2021-08-05 02:19:03 +02:00
var response = await _httpClientFactory.CreateClient(NamedClient.Default)
.SendAsync(requestMessage, HttpCompletionOption.ResponseHeadersRead, cancellationToken)
.ConfigureAwait(false);
response.EnsureSuccessStatusCode();
2023-12-05 18:26:29 +01:00
return await response.Content.ReadAsStreamAsync(cancellationToken).ConfigureAwait(false);
2016-02-19 07:20:18 +01:00
}
2021-01-08 23:57:27 +01:00
private async Task<List<ChannelInfo>> GetChannelsAsync(TextReader reader, string channelIdPrefix, string tunerHostId)
2016-02-19 07:20:18 +01:00
{
2017-08-20 21:10:00 +02:00
var channels = new List<ChannelInfo>();
string extInf = string.Empty;
2017-02-01 21:56:41 +01:00
2021-01-08 23:57:27 +01:00
await foreach (var line in reader.ReadAllLinesAsync().ConfigureAwait(false))
2016-02-19 07:20:18 +01:00
{
2021-01-08 23:57:27 +01:00
var trimmedLine = line.Trim();
if (string.IsNullOrWhiteSpace(trimmedLine))
2016-02-19 07:20:18 +01:00
{
continue;
}
2021-01-08 23:57:27 +01:00
if (trimmedLine.StartsWith("#EXTM3U", StringComparison.OrdinalIgnoreCase))
2016-02-19 07:20:18 +01:00
{
continue;
}
2021-01-08 23:57:27 +01:00
if (trimmedLine.StartsWith(ExtInfPrefix, StringComparison.OrdinalIgnoreCase))
2016-02-19 07:20:18 +01:00
{
2021-01-08 23:57:27 +01:00
extInf = trimmedLine.Substring(ExtInfPrefix.Length).Trim();
2016-02-19 07:20:18 +01:00
}
2021-01-08 23:57:27 +01:00
else if (!string.IsNullOrWhiteSpace(extInf) && !trimmedLine.StartsWith('#'))
2016-02-24 20:06:26 +01:00
{
2021-01-08 23:57:27 +01:00
var channel = GetChannelnfo(extInf, tunerHostId, trimmedLine);
channel.Id = channelIdPrefix + trimmedLine.GetMD5().ToString("N", CultureInfo.InvariantCulture);
2017-01-23 22:51:23 +01:00
2021-01-08 23:57:27 +01:00
channel.Path = trimmedLine;
channels.Add(channel);
_logger.LogInformation("Parsed channel: {ChannelName}", channel.Name);
extInf = string.Empty;
2016-02-19 07:20:18 +01:00
}
}
2017-02-01 21:56:41 +01:00
2016-02-19 07:20:18 +01:00
return channels;
}
2016-11-27 21:52:24 +01:00
2017-08-20 21:10:00 +02:00
private ChannelInfo GetChannelnfo(string extInf, string tunerHostId, string mediaUrl)
{
var channel = new ChannelInfo()
{
TunerHostId = tunerHostId
};
2016-11-27 21:52:24 +01:00
extInf = extInf.Trim();
var attributes = ParseExtInf(extInf, out string remaining);
2016-12-07 21:03:00 +01:00
extInf = remaining;
2016-11-27 21:52:24 +01:00
if (attributes.TryGetValue("tvg-logo", out string tvgLogo))
2016-12-07 21:03:00 +01:00
{
channel.ImageUrl = tvgLogo;
}
else if (attributes.TryGetValue("logo", out string logo))
{
channel.ImageUrl = logo;
2016-12-07 21:03:00 +01:00
}
2016-11-27 21:52:24 +01:00
2021-03-20 20:15:19 +01:00
if (attributes.TryGetValue("group-title", out string groupTitle))
{
channel.ChannelGroup = groupTitle;
}
2016-12-07 21:03:00 +01:00
channel.Name = GetChannelName(extInf, attributes);
channel.Number = GetChannelNumber(extInf, attributes, mediaUrl);
2016-11-27 21:52:24 +01:00
attributes.TryGetValue("tvg-id", out string tvgId);
2017-02-23 20:13:26 +01:00
attributes.TryGetValue("channel-id", out string channelId);
2017-02-23 20:13:26 +01:00
channel.TunerChannelId = string.IsNullOrWhiteSpace(tvgId) ? channelId : tvgId;
var channelIdValues = new List<string>();
2017-02-05 00:32:16 +01:00
if (!string.IsNullOrWhiteSpace(channelId))
2017-01-23 22:51:23 +01:00
{
2017-02-23 20:13:26 +01:00
channelIdValues.Add(channelId);
}
2017-02-23 20:13:26 +01:00
if (!string.IsNullOrWhiteSpace(tvgId))
{
channelIdValues.Add(tvgId);
}
2017-02-23 20:13:26 +01:00
if (channelIdValues.Count > 0)
{
2021-02-13 00:39:18 +01:00
channel.Id = string.Join('_', channelIdValues);
2017-01-23 22:51:23 +01:00
}
2016-11-27 21:52:24 +01:00
return channel;
}
2016-12-07 21:03:00 +01:00
private string GetChannelNumber(string extInf, Dictionary<string, string> attributes, string mediaUrl)
2016-11-27 21:52:24 +01:00
{
var nameParts = extInf.Split(',', StringSplitOptions.RemoveEmptyEntries);
var nameInExtInf = nameParts.Length > 1 ? nameParts[^1].AsSpan().Trim() : ReadOnlySpan<char>.Empty;
2016-11-27 21:52:24 +01:00
2017-01-14 04:46:02 +01:00
string numberString = null;
2016-11-27 21:52:24 +01:00
2023-04-06 19:21:29 +02:00
if (attributes.TryGetValue("tvg-chno", out var attributeValue)
&& double.TryParse(attributeValue, CultureInfo.InvariantCulture, out _))
{
numberString = attributeValue;
}
2017-02-05 00:32:16 +01:00
if (!IsValidChannelNumber(numberString))
{
if (attributes.TryGetValue("tvg-id", out attributeValue))
2016-12-07 21:03:00 +01:00
{
if (double.TryParse(attributeValue, CultureInfo.InvariantCulture, out _))
2017-01-14 04:46:02 +01:00
{
numberString = attributeValue;
}
else if (attributes.TryGetValue("channel-id", out attributeValue)
&& double.TryParse(attributeValue, CultureInfo.InvariantCulture, out _))
2017-01-14 04:46:02 +01:00
{
numberString = attributeValue;
2017-01-14 04:46:02 +01:00
}
2016-12-07 21:03:00 +01:00
}
if (string.IsNullOrWhiteSpace(numberString))
2016-12-07 21:03:00 +01:00
{
// Using this as a fallback now as this leads to Problems with channels like "5 USA"
// where 5 isn't meant to be the channel number
// Check for channel number with the format from SatIp
// #EXTINF:0,84. VOX Schweiz
// #EXTINF:0,84.0 - VOX Schweiz
if (!nameInExtInf.IsEmpty && !nameInExtInf.IsWhiteSpace())
{
var numberIndex = nameInExtInf.IndexOf(' ');
if (numberIndex > 0)
{
var numberPart = nameInExtInf.Slice(0, numberIndex).Trim(new[] { ' ', '.' });
if (double.TryParse(numberPart, CultureInfo.InvariantCulture, out _))
{
numberString = numberPart.ToString();
}
}
}
2016-12-07 21:03:00 +01:00
}
}
2017-02-05 00:32:16 +01:00
if (!IsValidChannelNumber(numberString))
2016-11-27 21:52:24 +01:00
{
numberString = null;
}
2016-12-07 21:03:00 +01:00
if (!string.IsNullOrWhiteSpace(numberString))
{
numberString = numberString.Trim();
}
else
{
2016-11-27 21:52:24 +01:00
if (string.IsNullOrWhiteSpace(mediaUrl))
{
numberString = null;
}
else
{
2017-06-06 08:13:49 +02:00
try
{
2021-09-19 20:53:31 +02:00
numberString = Path.GetFileNameWithoutExtension(mediaUrl.AsSpan().RightPart('/')).ToString();
2016-12-07 21:03:00 +01:00
2017-06-06 08:13:49 +02:00
if (!IsValidChannelNumber(numberString))
{
numberString = null;
}
}
catch
2016-12-07 21:03:00 +01:00
{
2017-06-06 08:13:49 +02:00
// Seeing occasional argument exception here
2016-12-07 21:03:00 +01:00
numberString = null;
}
2016-11-27 21:52:24 +01:00
}
}
2016-11-27 21:52:24 +01:00
return numberString;
}
private static bool IsValidChannelNumber(string numberString)
2017-02-05 00:32:16 +01:00
{
if (string.IsNullOrWhiteSpace(numberString)
|| string.Equals(numberString, "-1", StringComparison.Ordinal)
|| string.Equals(numberString, "0", StringComparison.Ordinal))
2017-02-05 00:32:16 +01:00
{
return false;
}
return double.TryParse(numberString, CultureInfo.InvariantCulture, out _);
2017-02-05 00:32:16 +01:00
}
private static string GetChannelName(string extInf, Dictionary<string, string> attributes)
2016-11-27 21:52:24 +01:00
{
var nameParts = extInf.Split(',', StringSplitOptions.RemoveEmptyEntries);
var nameInExtInf = nameParts.Length > 1 ? nameParts[^1].Trim() : null;
2016-11-27 21:52:24 +01:00
2017-01-14 05:31:43 +01:00
// Check for channel number with the format from SatIp
// #EXTINF:0,84. VOX Schweiz
// #EXTINF:0,84.0 - VOX Schweiz
2016-11-27 21:52:24 +01:00
if (!string.IsNullOrWhiteSpace(nameInExtInf))
{
2021-11-15 15:57:07 +01:00
var numberIndex = nameInExtInf.IndexOf(' ', StringComparison.Ordinal);
2016-11-27 21:52:24 +01:00
if (numberIndex > 0)
{
var numberPart = nameInExtInf.AsSpan(0, numberIndex).Trim(new[] { ' ', '.' });
2017-01-14 05:31:43 +01:00
if (double.TryParse(numberPart, CultureInfo.InvariantCulture, out _))
2016-11-27 21:52:24 +01:00
{
2020-06-14 11:11:11 +02:00
// channel.Number = number.ToString();
nameInExtInf = nameInExtInf.AsSpan(numberIndex + 1).Trim(new[] { ' ', '-' }).ToString();
2016-11-27 21:52:24 +01:00
}
}
}
string name = nameInExtInf;
2016-12-07 21:03:00 +01:00
2016-11-27 21:52:24 +01:00
if (string.IsNullOrWhiteSpace(name))
{
attributes.TryGetValue("tvg-name", out name);
}
2016-11-27 21:52:24 +01:00
if (string.IsNullOrWhiteSpace(name))
{
2019-01-19 22:04:09 +01:00
attributes.TryGetValue("tvg-id", out name);
}
2016-11-27 21:52:24 +01:00
if (string.IsNullOrWhiteSpace(name))
{
name = null;
}
2016-11-27 21:52:24 +01:00
return name;
}
2016-11-27 21:52:24 +01:00
private static Dictionary<string, string> ParseExtInf(string line, out string remaining)
2016-02-21 18:22:13 +01:00
{
2016-12-07 21:03:00 +01:00
var dict = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
2023-05-22 22:48:09 +02:00
var matches = KeyValueRegex().Matches(line);
2017-01-14 20:57:08 +01:00
remaining = line;
2016-02-21 18:22:13 +01:00
foreach (Match match in matches)
{
2017-01-14 20:57:08 +01:00
var key = match.Groups[1].Value;
var value = match.Groups[2].Value;
2016-12-07 21:03:00 +01:00
2023-05-22 22:48:09 +02:00
dict[key] = value;
2017-01-14 20:57:08 +01:00
remaining = remaining.Replace(key + "=\"" + value + "\"", string.Empty, StringComparison.OrdinalIgnoreCase);
2016-12-07 21:03:00 +01:00
}
return dict;
2016-02-21 18:22:13 +01:00
}
2016-02-19 07:20:18 +01:00
}
}