123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333 |
- #nullable disable
- #pragma warning disable CS1591
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- using System.Net.Http;
- using System.Text.RegularExpressions;
- using System.Threading;
- using System.Threading.Tasks;
- using Jellyfin.Extensions;
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Common.Net;
- using MediaBrowser.Controller;
- using MediaBrowser.Controller.LiveTv;
- using MediaBrowser.Model.LiveTv;
- using Microsoft.Extensions.Logging;
- namespace Emby.Server.Implementations.LiveTv.TunerHosts
- {
- public class M3uParser
- {
- private const string ExtInfPrefix = "#EXTINF:";
- private readonly ILogger _logger;
- private readonly IHttpClientFactory _httpClientFactory;
- public M3uParser(ILogger logger, IHttpClientFactory httpClientFactory)
- {
- _logger = logger;
- _httpClientFactory = httpClientFactory;
- }
- public async Task<List<ChannelInfo>> Parse(TunerHostInfo info, string channelIdPrefix, CancellationToken cancellationToken)
- {
- // Read the file and display it line by line.
- using (var reader = new StreamReader(await GetListingsStream(info, cancellationToken).ConfigureAwait(false)))
- {
- return await GetChannelsAsync(reader, channelIdPrefix, info.Id).ConfigureAwait(false);
- }
- }
- public async Task<Stream> GetListingsStream(TunerHostInfo info, CancellationToken cancellationToken)
- {
- if (info.Url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
- {
- using var requestMessage = new HttpRequestMessage(HttpMethod.Get, info.Url);
- if (!string.IsNullOrEmpty(info.UserAgent))
- {
- requestMessage.Headers.UserAgent.TryParseAdd(info.UserAgent);
- }
- var response = await _httpClientFactory.CreateClient(NamedClient.Default)
- .SendAsync(requestMessage, cancellationToken)
- .ConfigureAwait(false);
- response.EnsureSuccessStatusCode();
- return await response.Content.ReadAsStreamAsync(cancellationToken).ConfigureAwait(false);
- }
- return File.OpenRead(info.Url);
- }
- private async Task<List<ChannelInfo>> GetChannelsAsync(TextReader reader, string channelIdPrefix, string tunerHostId)
- {
- var channels = new List<ChannelInfo>();
- string extInf = string.Empty;
- await foreach (var line in reader.ReadAllLinesAsync().ConfigureAwait(false))
- {
- var trimmedLine = line.Trim();
- if (string.IsNullOrWhiteSpace(trimmedLine))
- {
- continue;
- }
- if (trimmedLine.StartsWith("#EXTM3U", StringComparison.OrdinalIgnoreCase))
- {
- continue;
- }
- if (trimmedLine.StartsWith(ExtInfPrefix, StringComparison.OrdinalIgnoreCase))
- {
- extInf = trimmedLine.Substring(ExtInfPrefix.Length).Trim();
- _logger.LogInformation("Found m3u channel: {0}", extInf);
- }
- else if (!string.IsNullOrWhiteSpace(extInf) && !trimmedLine.StartsWith('#'))
- {
- var channel = GetChannelnfo(extInf, tunerHostId, trimmedLine);
- if (string.IsNullOrWhiteSpace(channel.Id))
- {
- channel.Id = channelIdPrefix + trimmedLine.GetMD5().ToString("N", CultureInfo.InvariantCulture);
- }
- else
- {
- channel.Id = channelIdPrefix + channel.Id.GetMD5().ToString("N", CultureInfo.InvariantCulture);
- }
- channel.Path = trimmedLine;
- channels.Add(channel);
- extInf = string.Empty;
- }
- }
- return channels;
- }
- private ChannelInfo GetChannelnfo(string extInf, string tunerHostId, string mediaUrl)
- {
- var channel = new ChannelInfo()
- {
- TunerHostId = tunerHostId
- };
- extInf = extInf.Trim();
- var attributes = ParseExtInf(extInf, out string remaining);
- extInf = remaining;
- if (attributes.TryGetValue("tvg-logo", out string value))
- {
- channel.ImageUrl = value;
- }
- if (attributes.TryGetValue("group-title", out string groupTitle))
- {
- channel.ChannelGroup = groupTitle;
- }
- channel.Name = GetChannelName(extInf, attributes);
- channel.Number = GetChannelNumber(extInf, attributes, mediaUrl);
- attributes.TryGetValue("tvg-id", out string tvgId);
- attributes.TryGetValue("channel-id", out string channelId);
- channel.TunerChannelId = string.IsNullOrWhiteSpace(tvgId) ? channelId : tvgId;
- var channelIdValues = new List<string>();
- if (!string.IsNullOrWhiteSpace(channelId))
- {
- channelIdValues.Add(channelId);
- }
- if (!string.IsNullOrWhiteSpace(tvgId))
- {
- channelIdValues.Add(tvgId);
- }
- if (channelIdValues.Count > 0)
- {
- channel.Id = string.Join('_', channelIdValues);
- }
- return channel;
- }
- private string GetChannelNumber(string extInf, Dictionary<string, string> attributes, string mediaUrl)
- {
- var nameParts = extInf.Split(',', StringSplitOptions.RemoveEmptyEntries);
- var nameInExtInf = nameParts.Length > 1 ? nameParts[^1].AsSpan().Trim() : ReadOnlySpan<char>.Empty;
- string numberString = null;
- string attributeValue;
- if (attributes.TryGetValue("tvg-chno", out attributeValue))
- {
- if (double.TryParse(attributeValue, NumberStyles.Any, CultureInfo.InvariantCulture, out _))
- {
- numberString = attributeValue;
- }
- }
- if (!IsValidChannelNumber(numberString))
- {
- if (attributes.TryGetValue("tvg-id", out attributeValue))
- {
- if (double.TryParse(attributeValue, NumberStyles.Any, CultureInfo.InvariantCulture, out _))
- {
- numberString = attributeValue;
- }
- else if (attributes.TryGetValue("channel-id", out attributeValue))
- {
- if (double.TryParse(attributeValue, NumberStyles.Any, CultureInfo.InvariantCulture, out _))
- {
- numberString = attributeValue;
- }
- }
- }
- if (string.IsNullOrWhiteSpace(numberString))
- {
- // Using this as a fallback now as this leads to Problems with channels like "5 USA"
- // where 5 isn't ment 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, NumberStyles.Any, CultureInfo.InvariantCulture, out _))
- {
- numberString = numberPart.ToString();
- }
- }
- }
- }
- }
- if (!IsValidChannelNumber(numberString))
- {
- numberString = null;
- }
- if (!string.IsNullOrWhiteSpace(numberString))
- {
- numberString = numberString.Trim();
- }
- else
- {
- if (string.IsNullOrWhiteSpace(mediaUrl))
- {
- numberString = null;
- }
- else
- {
- try
- {
- numberString = Path.GetFileNameWithoutExtension(mediaUrl.Split('/')[^1]);
- if (!IsValidChannelNumber(numberString))
- {
- numberString = null;
- }
- }
- catch
- {
- // Seeing occasional argument exception here
- numberString = null;
- }
- }
- }
- return numberString;
- }
- private static bool IsValidChannelNumber(string numberString)
- {
- if (string.IsNullOrWhiteSpace(numberString) ||
- string.Equals(numberString, "-1", StringComparison.OrdinalIgnoreCase) ||
- string.Equals(numberString, "0", StringComparison.OrdinalIgnoreCase))
- {
- return false;
- }
- if (!double.TryParse(numberString, NumberStyles.Any, CultureInfo.InvariantCulture, out _))
- {
- return false;
- }
- return true;
- }
- private static string GetChannelName(string extInf, Dictionary<string, string> attributes)
- {
- var nameParts = extInf.Split(',', StringSplitOptions.RemoveEmptyEntries);
- var nameInExtInf = nameParts.Length > 1 ? nameParts[^1].Trim() : null;
- // Check for channel number with the format from SatIp
- // #EXTINF:0,84. VOX Schweiz
- // #EXTINF:0,84.0 - VOX Schweiz
- if (!string.IsNullOrWhiteSpace(nameInExtInf))
- {
- var numberIndex = nameInExtInf.IndexOf(' ');
- if (numberIndex > 0)
- {
- var numberPart = nameInExtInf.Substring(0, numberIndex).Trim(new[] { ' ', '.' });
- if (double.TryParse(numberPart, NumberStyles.Any, CultureInfo.InvariantCulture, out _))
- {
- // channel.Number = number.ToString();
- nameInExtInf = nameInExtInf.Substring(numberIndex + 1).Trim(new[] { ' ', '-' });
- }
- }
- }
- attributes.TryGetValue("tvg-name", out string name);
- if (string.IsNullOrWhiteSpace(name))
- {
- name = nameInExtInf;
- }
- if (string.IsNullOrWhiteSpace(name))
- {
- attributes.TryGetValue("tvg-id", out name);
- }
- if (string.IsNullOrWhiteSpace(name))
- {
- name = null;
- }
- return name;
- }
- private static Dictionary<string, string> ParseExtInf(string line, out string remaining)
- {
- var dict = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
- var reg = new Regex(@"([a-z0-9\-_]+)=\""([^""]+)\""", RegexOptions.IgnoreCase);
- var matches = reg.Matches(line);
- remaining = line;
- foreach (Match match in matches)
- {
- var key = match.Groups[1].Value;
- var value = match.Groups[2].Value;
- dict[match.Groups[1].Value] = match.Groups[2].Value;
- remaining = remaining.Replace(key + "=\"" + value + "\"", string.Empty, StringComparison.OrdinalIgnoreCase);
- }
- return dict;
- }
- }
- }
|