Finaly working version of livestreams
This commit is contained in:
parent
8e86db100a
commit
cfe510e15c
@ -25,6 +25,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly ILoggerFactory _loggerFactory;
|
||||||
private readonly ContentRefreshService _contentRefreshService;
|
private readonly ContentRefreshService _contentRefreshService;
|
||||||
private readonly StreamUrlResolver _streamResolver;
|
private readonly StreamUrlResolver _streamResolver;
|
||||||
|
private readonly StreamProxyService _proxyService;
|
||||||
private readonly CategoryService? _categoryService;
|
private readonly CategoryService? _categoryService;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -33,17 +34,20 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
/// <param name="contentRefreshService">The content refresh service.</param>
|
/// <param name="contentRefreshService">The content refresh service.</param>
|
||||||
/// <param name="streamResolver">The stream resolver.</param>
|
/// <param name="streamResolver">The stream resolver.</param>
|
||||||
|
/// <param name="proxyService">The stream proxy service.</param>
|
||||||
/// <param name="categoryService">The category service (optional).</param>
|
/// <param name="categoryService">The category service (optional).</param>
|
||||||
public SRFPlayChannel(
|
public SRFPlayChannel(
|
||||||
ILoggerFactory loggerFactory,
|
ILoggerFactory loggerFactory,
|
||||||
ContentRefreshService contentRefreshService,
|
ContentRefreshService contentRefreshService,
|
||||||
StreamUrlResolver streamResolver,
|
StreamUrlResolver streamResolver,
|
||||||
|
StreamProxyService proxyService,
|
||||||
CategoryService? categoryService = null)
|
CategoryService? categoryService = null)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
_loggerFactory = loggerFactory;
|
||||||
_logger = loggerFactory.CreateLogger<SRFPlayChannel>();
|
_logger = loggerFactory.CreateLogger<SRFPlayChannel>();
|
||||||
_contentRefreshService = contentRefreshService;
|
_contentRefreshService = contentRefreshService;
|
||||||
_streamResolver = streamResolver;
|
_streamResolver = streamResolver;
|
||||||
|
_proxyService = proxyService;
|
||||||
_categoryService = categoryService;
|
_categoryService = categoryService;
|
||||||
|
|
||||||
if (_categoryService == null)
|
if (_categoryService == null)
|
||||||
@ -61,7 +65,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
public string Description => "Swiss Radio and Television video-on-demand content";
|
public string Description => "Swiss Radio and Television video-on-demand content";
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public string DataVersion => "1.1";
|
public string DataVersion => "2.0"; // Back to authenticating at channel refresh with auto-refresh for fresh tokens
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public string HomePageUrl => "https://www.srf.ch/play";
|
public string HomePageUrl => "https://www.srf.ch/play";
|
||||||
@ -433,7 +437,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
// Generate deterministic GUID from URN
|
// Generate deterministic GUID from URN
|
||||||
var itemId = UrnToGuid(urn);
|
var itemId = UrnToGuid(urn);
|
||||||
|
|
||||||
// Get stream URL (unauthenticated - token will be added at playback time by SRFMediaProvider)
|
// Get stream URL and authenticate it
|
||||||
var streamUrl = _streamResolver.GetStreamUrl(chapter, config.QualityPreference);
|
var streamUrl = _streamResolver.GetStreamUrl(chapter, config.QualityPreference);
|
||||||
|
|
||||||
// Skip scheduled livestreams that haven't started yet (no stream URL available)
|
// Skip scheduled livestreams that haven't started yet (no stream URL available)
|
||||||
@ -447,6 +451,12 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Authenticate the stream URL with fresh token
|
||||||
|
if (!string.IsNullOrEmpty(streamUrl))
|
||||||
|
{
|
||||||
|
streamUrl = await _streamResolver.GetAuthenticatedStreamUrlAsync(streamUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
// Skip items without a valid stream URL
|
// Skip items without a valid stream URL
|
||||||
if (string.IsNullOrEmpty(streamUrl))
|
if (string.IsNullOrEmpty(streamUrl))
|
||||||
{
|
{
|
||||||
@ -458,6 +468,13 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Register stream with proxy service
|
||||||
|
_proxyService.RegisterStream(itemId, streamUrl);
|
||||||
|
|
||||||
|
// Create proxy URL as absolute HTTP URL (required for ffmpeg)
|
||||||
|
// Use localhost as Jellyfin should be able to access its own endpoints
|
||||||
|
var proxyUrl = $"http://localhost:8096/Plugins/SRFPlay/Proxy/{itemId}/master.m3u8";
|
||||||
|
|
||||||
// Build overview
|
// Build overview
|
||||||
var overview = chapter.Description ?? chapter.Lead;
|
var overview = chapter.Description ?? chapter.Lead;
|
||||||
|
|
||||||
@ -477,7 +494,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
// Use ValidFrom for premiere date if this is a scheduled livestream, otherwise use Date
|
// Use ValidFrom for premiere date if this is a scheduled livestream, otherwise use Date
|
||||||
var premiereDate = chapter.Type == "SCHEDULED_LIVESTREAM" ? chapter.ValidFrom?.ToUniversalTime() : chapter.Date?.ToUniversalTime();
|
var premiereDate = chapter.Type == "SCHEDULED_LIVESTREAM" ? chapter.ValidFrom?.ToUniversalTime() : chapter.Date?.ToUniversalTime();
|
||||||
|
|
||||||
// Store unauthenticated URL as placeholder - SRFMediaProvider will provide authenticated version at playback
|
// Store authenticated URL - tokens refresh automatically via scheduled channel scans
|
||||||
var item = new ChannelItemInfo
|
var item = new ChannelItemInfo
|
||||||
{
|
{
|
||||||
Id = itemId,
|
Id = itemId,
|
||||||
@ -501,15 +518,38 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
{
|
{
|
||||||
Id = itemId,
|
Id = itemId,
|
||||||
Name = chapter.Title,
|
Name = chapter.Title,
|
||||||
Path = streamUrl, // Unauthenticated URL - placeholder only
|
Path = proxyUrl, // Proxy URL instead of direct Akamai URL
|
||||||
Protocol = MediaBrowser.Model.MediaInfo.MediaProtocol.Http,
|
Protocol = MediaBrowser.Model.MediaInfo.MediaProtocol.Http,
|
||||||
Container = "m3u8",
|
Container = "hls",
|
||||||
SupportsDirectPlay = false, // Disable direct play - requires auth from provider
|
SupportsDirectStream = true,
|
||||||
SupportsDirectStream = false, // Disable direct stream - requires auth from provider
|
SupportsDirectPlay = true, // ✅ Enabled! Proxy handles auth
|
||||||
SupportsTranscoding = false, // Force use of IMediaSourceProvider
|
SupportsTranscoding = true,
|
||||||
IsRemote = true,
|
IsRemote = false, // False because it's a local proxy endpoint
|
||||||
Type = MediaBrowser.Model.Dto.MediaSourceType.Placeholder, // Mark as placeholder
|
Type = MediaBrowser.Model.Dto.MediaSourceType.Default,
|
||||||
VideoType = VideoType.VideoFile
|
VideoType = VideoType.VideoFile,
|
||||||
|
RequiresOpening = false,
|
||||||
|
RequiresClosing = false,
|
||||||
|
SupportsProbing = false, // Disable probing for proxy URLs
|
||||||
|
ReadAtNativeFramerate = false,
|
||||||
|
MediaStreams = new List<MediaBrowser.Model.Entities.MediaStream>
|
||||||
|
{
|
||||||
|
new MediaBrowser.Model.Entities.MediaStream
|
||||||
|
{
|
||||||
|
Type = MediaBrowser.Model.Entities.MediaStreamType.Video,
|
||||||
|
Codec = "h264",
|
||||||
|
Profile = "high",
|
||||||
|
IsInterlaced = false,
|
||||||
|
IsDefault = true,
|
||||||
|
Index = 0
|
||||||
|
},
|
||||||
|
new MediaBrowser.Model.Entities.MediaStream
|
||||||
|
{
|
||||||
|
Type = MediaBrowser.Model.Entities.MediaStreamType.Audio,
|
||||||
|
Codec = "aac",
|
||||||
|
IsDefault = true,
|
||||||
|
Index = 1
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -523,6 +563,13 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
items.Add(item);
|
items.Add(item);
|
||||||
successCount++;
|
successCount++;
|
||||||
_logger.LogInformation("URN {Urn}: Successfully converted to channel item - {Title}", urn, chapter.Title);
|
_logger.LogInformation("URN {Urn}: Successfully converted to channel item - {Title}", urn, chapter.Title);
|
||||||
|
_logger.LogInformation(
|
||||||
|
"URN {Urn}: MediaSource configured - DirectStream={DirectStream}, DirectPlay={DirectPlay}, Transcoding={Transcoding}, Container={Container}",
|
||||||
|
urn,
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
"hls");
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
|||||||
220
Jellyfin.Plugin.SRFPlay/Controllers/StreamProxyController.cs
Normal file
220
Jellyfin.Plugin.SRFPlay/Controllers/StreamProxyController.cs
Normal file
@ -0,0 +1,220 @@
|
|||||||
|
using System;
|
||||||
|
using System.Globalization;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
|
using Microsoft.AspNetCore.Http;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Controllers;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Controller for proxying SRF Play streams.
|
||||||
|
/// </summary>
|
||||||
|
[ApiController]
|
||||||
|
[Route("Plugins/SRFPlay/Proxy")]
|
||||||
|
public class StreamProxyController : ControllerBase
|
||||||
|
{
|
||||||
|
private readonly ILogger<StreamProxyController> _logger;
|
||||||
|
private readonly StreamProxyService _proxyService;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="StreamProxyController"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="logger">The logger.</param>
|
||||||
|
/// <param name="proxyService">The proxy service.</param>
|
||||||
|
public StreamProxyController(ILogger<StreamProxyController> logger, StreamProxyService proxyService)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
_proxyService = proxyService;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Proxies HLS master manifest requests.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The HLS manifest with rewritten URLs.</returns>
|
||||||
|
[HttpGet("{itemId}/master.m3u8")]
|
||||||
|
[AllowAnonymous] // Allow anonymous since Jellyfin handles auth upstream
|
||||||
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
|
public async Task<IActionResult> GetMasterManifest(
|
||||||
|
[FromRoute] string itemId,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("Proxy request for master manifest - ItemId: {ItemId}", itemId);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Build the base proxy URL for this item
|
||||||
|
var baseProxyUrl = $"{Request.Scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}";
|
||||||
|
|
||||||
|
var manifestContent = await _proxyService.GetRewrittenManifestAsync(itemId, baseProxyUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (manifestContent == null)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Manifest not found for item {ItemId}", itemId);
|
||||||
|
return NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogDebug("Returning master manifest for item {ItemId} ({Length} bytes)", itemId, manifestContent.Length);
|
||||||
|
return Content(manifestContent, "application/vnd.apple.mpegurl");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error proxying master manifest for item {ItemId}", itemId);
|
||||||
|
return StatusCode(StatusCodes.Status500InternalServerError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Proxies HLS variant manifest requests.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="manifestPath">The manifest path (e.g., "index_0_av").</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The HLS manifest with rewritten segment URLs.</returns>
|
||||||
|
[HttpGet("{itemId}/{manifestPath}.m3u8")]
|
||||||
|
[AllowAnonymous]
|
||||||
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
|
public async Task<IActionResult> GetVariantManifest(
|
||||||
|
[FromRoute] string itemId,
|
||||||
|
[FromRoute] string manifestPath,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var fullPath = $"{manifestPath}.m3u8";
|
||||||
|
_logger.LogInformation("Proxy request for variant manifest - ItemId: {ItemId}, Path: {Path}", itemId, fullPath);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Fetch the variant manifest as a segment
|
||||||
|
var manifestData = await _proxyService.GetSegmentAsync(itemId, fullPath, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (manifestData == null)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Variant manifest not found - ItemId: {ItemId}, Path: {Path}", itemId, fullPath);
|
||||||
|
return NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert to string and rewrite URLs
|
||||||
|
var manifestContent = System.Text.Encoding.UTF8.GetString(manifestData);
|
||||||
|
var baseProxyUrl = $"{Request.Scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}";
|
||||||
|
var rewrittenContent = RewriteSegmentUrls(manifestContent, baseProxyUrl);
|
||||||
|
|
||||||
|
_logger.LogDebug("Returning variant manifest for item {ItemId} ({Length} bytes)", itemId, rewrittenContent.Length);
|
||||||
|
return Content(rewrittenContent, "application/vnd.apple.mpegurl");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error proxying variant manifest - ItemId: {ItemId}, Path: {Path}", itemId, fullPath);
|
||||||
|
return StatusCode(StatusCodes.Status500InternalServerError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Proxies HLS segment requests (.ts, .mp4, .m4s, .aac files).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="segmentPath">The segment path.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The segment data.</returns>
|
||||||
|
[HttpGet("{itemId}/{*segmentPath}")]
|
||||||
|
[AllowAnonymous]
|
||||||
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
|
public async Task<IActionResult> GetSegment(
|
||||||
|
[FromRoute] string itemId,
|
||||||
|
[FromRoute] string segmentPath,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
_logger.LogDebug("Proxy request for segment - ItemId: {ItemId}, Path: {SegmentPath}", itemId, segmentPath);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var segmentData = await _proxyService.GetSegmentAsync(itemId, segmentPath, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (segmentData == null)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Segment not found - ItemId: {ItemId}, Path: {SegmentPath}", itemId, segmentPath);
|
||||||
|
return NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine content type based on file extension
|
||||||
|
var contentType = GetContentType(segmentPath);
|
||||||
|
|
||||||
|
_logger.LogDebug("Returning segment {SegmentPath} ({Length} bytes, {ContentType})", segmentPath, segmentData.Length, contentType);
|
||||||
|
return File(segmentData, contentType);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error proxying segment - ItemId: {ItemId}, Path: {SegmentPath}", itemId, segmentPath);
|
||||||
|
return StatusCode(StatusCodes.Status500InternalServerError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Rewrites segment URLs in a manifest to point to proxy.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="manifestContent">The manifest content.</param>
|
||||||
|
/// <param name="baseProxyUrl">The base proxy URL.</param>
|
||||||
|
/// <returns>The rewritten manifest.</returns>
|
||||||
|
private string RewriteSegmentUrls(string manifestContent, string baseProxyUrl)
|
||||||
|
{
|
||||||
|
var lines = manifestContent.Split('\n');
|
||||||
|
var result = new System.Text.StringBuilder();
|
||||||
|
|
||||||
|
foreach (var line in lines)
|
||||||
|
{
|
||||||
|
if (line.StartsWith('#') || string.IsNullOrWhiteSpace(line))
|
||||||
|
{
|
||||||
|
// Keep metadata and blank lines as-is
|
||||||
|
result.AppendLine(line);
|
||||||
|
}
|
||||||
|
else if (line.Contains("://", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
// Absolute URL - extract the path and rewrite
|
||||||
|
var uri = new Uri(line.Trim());
|
||||||
|
var segments = uri.AbsolutePath.Split('/');
|
||||||
|
var fileName = segments[^1];
|
||||||
|
result.AppendLine(CultureInfo.InvariantCulture, $"{baseProxyUrl}/{fileName}");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Relative URL - rewrite to proxy
|
||||||
|
result.AppendLine(CultureInfo.InvariantCulture, $"{baseProxyUrl}/{line.Trim()}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the content type for a segment based on file extension.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="path">The segment path.</param>
|
||||||
|
/// <returns>The MIME content type.</returns>
|
||||||
|
private static string GetContentType(string path)
|
||||||
|
{
|
||||||
|
if (path.EndsWith(".ts", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return "video/MP2T";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.EndsWith(".mp4", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
path.EndsWith(".m4s", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return "video/mp4";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.EndsWith(".aac", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return "audio/aac";
|
||||||
|
}
|
||||||
|
|
||||||
|
return "application/octet-stream";
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -23,6 +23,7 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly ILoggerFactory _loggerFactory;
|
||||||
private readonly MetadataCache _metadataCache;
|
private readonly MetadataCache _metadataCache;
|
||||||
private readonly StreamUrlResolver _streamResolver;
|
private readonly StreamUrlResolver _streamResolver;
|
||||||
|
private readonly StreamProxyService _proxyService;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="SRFMediaProvider"/> class.
|
/// Initializes a new instance of the <see cref="SRFMediaProvider"/> class.
|
||||||
@ -30,15 +31,18 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
/// <param name="metadataCache">The metadata cache.</param>
|
/// <param name="metadataCache">The metadata cache.</param>
|
||||||
/// <param name="streamResolver">The stream URL resolver.</param>
|
/// <param name="streamResolver">The stream URL resolver.</param>
|
||||||
|
/// <param name="proxyService">The stream proxy service.</param>
|
||||||
public SRFMediaProvider(
|
public SRFMediaProvider(
|
||||||
ILoggerFactory loggerFactory,
|
ILoggerFactory loggerFactory,
|
||||||
MetadataCache metadataCache,
|
MetadataCache metadataCache,
|
||||||
StreamUrlResolver streamResolver)
|
StreamUrlResolver streamResolver,
|
||||||
|
StreamProxyService proxyService)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
_loggerFactory = loggerFactory;
|
||||||
_logger = loggerFactory.CreateLogger<SRFMediaProvider>();
|
_logger = loggerFactory.CreateLogger<SRFMediaProvider>();
|
||||||
_metadataCache = metadataCache;
|
_metadataCache = metadataCache;
|
||||||
_streamResolver = streamResolver;
|
_streamResolver = streamResolver;
|
||||||
|
_proxyService = proxyService;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -52,24 +56,35 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
/// <param name="item">The item.</param>
|
/// <param name="item">The item.</param>
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
/// <returns>List of media sources.</returns>
|
/// <returns>List of media sources.</returns>
|
||||||
public Task<IEnumerable<MediaSourceInfo>> GetMediaSources(BaseItem item, CancellationToken cancellationToken)
|
public async Task<IEnumerable<MediaSourceInfo>> GetMediaSources(BaseItem item, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var sources = new List<MediaSourceInfo>();
|
var sources = new List<MediaSourceInfo>();
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
// Log detailed information about the request
|
||||||
|
var stackTrace = new System.Diagnostics.StackTrace(true);
|
||||||
|
var callingMethod = stackTrace.GetFrame(1)?.GetMethod();
|
||||||
|
_logger.LogInformation(
|
||||||
|
"GetMediaSources called - Item: {ItemName}, Type: {ItemType}, Id: {ItemId}, CalledBy: {CallingMethod}",
|
||||||
|
item.Name,
|
||||||
|
item.GetType().Name,
|
||||||
|
item.Id,
|
||||||
|
callingMethod?.DeclaringType?.Name + "." + callingMethod?.Name);
|
||||||
|
|
||||||
// Check if this is an SRF item
|
// Check if this is an SRF item
|
||||||
if (!item.ProviderIds.TryGetValue("SRF", out var urn) || string.IsNullOrEmpty(urn))
|
if (!item.ProviderIds.TryGetValue("SRF", out var urn) || string.IsNullOrEmpty(urn))
|
||||||
{
|
{
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
_logger.LogDebug("Item {ItemName} is not an SRF item, returning empty sources", item.Name);
|
||||||
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogDebug("Getting media sources for URN: {Urn}", urn);
|
_logger.LogInformation("Getting media sources for URN: {Urn}, Item: {ItemName}", urn, item.Name);
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var config = Plugin.Instance?.Configuration;
|
||||||
if (config == null)
|
if (config == null)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// For scheduled livestreams, use shorter cache TTL (5 minutes) so they refresh when they go live
|
// For scheduled livestreams, use shorter cache TTL (5 minutes) so they refresh when they go live
|
||||||
@ -85,7 +100,7 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
if (mediaComposition == null)
|
if (mediaComposition == null)
|
||||||
{
|
{
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
using var apiClient = new SRFApiClient(_loggerFactory);
|
||||||
mediaComposition = apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).GetAwaiter().GetResult();
|
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (mediaComposition != null)
|
if (mediaComposition != null)
|
||||||
{
|
{
|
||||||
@ -96,7 +111,7 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("No chapters found for URN: {Urn}", urn);
|
_logger.LogWarning("No chapters found for URN: {Urn}", urn);
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the first chapter (main video)
|
// Get the first chapter (main video)
|
||||||
@ -106,14 +121,14 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
if (_streamResolver.IsContentExpired(chapter))
|
if (_streamResolver.IsContentExpired(chapter))
|
||||||
{
|
{
|
||||||
_logger.LogWarning("Content expired for URN: {Urn}, ValidTo: {ValidTo}", urn, chapter.ValidTo);
|
_logger.LogWarning("Content expired for URN: {Urn}, ValidTo: {ValidTo}", urn, chapter.ValidTo);
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if content has playable streams
|
// Check if content has playable streams
|
||||||
if (!_streamResolver.HasPlayableContent(chapter))
|
if (!_streamResolver.HasPlayableContent(chapter))
|
||||||
{
|
{
|
||||||
_logger.LogWarning("No playable content found for URN: {Urn}", urn);
|
_logger.LogWarning("No playable content found for URN: {Urn}", urn);
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get stream URL based on quality preference
|
// Get stream URL based on quality preference
|
||||||
@ -125,7 +140,7 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
_logger.LogDebug("URN {Urn}: Scheduled livestream has no stream URL, fetching fresh data", urn);
|
_logger.LogDebug("URN {Urn}: Scheduled livestream has no stream URL, fetching fresh data", urn);
|
||||||
|
|
||||||
using var freshApiClient = new SRFApiClient(_loggerFactory);
|
using var freshApiClient = new SRFApiClient(_loggerFactory);
|
||||||
var freshMediaComposition = freshApiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).GetAwaiter().GetResult();
|
var freshMediaComposition = await freshApiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (freshMediaComposition?.ChapterList != null && freshMediaComposition.ChapterList.Count > 0)
|
if (freshMediaComposition?.ChapterList != null && freshMediaComposition.ChapterList.Count > 0)
|
||||||
{
|
{
|
||||||
@ -145,58 +160,91 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
if (string.IsNullOrEmpty(streamUrl))
|
if (string.IsNullOrEmpty(streamUrl))
|
||||||
{
|
{
|
||||||
_logger.LogWarning("Could not resolve stream URL for URN: {Urn}", urn);
|
_logger.LogWarning("Could not resolve stream URL for URN: {Urn}", urn);
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Authenticate the stream URL (required for all SRF streams, especially livestreams)
|
// Authenticate the stream URL (required for all SRF streams, especially livestreams)
|
||||||
if (!string.IsNullOrEmpty(streamUrl))
|
if (!string.IsNullOrEmpty(streamUrl))
|
||||||
{
|
{
|
||||||
streamUrl = _streamResolver.GetAuthenticatedStreamUrlAsync(streamUrl, cancellationToken).GetAwaiter().GetResult();
|
streamUrl = await _streamResolver.GetAuthenticatedStreamUrlAsync(streamUrl, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogDebug("Authenticated stream URL for URN: {Urn}", urn);
|
_logger.LogDebug("Authenticated stream URL for URN: {Urn}", urn);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create media source
|
// Register stream with proxy service
|
||||||
|
var itemIdStr = item.Id.ToString("N"); // Use hex format without dashes
|
||||||
|
_proxyService.RegisterStream(itemIdStr, streamUrl);
|
||||||
|
|
||||||
|
// Create proxy URL as absolute HTTP URL (required for ffmpeg)
|
||||||
|
// Use localhost as Jellyfin should be able to access its own endpoints
|
||||||
|
var proxyUrl = $"http://localhost:8096/Plugins/SRFPlay/Proxy/{itemIdStr}/master.m3u8";
|
||||||
|
|
||||||
|
_logger.LogInformation("Using proxy URL for item {ItemId}: {ProxyUrl}", itemIdStr, proxyUrl);
|
||||||
|
|
||||||
|
// Create media source using proxy URL - enables DirectPlay!
|
||||||
var mediaSource = new MediaSourceInfo
|
var mediaSource = new MediaSourceInfo
|
||||||
{
|
{
|
||||||
Id = item.Id.ToString(), // Use item GUID, not URN string (required for transcoding)
|
Id = item.Id.ToString(), // Use item GUID, not URN string (required for transcoding)
|
||||||
Name = chapter.Title,
|
Name = chapter.Title,
|
||||||
Path = streamUrl,
|
Path = proxyUrl, // Proxy URL instead of direct Akamai URL
|
||||||
Protocol = MediaProtocol.Http,
|
Protocol = MediaProtocol.Http,
|
||||||
Container = "m3u8",
|
Container = "hls",
|
||||||
SupportsDirectStream = true,
|
SupportsDirectStream = true,
|
||||||
SupportsDirectPlay = false, // Disabled: auth tokens don't carry over to HLS segments in browser
|
SupportsDirectPlay = true, // ✅ Enabled! Proxy handles auth
|
||||||
SupportsTranscoding = true,
|
SupportsTranscoding = true,
|
||||||
IsRemote = true,
|
IsRemote = false, // False because it's a local proxy endpoint
|
||||||
Type = MediaSourceType.Default,
|
Type = MediaSourceType.Default,
|
||||||
RunTimeTicks = chapter.Duration > 0 ? TimeSpan.FromMilliseconds(chapter.Duration).Ticks : null,
|
RunTimeTicks = chapter.Duration > 0 ? TimeSpan.FromMilliseconds(chapter.Duration).Ticks : null,
|
||||||
VideoType = VideoType.VideoFile,
|
VideoType = VideoType.VideoFile,
|
||||||
IsInfiniteStream = false,
|
IsInfiniteStream = false,
|
||||||
RequiresOpening = false,
|
RequiresOpening = false,
|
||||||
RequiresClosing = false,
|
RequiresClosing = false,
|
||||||
SupportsProbing = true
|
SupportsProbing = false, // Disable probing for proxy URLs
|
||||||
};
|
ReadAtNativeFramerate = false,
|
||||||
|
MediaStreams = new List<MediaBrowser.Model.Entities.MediaStream>
|
||||||
// Add video stream info
|
|
||||||
mediaSource.MediaStreams = new List<MediaStream>
|
|
||||||
{
|
|
||||||
new MediaStream
|
|
||||||
{
|
{
|
||||||
Type = MediaStreamType.Video,
|
new MediaBrowser.Model.Entities.MediaStream
|
||||||
Codec = "h264",
|
{
|
||||||
IsInterlaced = false,
|
Type = MediaStreamType.Video,
|
||||||
IsDefault = true
|
Codec = "h264",
|
||||||
|
Profile = "high",
|
||||||
|
IsInterlaced = false,
|
||||||
|
IsDefault = true,
|
||||||
|
Index = 0
|
||||||
|
},
|
||||||
|
new MediaBrowser.Model.Entities.MediaStream
|
||||||
|
{
|
||||||
|
Type = MediaStreamType.Audio,
|
||||||
|
Codec = "aac",
|
||||||
|
IsDefault = true,
|
||||||
|
Index = 1
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
sources.Add(mediaSource);
|
sources.Add(mediaSource);
|
||||||
_logger.LogInformation("Resolved stream URL for {Title}: {Url}", chapter.Title, streamUrl);
|
_logger.LogInformation("Resolved stream URL for {Title}: {Url}", chapter.Title, streamUrl);
|
||||||
|
_logger.LogInformation(
|
||||||
|
"MediaSource created - Id={Id}, DirectStream={DirectStream}, DirectPlay={DirectPlay}, Probing={Probing}, Container={Container}, Protocol={Protocol}, IsRemote={IsRemote}",
|
||||||
|
mediaSource.Id,
|
||||||
|
mediaSource.SupportsDirectStream,
|
||||||
|
mediaSource.SupportsDirectPlay,
|
||||||
|
mediaSource.SupportsProbing,
|
||||||
|
mediaSource.Container,
|
||||||
|
mediaSource.Protocol,
|
||||||
|
mediaSource.IsRemote);
|
||||||
|
_logger.LogInformation(
|
||||||
|
"MediaSource capabilities - SupportsTranscoding={Transcoding}, RequiresOpening={RequiresOpening}, RequiresClosing={RequiresClosing}, Type={Type}",
|
||||||
|
mediaSource.SupportsTranscoding,
|
||||||
|
mediaSource.RequiresOpening,
|
||||||
|
mediaSource.RequiresClosing,
|
||||||
|
mediaSource.Type);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Error getting media sources for item: {Name}", item.Name);
|
_logger.LogError(ex, "Error getting media sources for item: {Name}", item.Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Task.FromResult<IEnumerable<MediaSourceInfo>>(sources);
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -207,6 +255,7 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
/// <returns>The direct stream provider.</returns>
|
/// <returns>The direct stream provider.</returns>
|
||||||
public Task<IDirectStreamProvider?> GetDirectStreamProviderByUniqueId(string uniqueId, CancellationToken cancellationToken)
|
public Task<IDirectStreamProvider?> GetDirectStreamProviderByUniqueId(string uniqueId, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
_logger.LogInformation("GetDirectStreamProviderByUniqueId called with uniqueId: {UniqueId}", uniqueId);
|
||||||
// Not needed for HTTP streams
|
// Not needed for HTTP streams
|
||||||
return Task.FromResult<IDirectStreamProvider?>(null);
|
return Task.FromResult<IDirectStreamProvider?>(null);
|
||||||
}
|
}
|
||||||
@ -214,6 +263,7 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public Task<ILiveStream> OpenMediaSource(string openToken, List<ILiveStream> currentLiveStreams, CancellationToken cancellationToken)
|
public Task<ILiveStream> OpenMediaSource(string openToken, List<ILiveStream> currentLiveStreams, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
_logger.LogWarning("OpenMediaSource called with openToken: {OpenToken} - This should not be called for HTTP streams!", openToken);
|
||||||
// Not needed for static HTTP streams
|
// Not needed for static HTTP streams
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
}
|
}
|
||||||
|
|||||||
@ -25,6 +25,7 @@ public class ServiceRegistrator : IPluginServiceRegistrator
|
|||||||
serviceCollection.AddSingleton<ContentExpirationService>();
|
serviceCollection.AddSingleton<ContentExpirationService>();
|
||||||
serviceCollection.AddSingleton<ContentRefreshService>();
|
serviceCollection.AddSingleton<ContentRefreshService>();
|
||||||
serviceCollection.AddSingleton<CategoryService>();
|
serviceCollection.AddSingleton<CategoryService>();
|
||||||
|
serviceCollection.AddSingleton<StreamProxyService>(); // Stream proxy service
|
||||||
|
|
||||||
// Register metadata providers
|
// Register metadata providers
|
||||||
serviceCollection.AddSingleton<SRFSeriesProvider>();
|
serviceCollection.AddSingleton<SRFSeriesProvider>();
|
||||||
|
|||||||
250
Jellyfin.Plugin.SRFPlay/Services/StreamProxyService.cs
Normal file
250
Jellyfin.Plugin.SRFPlay/Services/StreamProxyService.cs
Normal file
@ -0,0 +1,250 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Service for proxying SRF Play streams and managing authentication.
|
||||||
|
/// </summary>
|
||||||
|
public class StreamProxyService : IDisposable
|
||||||
|
{
|
||||||
|
private readonly ILogger<StreamProxyService> _logger;
|
||||||
|
private readonly StreamUrlResolver _streamResolver;
|
||||||
|
private readonly HttpClient _httpClient;
|
||||||
|
private readonly ConcurrentDictionary<string, StreamInfo> _streamMappings;
|
||||||
|
private bool _disposed;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="StreamProxyService"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="logger">The logger.</param>
|
||||||
|
/// <param name="streamResolver">The stream URL resolver.</param>
|
||||||
|
public StreamProxyService(ILogger<StreamProxyService> logger, StreamUrlResolver streamResolver)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
_streamResolver = streamResolver;
|
||||||
|
_httpClient = new HttpClient
|
||||||
|
{
|
||||||
|
Timeout = TimeSpan.FromSeconds(30)
|
||||||
|
};
|
||||||
|
_streamMappings = new ConcurrentDictionary<string, StreamInfo>();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Registers a stream for proxying.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="authenticatedUrl">The authenticated stream URL.</param>
|
||||||
|
public void RegisterStream(string itemId, string authenticatedUrl)
|
||||||
|
{
|
||||||
|
var streamInfo = new StreamInfo
|
||||||
|
{
|
||||||
|
AuthenticatedUrl = authenticatedUrl,
|
||||||
|
RegisteredAt = DateTime.UtcNow
|
||||||
|
};
|
||||||
|
|
||||||
|
_streamMappings.AddOrUpdate(itemId, streamInfo, (key, old) => streamInfo);
|
||||||
|
_logger.LogInformation("Registered stream for item {ItemId}: {Url}", itemId, authenticatedUrl);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the authenticated URL for an item.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <returns>The authenticated URL, or null if not found.</returns>
|
||||||
|
public string? GetAuthenticatedUrl(string itemId)
|
||||||
|
{
|
||||||
|
if (_streamMappings.TryGetValue(itemId, out var streamInfo))
|
||||||
|
{
|
||||||
|
return streamInfo.AuthenticatedUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogWarning("No stream mapping found for item {ItemId}", itemId);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fetches and rewrites an HLS manifest to use proxy URLs.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="baseProxyUrl">The base proxy URL (e.g., https://jellyfin-server/Plugins/SRFPlay/Proxy/{itemId}).</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The rewritten manifest content.</returns>
|
||||||
|
public async Task<string?> GetRewrittenManifestAsync(
|
||||||
|
string itemId,
|
||||||
|
string baseProxyUrl,
|
||||||
|
CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
var authenticatedUrl = GetAuthenticatedUrl(itemId);
|
||||||
|
if (authenticatedUrl == null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
_logger.LogDebug("Fetching manifest from: {Url}", authenticatedUrl);
|
||||||
|
var manifestContent = await _httpClient.GetStringAsync(authenticatedUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Rewrite the manifest to replace Akamai URLs with proxy URLs
|
||||||
|
var rewrittenContent = RewriteManifestUrls(manifestContent, authenticatedUrl, baseProxyUrl);
|
||||||
|
|
||||||
|
_logger.LogDebug("Successfully rewrote manifest for item {ItemId}", itemId);
|
||||||
|
return rewrittenContent;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Failed to fetch manifest for item {ItemId} from {Url}", itemId, authenticatedUrl);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fetches a segment from the original source.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="segmentPath">The segment path.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The segment content as bytes.</returns>
|
||||||
|
public async Task<byte[]?> GetSegmentAsync(
|
||||||
|
string itemId,
|
||||||
|
string segmentPath,
|
||||||
|
CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
var authenticatedUrl = GetAuthenticatedUrl(itemId);
|
||||||
|
if (authenticatedUrl == null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Build the full segment URL by combining the base URL with the segment path
|
||||||
|
var baseUri = new Uri(authenticatedUrl);
|
||||||
|
var baseUrl = $"{baseUri.Scheme}://{baseUri.Host}{string.Join('/', baseUri.AbsolutePath.Split('/')[..^1])}";
|
||||||
|
|
||||||
|
// Extract query parameters (auth tokens) from authenticated URL
|
||||||
|
var queryParams = baseUri.Query;
|
||||||
|
|
||||||
|
// Build full segment URL
|
||||||
|
var segmentUrl = $"{baseUrl}/{segmentPath}{queryParams}";
|
||||||
|
|
||||||
|
_logger.LogDebug("Fetching segment: {SegmentUrl}", segmentUrl);
|
||||||
|
var segmentData = await _httpClient.GetByteArrayAsync(segmentUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
_logger.LogDebug("Successfully fetched segment {SegmentPath} ({Size} bytes)", segmentPath, segmentData.Length);
|
||||||
|
return segmentData;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Failed to fetch segment {SegmentPath} for item {ItemId}", segmentPath, itemId);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Rewrites URLs in HLS manifest to point to proxy.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="manifestContent">The original manifest content.</param>
|
||||||
|
/// <param name="originalBaseUrl">The original base URL.</param>
|
||||||
|
/// <param name="proxyBaseUrl">The proxy base URL.</param>
|
||||||
|
/// <returns>The rewritten manifest.</returns>
|
||||||
|
private string RewriteManifestUrls(string manifestContent, string originalBaseUrl, string proxyBaseUrl)
|
||||||
|
{
|
||||||
|
var baseUri = new Uri(originalBaseUrl);
|
||||||
|
var baseUrl = $"{baseUri.Scheme}://{baseUri.Host}{string.Join('/', baseUri.AbsolutePath.Split('/')[..^1])}";
|
||||||
|
|
||||||
|
// Pattern to match .m3u8 and .ts/.mp4 segment references
|
||||||
|
var pattern = @"(?:^|\n)([^#\n][^\n]*\.(?:m3u8|ts|mp4|m4s|aac)[^\n]*)";
|
||||||
|
|
||||||
|
var rewritten = Regex.Replace(manifestContent, pattern, match =>
|
||||||
|
{
|
||||||
|
var url = match.Groups[1].Value.Trim();
|
||||||
|
|
||||||
|
// Skip if it's already an absolute URL
|
||||||
|
if (url.StartsWith("http://", StringComparison.OrdinalIgnoreCase) ||
|
||||||
|
url.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
// Rewrite absolute URLs to proxy
|
||||||
|
var relativePath = url.Replace(baseUrl + "/", string.Empty, StringComparison.Ordinal);
|
||||||
|
return $"\n{proxyBaseUrl}/{relativePath}";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Relative URL - rewrite to proxy
|
||||||
|
return $"\n{proxyBaseUrl}/{url}";
|
||||||
|
});
|
||||||
|
|
||||||
|
return rewritten;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Cleans up old stream mappings.
|
||||||
|
/// </summary>
|
||||||
|
public void CleanupOldMappings()
|
||||||
|
{
|
||||||
|
var cutoff = DateTime.UtcNow.AddHours(-24);
|
||||||
|
var keysToRemove = new System.Collections.Generic.List<string>();
|
||||||
|
|
||||||
|
foreach (var kvp in _streamMappings)
|
||||||
|
{
|
||||||
|
if (kvp.Value.RegisteredAt < cutoff)
|
||||||
|
{
|
||||||
|
keysToRemove.Add(kvp.Key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var key in keysToRemove)
|
||||||
|
{
|
||||||
|
_streamMappings.TryRemove(key, out _);
|
||||||
|
_logger.LogDebug("Removed old stream mapping for item {ItemId}", key);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (keysToRemove.Count > 0)
|
||||||
|
{
|
||||||
|
_logger.LogInformation("Cleaned up {Count} old stream mappings", keysToRemove.Count);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Disposes the service.
|
||||||
|
/// </summary>
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
Dispose(true);
|
||||||
|
GC.SuppressFinalize(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Disposes the service.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="disposing">True if disposing.</param>
|
||||||
|
protected virtual void Dispose(bool disposing)
|
||||||
|
{
|
||||||
|
if (_disposed)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (disposing)
|
||||||
|
{
|
||||||
|
_httpClient?.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
_disposed = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Stream information.
|
||||||
|
/// </summary>
|
||||||
|
private sealed class StreamInfo
|
||||||
|
{
|
||||||
|
public string AuthenticatedUrl { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
public DateTime RegisteredAt { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user