|
|
|
@ -14,6 +14,7 @@ using MediaBrowser.Controller;
|
|
|
|
|
using MediaBrowser.Controller.IO;
|
|
|
|
|
using MediaBrowser.Controller.LiveTv;
|
|
|
|
|
using MediaBrowser.Model.Logging;
|
|
|
|
|
using MediaBrowser.Model.Extensions;
|
|
|
|
|
|
|
|
|
|
namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
{
|
|
|
|
@ -43,6 +44,17 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<M3UChannel> ParseString(string text, string channelIdPrefix, string tunerHostId)
|
|
|
|
|
{
|
|
|
|
|
var urlHash = "text".GetMD5().ToString("N");
|
|
|
|
|
|
|
|
|
|
// Read the file and display it line by line.
|
|
|
|
|
using (var reader = new StringReader(text))
|
|
|
|
|
{
|
|
|
|
|
return GetChannels(reader, urlHash, channelIdPrefix, tunerHostId);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<Stream> GetListingsStream(string url, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
if (url.StartsWith("http", StringComparison.OrdinalIgnoreCase))
|
|
|
|
@ -59,7 +71,7 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const string ExtInfPrefix = "#EXTINF:";
|
|
|
|
|
private List<M3UChannel> GetChannels(StreamReader reader, string urlHash, string channelIdPrefix, string tunerHostId)
|
|
|
|
|
private List<M3UChannel> GetChannels(TextReader reader, string urlHash, string channelIdPrefix, string tunerHostId)
|
|
|
|
|
{
|
|
|
|
|
var channels = new List<M3UChannel>();
|
|
|
|
|
string line;
|
|
|
|
@ -122,18 +134,22 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
var nameParts = extInf.Split(new[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
|
var nameInExtInf = nameParts.Length > 1 ? nameParts.Last().Trim() : null;
|
|
|
|
|
|
|
|
|
|
var numberString = nameParts[0];
|
|
|
|
|
string numberString = null;
|
|
|
|
|
|
|
|
|
|
//Check for channel number with the format from SatIp
|
|
|
|
|
int number;
|
|
|
|
|
// 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('.');
|
|
|
|
|
var numberIndex = nameInExtInf.IndexOf(' ');
|
|
|
|
|
if (numberIndex > 0)
|
|
|
|
|
{
|
|
|
|
|
if (int.TryParse(nameInExtInf.Substring(0, numberIndex), out number))
|
|
|
|
|
var numberPart = nameInExtInf.Substring(0, numberIndex).Trim(new[] { ' ', '.' });
|
|
|
|
|
|
|
|
|
|
double number;
|
|
|
|
|
if (double.TryParse(numberPart, NumberStyles.Any, CultureInfo.InvariantCulture, out number))
|
|
|
|
|
{
|
|
|
|
|
numberString = number.ToString();
|
|
|
|
|
numberString = numberPart;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -150,7 +166,11 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
string value;
|
|
|
|
|
if (attributes.TryGetValue("tvg-id", out value))
|
|
|
|
|
{
|
|
|
|
|
numberString = value;
|
|
|
|
|
double doubleValue;
|
|
|
|
|
if (double.TryParse(value, NumberStyles.Any, CultureInfo.InvariantCulture, out doubleValue))
|
|
|
|
|
{
|
|
|
|
|
numberString = value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -208,17 +228,21 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
var nameParts = extInf.Split(new[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
|
var nameInExtInf = nameParts.Length > 1 ? nameParts.Last().Trim() : null;
|
|
|
|
|
|
|
|
|
|
//Check for channel number with the format from SatIp
|
|
|
|
|
int number;
|
|
|
|
|
// 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('.');
|
|
|
|
|
var numberIndex = nameInExtInf.IndexOf(' ');
|
|
|
|
|
if (numberIndex > 0)
|
|
|
|
|
{
|
|
|
|
|
if (int.TryParse(nameInExtInf.Substring(0, numberIndex), out number))
|
|
|
|
|
var numberPart = nameInExtInf.Substring(0, numberIndex).Trim(new[] { ' ', '.' });
|
|
|
|
|
|
|
|
|
|
double number;
|
|
|
|
|
if (double.TryParse(numberPart, NumberStyles.Any, CultureInfo.InvariantCulture, out number))
|
|
|
|
|
{
|
|
|
|
|
//channel.Number = number.ToString();
|
|
|
|
|
nameInExtInf = nameInExtInf.Substring(numberIndex + 1);
|
|
|
|
|
nameInExtInf = nameInExtInf.Substring(numberIndex + 1).Trim(new[] { ' ', '-' });
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -250,20 +274,18 @@ namespace Emby.Server.Implementations.LiveTv.TunerHosts
|
|
|
|
|
|
|
|
|
|
var reg = new Regex(@"([a-z0-9\-_]+)=\""([^""]+)\""", RegexOptions.IgnoreCase);
|
|
|
|
|
var matches = reg.Matches(line);
|
|
|
|
|
var minIndex = int.MaxValue;
|
|
|
|
|
|
|
|
|
|
remaining = line;
|
|
|
|
|
|
|
|
|
|
foreach (Match match in matches)
|
|
|
|
|
{
|
|
|
|
|
dict[match.Groups[1].Value] = match.Groups[2].Value;
|
|
|
|
|
minIndex = Math.Min(minIndex, match.Index);
|
|
|
|
|
}
|
|
|
|
|
var key = match.Groups[1].Value;
|
|
|
|
|
var value = match.Groups[2].Value;
|
|
|
|
|
|
|
|
|
|
if (minIndex > 0 && minIndex < line.Length)
|
|
|
|
|
{
|
|
|
|
|
line = line.Substring(0, minIndex);
|
|
|
|
|
dict[match.Groups[1].Value] = match.Groups[2].Value;
|
|
|
|
|
remaining = remaining.Replace(key + "=\"" + value + "\"", string.Empty, StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
remaining = line;
|
|
|
|
|
|
|
|
|
|
return dict;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|