Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d313b68975 | ||
| 60434abd01 | |||
|
|
5ace3f4296 | ||
| 757aab1943 | |||
|
|
23d8da9ae7 | ||
| 2631c93444 | |||
| 7f71d3419c | |||
| dbbdd7eb6d | |||
| 9146830546 | |||
| 0548fe7dec | |||
| 198fc4c58d | |||
| ed4cc0990c | |||
| 4f9ebe2bce | |||
| 14b6c33542 | |||
| a0e7663323 | |||
| 0fea57a4f9 | |||
| 89c41842a7 | |||
| 89a911b9c4 | |||
| b8ac466c90 | |||
| e26f2a2ab1 |
@ -70,6 +70,13 @@ jobs:
|
|||||||
echo "artifact_name=${ARTIFACT_NAME}" >> $GITHUB_OUTPUT
|
echo "artifact_name=${ARTIFACT_NAME}" >> $GITHUB_OUTPUT
|
||||||
echo "Found artifact: ${ARTIFACT}"
|
echo "Found artifact: ${ARTIFACT}"
|
||||||
|
|
||||||
|
- name: Calculate checksum
|
||||||
|
id: checksum
|
||||||
|
run: |
|
||||||
|
CHECKSUM=$(md5sum "${{ steps.jprm.outputs.artifact }}" | awk '{print $1}')
|
||||||
|
echo "checksum=${CHECKSUM}" >> $GITHUB_OUTPUT
|
||||||
|
echo "Checksum: ${CHECKSUM}"
|
||||||
|
|
||||||
- name: Create Release
|
- name: Create Release
|
||||||
env:
|
env:
|
||||||
GITEA_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITEA_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
@ -126,3 +133,38 @@ jobs:
|
|||||||
|
|
||||||
echo "✅ Release created successfully!"
|
echo "✅ Release created successfully!"
|
||||||
echo "View at: ${GITEA_URL}/${REPO_OWNER}/${REPO_NAME}/releases/tag/${{ steps.get_version.outputs.version }}"
|
echo "View at: ${GITEA_URL}/${REPO_OWNER}/${REPO_NAME}/releases/tag/${{ steps.get_version.outputs.version }}"
|
||||||
|
|
||||||
|
- name: Update manifest.json
|
||||||
|
env:
|
||||||
|
GITEA_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
run: |
|
||||||
|
REPO_OWNER="${{ github.repository_owner }}"
|
||||||
|
REPO_NAME="${{ github.event.repository.name }}"
|
||||||
|
GITEA_URL="${{ github.server_url }}"
|
||||||
|
VERSION="${{ steps.get_version.outputs.version_number }}"
|
||||||
|
CHECKSUM="${{ steps.checksum.outputs.checksum }}"
|
||||||
|
ARTIFACT_NAME="${{ steps.jprm.outputs.artifact_name }}"
|
||||||
|
TIMESTAMP=$(date -u +"%Y-%m-%dT%H:%M:%SZ")
|
||||||
|
DOWNLOAD_URL="${GITEA_URL}/${REPO_OWNER}/${REPO_NAME}/releases/download/${{ steps.get_version.outputs.version }}/${ARTIFACT_NAME}"
|
||||||
|
|
||||||
|
git config user.name "Gitea Actions"
|
||||||
|
git config user.email "actions@gitea.tourolle.paris"
|
||||||
|
git fetch origin master
|
||||||
|
git checkout master
|
||||||
|
|
||||||
|
NEW_VERSION=$(cat <<EOF
|
||||||
|
{
|
||||||
|
"version": "${VERSION}",
|
||||||
|
"changelog": "Release ${VERSION}",
|
||||||
|
"targetAbi": "10.9.0.0",
|
||||||
|
"sourceUrl": "${DOWNLOAD_URL}",
|
||||||
|
"checksum": "${CHECKSUM}",
|
||||||
|
"timestamp": "${TIMESTAMP}"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
|
||||||
|
jq --argjson newver "${NEW_VERSION}" '.[0].versions = [$newver] + .[0].versions' manifest.json > manifest.tmp && mv manifest.tmp manifest.json
|
||||||
|
git add manifest.json
|
||||||
|
git commit -m "Update manifest.json for version ${VERSION}"
|
||||||
|
git push origin master
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@ -3,3 +3,4 @@ obj/
|
|||||||
.vs/
|
.vs/
|
||||||
.idea/
|
.idea/
|
||||||
artifacts
|
artifacts
|
||||||
|
*.log
|
||||||
|
|||||||
@ -12,6 +12,14 @@ using Microsoft.Extensions.Logging;
|
|||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Tests
|
namespace Jellyfin.Plugin.SRFPlay.Tests
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Simple IHttpClientFactory implementation for tests.
|
||||||
|
/// </summary>
|
||||||
|
internal sealed class TestHttpClientFactory : IHttpClientFactory
|
||||||
|
{
|
||||||
|
public HttpClient CreateClient(string name) => new HttpClient();
|
||||||
|
}
|
||||||
|
|
||||||
class Program
|
class Program
|
||||||
{
|
{
|
||||||
static async Task Main(string[] args)
|
static async Task Main(string[] args)
|
||||||
@ -33,8 +41,9 @@ namespace Jellyfin.Plugin.SRFPlay.Tests
|
|||||||
builder.SetMinimumLevel(LogLevel.Warning); // Only show warnings and errors
|
builder.SetMinimumLevel(LogLevel.Warning); // Only show warnings and errors
|
||||||
});
|
});
|
||||||
|
|
||||||
|
var httpClientFactory = new TestHttpClientFactory();
|
||||||
var apiClient = new SRFApiClient(loggerFactory);
|
var apiClient = new SRFApiClient(loggerFactory);
|
||||||
var streamResolver = new StreamUrlResolver(loggerFactory.CreateLogger<StreamUrlResolver>());
|
var streamResolver = new StreamUrlResolver(loggerFactory.CreateLogger<StreamUrlResolver>(), httpClientFactory);
|
||||||
|
|
||||||
var cancellationToken = CancellationToken.None;
|
var cancellationToken = CancellationToken.None;
|
||||||
|
|
||||||
|
|||||||
13
Jellyfin.Plugin.SRFPlay/Api/ISRFApiClientFactory.cs
Normal file
13
Jellyfin.Plugin.SRFPlay/Api/ISRFApiClientFactory.cs
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
namespace Jellyfin.Plugin.SRFPlay.Api;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Factory interface for creating SRF API clients.
|
||||||
|
/// </summary>
|
||||||
|
public interface ISRFApiClientFactory
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a new instance of the SRF API client.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A new SRFApiClient instance.</returns>
|
||||||
|
SRFApiClient CreateClient();
|
||||||
|
}
|
||||||
@ -1,16 +0,0 @@
|
|||||||
using System.Collections.Generic;
|
|
||||||
using System.Text.Json.Serialization;
|
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Api.Models.PlayV3;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// TV Program Guide response from Play v3 API.
|
|
||||||
/// </summary>
|
|
||||||
public class PlayV3TvProgramGuideResponse
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Gets the list of TV program entries.
|
|
||||||
/// </summary>
|
|
||||||
[JsonPropertyName("data")]
|
|
||||||
public IReadOnlyList<PlayV3TvProgram>? Data { get; init; }
|
|
||||||
}
|
|
||||||
@ -2,12 +2,15 @@ using System;
|
|||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
|
using System.Text;
|
||||||
|
using System.Text.Encodings.Web;
|
||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api.Models.PlayV3;
|
using Jellyfin.Plugin.SRFPlay.Api.Models.PlayV3;
|
||||||
using Jellyfin.Plugin.SRFPlay.Configuration;
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Constants;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Api;
|
namespace Jellyfin.Plugin.SRFPlay.Api;
|
||||||
@ -17,9 +20,7 @@ namespace Jellyfin.Plugin.SRFPlay.Api;
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class SRFApiClient : IDisposable
|
public class SRFApiClient : IDisposable
|
||||||
{
|
{
|
||||||
private const string BaseUrl = "https://il.srgssr.ch/integrationlayer/2.0";
|
private static readonly System.Text.CompositeFormat PlayV3UrlFormat = System.Text.CompositeFormat.Parse(ApiEndpoints.PlayV3BaseUrlTemplate);
|
||||||
private const string PlayV3BaseUrlTemplate = "https://www.{0}.ch/play/v3/api/{0}/production/";
|
|
||||||
private static readonly System.Text.CompositeFormat PlayV3UrlFormat = System.Text.CompositeFormat.Parse(PlayV3BaseUrlTemplate);
|
|
||||||
private readonly HttpClient _httpClient;
|
private readonly HttpClient _httpClient;
|
||||||
private readonly HttpClient _playV3HttpClient;
|
private readonly HttpClient _playV3HttpClient;
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
@ -47,15 +48,28 @@ public class SRFApiClient : IDisposable
|
|||||||
_logger.LogInformation("SRFApiClient initializing without proxy");
|
_logger.LogInformation("SRFApiClient initializing without proxy");
|
||||||
}
|
}
|
||||||
|
|
||||||
_httpClient = CreateHttpClient(BaseUrl);
|
_httpClient = CreateHttpClient(ApiEndpoints.IntegrationLayerBaseUrl);
|
||||||
_playV3HttpClient = CreateHttpClient(null);
|
_playV3HttpClient = CreateHttpClient(null);
|
||||||
|
|
||||||
_jsonOptions = new JsonSerializerOptions
|
_jsonOptions = new JsonSerializerOptions
|
||||||
{
|
{
|
||||||
PropertyNameCaseInsensitive = true
|
PropertyNameCaseInsensitive = true,
|
||||||
|
Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Reads HTTP response content as UTF-8 string.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="content">The HTTP content to read.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>The content as a UTF-8 decoded string.</returns>
|
||||||
|
private async Task<string> ReadAsUtf8StringAsync(HttpContent content, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
var bytes = await content.ReadAsByteArrayAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
return Encoding.UTF8.GetString(bytes);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates an HttpClient with optional proxy configuration.
|
/// Creates an HttpClient with optional proxy configuration.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -111,6 +125,7 @@ public class SRFApiClient : IDisposable
|
|||||||
client.DefaultRequestHeaders.UserAgent.ParseAdd("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36");
|
client.DefaultRequestHeaders.UserAgent.ParseAdd("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36");
|
||||||
client.DefaultRequestHeaders.Accept.ParseAdd("*/*");
|
client.DefaultRequestHeaders.Accept.ParseAdd("*/*");
|
||||||
client.DefaultRequestHeaders.AcceptLanguage.ParseAdd("en-US,en;q=0.9");
|
client.DefaultRequestHeaders.AcceptLanguage.ParseAdd("en-US,en;q=0.9");
|
||||||
|
client.DefaultRequestHeaders.Add("Accept-Charset", "utf-8");
|
||||||
|
|
||||||
_logger.LogInformation(
|
_logger.LogInformation(
|
||||||
"HttpClient created with HTTP/1.1 and headers - User-Agent: {UserAgent}, Accept: {Accept}, Accept-Language: {AcceptLanguage}",
|
"HttpClient created with HTTP/1.1 and headers - User-Agent: {UserAgent}, Accept: {Accept}, Accept-Language: {AcceptLanguage}",
|
||||||
@ -136,58 +151,10 @@ public class SRFApiClient : IDisposable
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var url = $"/mediaComposition/byUrn/{urn}.json";
|
var fullUrl = $"{ApiEndpoints.IntegrationLayerBaseUrl}/mediaComposition/byUrn/{urn}.json";
|
||||||
var fullUrl = $"{BaseUrl}{url}";
|
_logger.LogDebug("Fetching media composition for URN: {Urn} from {Url}", urn, fullUrl);
|
||||||
_logger.LogInformation("Fetching media composition for URN: {Urn} from {Url}", urn, fullUrl);
|
|
||||||
|
|
||||||
var response = await _httpClient.GetAsync(url, cancellationToken).ConfigureAwait(false);
|
// Use curl - HttpClient returns 404 due to server routing/network configuration
|
||||||
|
|
||||||
// Log response headers to diagnose geo-blocking
|
|
||||||
var xLocation = response.Headers.Contains("x-location")
|
|
||||||
? string.Join(", ", response.Headers.GetValues("x-location"))
|
|
||||||
: "not present";
|
|
||||||
|
|
||||||
_logger.LogInformation(
|
|
||||||
"Media composition response for URN {Urn}: StatusCode={StatusCode}, x-location={XLocation}",
|
|
||||||
urn,
|
|
||||||
response.StatusCode,
|
|
||||||
xLocation);
|
|
||||||
|
|
||||||
// If HttpClient fails, try curl as fallback
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
{
|
|
||||||
_logger.LogWarning("HttpClient failed with {StatusCode}, trying curl fallback", response.StatusCode);
|
|
||||||
return await FetchWithCurlAsync(fullUrl, cancellationToken).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
response.EnsureSuccessStatusCode();
|
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
|
||||||
var result = JsonSerializer.Deserialize<MediaComposition>(content, _jsonOptions);
|
|
||||||
|
|
||||||
if (result?.ChapterList != null && result.ChapterList.Count > 0)
|
|
||||||
{
|
|
||||||
_logger.LogInformation(
|
|
||||||
"Successfully fetched media composition for URN: {Urn} - Chapters: {ChapterCount}",
|
|
||||||
urn,
|
|
||||||
result.ChapterList.Count);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogWarning("Media composition for URN {Urn} has no chapters", urn);
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
catch (HttpRequestException ex)
|
|
||||||
{
|
|
||||||
_logger.LogError(
|
|
||||||
ex,
|
|
||||||
"HTTP error fetching media composition for URN: {Urn} - StatusCode: {StatusCode}, trying curl fallback",
|
|
||||||
urn,
|
|
||||||
ex.StatusCode);
|
|
||||||
|
|
||||||
var fullUrl = $"{BaseUrl}/mediaComposition/byUrn/{urn}.json";
|
|
||||||
return await FetchWithCurlAsync(fullUrl, cancellationToken).ConfigureAwait(false);
|
return await FetchWithCurlAsync(fullUrl, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@ -233,7 +200,8 @@ public class SRFApiClient : IDisposable
|
|||||||
RedirectStandardOutput = true,
|
RedirectStandardOutput = true,
|
||||||
RedirectStandardError = true,
|
RedirectStandardError = true,
|
||||||
UseShellExecute = false,
|
UseShellExecute = false,
|
||||||
CreateNoWindow = true
|
CreateNoWindow = true,
|
||||||
|
StandardOutputEncoding = Encoding.UTF8
|
||||||
};
|
};
|
||||||
|
|
||||||
using var process = new System.Diagnostics.Process { StartInfo = processStartInfo };
|
using var process = new System.Diagnostics.Process { StartInfo = processStartInfo };
|
||||||
@ -293,12 +261,12 @@ public class SRFApiClient : IDisposable
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var errorContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var errorContent = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogDebug("Latest videos response length: {Length}", content.Length);
|
_logger.LogDebug("Latest videos response length: {Length}", content.Length);
|
||||||
|
|
||||||
var result = JsonSerializer.Deserialize<MediaComposition>(content, _jsonOptions);
|
var result = JsonSerializer.Deserialize<MediaComposition>(content, _jsonOptions);
|
||||||
@ -332,12 +300,12 @@ public class SRFApiClient : IDisposable
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var errorContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var errorContent = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogDebug("Trending videos response length: {Length}", content.Length);
|
_logger.LogDebug("Trending videos response length: {Length}", content.Length);
|
||||||
|
|
||||||
var result = JsonSerializer.Deserialize<MediaComposition>(content, _jsonOptions);
|
var result = JsonSerializer.Deserialize<MediaComposition>(content, _jsonOptions);
|
||||||
@ -367,7 +335,7 @@ public class SRFApiClient : IDisposable
|
|||||||
var response = await _httpClient.GetAsync(url, cancellationToken).ConfigureAwait(false);
|
var response = await _httpClient.GetAsync(url, cancellationToken).ConfigureAwait(false);
|
||||||
response.EnsureSuccessStatusCode();
|
response.EnsureSuccessStatusCode();
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
return content;
|
return content;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@ -395,12 +363,12 @@ public class SRFApiClient : IDisposable
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var errorContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var errorContent = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
var result = JsonSerializer.Deserialize<PlayV3DirectResponse<PlayV3Show>>(content, _jsonOptions);
|
var result = JsonSerializer.Deserialize<PlayV3DirectResponse<PlayV3Show>>(content, _jsonOptions);
|
||||||
|
|
||||||
_logger.LogInformation("Successfully fetched {Count} shows for business unit: {BusinessUnit}", result?.Data?.Count ?? 0, businessUnit);
|
_logger.LogInformation("Successfully fetched {Count} shows for business unit: {BusinessUnit}", result?.Data?.Count ?? 0, businessUnit);
|
||||||
@ -431,12 +399,12 @@ public class SRFApiClient : IDisposable
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var errorContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var errorContent = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
var result = JsonSerializer.Deserialize<PlayV3DirectResponse<PlayV3Topic>>(content, _jsonOptions);
|
var result = JsonSerializer.Deserialize<PlayV3DirectResponse<PlayV3Topic>>(content, _jsonOptions);
|
||||||
|
|
||||||
_logger.LogInformation("Successfully fetched {Count} topics for business unit: {BusinessUnit}", result?.Data?.Count ?? 0, businessUnit);
|
_logger.LogInformation("Successfully fetched {Count} topics for business unit: {BusinessUnit}", result?.Data?.Count ?? 0, businessUnit);
|
||||||
@ -468,12 +436,12 @@ public class SRFApiClient : IDisposable
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var errorContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var errorContent = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
var result = JsonSerializer.Deserialize<PlayV3Response<PlayV3Video>>(content, _jsonOptions);
|
var result = JsonSerializer.Deserialize<PlayV3Response<PlayV3Video>>(content, _jsonOptions);
|
||||||
|
|
||||||
_logger.LogDebug("Successfully fetched {Count} videos for show {ShowId}", result?.Data?.Data?.Count ?? 0, showId);
|
_logger.LogDebug("Successfully fetched {Count} videos for show {ShowId}", result?.Data?.Data?.Count ?? 0, showId);
|
||||||
@ -508,12 +476,12 @@ public class SRFApiClient : IDisposable
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
var errorContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var errorContent = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
_logger.LogError("API returned error {StatusCode}: {Error}", response.StatusCode, errorContent);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var content = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var content = await ReadAsUtf8StringAsync(response.Content, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
// The response structure is: { "data": { "scheduledLivestreams": [...] } }
|
// The response structure is: { "data": { "scheduledLivestreams": [...] } }
|
||||||
var jsonDoc = JsonSerializer.Deserialize<JsonElement>(content, _jsonOptions);
|
var jsonDoc = JsonSerializer.Deserialize<JsonElement>(content, _jsonOptions);
|
||||||
|
|||||||
26
Jellyfin.Plugin.SRFPlay/Api/SRFApiClientFactory.cs
Normal file
26
Jellyfin.Plugin.SRFPlay/Api/SRFApiClientFactory.cs
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Api;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Factory for creating SRF API clients.
|
||||||
|
/// </summary>
|
||||||
|
public class SRFApiClientFactory : ISRFApiClientFactory
|
||||||
|
{
|
||||||
|
private readonly ILoggerFactory _loggerFactory;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="SRFApiClientFactory"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
|
public SRFApiClientFactory(ILoggerFactory loggerFactory)
|
||||||
|
{
|
||||||
|
_loggerFactory = loggerFactory;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public SRFApiClient CreateClient()
|
||||||
|
{
|
||||||
|
return new SRFApiClient(_loggerFactory);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,11 +2,14 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Security.Cryptography;
|
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
using Jellyfin.Plugin.SRFPlay.Api;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Constants;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
|
using MediaBrowser.Controller;
|
||||||
using MediaBrowser.Controller.Channels;
|
using MediaBrowser.Controller.Channels;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
using MediaBrowser.Model.Channels;
|
using MediaBrowser.Model.Channels;
|
||||||
@ -22,11 +25,11 @@ namespace Jellyfin.Plugin.SRFPlay.Channels;
|
|||||||
public class SRFPlayChannel : IChannel, IHasCacheKey
|
public class SRFPlayChannel : IChannel, IHasCacheKey
|
||||||
{
|
{
|
||||||
private readonly ILogger<SRFPlayChannel> _logger;
|
private readonly ILogger<SRFPlayChannel> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly IContentRefreshService _contentRefreshService;
|
||||||
private readonly ContentRefreshService _contentRefreshService;
|
private readonly IStreamUrlResolver _streamResolver;
|
||||||
private readonly StreamUrlResolver _streamResolver;
|
private readonly IMediaSourceFactory _mediaSourceFactory;
|
||||||
private readonly StreamProxyService _proxyService;
|
private readonly ICategoryService? _categoryService;
|
||||||
private readonly CategoryService? _categoryService;
|
private readonly ISRFApiClientFactory _apiClientFactory;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="SRFPlayChannel"/> class.
|
/// Initializes a new instance of the <see cref="SRFPlayChannel"/> class.
|
||||||
@ -34,28 +37,30 @@ 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="mediaSourceFactory">The media source factory.</param>
|
||||||
/// <param name="categoryService">The category service (optional).</param>
|
/// <param name="categoryService">The category service (optional).</param>
|
||||||
|
/// <param name="apiClientFactory">The API client factory.</param>
|
||||||
public SRFPlayChannel(
|
public SRFPlayChannel(
|
||||||
ILoggerFactory loggerFactory,
|
ILoggerFactory loggerFactory,
|
||||||
ContentRefreshService contentRefreshService,
|
IContentRefreshService contentRefreshService,
|
||||||
StreamUrlResolver streamResolver,
|
IStreamUrlResolver streamResolver,
|
||||||
StreamProxyService proxyService,
|
IMediaSourceFactory mediaSourceFactory,
|
||||||
CategoryService? categoryService = null)
|
ICategoryService? categoryService,
|
||||||
|
ISRFApiClientFactory apiClientFactory)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
|
||||||
_logger = loggerFactory.CreateLogger<SRFPlayChannel>();
|
_logger = loggerFactory.CreateLogger<SRFPlayChannel>();
|
||||||
_contentRefreshService = contentRefreshService;
|
_contentRefreshService = contentRefreshService;
|
||||||
_streamResolver = streamResolver;
|
_streamResolver = streamResolver;
|
||||||
_proxyService = proxyService;
|
_mediaSourceFactory = mediaSourceFactory;
|
||||||
_categoryService = categoryService;
|
_categoryService = categoryService;
|
||||||
|
_apiClientFactory = apiClientFactory;
|
||||||
|
|
||||||
if (_categoryService == null)
|
if (_categoryService == null)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("CategoryService not available - category folders will be disabled");
|
_logger.LogWarning("CategoryService not available - category folders will be disabled");
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogInformation("=== SRFPlayChannel constructor called! Channel is being instantiated ===");
|
_logger.LogDebug("SRFPlayChannel initialized");
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -68,7 +73,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
public string DataVersion => "2.0"; // Back to authenticating at channel refresh with auto-refresh for fresh tokens
|
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 => ApiEndpoints.SrfPlayHomepage;
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public ChannelParentalRating ParentalRating => ChannelParentalRating.GeneralAudience;
|
public ChannelParentalRating ParentalRating => ChannelParentalRating.GeneralAudience;
|
||||||
@ -76,7 +81,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public InternalChannelFeatures GetChannelFeatures()
|
public InternalChannelFeatures GetChannelFeatures()
|
||||||
{
|
{
|
||||||
_logger.LogInformation("=== GetChannelFeatures called for SRF Play channel ===");
|
_logger.LogDebug("GetChannelFeatures called");
|
||||||
|
|
||||||
return new InternalChannelFeatures
|
return new InternalChannelFeatures
|
||||||
{
|
{
|
||||||
@ -102,10 +107,19 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public Task<DynamicImageResponse> GetChannelImage(ImageType type, CancellationToken cancellationToken)
|
public Task<DynamicImageResponse> GetChannelImage(ImageType type, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
// Could provide a channel logo here
|
var assembly = GetType().Assembly;
|
||||||
|
var resourceStream = assembly.GetManifestResourceStream("Jellyfin.Plugin.SRFPlay.Images.logo.png");
|
||||||
|
|
||||||
|
if (resourceStream == null)
|
||||||
|
{
|
||||||
|
return Task.FromResult(new DynamicImageResponse { HasImage = false });
|
||||||
|
}
|
||||||
|
|
||||||
return Task.FromResult(new DynamicImageResponse
|
return Task.FromResult(new DynamicImageResponse
|
||||||
{
|
{
|
||||||
HasImage = false
|
HasImage = true,
|
||||||
|
Stream = resourceStream,
|
||||||
|
Format = MediaBrowser.Model.Drawing.ImageFormat.Png
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,254 +136,263 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public async Task<ChannelItemResult> GetChannelItems(InternalChannelItemQuery query, CancellationToken cancellationToken)
|
public async Task<ChannelItemResult> GetChannelItems(InternalChannelItemQuery query, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
_logger.LogInformation("=== GetChannelItems called! FolderId: {FolderId} ===", query.FolderId);
|
_logger.LogDebug("GetChannelItems called for folder {FolderId}", query.FolderId);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var items = await GetFolderItemsAsync(query.FolderId, cancellationToken).ConfigureAwait(false);
|
||||||
|
_logger.LogDebug("Returning {Count} channel items for folder {FolderId}", items.Count, query.FolderId);
|
||||||
|
|
||||||
|
return new ChannelItemResult
|
||||||
|
{
|
||||||
|
Items = items,
|
||||||
|
TotalRecordCount = items.Count
|
||||||
|
};
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error getting channel items for folder {FolderId}", query.FolderId);
|
||||||
|
return new ChannelItemResult { Items = new List<ChannelItemInfo>(), TotalRecordCount = 0 };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<ChannelItemInfo>> GetFolderItemsAsync(string? folderId, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
// Root level - show folder list
|
||||||
|
if (string.IsNullOrEmpty(folderId))
|
||||||
|
{
|
||||||
|
return await GetRootFoldersAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle known folder types
|
||||||
|
return folderId switch
|
||||||
|
{
|
||||||
|
"latest" => await GetLatestVideosAsync(cancellationToken).ConfigureAwait(false),
|
||||||
|
"trending" => await GetTrendingVideosAsync(cancellationToken).ConfigureAwait(false),
|
||||||
|
"live_sports" => await GetLiveSportsAsync(cancellationToken).ConfigureAwait(false),
|
||||||
|
_ when folderId.StartsWith("category_", StringComparison.Ordinal) => await GetCategoryVideosAsync(folderId, cancellationToken).ConfigureAwait(false),
|
||||||
|
_ => new List<ChannelItemInfo>()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<ChannelItemInfo>> GetRootFoldersAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var items = new List<ChannelItemInfo>
|
||||||
|
{
|
||||||
|
CreateFolder("latest", "Latest Videos"),
|
||||||
|
CreateFolder("trending", "Trending Videos"),
|
||||||
|
CreateFolder("live_sports", "Live Sports & Events")
|
||||||
|
};
|
||||||
|
|
||||||
|
// Add category folders if enabled
|
||||||
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
if (config?.EnableCategoryFolders == true && _categoryService != null)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var businessUnit = config.BusinessUnit.ToString().ToLowerInvariant();
|
||||||
|
var topics = await _categoryService.GetTopicsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
foreach (var topic in topics.Where(t => !string.IsNullOrEmpty(t.Id)))
|
||||||
|
{
|
||||||
|
if (config.EnabledTopics != null && config.EnabledTopics.Count > 0 && !config.EnabledTopics.Contains(topic.Id!))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
items.Add(CreateFolder($"category_{topic.Id}", topic.Title ?? topic.Id!, topic.Lead));
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation("Added {Count} category folders", topics.Count);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Failed to load category folders - continuing without categories");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ChannelItemInfo CreateFolder(string id, string name, string? overview = null)
|
||||||
|
{
|
||||||
|
return new ChannelItemInfo
|
||||||
|
{
|
||||||
|
Id = id,
|
||||||
|
Name = name,
|
||||||
|
Type = ChannelItemType.Folder,
|
||||||
|
FolderType = ChannelFolderType.Container,
|
||||||
|
ImageUrl = null,
|
||||||
|
Overview = overview
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<ChannelItemInfo>> GetLatestVideosAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var urns = await _contentRefreshService.RefreshLatestContentAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
return await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<ChannelItemInfo>> GetTrendingVideosAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var urns = await _contentRefreshService.RefreshTrendingContentAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
return await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<ChannelItemInfo>> GetLiveSportsAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
var items = new List<ChannelItemInfo>();
|
var items = new List<ChannelItemInfo>();
|
||||||
var config = Plugin.Instance?.Configuration;
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// Root level - show categories
|
var businessUnit = config?.BusinessUnit.ToString().ToLowerInvariant() ?? "srf";
|
||||||
if (string.IsNullOrEmpty(query.FolderId))
|
|
||||||
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
|
var scheduledLivestreams = await apiClient.GetScheduledLivestreamsAsync(businessUnit, "SPORT", cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (scheduledLivestreams == null)
|
||||||
{
|
{
|
||||||
items.Add(new ChannelItemInfo
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filter for upcoming/current events (within next 7 days)
|
||||||
|
var now = DateTime.UtcNow;
|
||||||
|
var weekFromNow = now.AddDays(7);
|
||||||
|
|
||||||
|
var upcomingEvents = scheduledLivestreams
|
||||||
|
.Where(p => p.Urn != null &&
|
||||||
|
!string.IsNullOrEmpty(p.Title) &&
|
||||||
|
p.ValidFrom != null &&
|
||||||
|
p.ValidFrom.Value.ToUniversalTime() <= weekFromNow &&
|
||||||
|
(p.ValidTo == null || p.ValidTo.Value.ToUniversalTime() > now))
|
||||||
|
.OrderBy(p => p.ValidFrom)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
_logger.LogInformation("Found {Count} scheduled live sports events", upcomingEvents.Count);
|
||||||
|
|
||||||
|
var urns = upcomingEvents.Select(e => e.Urn!).ToList();
|
||||||
|
items = await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Enhance items with scheduled time information
|
||||||
|
foreach (var item in items)
|
||||||
|
{
|
||||||
|
var matchingEvent = upcomingEvents.FirstOrDefault(e => item.ProviderIds.ContainsKey("SRF") && item.ProviderIds["SRF"] == e.Urn);
|
||||||
|
if (matchingEvent?.ValidFrom != null)
|
||||||
{
|
{
|
||||||
Id = "latest",
|
var eventTime = matchingEvent.ValidFrom.Value;
|
||||||
Name = "Latest Videos",
|
item.Name = $"[{eventTime:dd.MM HH:mm}] {matchingEvent.Title}";
|
||||||
Type = ChannelItemType.Folder,
|
item.PremiereDate = eventTime;
|
||||||
FolderType = ChannelFolderType.Container,
|
}
|
||||||
ImageUrl = null
|
}
|
||||||
});
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Failed to load live sports events");
|
||||||
|
}
|
||||||
|
|
||||||
items.Add(new ChannelItemInfo
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<List<ChannelItemInfo>> GetCategoryVideosAsync(string folderId, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var items = new List<ChannelItemInfo>();
|
||||||
|
|
||||||
|
if (_categoryService == null)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("CategoryService not available - cannot display category folder");
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
var topicId = folderId.Substring("category_".Length);
|
||||||
|
var businessUnit = config?.BusinessUnit.ToString().ToLowerInvariant() ?? "srf";
|
||||||
|
|
||||||
|
var shows = await _categoryService.GetShowsByTopicAsync(topicId, businessUnit, 20, cancellationToken).ConfigureAwait(false);
|
||||||
|
var urns = new List<string>();
|
||||||
|
|
||||||
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
|
|
||||||
|
foreach (var show in shows)
|
||||||
|
{
|
||||||
|
if (show.Id == null || cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
Id = "trending",
|
continue;
|
||||||
Name = "Trending Videos",
|
|
||||||
Type = ChannelItemType.Folder,
|
|
||||||
FolderType = ChannelFolderType.Container,
|
|
||||||
ImageUrl = null
|
|
||||||
});
|
|
||||||
|
|
||||||
items.Add(new ChannelItemInfo
|
|
||||||
{
|
|
||||||
Id = "live_sports",
|
|
||||||
Name = "Live Sports & Events",
|
|
||||||
Type = ChannelItemType.Folder,
|
|
||||||
FolderType = ChannelFolderType.Container,
|
|
||||||
ImageUrl = null
|
|
||||||
});
|
|
||||||
|
|
||||||
// Add category folders if enabled and CategoryService is available
|
|
||||||
if (config?.EnableCategoryFolders == true && _categoryService != null)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var businessUnit = config.BusinessUnit.ToString().ToLowerInvariant();
|
|
||||||
var topics = await _categoryService.GetTopicsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
foreach (var topic in topics.Where(t => !string.IsNullOrEmpty(t.Id)))
|
|
||||||
{
|
|
||||||
// Filter by enabled topics if configured
|
|
||||||
if (config.EnabledTopics != null && config.EnabledTopics.Count > 0 && !config.EnabledTopics.Contains(topic.Id!))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
items.Add(new ChannelItemInfo
|
|
||||||
{
|
|
||||||
Id = $"category_{topic.Id}",
|
|
||||||
Name = topic.Title ?? topic.Id!,
|
|
||||||
Type = ChannelItemType.Folder,
|
|
||||||
FolderType = ChannelFolderType.Container,
|
|
||||||
ImageUrl = null,
|
|
||||||
Overview = topic.Lead
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogInformation("Added {Count} category folders", topics.Count);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
_logger.LogError(ex, "Failed to load category folders - continuing without categories");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ChannelItemResult
|
|
||||||
{
|
|
||||||
Items = items,
|
|
||||||
TotalRecordCount = items.Count
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Latest videos
|
|
||||||
if (query.FolderId == "latest")
|
|
||||||
{
|
|
||||||
var urns = await _contentRefreshService.RefreshLatestContentAsync(cancellationToken).ConfigureAwait(false);
|
|
||||||
items = await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Trending videos
|
|
||||||
else if (query.FolderId == "trending")
|
|
||||||
{
|
|
||||||
var urns = await _contentRefreshService.RefreshTrendingContentAsync(cancellationToken).ConfigureAwait(false);
|
|
||||||
items = await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Live Sports & Events
|
|
||||||
else if (query.FolderId == "live_sports")
|
|
||||||
{
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var businessUnit = config?.BusinessUnit.ToString().ToLowerInvariant() ?? "srf";
|
var latestUrn = await GetLatestVideoUrnForShowAsync(apiClient, businessUnit, show, topicId, cancellationToken).ConfigureAwait(false);
|
||||||
|
if (latestUrn != null)
|
||||||
using var apiClient = new Api.SRFApiClient(_loggerFactory);
|
|
||||||
var scheduledLivestreams = await apiClient.GetScheduledLivestreamsAsync(businessUnit, "SPORT", cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (scheduledLivestreams != null)
|
|
||||||
{
|
{
|
||||||
// Filter for upcoming/current events (within next 7 days) that have URNs
|
urns.Add(latestUrn);
|
||||||
var now = DateTime.UtcNow;
|
|
||||||
var weekFromNow = now.AddDays(7);
|
|
||||||
|
|
||||||
var upcomingEvents = scheduledLivestreams
|
|
||||||
.Where(p => p.Urn != null &&
|
|
||||||
!string.IsNullOrEmpty(p.Title) &&
|
|
||||||
p.ValidFrom != null &&
|
|
||||||
p.ValidFrom.Value.ToUniversalTime() <= weekFromNow &&
|
|
||||||
(p.ValidTo == null || p.ValidTo.Value.ToUniversalTime() > now))
|
|
||||||
.OrderBy(p => p.ValidFrom)
|
|
||||||
.ToList();
|
|
||||||
|
|
||||||
_logger.LogInformation("Found {Count} scheduled live sports events", upcomingEvents.Count);
|
|
||||||
|
|
||||||
var urns = upcomingEvents.Select(e => e.Urn!).ToList();
|
|
||||||
items = await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
// Enhance items with scheduled time information
|
|
||||||
foreach (var item in items)
|
|
||||||
{
|
|
||||||
var matchingEvent = upcomingEvents.FirstOrDefault(e => item.ProviderIds.ContainsKey("SRF") && item.ProviderIds["SRF"] == e.Urn);
|
|
||||||
if (matchingEvent?.ValidFrom != null)
|
|
||||||
{
|
|
||||||
var eventTime = matchingEvent.ValidFrom.Value;
|
|
||||||
item.Name = $"[{eventTime:dd.MM HH:mm}] {matchingEvent.Title}";
|
|
||||||
item.PremiereDate = eventTime;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Failed to load live sports events");
|
_logger.LogWarning(ex, "Error fetching videos for show {ShowId} in category {TopicId}", show.Id, topicId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Category folder - show videos for this category
|
items = await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
||||||
else if (query.FolderId?.StartsWith("category_", StringComparison.Ordinal) == true)
|
_logger.LogInformation("Found {Count} videos for category {TopicId}", items.Count, topicId);
|
||||||
{
|
|
||||||
if (_categoryService == null)
|
|
||||||
{
|
|
||||||
_logger.LogWarning("CategoryService not available - cannot display category folder");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var topicId = query.FolderId.Substring("category_".Length);
|
|
||||||
var businessUnit = config?.BusinessUnit.ToString().ToLowerInvariant() ?? "srf";
|
|
||||||
|
|
||||||
var shows = await _categoryService.GetShowsByTopicAsync(topicId, businessUnit, 20, cancellationToken).ConfigureAwait(false);
|
|
||||||
var urns = new List<string>();
|
|
||||||
|
|
||||||
using var apiClient = new Api.SRFApiClient(_loggerFactory);
|
|
||||||
|
|
||||||
foreach (var show in shows)
|
|
||||||
{
|
|
||||||
if (show.Id == null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var videos = await apiClient.GetVideosForShowAsync(businessUnit, show.Id, cancellationToken).ConfigureAwait(false);
|
|
||||||
if (videos != null && videos.Count > 0)
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Category {TopicId}, Show {Show} ({ShowId}): Found {Count} videos", topicId, show.Title, show.Id, videos.Count);
|
|
||||||
|
|
||||||
// Filter to videos that are actually published and not expired
|
|
||||||
var now = DateTime.UtcNow;
|
|
||||||
var availableVideos = videos.Where(v =>
|
|
||||||
(v.ValidFrom == null || v.ValidFrom.Value.ToUniversalTime() <= now) &&
|
|
||||||
(v.ValidTo == null || v.ValidTo.Value.ToUniversalTime() > now)).ToList();
|
|
||||||
|
|
||||||
_logger.LogDebug("Category {TopicId}, Show {Show}: {AvailableCount} available out of {TotalCount} videos", topicId, show.Title, availableVideos.Count, videos.Count);
|
|
||||||
|
|
||||||
if (availableVideos.Count > 0)
|
|
||||||
{
|
|
||||||
// Get most recent available video from this show
|
|
||||||
var latestVideo = availableVideos.OrderByDescending(v => v.Date).FirstOrDefault();
|
|
||||||
if (latestVideo?.Urn != null)
|
|
||||||
{
|
|
||||||
urns.Add(latestVideo.Urn);
|
|
||||||
_logger.LogInformation(
|
|
||||||
"Category {TopicId}: Added video from show {Show}: {Title} (URN: {Urn}, Date: {Date}, ValidFrom: {ValidFrom}, ValidTo: {ValidTo})",
|
|
||||||
topicId,
|
|
||||||
show.Title,
|
|
||||||
latestVideo.Title,
|
|
||||||
latestVideo.Urn,
|
|
||||||
latestVideo.Date,
|
|
||||||
latestVideo.ValidFrom,
|
|
||||||
latestVideo.ValidTo);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogWarning("Category {TopicId}, Show {Show}: Latest available video has null URN", topicId, show.Title);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Category {TopicId}, Show {Show}: No available videos (all expired or not yet published)", topicId, show.Title);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Category {TopicId}, Show {Show} ({ShowId}): No videos returned from API", topicId, show.Title, show.Id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
_logger.LogWarning(ex, "Error fetching videos for show {ShowId} in category {TopicId}", show.Id, topicId);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
items = await ConvertUrnsToChannelItems(urns, cancellationToken).ConfigureAwait(false);
|
|
||||||
_logger.LogInformation("Found {Count} videos for category {TopicId}", items.Count, topicId);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
_logger.LogError(ex, "Failed to load category videos");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogInformation("Returning {Count} channel items for folder {FolderId}", items.Count, query.FolderId);
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Error getting channel items for folder {FolderId}", query.FolderId);
|
_logger.LogError(ex, "Failed to load category videos");
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ChannelItemResult
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<string?> GetLatestVideoUrnForShowAsync(
|
||||||
|
Api.SRFApiClient apiClient,
|
||||||
|
string businessUnit,
|
||||||
|
Api.Models.PlayV3Show show,
|
||||||
|
string topicId,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var videos = await apiClient.GetVideosForShowAsync(businessUnit, show.Id!, cancellationToken).ConfigureAwait(false);
|
||||||
|
if (videos == null || videos.Count == 0)
|
||||||
{
|
{
|
||||||
Items = items,
|
_logger.LogDebug("Category {TopicId}, Show {Show} ({ShowId}): No videos returned from API", topicId, show.Title, show.Id);
|
||||||
TotalRecordCount = items.Count
|
return null;
|
||||||
};
|
}
|
||||||
|
|
||||||
|
_logger.LogDebug("Category {TopicId}, Show {Show} ({ShowId}): Found {Count} videos", topicId, show.Title, show.Id, videos.Count);
|
||||||
|
|
||||||
|
// Filter to available videos
|
||||||
|
var now = DateTime.UtcNow;
|
||||||
|
var availableVideos = videos.Where(v =>
|
||||||
|
(v.ValidFrom == null || v.ValidFrom.Value.ToUniversalTime() <= now) &&
|
||||||
|
(v.ValidTo == null || v.ValidTo.Value.ToUniversalTime() > now)).ToList();
|
||||||
|
|
||||||
|
_logger.LogDebug("Category {TopicId}, Show {Show}: {AvailableCount} available out of {TotalCount} videos", topicId, show.Title, availableVideos.Count, videos.Count);
|
||||||
|
|
||||||
|
if (availableVideos.Count == 0)
|
||||||
|
{
|
||||||
|
_logger.LogDebug("Category {TopicId}, Show {Show}: No available videos (all expired or not yet published)", topicId, show.Title);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var latestVideo = availableVideos.OrderByDescending(v => v.Date).FirstOrDefault();
|
||||||
|
if (latestVideo?.Urn == null)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Category {TopicId}, Show {Show}: Latest available video has null URN", topicId, show.Title);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Category {TopicId}: Added video from show {Show}: {Title} (URN: {Urn}, Date: {Date})",
|
||||||
|
topicId,
|
||||||
|
show.Title,
|
||||||
|
latestVideo.Title,
|
||||||
|
latestVideo.Urn,
|
||||||
|
latestVideo.Date);
|
||||||
|
|
||||||
|
return latestVideo.Urn;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -379,8 +402,14 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
var enabledTopics = config?.EnabledTopics != null && config.EnabledTopics.Count > 0
|
var enabledTopics = config?.EnabledTopics != null && config.EnabledTopics.Count > 0
|
||||||
? string.Join(",", config.EnabledTopics)
|
? string.Join(",", config.EnabledTopics)
|
||||||
: "all";
|
: "all";
|
||||||
var date = DateTime.Now.ToString("yyyy-MM-dd", CultureInfo.InvariantCulture);
|
|
||||||
return $"{config?.BusinessUnit}_{config?.EnableLatestContent}_{config?.EnableTrendingContent}_{config?.EnableCategoryFolders}_{enabledTopics}_{date}";
|
// Use 15-minute time buckets for cache key so live content refreshes frequently
|
||||||
|
// This ensures livestream folders update as programs start/end throughout the day
|
||||||
|
var now = DateTime.Now;
|
||||||
|
var timeBucket = new DateTime(now.Year, now.Month, now.Day, now.Hour, (now.Minute / 15) * 15, 0);
|
||||||
|
var timeKey = timeBucket.ToString("yyyy-MM-dd-HH-mm", CultureInfo.InvariantCulture);
|
||||||
|
|
||||||
|
return $"{config?.BusinessUnit}_{config?.EnableLatestContent}_{config?.EnableTrendingContent}_{config?.EnableCategoryFolders}_{enabledTopics}_{timeKey}";
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<List<ChannelItemInfo>> ConvertUrnsToChannelItems(List<string> urns, CancellationToken cancellationToken)
|
private async Task<List<ChannelItemInfo>> ConvertUrnsToChannelItems(List<string> urns, CancellationToken cancellationToken)
|
||||||
@ -396,7 +425,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
|
|
||||||
_logger.LogInformation("Converting {Count} URNs to channel items", urns.Count);
|
_logger.LogInformation("Converting {Count} URNs to channel items", urns.Count);
|
||||||
|
|
||||||
using var apiClient = new Api.SRFApiClient(_loggerFactory);
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
int successCount = 0;
|
int successCount = 0;
|
||||||
int failedCount = 0;
|
int failedCount = 0;
|
||||||
int expiredCount = 0;
|
int expiredCount = 0;
|
||||||
@ -435,60 +464,71 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Generate deterministic GUID from URN
|
// Generate deterministic GUID from URN
|
||||||
var itemId = UrnToGuid(urn);
|
var itemId = UrnHelper.ToGuid(urn);
|
||||||
|
|
||||||
// Get stream URL and authenticate it
|
// Use factory to create MediaSourceInfo (handles stream URL, auth, proxy registration)
|
||||||
var streamUrl = _streamResolver.GetStreamUrl(chapter, config.QualityPreference);
|
var mediaSource = await _mediaSourceFactory.CreateMediaSourceAsync(
|
||||||
|
chapter,
|
||||||
|
itemId,
|
||||||
|
urn,
|
||||||
|
config.QualityPreference,
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
// Skip scheduled livestreams that haven't started yet (no stream URL available)
|
// Skip items without a valid media source (no stream URL available)
|
||||||
if (chapter.Type == "SCHEDULED_LIVESTREAM" && string.IsNullOrEmpty(streamUrl))
|
if (mediaSource == null)
|
||||||
{
|
{
|
||||||
_logger.LogDebug(
|
if (chapter.Type == "SCHEDULED_LIVESTREAM")
|
||||||
"URN {Urn}: Skipping upcoming livestream '{Title}' - stream not yet available (starts at {ValidFrom})",
|
{
|
||||||
urn,
|
_logger.LogDebug(
|
||||||
chapter.Title,
|
"URN {Urn}: Skipping upcoming livestream '{Title}' - stream not yet available (starts at {ValidFrom})",
|
||||||
chapter.ValidFrom);
|
urn,
|
||||||
|
chapter.Title,
|
||||||
|
chapter.ValidFrom);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogWarning(
|
||||||
|
"URN {Urn}: Skipping '{Title}' - no valid stream URL available",
|
||||||
|
urn,
|
||||||
|
chapter.Title);
|
||||||
|
noStreamCount++;
|
||||||
|
}
|
||||||
|
|
||||||
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
|
|
||||||
if (string.IsNullOrEmpty(streamUrl))
|
|
||||||
{
|
|
||||||
_logger.LogWarning(
|
|
||||||
"URN {Urn}: Skipping '{Title}' - no valid stream URL available",
|
|
||||||
urn,
|
|
||||||
chapter.Title);
|
|
||||||
noStreamCount++;
|
|
||||||
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;
|
||||||
|
|
||||||
// Get image URL - prefer chapter image, fall back to show image if available
|
// Determine image URL based on configuration
|
||||||
var imageUrl = chapter.ImageUrl;
|
string? imageUrl;
|
||||||
if (string.IsNullOrEmpty(imageUrl) && mediaComposition.Show != null)
|
var generateTitleCards = config.GenerateTitleCards;
|
||||||
{
|
|
||||||
imageUrl = mediaComposition.Show.ImageUrl;
|
|
||||||
_logger.LogDebug("URN {Urn}: Using show image as fallback", urn);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(imageUrl))
|
if (generateTitleCards)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("URN {Urn}: No image URL available for '{Title}'", urn, chapter.Title);
|
// Generate title card with content name
|
||||||
|
_logger.LogDebug("URN {Urn}: Generating title card for '{Title}'", urn, chapter.Title);
|
||||||
|
imageUrl = CreatePlaceholderImageUrl(chapter.Title ?? "SRF Play", _mediaSourceFactory.GetServerBaseUrl());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Use SRF-provided images - prefer chapter image, fall back to show image
|
||||||
|
var originalImageUrl = chapter.ImageUrl;
|
||||||
|
if (string.IsNullOrEmpty(originalImageUrl) && mediaComposition.Show != null)
|
||||||
|
{
|
||||||
|
originalImageUrl = mediaComposition.Show.ImageUrl;
|
||||||
|
_logger.LogDebug("URN {Urn}: Using show image as fallback", urn);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(originalImageUrl))
|
||||||
|
{
|
||||||
|
_logger.LogDebug("URN {Urn}: No image URL available for '{Title}', using placeholder", urn, chapter.Title);
|
||||||
|
imageUrl = CreatePlaceholderImageUrl(chapter.Title ?? "SRF Play", _mediaSourceFactory.GetServerBaseUrl());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
imageUrl = CreateProxiedImageUrl(originalImageUrl, _mediaSourceFactory.GetServerBaseUrl());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 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
|
||||||
@ -512,46 +552,7 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
{
|
{
|
||||||
{ "SRF", urn }
|
{ "SRF", urn }
|
||||||
},
|
},
|
||||||
MediaSources = new List<MediaSourceInfo>
|
MediaSources = new List<MediaSourceInfo> { mediaSource }
|
||||||
{
|
|
||||||
new MediaSourceInfo
|
|
||||||
{
|
|
||||||
Id = itemId,
|
|
||||||
Name = chapter.Title,
|
|
||||||
Path = proxyUrl, // Proxy URL instead of direct Akamai URL
|
|
||||||
Protocol = MediaBrowser.Model.MediaInfo.MediaProtocol.Http,
|
|
||||||
Container = "hls",
|
|
||||||
SupportsDirectStream = true,
|
|
||||||
SupportsDirectPlay = true, // ✅ Enabled! Proxy handles auth
|
|
||||||
SupportsTranscoding = true,
|
|
||||||
IsRemote = false, // False because it's a local proxy endpoint
|
|
||||||
Type = MediaBrowser.Model.Dto.MediaSourceType.Default,
|
|
||||||
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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Add series info if available
|
// Add series info if available
|
||||||
@ -563,13 +564,11 @@ 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(
|
_logger.LogDebug(
|
||||||
"URN {Urn}: MediaSource configured - DirectStream={DirectStream}, DirectPlay={DirectPlay}, Transcoding={Transcoding}, Container={Container}",
|
"URN {Urn}: MediaSource created via factory - DirectPlay={DirectPlay}, Transcoding={Transcoding}",
|
||||||
urn,
|
urn,
|
||||||
true,
|
mediaSource.SupportsDirectPlay,
|
||||||
true,
|
mediaSource.SupportsTranscoding);
|
||||||
true,
|
|
||||||
"hls");
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@ -589,23 +588,34 @@ public class SRFPlayChannel : IChannel, IHasCacheKey
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Generates a deterministic GUID from a URN.
|
/// Creates a proxy URL for an image to fix Content-Type headers from SRF CDN.
|
||||||
/// This ensures the same URN always produces the same GUID.
|
|
||||||
/// MD5 is used for non-cryptographic purposes only (generating IDs).
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="urn">The URN to convert.</param>
|
/// <param name="originalUrl">The original SRF image URL.</param>
|
||||||
/// <returns>A deterministic GUID.</returns>
|
/// <param name="serverUrl">The server base URL.</param>
|
||||||
#pragma warning disable CA5351 // MD5 is used for non-cryptographic purposes (ID generation)
|
/// <returns>The proxied image URL, or null if original is null.</returns>
|
||||||
private static string UrnToGuid(string urn)
|
private static string? CreateProxiedImageUrl(string? originalUrl, string serverUrl)
|
||||||
{
|
{
|
||||||
// Use MD5 to generate a deterministic hash from the URN
|
if (string.IsNullOrEmpty(originalUrl))
|
||||||
var hash = MD5.HashData(Encoding.UTF8.GetBytes(urn));
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
// Convert the first 16 bytes to a GUID
|
// Encode the original URL as base64 for safe transport
|
||||||
var guid = new Guid(hash);
|
var encodedUrl = Convert.ToBase64String(Encoding.UTF8.GetBytes(originalUrl));
|
||||||
return guid.ToString();
|
return $"{serverUrl}/Plugins/SRFPlay/Proxy/Image?url={encodedUrl}";
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a placeholder image URL with the given text.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="text">The text to display on the placeholder.</param>
|
||||||
|
/// <param name="serverUrl">The server base URL.</param>
|
||||||
|
/// <returns>The placeholder image URL.</returns>
|
||||||
|
private static string CreatePlaceholderImageUrl(string text, string serverUrl)
|
||||||
|
{
|
||||||
|
var encodedText = Convert.ToBase64String(Encoding.UTF8.GetBytes(text));
|
||||||
|
return $"{serverUrl}/Plugins/SRFPlay/Proxy/Placeholder?text={encodedText}";
|
||||||
}
|
}
|
||||||
#pragma warning restore CA5351
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public bool IsEnabledFor(string userId)
|
public bool IsEnabledFor(string userId)
|
||||||
|
|||||||
@ -74,6 +74,7 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
EnableTrendingContent = true;
|
EnableTrendingContent = true;
|
||||||
EnableCategoryFolders = true;
|
EnableCategoryFolders = true;
|
||||||
EnabledTopics = new System.Collections.Generic.List<string>();
|
EnabledTopics = new System.Collections.Generic.List<string>();
|
||||||
|
GenerateTitleCards = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -142,4 +143,17 @@ public class PluginConfiguration : BasePluginConfiguration
|
|||||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("Usage", "CA2227:Collection properties should be read only", Justification = "Required for configuration serialization")]
|
[System.Diagnostics.CodeAnalysis.SuppressMessage("Usage", "CA2227:Collection properties should be read only", Justification = "Required for configuration serialization")]
|
||||||
[System.Diagnostics.CodeAnalysis.SuppressMessage("Design", "CA1002:Do not expose generic lists", Justification = "Configuration DTO")]
|
[System.Diagnostics.CodeAnalysis.SuppressMessage("Design", "CA1002:Do not expose generic lists", Justification = "Configuration DTO")]
|
||||||
public System.Collections.Generic.List<string> EnabledTopics { get; set; }
|
public System.Collections.Generic.List<string> EnabledTopics { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the public/external server URL for remote clients (e.g., https://jellyfin.example.com:8920).
|
||||||
|
/// If not set, the plugin will use Jellyfin's GetSmartApiUrl() which may return local addresses.
|
||||||
|
/// This is important for Android and other remote clients to access streams.
|
||||||
|
/// </summary>
|
||||||
|
public string PublicServerUrl { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a value indicating whether to generate title card images with the content title.
|
||||||
|
/// When enabled, generates custom thumbnails instead of using SRF-provided images.
|
||||||
|
/// </summary>
|
||||||
|
public bool GenerateTitleCards { get; set; }
|
||||||
}
|
}
|
||||||
|
|||||||
@ -58,6 +58,13 @@
|
|||||||
</label>
|
</label>
|
||||||
<div class="fieldDescription">Automatically discover and add trending videos</div>
|
<div class="fieldDescription">Automatically discover and add trending videos</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
<label class="emby-checkbox-label">
|
||||||
|
<input id="GenerateTitleCards" name="GenerateTitleCards" type="checkbox" is="emby-checkbox" />
|
||||||
|
<span>Generate Title Cards</span>
|
||||||
|
</label>
|
||||||
|
<div class="fieldDescription">Generate custom thumbnails with the content title instead of using SRF-provided images</div>
|
||||||
|
</div>
|
||||||
<br />
|
<br />
|
||||||
<h2>Proxy Settings</h2>
|
<h2>Proxy Settings</h2>
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription">
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
@ -82,6 +89,17 @@
|
|||||||
<input id="ProxyPassword" name="ProxyPassword" type="password" is="emby-input" autocomplete="off" />
|
<input id="ProxyPassword" name="ProxyPassword" type="password" is="emby-input" autocomplete="off" />
|
||||||
<div class="fieldDescription">Password for proxy authentication (leave empty if not required)</div>
|
<div class="fieldDescription">Password for proxy authentication (leave empty if not required)</div>
|
||||||
</div>
|
</div>
|
||||||
|
<br />
|
||||||
|
<h2>Network Settings</h2>
|
||||||
|
<div class="inputContainer">
|
||||||
|
<label class="inputLabel inputLabelUnfocused" for="PublicServerUrl">Public Server URL (Optional)</label>
|
||||||
|
<input id="PublicServerUrl" name="PublicServerUrl" type="text" is="emby-input" placeholder="e.g., https://jellyfin.example.com:8920" />
|
||||||
|
<div class="fieldDescription">
|
||||||
|
The public/external URL for remote clients (Android, iOS, etc.) to access streaming proxy.
|
||||||
|
<br />If not set, the plugin will use Jellyfin's automatic URL detection which may return local addresses.
|
||||||
|
<br /><strong>Important for Android/remote playback!</strong>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block emby-button">
|
<button is="emby-button" type="submit" class="raised button-submit block emby-button">
|
||||||
<span>Save</span>
|
<span>Save</span>
|
||||||
@ -106,10 +124,12 @@
|
|||||||
document.querySelector('#CacheDurationMinutes').value = config.CacheDurationMinutes;
|
document.querySelector('#CacheDurationMinutes').value = config.CacheDurationMinutes;
|
||||||
document.querySelector('#EnableLatestContent').checked = config.EnableLatestContent;
|
document.querySelector('#EnableLatestContent').checked = config.EnableLatestContent;
|
||||||
document.querySelector('#EnableTrendingContent').checked = config.EnableTrendingContent;
|
document.querySelector('#EnableTrendingContent').checked = config.EnableTrendingContent;
|
||||||
|
document.querySelector('#GenerateTitleCards').checked = config.GenerateTitleCards !== false;
|
||||||
document.querySelector('#UseProxy').checked = config.UseProxy || false;
|
document.querySelector('#UseProxy').checked = config.UseProxy || false;
|
||||||
document.querySelector('#ProxyAddress').value = config.ProxyAddress || '';
|
document.querySelector('#ProxyAddress').value = config.ProxyAddress || '';
|
||||||
document.querySelector('#ProxyUsername').value = config.ProxyUsername || '';
|
document.querySelector('#ProxyUsername').value = config.ProxyUsername || '';
|
||||||
document.querySelector('#ProxyPassword').value = config.ProxyPassword || '';
|
document.querySelector('#ProxyPassword').value = config.ProxyPassword || '';
|
||||||
|
document.querySelector('#PublicServerUrl').value = config.PublicServerUrl || '';
|
||||||
Dashboard.hideLoadingMsg();
|
Dashboard.hideLoadingMsg();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -125,10 +145,12 @@
|
|||||||
config.CacheDurationMinutes = parseInt(document.querySelector('#CacheDurationMinutes').value);
|
config.CacheDurationMinutes = parseInt(document.querySelector('#CacheDurationMinutes').value);
|
||||||
config.EnableLatestContent = document.querySelector('#EnableLatestContent').checked;
|
config.EnableLatestContent = document.querySelector('#EnableLatestContent').checked;
|
||||||
config.EnableTrendingContent = document.querySelector('#EnableTrendingContent').checked;
|
config.EnableTrendingContent = document.querySelector('#EnableTrendingContent').checked;
|
||||||
|
config.GenerateTitleCards = document.querySelector('#GenerateTitleCards').checked;
|
||||||
config.UseProxy = document.querySelector('#UseProxy').checked;
|
config.UseProxy = document.querySelector('#UseProxy').checked;
|
||||||
config.ProxyAddress = document.querySelector('#ProxyAddress').value;
|
config.ProxyAddress = document.querySelector('#ProxyAddress').value;
|
||||||
config.ProxyUsername = document.querySelector('#ProxyUsername').value;
|
config.ProxyUsername = document.querySelector('#ProxyUsername').value;
|
||||||
config.ProxyPassword = document.querySelector('#ProxyPassword').value;
|
config.ProxyPassword = document.querySelector('#ProxyPassword').value;
|
||||||
|
config.PublicServerUrl = document.querySelector('#PublicServerUrl').value;
|
||||||
ApiClient.updatePluginConfiguration(SRFPlayConfig.pluginUniqueId, config).then(function (result) {
|
ApiClient.updatePluginConfiguration(SRFPlayConfig.pluginUniqueId, config).then(function (result) {
|
||||||
Dashboard.processPluginConfigurationUpdateResult(result);
|
Dashboard.processPluginConfigurationUpdateResult(result);
|
||||||
});
|
});
|
||||||
|
|||||||
73
Jellyfin.Plugin.SRFPlay/Constants/ApiEndpoints.cs
Normal file
73
Jellyfin.Plugin.SRFPlay/Constants/ApiEndpoints.cs
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
namespace Jellyfin.Plugin.SRFPlay.Constants;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Centralized API endpoints and URL templates used throughout the plugin.
|
||||||
|
/// </summary>
|
||||||
|
public static class ApiEndpoints
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// SRG SSR Integration Layer API base URL.
|
||||||
|
/// Used for fetching media compositions, video metadata, and stream URLs.
|
||||||
|
/// </summary>
|
||||||
|
public const string IntegrationLayerBaseUrl = "https://il.srgssr.ch/integrationlayer/2.0";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Play V3 API URL template. Format: {0} = business unit (srf, rts, rsi, rtr, swi).
|
||||||
|
/// Used for fetching shows, topics, and latest content.
|
||||||
|
/// </summary>
|
||||||
|
public const string PlayV3BaseUrlTemplate = "https://www.{0}.ch/play/v3/api/{0}/production/";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Akamai token authentication endpoint.
|
||||||
|
/// Used to get hdnts tokens for stream authentication.
|
||||||
|
/// </summary>
|
||||||
|
public const string AkamaiTokenEndpoint = "https://tp.srgssr.ch/akahd/token";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// SRF Play homepage URL.
|
||||||
|
/// </summary>
|
||||||
|
public const string SrfPlayHomepage = "https://www.srf.ch/play";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Media composition endpoint template (relative to IntegrationLayerBaseUrl).
|
||||||
|
/// Format: {0} = URN.
|
||||||
|
/// </summary>
|
||||||
|
public const string MediaCompositionByUrnPath = "/mediaComposition/byUrn/{0}.json";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Base proxy route for stream proxying (relative to server root).
|
||||||
|
/// </summary>
|
||||||
|
public const string ProxyBasePath = "/Plugins/SRFPlay/Proxy";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// HLS master manifest route template (relative to server root).
|
||||||
|
/// Format: {0} = item ID.
|
||||||
|
/// </summary>
|
||||||
|
public const string ProxyMasterManifestPath = "/Plugins/SRFPlay/Proxy/{0}/master.m3u8";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Image proxy route template (relative to server root).
|
||||||
|
/// Format: {0} = base64-encoded original URL.
|
||||||
|
/// </summary>
|
||||||
|
public const string ImageProxyPath = "/Plugins/SRFPlay/Image/{0}";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Play V3 shows endpoint (relative to PlayV3 base URL).
|
||||||
|
/// </summary>
|
||||||
|
public const string PlayV3ShowsPath = "shows";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Play V3 topics endpoint (relative to PlayV3 base URL).
|
||||||
|
/// </summary>
|
||||||
|
public const string PlayV3TopicsPath = "topics";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Play V3 livestreams endpoint (relative to PlayV3 base URL).
|
||||||
|
/// </summary>
|
||||||
|
public const string PlayV3LivestreamsPath = "livestreams";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Play V3 scheduled livestreams endpoint (relative to PlayV3 base URL).
|
||||||
|
/// </summary>
|
||||||
|
public const string PlayV3ScheduledLivestreamsPath = "scheduled-livestreams";
|
||||||
|
}
|
||||||
@ -1,8 +1,12 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Net.Http.Headers;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
|
using MediaBrowser.Common.Net;
|
||||||
using Microsoft.AspNetCore.Authorization;
|
using Microsoft.AspNetCore.Authorization;
|
||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
@ -18,17 +22,72 @@ namespace Jellyfin.Plugin.SRFPlay.Controllers;
|
|||||||
public class StreamProxyController : ControllerBase
|
public class StreamProxyController : ControllerBase
|
||||||
{
|
{
|
||||||
private readonly ILogger<StreamProxyController> _logger;
|
private readonly ILogger<StreamProxyController> _logger;
|
||||||
private readonly StreamProxyService _proxyService;
|
private readonly IStreamProxyService _proxyService;
|
||||||
|
private readonly IHttpClientFactory _httpClientFactory;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="StreamProxyController"/> class.
|
/// Initializes a new instance of the <see cref="StreamProxyController"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="logger">The logger.</param>
|
/// <param name="logger">The logger.</param>
|
||||||
/// <param name="proxyService">The proxy service.</param>
|
/// <param name="proxyService">The proxy service.</param>
|
||||||
public StreamProxyController(ILogger<StreamProxyController> logger, StreamProxyService proxyService)
|
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
||||||
|
public StreamProxyController(
|
||||||
|
ILogger<StreamProxyController> logger,
|
||||||
|
IStreamProxyService proxyService,
|
||||||
|
IHttpClientFactory httpClientFactory)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_proxyService = proxyService;
|
_proxyService = proxyService;
|
||||||
|
_httpClientFactory = httpClientFactory;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adds CORS headers to allow cross-origin requests from hls.js in browsers.
|
||||||
|
/// </summary>
|
||||||
|
private void AddCorsHeaders()
|
||||||
|
{
|
||||||
|
Response.Headers["Access-Control-Allow-Origin"] = "*";
|
||||||
|
Response.Headers["Access-Control-Allow-Methods"] = "GET, HEAD, OPTIONS";
|
||||||
|
Response.Headers["Access-Control-Allow-Headers"] = "Content-Type, Range, Accept, Origin";
|
||||||
|
Response.Headers["Access-Control-Expose-Headers"] = "Content-Length, Content-Range, Accept-Ranges";
|
||||||
|
Response.Headers["Access-Control-Max-Age"] = "86400"; // Cache preflight for 24 hours
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adds Cache-Control headers appropriate for the stream type.
|
||||||
|
/// Livestreams need frequent manifest refresh, VOD can be cached longer.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID to check.</param>
|
||||||
|
private void AddManifestCacheHeaders(string itemId)
|
||||||
|
{
|
||||||
|
var metadata = _proxyService.GetStreamMetadata(itemId);
|
||||||
|
var isLiveStream = metadata?.IsLiveStream ?? false;
|
||||||
|
|
||||||
|
if (isLiveStream)
|
||||||
|
{
|
||||||
|
// Livestreams need frequent manifest refresh (segments rotate every ~6-10s)
|
||||||
|
Response.Headers["Cache-Control"] = "max-age=2, must-revalidate";
|
||||||
|
_logger.LogDebug("Setting livestream cache headers for {ItemId}", itemId);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// VOD manifests are static, can cache longer
|
||||||
|
Response.Headers["Cache-Control"] = "max-age=3600";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Handles CORS preflight OPTIONS requests for all proxy endpoints.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>Empty response with CORS headers.</returns>
|
||||||
|
[HttpOptions("{itemId}/master.m3u8")]
|
||||||
|
[HttpOptions("{itemId}/{manifestPath}.m3u8")]
|
||||||
|
[HttpOptions("{itemId}/{*segmentPath}")]
|
||||||
|
[AllowAnonymous]
|
||||||
|
public IActionResult HandleOptions()
|
||||||
|
{
|
||||||
|
AddCorsHeaders();
|
||||||
|
return Ok();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -45,26 +104,51 @@ public class StreamProxyController : ControllerBase
|
|||||||
[FromRoute] string itemId,
|
[FromRoute] string itemId,
|
||||||
CancellationToken cancellationToken)
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
_logger.LogInformation("Proxy request for master manifest - ItemId: {ItemId}", itemId);
|
AddCorsHeaders();
|
||||||
|
_logger.LogInformation("Proxy request for master manifest - Path ItemId: {PathItemId}, Query params: {QueryString}", itemId, Request.QueryString);
|
||||||
|
|
||||||
// Try to resolve the actual item ID (path ID might be a session ID during transcoding)
|
// Try to resolve the actual item ID (path ID might be a session ID during transcoding)
|
||||||
var actualItemId = ResolveItemId(itemId);
|
var actualItemId = ResolveItemId(itemId);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// Build the base proxy URL for this item (use original itemId from path to maintain URL structure)
|
// Get the correct scheme (https if configured, otherwise use request scheme)
|
||||||
var baseProxyUrl = $"{Request.Scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}";
|
var scheme = GetProxyScheme();
|
||||||
|
|
||||||
|
// Build the base proxy URL for this item
|
||||||
|
// Preserve query parameters (token or itemId) from the original request
|
||||||
|
string baseProxyUrl;
|
||||||
|
if (Request.Query.TryGetValue("token", out var token) && !string.IsNullOrEmpty(token))
|
||||||
|
{
|
||||||
|
baseProxyUrl = $"{scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}?token={token}";
|
||||||
|
_logger.LogDebug("Using token-based proxy URL with token: {Token}", token.ToString());
|
||||||
|
}
|
||||||
|
else if (actualItemId != itemId)
|
||||||
|
{
|
||||||
|
// Legacy: If path ID differs from resolved ID, add itemId query parameter
|
||||||
|
baseProxyUrl = $"{scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}?itemId={actualItemId}";
|
||||||
|
_logger.LogInformation("Path itemId {PathId} differs from resolved itemId {ResolvedId}, adding query parameter", itemId, actualItemId);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Simple case: no query parameters needed
|
||||||
|
baseProxyUrl = $"{scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}";
|
||||||
|
_logger.LogDebug("Path itemId matches resolved itemId: {ItemId}", itemId);
|
||||||
|
}
|
||||||
|
|
||||||
var manifestContent = await _proxyService.GetRewrittenManifestAsync(actualItemId, baseProxyUrl, cancellationToken).ConfigureAwait(false);
|
var manifestContent = await _proxyService.GetRewrittenManifestAsync(actualItemId, baseProxyUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (manifestContent == null)
|
if (manifestContent == null)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("Manifest not found for item {ItemId}", itemId);
|
_logger.LogWarning("Manifest not found for path itemId {PathItemId}, resolved itemId {ResolvedItemId} - stream may not be registered", itemId, actualItemId);
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Set cache headers based on stream type (live vs VOD)
|
||||||
|
AddManifestCacheHeaders(actualItemId);
|
||||||
|
|
||||||
_logger.LogDebug("Returning master manifest for item {ItemId} ({Length} bytes)", itemId, manifestContent.Length);
|
_logger.LogDebug("Returning master manifest for item {ItemId} ({Length} bytes)", itemId, manifestContent.Length);
|
||||||
return Content(manifestContent, "application/vnd.apple.mpegurl");
|
return Content(manifestContent, "application/vnd.apple.mpegurl; charset=utf-8");
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@ -89,6 +173,7 @@ public class StreamProxyController : ControllerBase
|
|||||||
[FromRoute] string manifestPath,
|
[FromRoute] string manifestPath,
|
||||||
CancellationToken cancellationToken)
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
AddCorsHeaders();
|
||||||
var fullPath = $"{manifestPath}.m3u8";
|
var fullPath = $"{manifestPath}.m3u8";
|
||||||
_logger.LogInformation("Proxy request for variant manifest - ItemId: {ItemId}, Path: {Path}", itemId, fullPath);
|
_logger.LogInformation("Proxy request for variant manifest - ItemId: {ItemId}, Path: {Path}", itemId, fullPath);
|
||||||
|
|
||||||
@ -108,11 +193,15 @@ public class StreamProxyController : ControllerBase
|
|||||||
|
|
||||||
// Convert to string and rewrite URLs
|
// Convert to string and rewrite URLs
|
||||||
var manifestContent = System.Text.Encoding.UTF8.GetString(manifestData);
|
var manifestContent = System.Text.Encoding.UTF8.GetString(manifestData);
|
||||||
var baseProxyUrl = $"{Request.Scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}";
|
var scheme = GetProxyScheme();
|
||||||
|
var baseProxyUrl = $"{scheme}://{Request.Host}/Plugins/SRFPlay/Proxy/{itemId}";
|
||||||
var rewrittenContent = RewriteSegmentUrls(manifestContent, baseProxyUrl);
|
var rewrittenContent = RewriteSegmentUrls(manifestContent, baseProxyUrl);
|
||||||
|
|
||||||
|
// Set cache headers based on stream type (live vs VOD)
|
||||||
|
AddManifestCacheHeaders(actualItemId);
|
||||||
|
|
||||||
_logger.LogDebug("Returning variant manifest for item {ItemId} ({Length} bytes)", itemId, rewrittenContent.Length);
|
_logger.LogDebug("Returning variant manifest for item {ItemId} ({Length} bytes)", itemId, rewrittenContent.Length);
|
||||||
return Content(rewrittenContent, "application/vnd.apple.mpegurl");
|
return Content(rewrittenContent, "application/vnd.apple.mpegurl; charset=utf-8");
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@ -137,6 +226,7 @@ public class StreamProxyController : ControllerBase
|
|||||||
[FromRoute] string segmentPath,
|
[FromRoute] string segmentPath,
|
||||||
CancellationToken cancellationToken)
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
AddCorsHeaders();
|
||||||
_logger.LogDebug("Proxy request for segment - ItemId: {ItemId}, Path: {SegmentPath}", itemId, segmentPath);
|
_logger.LogDebug("Proxy request for segment - ItemId: {ItemId}, Path: {SegmentPath}", itemId, segmentPath);
|
||||||
|
|
||||||
// Try to resolve the actual item ID
|
// Try to resolve the actual item ID
|
||||||
@ -153,7 +243,7 @@ public class StreamProxyController : ControllerBase
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Determine content type based on file extension
|
// Determine content type based on file extension
|
||||||
var contentType = GetContentType(segmentPath);
|
var contentType = MimeTypeHelper.GetSegmentContentType(segmentPath);
|
||||||
|
|
||||||
_logger.LogDebug("Returning segment {SegmentPath} ({Length} bytes, {ContentType})", segmentPath, segmentData.Length, contentType);
|
_logger.LogDebug("Returning segment {SegmentPath} ({Length} bytes, {ContentType})", segmentPath, segmentData.Length, contentType);
|
||||||
return File(segmentData, contentType);
|
return File(segmentData, contentType);
|
||||||
@ -165,6 +255,31 @@ public class StreamProxyController : ControllerBase
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the correct scheme for proxy URLs (https if public URL is configured with https).
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The scheme to use (http or https).</returns>
|
||||||
|
private string GetProxyScheme()
|
||||||
|
{
|
||||||
|
// Check if PublicServerUrl is configured and uses HTTPS
|
||||||
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
if (config != null && !string.IsNullOrWhiteSpace(config.PublicServerUrl))
|
||||||
|
{
|
||||||
|
if (config.PublicServerUrl.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return "https";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fall back to request scheme, but prefer https if forwarded headers indicate it
|
||||||
|
if (Request.Headers.TryGetValue("X-Forwarded-Proto", out var forwardedProto))
|
||||||
|
{
|
||||||
|
return forwardedProto.ToString().ToLowerInvariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
return Request.Scheme;
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Resolves the actual item ID from the request.
|
/// Resolves the actual item ID from the request.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -172,14 +287,24 @@ public class StreamProxyController : ControllerBase
|
|||||||
/// <returns>The resolved item ID.</returns>
|
/// <returns>The resolved item ID.</returns>
|
||||||
private string ResolveItemId(string pathItemId)
|
private string ResolveItemId(string pathItemId)
|
||||||
{
|
{
|
||||||
// Check if there's an itemId query parameter (fallback for transcoding sessions)
|
// Check for token parameter first (preferred method)
|
||||||
|
if (Request.Query.TryGetValue("token", out var token) && !string.IsNullOrEmpty(token))
|
||||||
|
{
|
||||||
|
// Try to resolve the original item ID from the token via the proxy service
|
||||||
|
// We'll need to add a method to StreamProxyService to look up by token
|
||||||
|
_logger.LogInformation("Found token parameter: {Token}, will use path ID {PathItemId} for lookup", token.ToString(), pathItemId);
|
||||||
|
return pathItemId; // Use path ID for now; token prevents Jellyfin from rewriting the URL
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if there's an itemId query parameter (legacy fallback)
|
||||||
if (Request.Query.TryGetValue("itemId", out var queryItemId) && !string.IsNullOrEmpty(queryItemId))
|
if (Request.Query.TryGetValue("itemId", out var queryItemId) && !string.IsNullOrEmpty(queryItemId))
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Using itemId from query parameter: {QueryItemId} (path had: {PathItemId})", queryItemId.ToString(), pathItemId);
|
_logger.LogInformation("Using itemId from query parameter: {QueryItemId} (path had: {PathItemId})", queryItemId.ToString(), pathItemId);
|
||||||
return queryItemId.ToString();
|
return queryItemId.ToString();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use the path item ID as-is
|
// No query parameters - use path ID as-is (normal case for segments and transcoding sessions)
|
||||||
|
_logger.LogDebug("No query parameters, using path ID as-is: {PathItemId}", pathItemId);
|
||||||
return pathItemId;
|
return pathItemId;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,28 +316,67 @@ public class StreamProxyController : ControllerBase
|
|||||||
/// <returns>The rewritten manifest.</returns>
|
/// <returns>The rewritten manifest.</returns>
|
||||||
private string RewriteSegmentUrls(string manifestContent, string baseProxyUrl)
|
private string RewriteSegmentUrls(string manifestContent, string baseProxyUrl)
|
||||||
{
|
{
|
||||||
|
// Extract query parameters from the current request to propagate them
|
||||||
|
string queryParams;
|
||||||
|
if (Request.Query.TryGetValue("token", out var token) && !string.IsNullOrEmpty(token))
|
||||||
|
{
|
||||||
|
queryParams = $"?token={token}";
|
||||||
|
}
|
||||||
|
else if (Request.Query.TryGetValue("itemId", out var itemId) && !string.IsNullOrEmpty(itemId))
|
||||||
|
{
|
||||||
|
queryParams = $"?itemId={itemId}";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
queryParams = string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper function to rewrite a single URL
|
||||||
|
string RewriteUrl(string url)
|
||||||
|
{
|
||||||
|
if (url.Contains("://", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
// Absolute URL - extract filename and rewrite
|
||||||
|
var uri = new Uri(url.Trim());
|
||||||
|
var segments = uri.AbsolutePath.Split('/');
|
||||||
|
var fileName = segments[^1];
|
||||||
|
return $"{baseProxyUrl}/{fileName}{queryParams}";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Relative URL - rewrite to proxy
|
||||||
|
return $"{baseProxyUrl}/{url.Trim()}{queryParams}";
|
||||||
|
}
|
||||||
|
|
||||||
var lines = manifestContent.Split('\n');
|
var lines = manifestContent.Split('\n');
|
||||||
var result = new System.Text.StringBuilder();
|
var result = new System.Text.StringBuilder();
|
||||||
|
|
||||||
foreach (var line in lines)
|
foreach (var line in lines)
|
||||||
{
|
{
|
||||||
if (line.StartsWith('#') || string.IsNullOrWhiteSpace(line))
|
if (string.IsNullOrWhiteSpace(line))
|
||||||
{
|
{
|
||||||
// Keep metadata and blank lines as-is
|
|
||||||
result.AppendLine(line);
|
result.AppendLine(line);
|
||||||
}
|
}
|
||||||
else if (line.Contains("://", StringComparison.Ordinal))
|
else if (line.StartsWith('#'))
|
||||||
{
|
{
|
||||||
// Absolute URL - extract the path and rewrite
|
// HLS tag line - check for URI="..." attributes (e.g., #EXT-X-MAP:URI="init.mp4")
|
||||||
var uri = new Uri(line.Trim());
|
if (line.Contains("URI=\"", StringComparison.Ordinal))
|
||||||
var segments = uri.AbsolutePath.Split('/');
|
{
|
||||||
var fileName = segments[^1];
|
var rewrittenLine = System.Text.RegularExpressions.Regex.Replace(
|
||||||
result.AppendLine(CultureInfo.InvariantCulture, $"{baseProxyUrl}/{fileName}");
|
line,
|
||||||
|
@"URI=""([^""]+)""",
|
||||||
|
match => $"URI=\"{RewriteUrl(match.Groups[1].Value)}\"");
|
||||||
|
result.AppendLine(rewrittenLine);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Keep other metadata lines as-is
|
||||||
|
result.AppendLine(line);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Relative URL - rewrite to proxy
|
// Non-tag line with URL - rewrite it
|
||||||
result.AppendLine(CultureInfo.InvariantCulture, $"{baseProxyUrl}/{line.Trim()}");
|
result.AppendLine(RewriteUrl(line));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -220,28 +384,115 @@ public class StreamProxyController : ControllerBase
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the content type for a segment based on file extension.
|
/// Proxies image requests from SRF CDN, fixing Content-Type headers.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="path">The segment path.</param>
|
/// <param name="url">The original image URL (base64 encoded).</param>
|
||||||
/// <returns>The MIME content type.</returns>
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
private static string GetContentType(string path)
|
/// <returns>The image data with correct Content-Type.</returns>
|
||||||
|
[HttpGet("Image")]
|
||||||
|
[AllowAnonymous]
|
||||||
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status404NotFound)]
|
||||||
|
public async Task<IActionResult> GetImage(
|
||||||
|
[FromQuery] string url,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
if (path.EndsWith(".ts", StringComparison.OrdinalIgnoreCase))
|
if (string.IsNullOrEmpty(url))
|
||||||
{
|
{
|
||||||
return "video/MP2T";
|
return BadRequest("URL parameter is required");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (path.EndsWith(".mp4", StringComparison.OrdinalIgnoreCase) ||
|
string decodedUrl;
|
||||||
path.EndsWith(".m4s", StringComparison.OrdinalIgnoreCase))
|
try
|
||||||
{
|
{
|
||||||
return "video/mp4";
|
// Decode base64 URL
|
||||||
|
var bytes = Convert.FromBase64String(url);
|
||||||
|
decodedUrl = System.Text.Encoding.UTF8.GetString(bytes);
|
||||||
|
}
|
||||||
|
catch (FormatException)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Invalid base64 URL parameter: {Url}", url);
|
||||||
|
return BadRequest("Invalid URL encoding");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (path.EndsWith(".aac", StringComparison.OrdinalIgnoreCase))
|
_logger.LogDebug("Proxying image from: {Url}", decodedUrl);
|
||||||
|
|
||||||
|
try
|
||||||
{
|
{
|
||||||
return "audio/aac";
|
var httpClient = _httpClientFactory.CreateClient(NamedClient.Default);
|
||||||
|
|
||||||
|
// Create request with proper headers
|
||||||
|
var request = new HttpRequestMessage(HttpMethod.Get, new Uri(decodedUrl));
|
||||||
|
request.Headers.UserAgent.ParseAdd(
|
||||||
|
"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36");
|
||||||
|
request.Headers.Accept.ParseAdd("image/jpeg, image/png, image/webp, image/*;q=0.8");
|
||||||
|
|
||||||
|
var response = await httpClient.SendAsync(
|
||||||
|
request,
|
||||||
|
HttpCompletionOption.ResponseHeadersRead,
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (!response.IsSuccessStatusCode)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(
|
||||||
|
"Image fetch failed with status {StatusCode} for {Url}",
|
||||||
|
response.StatusCode,
|
||||||
|
decodedUrl);
|
||||||
|
return NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
var imageData = await response.Content.ReadAsByteArrayAsync(cancellationToken)
|
||||||
|
.ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Determine correct content type
|
||||||
|
var contentType = MimeTypeHelper.GetImageContentType(decodedUrl);
|
||||||
|
|
||||||
|
_logger.LogDebug(
|
||||||
|
"Returning proxied image ({Length} bytes, {ContentType})",
|
||||||
|
imageData.Length,
|
||||||
|
contentType);
|
||||||
|
|
||||||
|
return File(imageData, contentType);
|
||||||
|
}
|
||||||
|
catch (HttpRequestException ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error fetching image from {Url}", decodedUrl);
|
||||||
|
return StatusCode(StatusCodes.Status502BadGateway);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Generates a placeholder image with the given text centered.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="text">The text to display (base64 encoded).</param>
|
||||||
|
/// <returns>A PNG image with the text centered.</returns>
|
||||||
|
[HttpGet("Placeholder")]
|
||||||
|
[AllowAnonymous]
|
||||||
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
||||||
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
||||||
|
public IActionResult GetPlaceholder([FromQuery] string text)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(text))
|
||||||
|
{
|
||||||
|
return BadRequest("Text parameter is required");
|
||||||
}
|
}
|
||||||
|
|
||||||
return "application/octet-stream";
|
string decodedText;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var bytes = Convert.FromBase64String(text);
|
||||||
|
decodedText = System.Text.Encoding.UTF8.GetString(bytes);
|
||||||
|
}
|
||||||
|
catch (FormatException)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Invalid base64 text parameter: {Text}", text);
|
||||||
|
return BadRequest("Invalid text encoding");
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogDebug("Generating placeholder image for: {Text}", decodedText);
|
||||||
|
|
||||||
|
var imageStream = PlaceholderImageGenerator.GeneratePlaceholder(decodedText);
|
||||||
|
return File(imageStream, "image/png");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -14,6 +14,7 @@
|
|||||||
<PackageReference Include="Jellyfin.Controller" Version="10.9.11" />
|
<PackageReference Include="Jellyfin.Controller" Version="10.9.11" />
|
||||||
<PackageReference Include="Jellyfin.Model" Version="10.9.11" />
|
<PackageReference Include="Jellyfin.Model" Version="10.9.11" />
|
||||||
<PackageReference Include="Socks5" Version="1.1.0" />
|
<PackageReference Include="Socks5" Version="1.1.0" />
|
||||||
|
<PackageReference Include="SkiaSharp" Version="2.88.8" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
@ -27,4 +28,9 @@
|
|||||||
<EmbeddedResource Include="Configuration\configPage.html" />
|
<EmbeddedResource Include="Configuration\configPage.html" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<None Remove="..\assests\main logo.png" />
|
||||||
|
<EmbeddedResource Include="..\assests\main logo.png" LogicalName="Jellyfin.Plugin.SRFPlay.Images.logo.png" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@ -1,11 +1,9 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
|
||||||
using MediaBrowser.Controller.Entities.TV;
|
using MediaBrowser.Controller.Entities.TV;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
using MediaBrowser.Model.Entities;
|
using MediaBrowser.Model.Entities;
|
||||||
@ -20,25 +18,19 @@ namespace Jellyfin.Plugin.SRFPlay.Providers;
|
|||||||
public class SRFEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>
|
public class SRFEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>
|
||||||
{
|
{
|
||||||
private readonly ILogger<SRFEpisodeProvider> _logger;
|
private readonly ILogger<SRFEpisodeProvider> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly IMediaCompositionFetcher _compositionFetcher;
|
||||||
private readonly IHttpClientFactory _httpClientFactory;
|
|
||||||
private readonly MetadataCache _metadataCache;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="SRFEpisodeProvider"/> class.
|
/// Initializes a new instance of the <see cref="SRFEpisodeProvider"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="logger">The logger.</param>
|
||||||
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
/// <param name="compositionFetcher">The media composition fetcher.</param>
|
||||||
/// <param name="metadataCache">The metadata cache.</param>
|
|
||||||
public SRFEpisodeProvider(
|
public SRFEpisodeProvider(
|
||||||
ILoggerFactory loggerFactory,
|
ILogger<SRFEpisodeProvider> logger,
|
||||||
IHttpClientFactory httpClientFactory,
|
IMediaCompositionFetcher compositionFetcher)
|
||||||
MetadataCache metadataCache)
|
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
_logger = logger;
|
||||||
_logger = loggerFactory.CreateLogger<SRFEpisodeProvider>();
|
_compositionFetcher = compositionFetcher;
|
||||||
_httpClientFactory = httpClientFactory;
|
|
||||||
_metadataCache = metadataCache;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -56,26 +48,7 @@ public class SRFEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>
|
|||||||
{
|
{
|
||||||
_logger.LogDebug("Searching for episode with URN: {Urn}", urn);
|
_logger.LogDebug("Searching for episode with URN: {Urn}", urn);
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
return results;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try cache first
|
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, config.CacheDurationMinutes);
|
|
||||||
|
|
||||||
// If not in cache, fetch from API
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.ChapterList != null && mediaComposition.ChapterList.Count > 0)
|
if (mediaComposition?.ChapterList != null && mediaComposition.ChapterList.Count > 0)
|
||||||
{
|
{
|
||||||
@ -118,26 +91,7 @@ public class SRFEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>
|
|||||||
|
|
||||||
_logger.LogDebug("Fetching metadata for episode URN: {Urn}", urn);
|
_logger.LogDebug("Fetching metadata for episode URN: {Urn}", urn);
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try cache first
|
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, config.CacheDurationMinutes);
|
|
||||||
|
|
||||||
// If not in cache, fetch from API
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
||||||
{
|
{
|
||||||
|
|||||||
@ -1,10 +1,11 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
|
using System.Net.Http.Headers;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
using MediaBrowser.Common.Net;
|
using MediaBrowser.Common.Net;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
@ -21,18 +22,22 @@ public class SRFImageProvider : IRemoteImageProvider, IHasOrder
|
|||||||
{
|
{
|
||||||
private readonly IHttpClientFactory _httpClientFactory;
|
private readonly IHttpClientFactory _httpClientFactory;
|
||||||
private readonly ILogger<SRFImageProvider> _logger;
|
private readonly ILogger<SRFImageProvider> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly IMediaCompositionFetcher _compositionFetcher;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="SRFImageProvider"/> class.
|
/// Initializes a new instance of the <see cref="SRFImageProvider"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
public SRFImageProvider(IHttpClientFactory httpClientFactory, ILoggerFactory loggerFactory)
|
/// <param name="compositionFetcher">The media composition fetcher.</param>
|
||||||
|
public SRFImageProvider(
|
||||||
|
IHttpClientFactory httpClientFactory,
|
||||||
|
ILoggerFactory loggerFactory,
|
||||||
|
IMediaCompositionFetcher compositionFetcher)
|
||||||
{
|
{
|
||||||
_httpClientFactory = httpClientFactory;
|
_httpClientFactory = httpClientFactory;
|
||||||
_loggerFactory = loggerFactory;
|
|
||||||
_logger = loggerFactory.CreateLogger<SRFImageProvider>();
|
_logger = loggerFactory.CreateLogger<SRFImageProvider>();
|
||||||
|
_compositionFetcher = compositionFetcher;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -78,8 +83,7 @@ public class SRFImageProvider : IRemoteImageProvider, IHasOrder
|
|||||||
_logger.LogDebug("Fetching images for SRF URN: {Urn}", urn);
|
_logger.LogDebug("Fetching images for SRF URN: {Urn}", urn);
|
||||||
|
|
||||||
// Fetch media composition to get image URLs
|
// Fetch media composition to get image URLs
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
var mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition == null)
|
if (mediaComposition == null)
|
||||||
{
|
{
|
||||||
@ -151,9 +155,51 @@ public class SRFImageProvider : IRemoteImageProvider, IHasOrder
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
|
public async Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
_logger.LogDebug("Fetching image from URL: {Url}", url);
|
||||||
|
|
||||||
var httpClient = _httpClientFactory.CreateClient(NamedClient.Default);
|
var httpClient = _httpClientFactory.CreateClient(NamedClient.Default);
|
||||||
return httpClient.GetAsync(new Uri(url), cancellationToken);
|
|
||||||
|
// Create request with proper headers - SRF CDN requires User-Agent
|
||||||
|
var request = new HttpRequestMessage(HttpMethod.Get, new Uri(url));
|
||||||
|
request.Headers.UserAgent.ParseAdd("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36");
|
||||||
|
request.Headers.Accept.ParseAdd("image/jpeg, image/png, image/webp, image/*;q=0.8");
|
||||||
|
|
||||||
|
var response = await httpClient.SendAsync(request, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
var originalContentType = response.Content.Headers.ContentType?.MediaType;
|
||||||
|
_logger.LogDebug(
|
||||||
|
"Image response status: {StatusCode}, Content-Type: {ContentType}, Content-Length: {Length}",
|
||||||
|
response.StatusCode,
|
||||||
|
originalContentType ?? "null",
|
||||||
|
response.Content.Headers.ContentLength);
|
||||||
|
|
||||||
|
// Fix Content-Type if it's not a proper image type - SRF CDN often returns wrong content type
|
||||||
|
// Jellyfin needs correct Content-Type to process images
|
||||||
|
if (response.IsSuccessStatusCode)
|
||||||
|
{
|
||||||
|
var needsContentTypeFix = string.IsNullOrEmpty(originalContentType) ||
|
||||||
|
originalContentType == "binary/octet-stream" ||
|
||||||
|
originalContentType == "application/octet-stream" ||
|
||||||
|
!originalContentType.StartsWith("image/", StringComparison.OrdinalIgnoreCase);
|
||||||
|
|
||||||
|
if (needsContentTypeFix)
|
||||||
|
{
|
||||||
|
// Determine correct content type from URL extension or default to JPEG
|
||||||
|
var contentType = MimeTypeHelper.GetImageContentType(url);
|
||||||
|
if (!string.IsNullOrEmpty(contentType))
|
||||||
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Fixing Content-Type from '{OriginalType}' to '{NewType}' for {Url}",
|
||||||
|
originalContentType ?? "null",
|
||||||
|
contentType,
|
||||||
|
url);
|
||||||
|
response.Content.Headers.ContentType = new MediaTypeHeaderValue(contentType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return response;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,15 +1,12 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
using MediaBrowser.Model.Dto;
|
using MediaBrowser.Model.Dto;
|
||||||
using MediaBrowser.Model.Entities;
|
|
||||||
using MediaBrowser.Model.MediaInfo;
|
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Providers;
|
namespace Jellyfin.Plugin.SRFPlay.Providers;
|
||||||
@ -20,29 +17,27 @@ namespace Jellyfin.Plugin.SRFPlay.Providers;
|
|||||||
public class SRFMediaProvider : IMediaSourceProvider
|
public class SRFMediaProvider : IMediaSourceProvider
|
||||||
{
|
{
|
||||||
private readonly ILogger<SRFMediaProvider> _logger;
|
private readonly ILogger<SRFMediaProvider> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly IMediaCompositionFetcher _compositionFetcher;
|
||||||
private readonly MetadataCache _metadataCache;
|
private readonly IStreamUrlResolver _streamResolver;
|
||||||
private readonly StreamUrlResolver _streamResolver;
|
private readonly IMediaSourceFactory _mediaSourceFactory;
|
||||||
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.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
/// <param name="metadataCache">The metadata cache.</param>
|
/// <param name="compositionFetcher">The media composition fetcher.</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>
|
/// <param name="mediaSourceFactory">The media source factory.</param>
|
||||||
public SRFMediaProvider(
|
public SRFMediaProvider(
|
||||||
ILoggerFactory loggerFactory,
|
ILoggerFactory loggerFactory,
|
||||||
MetadataCache metadataCache,
|
IMediaCompositionFetcher compositionFetcher,
|
||||||
StreamUrlResolver streamResolver,
|
IStreamUrlResolver streamResolver,
|
||||||
StreamProxyService proxyService)
|
IMediaSourceFactory mediaSourceFactory)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
|
||||||
_logger = loggerFactory.CreateLogger<SRFMediaProvider>();
|
_logger = loggerFactory.CreateLogger<SRFMediaProvider>();
|
||||||
_metadataCache = metadataCache;
|
_compositionFetcher = compositionFetcher;
|
||||||
_streamResolver = streamResolver;
|
_streamResolver = streamResolver;
|
||||||
_proxyService = proxyService;
|
_mediaSourceFactory = mediaSourceFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -62,51 +57,18 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
|
|
||||||
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))
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Item {ItemName} is not an SRF item, returning empty sources", item.Name);
|
|
||||||
return sources;
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogInformation("Getting media sources for URN: {Urn}, Item: {ItemName}", urn, item.Name);
|
_logger.LogDebug("GetMediaSources for URN: {Urn}, Item: {ItemName}", urn, item.Name);
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
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
|
||||||
// For regular content, use configured cache duration
|
var cacheDuration = urn.Contains("scheduled_livestream", StringComparison.OrdinalIgnoreCase) ? 5 : (int?)null;
|
||||||
var cacheDuration = urn.Contains("scheduled_livestream", StringComparison.OrdinalIgnoreCase)
|
|
||||||
? 5
|
|
||||||
: config.CacheDurationMinutes;
|
|
||||||
|
|
||||||
// Try cache first
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken, cacheDuration).ConfigureAwait(false);
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, cacheDuration);
|
|
||||||
|
|
||||||
// If not in cache, fetch from API
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
||||||
{
|
{
|
||||||
@ -131,119 +93,60 @@ public class SRFMediaProvider : IMediaSourceProvider
|
|||||||
return sources;
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get stream URL based on quality preference
|
// Get quality preference from config
|
||||||
var streamUrl = _streamResolver.GetStreamUrl(chapter, config.QualityPreference);
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
var qualityPref = config?.QualityPreference ?? QualityPreference.HD;
|
||||||
|
|
||||||
// For scheduled livestreams, always fetch fresh data to ensure stream URL is current
|
// Use item ID in hex format without dashes
|
||||||
if (chapter.Type == "SCHEDULED_LIVESTREAM" && string.IsNullOrEmpty(streamUrl))
|
var itemIdStr = item.Id.ToString("N");
|
||||||
|
|
||||||
|
// Use factory to create MediaSourceInfo
|
||||||
|
var mediaSource = await _mediaSourceFactory.CreateMediaSourceAsync(
|
||||||
|
chapter,
|
||||||
|
itemIdStr,
|
||||||
|
urn,
|
||||||
|
qualityPref,
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// For scheduled livestreams, retry with fresh data if no stream URL
|
||||||
|
if (mediaSource == null && chapter.Type == "SCHEDULED_LIVESTREAM")
|
||||||
{
|
{
|
||||||
_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);
|
// Force fresh fetch with short cache duration
|
||||||
var freshMediaComposition = await freshApiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
var freshMediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken, 1).ConfigureAwait(false);
|
||||||
|
|
||||||
if (freshMediaComposition?.ChapterList != null && freshMediaComposition.ChapterList.Count > 0)
|
if (freshMediaComposition?.ChapterList != null && freshMediaComposition.ChapterList.Count > 0)
|
||||||
{
|
{
|
||||||
var freshChapter = freshMediaComposition.ChapterList[0];
|
var freshChapter = freshMediaComposition.ChapterList[0];
|
||||||
streamUrl = _streamResolver.GetStreamUrl(freshChapter, config.QualityPreference);
|
mediaSource = await _mediaSourceFactory.CreateMediaSourceAsync(
|
||||||
|
freshChapter,
|
||||||
|
itemIdStr,
|
||||||
|
urn,
|
||||||
|
qualityPref,
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(streamUrl))
|
if (mediaSource != null)
|
||||||
{
|
{
|
||||||
// Update cache with fresh data
|
|
||||||
_metadataCache.SetMediaComposition(urn, freshMediaComposition);
|
|
||||||
chapter = freshChapter;
|
chapter = freshChapter;
|
||||||
_logger.LogInformation("URN {Urn}: Got fresh stream URL for scheduled livestream", urn);
|
_logger.LogInformation("URN {Urn}: Got fresh stream URL for scheduled livestream", urn);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(streamUrl))
|
if (mediaSource == null)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("Could not resolve stream URL for URN: {Urn}", urn);
|
_logger.LogWarning("Could not resolve stream URL for URN: {Urn}", urn);
|
||||||
return sources;
|
return sources;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Authenticate the stream URL (required for all SRF streams, especially livestreams)
|
|
||||||
if (!string.IsNullOrEmpty(streamUrl))
|
|
||||||
{
|
|
||||||
streamUrl = await _streamResolver.GetAuthenticatedStreamUrlAsync(streamUrl, cancellationToken).ConfigureAwait(false);
|
|
||||||
_logger.LogDebug("Authenticated stream URL for URN: {Urn}", urn);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 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
|
|
||||||
// Include item ID as query parameter to preserve it during transcoding
|
|
||||||
var proxyUrl = $"http://localhost:8096/Plugins/SRFPlay/Proxy/{itemIdStr}/master.m3u8?itemId={itemIdStr}";
|
|
||||||
|
|
||||||
_logger.LogInformation("Using proxy URL for item {ItemId}: {ProxyUrl}", itemIdStr, proxyUrl);
|
|
||||||
|
|
||||||
// Detect if this is a live stream
|
|
||||||
var isLiveStream = chapter.Type == "SCHEDULED_LIVESTREAM" || urn.Contains("livestream", StringComparison.OrdinalIgnoreCase);
|
|
||||||
|
|
||||||
// Create media source using proxy URL - enables DirectPlay!
|
|
||||||
var mediaSource = new MediaSourceInfo
|
|
||||||
{
|
|
||||||
Id = itemIdStr, // Must match the ID used in proxy URL registration
|
|
||||||
Name = chapter.Title,
|
|
||||||
Path = proxyUrl, // Proxy URL instead of direct Akamai URL
|
|
||||||
Protocol = MediaProtocol.Http,
|
|
||||||
Container = "hls",
|
|
||||||
SupportsDirectStream = true,
|
|
||||||
SupportsDirectPlay = true, // ✅ Enabled! Proxy handles auth
|
|
||||||
SupportsTranscoding = true,
|
|
||||||
IsRemote = false, // False because it's a local proxy endpoint
|
|
||||||
Type = MediaSourceType.Default,
|
|
||||||
RunTimeTicks = chapter.Duration > 0 ? TimeSpan.FromMilliseconds(chapter.Duration).Ticks : null,
|
|
||||||
VideoType = VideoType.VideoFile,
|
|
||||||
IsInfiniteStream = isLiveStream, // True for live streams!
|
|
||||||
RequiresOpening = false,
|
|
||||||
RequiresClosing = false,
|
|
||||||
SupportsProbing = false, // Disable probing for proxy URLs
|
|
||||||
ReadAtNativeFramerate = isLiveStream, // Read at native framerate for live streams
|
|
||||||
MediaStreams = new List<MediaBrowser.Model.Entities.MediaStream>
|
|
||||||
{
|
|
||||||
new MediaBrowser.Model.Entities.MediaStream
|
|
||||||
{
|
|
||||||
Type = MediaStreamType.Video,
|
|
||||||
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.LogDebug(
|
||||||
_logger.LogInformation(
|
"MediaSource created for {Title} - Id={Id}, DirectPlay={DirectPlay}, Transcoding={Transcoding}",
|
||||||
"MediaSource created - Id={Id}, DirectStream={DirectStream}, DirectPlay={DirectPlay}, Probing={Probing}, Container={Container}, Protocol={Protocol}, IsRemote={IsRemote}, IsLiveStream={IsLiveStream}",
|
chapter.Title,
|
||||||
mediaSource.Id,
|
mediaSource.Id,
|
||||||
mediaSource.SupportsDirectStream,
|
|
||||||
mediaSource.SupportsDirectPlay,
|
mediaSource.SupportsDirectPlay,
|
||||||
mediaSource.SupportsProbing,
|
mediaSource.SupportsTranscoding);
|
||||||
mediaSource.Container,
|
|
||||||
mediaSource.Protocol,
|
|
||||||
mediaSource.IsRemote,
|
|
||||||
isLiveStream);
|
|
||||||
_logger.LogInformation(
|
|
||||||
"MediaSource capabilities - SupportsTranscoding={Transcoding}, RequiresOpening={RequiresOpening}, RequiresClosing={RequiresClosing}, Type={Type}, IsInfiniteStream={IsInfiniteStream}",
|
|
||||||
mediaSource.SupportsTranscoding,
|
|
||||||
mediaSource.RequiresOpening,
|
|
||||||
mediaSource.RequiresClosing,
|
|
||||||
mediaSource.Type,
|
|
||||||
mediaSource.IsInfiniteStream);
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@ -269,8 +172,8 @@ 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 used - RequiresOpening is false, proxy handles authentication directly
|
||||||
// Not needed for static HTTP streams
|
_logger.LogWarning("OpenMediaSource unexpectedly called with openToken: {OpenToken}", openToken);
|
||||||
throw new NotImplementedException();
|
throw new NotSupportedException("OpenMediaSource not supported - streams use direct proxy access");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,14 +1,11 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
|
||||||
using MediaBrowser.Controller.Entities.TV;
|
using MediaBrowser.Controller.Entities.TV;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
using MediaBrowser.Model.Entities;
|
|
||||||
using MediaBrowser.Model.Providers;
|
using MediaBrowser.Model.Providers;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
@ -20,27 +17,19 @@ namespace Jellyfin.Plugin.SRFPlay.Providers;
|
|||||||
public class SRFSeriesProvider : IRemoteMetadataProvider<Series, SeriesInfo>
|
public class SRFSeriesProvider : IRemoteMetadataProvider<Series, SeriesInfo>
|
||||||
{
|
{
|
||||||
private readonly ILogger<SRFSeriesProvider> _logger;
|
private readonly ILogger<SRFSeriesProvider> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly IMediaCompositionFetcher _compositionFetcher;
|
||||||
private readonly IHttpClientFactory _httpClientFactory;
|
|
||||||
private readonly MetadataCache _metadataCache;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="SRFSeriesProvider"/> class.
|
/// Initializes a new instance of the <see cref="SRFSeriesProvider"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="logger">The logger instance.</param>
|
/// <param name="logger">The logger.</param>
|
||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="compositionFetcher">The media composition fetcher.</param>
|
||||||
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
|
||||||
/// <param name="metadataCache">The metadata cache.</param>
|
|
||||||
public SRFSeriesProvider(
|
public SRFSeriesProvider(
|
||||||
ILogger<SRFSeriesProvider> logger,
|
ILogger<SRFSeriesProvider> logger,
|
||||||
ILoggerFactory loggerFactory,
|
IMediaCompositionFetcher compositionFetcher)
|
||||||
IHttpClientFactory httpClientFactory,
|
|
||||||
MetadataCache metadataCache)
|
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_loggerFactory = loggerFactory;
|
_compositionFetcher = compositionFetcher;
|
||||||
_httpClientFactory = httpClientFactory;
|
|
||||||
_metadataCache = metadataCache;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -58,26 +47,7 @@ public class SRFSeriesProvider : IRemoteMetadataProvider<Series, SeriesInfo>
|
|||||||
{
|
{
|
||||||
_logger.LogDebug("Searching for series with URN: {Urn}", urn);
|
_logger.LogDebug("Searching for series with URN: {Urn}", urn);
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
return results;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try cache first
|
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, config.CacheDurationMinutes);
|
|
||||||
|
|
||||||
// If not in cache, fetch from API
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.Show != null)
|
if (mediaComposition?.Show != null)
|
||||||
{
|
{
|
||||||
@ -125,26 +95,7 @@ public class SRFSeriesProvider : IRemoteMetadataProvider<Series, SeriesInfo>
|
|||||||
|
|
||||||
_logger.LogDebug("Fetching metadata for series URN: {Urn}", urn);
|
_logger.LogDebug("Fetching metadata for series URN: {Urn}", urn);
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try cache first
|
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, config.CacheDurationMinutes);
|
|
||||||
|
|
||||||
// If not in cache, fetch from API
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.Show == null)
|
if (mediaComposition?.Show == null)
|
||||||
{
|
{
|
||||||
|
|||||||
@ -2,7 +2,7 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using MediaBrowser.Model.Tasks;
|
using MediaBrowser.Model.Tasks;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
@ -14,7 +14,7 @@ namespace Jellyfin.Plugin.SRFPlay.ScheduledTasks;
|
|||||||
public class ContentRefreshTask : IScheduledTask
|
public class ContentRefreshTask : IScheduledTask
|
||||||
{
|
{
|
||||||
private readonly ILogger<ContentRefreshTask> _logger;
|
private readonly ILogger<ContentRefreshTask> _logger;
|
||||||
private readonly ContentRefreshService _contentRefreshService;
|
private readonly IContentRefreshService _contentRefreshService;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="ContentRefreshTask"/> class.
|
/// Initializes a new instance of the <see cref="ContentRefreshTask"/> class.
|
||||||
@ -23,7 +23,7 @@ public class ContentRefreshTask : IScheduledTask
|
|||||||
/// <param name="contentRefreshService">The content refresh service.</param>
|
/// <param name="contentRefreshService">The content refresh service.</param>
|
||||||
public ContentRefreshTask(
|
public ContentRefreshTask(
|
||||||
ILogger<ContentRefreshTask> logger,
|
ILogger<ContentRefreshTask> logger,
|
||||||
ContentRefreshService contentRefreshService)
|
IContentRefreshService contentRefreshService)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_contentRefreshService = contentRefreshService;
|
_contentRefreshService = contentRefreshService;
|
||||||
|
|||||||
@ -2,7 +2,7 @@ using System;
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using MediaBrowser.Model.Tasks;
|
using MediaBrowser.Model.Tasks;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
@ -14,19 +14,23 @@ namespace Jellyfin.Plugin.SRFPlay.ScheduledTasks;
|
|||||||
public class ExpirationCheckTask : IScheduledTask
|
public class ExpirationCheckTask : IScheduledTask
|
||||||
{
|
{
|
||||||
private readonly ILogger<ExpirationCheckTask> _logger;
|
private readonly ILogger<ExpirationCheckTask> _logger;
|
||||||
private readonly ContentExpirationService _expirationService;
|
private readonly IContentExpirationService _expirationService;
|
||||||
|
private readonly IStreamProxyService _streamProxyService;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="ExpirationCheckTask"/> class.
|
/// Initializes a new instance of the <see cref="ExpirationCheckTask"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="logger">The logger instance.</param>
|
/// <param name="logger">The logger instance.</param>
|
||||||
/// <param name="expirationService">The content expiration service.</param>
|
/// <param name="expirationService">The content expiration service.</param>
|
||||||
|
/// <param name="streamProxyService">The stream proxy service.</param>
|
||||||
public ExpirationCheckTask(
|
public ExpirationCheckTask(
|
||||||
ILogger<ExpirationCheckTask> logger,
|
ILogger<ExpirationCheckTask> logger,
|
||||||
ContentExpirationService expirationService)
|
IContentExpirationService expirationService,
|
||||||
|
IStreamProxyService streamProxyService)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_expirationService = expirationService;
|
_expirationService = expirationService;
|
||||||
|
_streamProxyService = streamProxyService;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -77,6 +81,10 @@ public class ExpirationCheckTask : IScheduledTask
|
|||||||
progress?.Report(50);
|
progress?.Report(50);
|
||||||
var removedCount = await _expirationService.CheckAndRemoveExpiredContentAsync(cancellationToken).ConfigureAwait(false);
|
var removedCount = await _expirationService.CheckAndRemoveExpiredContentAsync(cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Clean up old stream proxy mappings
|
||||||
|
progress?.Report(75);
|
||||||
|
_streamProxyService.CleanupOldMappings();
|
||||||
|
|
||||||
progress?.Report(100);
|
progress?.Report(100);
|
||||||
_logger.LogInformation("SRF Play expiration check task completed. Removed {Count} expired items", removedCount);
|
_logger.LogInformation("SRF Play expiration check task completed. Removed {Count} expired items", removedCount);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,13 +1,14 @@
|
|||||||
|
using Jellyfin.Plugin.SRFPlay.Api;
|
||||||
using Jellyfin.Plugin.SRFPlay.Channels;
|
using Jellyfin.Plugin.SRFPlay.Channels;
|
||||||
using Jellyfin.Plugin.SRFPlay.Providers;
|
using Jellyfin.Plugin.SRFPlay.Providers;
|
||||||
using Jellyfin.Plugin.SRFPlay.ScheduledTasks;
|
using Jellyfin.Plugin.SRFPlay.ScheduledTasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Services;
|
using Jellyfin.Plugin.SRFPlay.Services;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using MediaBrowser.Controller;
|
using MediaBrowser.Controller;
|
||||||
using MediaBrowser.Controller.Channels;
|
using MediaBrowser.Controller.Channels;
|
||||||
using MediaBrowser.Controller.Plugins;
|
using MediaBrowser.Controller.Plugins;
|
||||||
using MediaBrowser.Model.Tasks;
|
using MediaBrowser.Model.Tasks;
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay;
|
namespace Jellyfin.Plugin.SRFPlay;
|
||||||
|
|
||||||
@ -19,13 +20,18 @@ public class ServiceRegistrator : IPluginServiceRegistrator
|
|||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public void RegisterServices(IServiceCollection serviceCollection, IServerApplicationHost applicationHost)
|
public void RegisterServices(IServiceCollection serviceCollection, IServerApplicationHost applicationHost)
|
||||||
{
|
{
|
||||||
// Register services as singletons
|
// Register API client factory
|
||||||
serviceCollection.AddSingleton<MetadataCache>();
|
serviceCollection.AddSingleton<ISRFApiClientFactory, SRFApiClientFactory>();
|
||||||
serviceCollection.AddSingleton<StreamUrlResolver>();
|
|
||||||
serviceCollection.AddSingleton<ContentExpirationService>();
|
// Register core services with interfaces
|
||||||
serviceCollection.AddSingleton<ContentRefreshService>();
|
serviceCollection.AddSingleton<IMetadataCache, MetadataCache>();
|
||||||
serviceCollection.AddSingleton<CategoryService>();
|
serviceCollection.AddSingleton<IStreamUrlResolver, StreamUrlResolver>();
|
||||||
serviceCollection.AddSingleton<StreamProxyService>(); // Stream proxy service
|
serviceCollection.AddSingleton<IMediaCompositionFetcher, MediaCompositionFetcher>();
|
||||||
|
serviceCollection.AddSingleton<IStreamProxyService, StreamProxyService>();
|
||||||
|
serviceCollection.AddSingleton<IMediaSourceFactory, MediaSourceFactory>();
|
||||||
|
serviceCollection.AddSingleton<IContentExpirationService, ContentExpirationService>();
|
||||||
|
serviceCollection.AddSingleton<IContentRefreshService, ContentRefreshService>();
|
||||||
|
serviceCollection.AddSingleton<ICategoryService, CategoryService>();
|
||||||
|
|
||||||
// Register metadata providers
|
// Register metadata providers
|
||||||
serviceCollection.AddSingleton<SRFSeriesProvider>();
|
serviceCollection.AddSingleton<SRFSeriesProvider>();
|
||||||
|
|||||||
@ -1,11 +1,11 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics.CodeAnalysis;
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Api;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Services;
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
@ -13,10 +13,10 @@ namespace Jellyfin.Plugin.SRFPlay.Services;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Service for managing topic/category data and filtering.
|
/// Service for managing topic/category data and filtering.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class CategoryService
|
public class CategoryService : ICategoryService
|
||||||
{
|
{
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger<CategoryService> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly ISRFApiClientFactory _apiClientFactory;
|
||||||
private readonly TimeSpan _topicsCacheDuration = TimeSpan.FromHours(24);
|
private readonly TimeSpan _topicsCacheDuration = TimeSpan.FromHours(24);
|
||||||
private Dictionary<string, PlayV3Topic>? _topicsCache;
|
private Dictionary<string, PlayV3Topic>? _topicsCache;
|
||||||
private DateTime _topicsCacheExpiry = DateTime.MinValue;
|
private DateTime _topicsCacheExpiry = DateTime.MinValue;
|
||||||
@ -24,19 +24,15 @@ public class CategoryService
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="CategoryService"/> class.
|
/// Initializes a new instance of the <see cref="CategoryService"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="logger">The logger.</param>
|
||||||
public CategoryService(ILoggerFactory loggerFactory)
|
/// <param name="apiClientFactory">The API client factory.</param>
|
||||||
|
public CategoryService(ILogger<CategoryService> logger, ISRFApiClientFactory apiClientFactory)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
_logger = logger;
|
||||||
_logger = loggerFactory.CreateLogger<CategoryService>();
|
_apiClientFactory = apiClientFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <inheritdoc />
|
||||||
/// Gets all topics for a business unit.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="businessUnit">The business unit.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>List of topics.</returns>
|
|
||||||
public async Task<List<PlayV3Topic>> GetTopicsAsync(string businessUnit, CancellationToken cancellationToken = default)
|
public async Task<List<PlayV3Topic>> GetTopicsAsync(string businessUnit, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
// Return cached topics if still valid
|
// Return cached topics if still valid
|
||||||
@ -47,7 +43,7 @@ public class CategoryService
|
|||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogInformation("Fetching topics for business unit: {BusinessUnit}", businessUnit);
|
_logger.LogInformation("Fetching topics for business unit: {BusinessUnit}", businessUnit);
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
var topics = await apiClient.GetAllTopicsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
var topics = await apiClient.GetAllTopicsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (topics != null && topics.Count > 0)
|
if (topics != null && topics.Count > 0)
|
||||||
@ -64,13 +60,7 @@ public class CategoryService
|
|||||||
return topics ?? new List<PlayV3Topic>();
|
return topics ?? new List<PlayV3Topic>();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <inheritdoc />
|
||||||
/// Gets a topic by ID.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="topicId">The topic ID.</param>
|
|
||||||
/// <param name="businessUnit">The business unit.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>The topic, or null if not found.</returns>
|
|
||||||
public async Task<PlayV3Topic?> GetTopicByIdAsync(string topicId, string businessUnit, CancellationToken cancellationToken = default)
|
public async Task<PlayV3Topic?> GetTopicByIdAsync(string topicId, string businessUnit, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
// Ensure topics are loaded
|
// Ensure topics are loaded
|
||||||
@ -82,70 +72,14 @@ public class CategoryService
|
|||||||
return _topicsCache?.GetValueOrDefault(topicId);
|
return _topicsCache?.GetValueOrDefault(topicId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <inheritdoc />
|
||||||
/// Filters shows by topic ID.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="shows">The shows to filter.</param>
|
|
||||||
/// <param name="topicId">The topic ID to filter by.</param>
|
|
||||||
/// <returns>Filtered list of shows.</returns>
|
|
||||||
public IReadOnlyList<PlayV3Show> FilterShowsByTopic(IReadOnlyList<PlayV3Show> shows, string topicId)
|
|
||||||
{
|
|
||||||
if (string.IsNullOrEmpty(topicId))
|
|
||||||
{
|
|
||||||
return shows;
|
|
||||||
}
|
|
||||||
|
|
||||||
return shows
|
|
||||||
.Where(s => s.TopicList != null && s.TopicList.Contains(topicId))
|
|
||||||
.ToList();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Groups shows by their topics.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="shows">The shows to group.</param>
|
|
||||||
/// <returns>Dictionary mapping topic IDs to shows.</returns>
|
|
||||||
public IReadOnlyDictionary<string, List<PlayV3Show>> GroupShowsByTopics(IReadOnlyList<PlayV3Show> shows)
|
|
||||||
{
|
|
||||||
var groupedShows = new Dictionary<string, List<PlayV3Show>>();
|
|
||||||
|
|
||||||
foreach (var show in shows)
|
|
||||||
{
|
|
||||||
if (show.TopicList == null || show.TopicList.Count == 0)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var topicId in show.TopicList)
|
|
||||||
{
|
|
||||||
if (!groupedShows.TryGetValue(topicId, out var showList))
|
|
||||||
{
|
|
||||||
showList = new List<PlayV3Show>();
|
|
||||||
groupedShows[topicId] = showList;
|
|
||||||
}
|
|
||||||
|
|
||||||
showList.Add(show);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return groupedShows;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets shows for a specific topic, sorted by number of episodes.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="topicId">The topic ID.</param>
|
|
||||||
/// <param name="businessUnit">The business unit.</param>
|
|
||||||
/// <param name="maxResults">Maximum number of results to return.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>List of shows for the topic.</returns>
|
|
||||||
public async Task<List<PlayV3Show>> GetShowsByTopicAsync(
|
public async Task<List<PlayV3Show>> GetShowsByTopicAsync(
|
||||||
string topicId,
|
string topicId,
|
||||||
string businessUnit,
|
string businessUnit,
|
||||||
int maxResults = 50,
|
int maxResults = 50,
|
||||||
CancellationToken cancellationToken = default)
|
CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
var allShows = await apiClient.GetAllShowsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
var allShows = await apiClient.GetAllShowsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (allShows == null || allShows.Count == 0)
|
if (allShows == null || allShows.Count == 0)
|
||||||
@ -164,43 +98,29 @@ public class CategoryService
|
|||||||
return filteredShows;
|
return filteredShows;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <inheritdoc />
|
||||||
/// Gets video count for each topic.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="shows">The shows to analyze.</param>
|
|
||||||
/// <returns>Dictionary mapping topic IDs to video counts.</returns>
|
|
||||||
public IReadOnlyDictionary<string, int> GetVideoCountByTopic(IReadOnlyList<PlayV3Show> shows)
|
|
||||||
{
|
|
||||||
var topicCounts = new Dictionary<string, int>();
|
|
||||||
|
|
||||||
foreach (var show in shows)
|
|
||||||
{
|
|
||||||
if (show.TopicList == null || show.TopicList.Count == 0)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var topicId in show.TopicList)
|
|
||||||
{
|
|
||||||
if (!topicCounts.TryGetValue(topicId, out var count))
|
|
||||||
{
|
|
||||||
count = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
topicCounts[topicId] = count + show.NumberOfEpisodes;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return topicCounts;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Clears the topics cache.
|
|
||||||
/// </summary>
|
|
||||||
public void ClearCache()
|
public void ClearCache()
|
||||||
{
|
{
|
||||||
_topicsCache = null;
|
_topicsCache = null;
|
||||||
_topicsCacheExpiry = DateTime.MinValue;
|
_topicsCacheExpiry = DateTime.MinValue;
|
||||||
_logger.LogInformation("Topics cache cleared");
|
_logger.LogInformation("Topics cache cleared");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Filters shows by topic ID.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="shows">The shows to filter.</param>
|
||||||
|
/// <param name="topicId">The topic ID to filter by.</param>
|
||||||
|
/// <returns>Filtered list of shows.</returns>
|
||||||
|
private static IReadOnlyList<PlayV3Show> FilterShowsByTopic(IReadOnlyList<PlayV3Show> shows, string topicId)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(topicId))
|
||||||
|
{
|
||||||
|
return shows;
|
||||||
|
}
|
||||||
|
|
||||||
|
return shows
|
||||||
|
.Where(s => s.TopicList != null && s.TopicList.Contains(topicId))
|
||||||
|
.ToList();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -3,10 +3,9 @@ using System.Collections.Generic;
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
using MediaBrowser.Model.IO;
|
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Services;
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
@ -14,13 +13,12 @@ namespace Jellyfin.Plugin.SRFPlay.Services;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Service for managing content expiration.
|
/// Service for managing content expiration.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class ContentExpirationService
|
public class ContentExpirationService : IContentExpirationService
|
||||||
{
|
{
|
||||||
private readonly ILogger<ContentExpirationService> _logger;
|
private readonly ILogger<ContentExpirationService> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
|
||||||
private readonly ILibraryManager _libraryManager;
|
private readonly ILibraryManager _libraryManager;
|
||||||
private readonly StreamUrlResolver _streamResolver;
|
private readonly IStreamUrlResolver _streamResolver;
|
||||||
private readonly MetadataCache _metadataCache;
|
private readonly IMediaCompositionFetcher _compositionFetcher;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="ContentExpirationService"/> class.
|
/// Initializes a new instance of the <see cref="ContentExpirationService"/> class.
|
||||||
@ -28,18 +26,17 @@ public class ContentExpirationService
|
|||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
/// <param name="libraryManager">The library manager.</param>
|
/// <param name="libraryManager">The library manager.</param>
|
||||||
/// <param name="streamResolver">The stream URL resolver.</param>
|
/// <param name="streamResolver">The stream URL resolver.</param>
|
||||||
/// <param name="metadataCache">The metadata cache.</param>
|
/// <param name="compositionFetcher">The media composition fetcher.</param>
|
||||||
public ContentExpirationService(
|
public ContentExpirationService(
|
||||||
ILoggerFactory loggerFactory,
|
ILoggerFactory loggerFactory,
|
||||||
ILibraryManager libraryManager,
|
ILibraryManager libraryManager,
|
||||||
StreamUrlResolver streamResolver,
|
IStreamUrlResolver streamResolver,
|
||||||
MetadataCache metadataCache)
|
IMediaCompositionFetcher compositionFetcher)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
|
||||||
_logger = loggerFactory.CreateLogger<ContentExpirationService>();
|
_logger = loggerFactory.CreateLogger<ContentExpirationService>();
|
||||||
_libraryManager = libraryManager;
|
_libraryManager = libraryManager;
|
||||||
_streamResolver = streamResolver;
|
_streamResolver = streamResolver;
|
||||||
_metadataCache = metadataCache;
|
_compositionFetcher = compositionFetcher;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -119,26 +116,7 @@ public class ContentExpirationService
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Try cache first
|
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, config.CacheDurationMinutes);
|
|
||||||
|
|
||||||
// If not in cache, fetch from API
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
||||||
{
|
{
|
||||||
@ -196,24 +174,7 @@ public class ContentExpirationService
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
var config = Plugin.Instance?.Configuration;
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
var mediaComposition = _metadataCache.GetMediaComposition(urn, config.CacheDurationMinutes);
|
|
||||||
|
|
||||||
if (mediaComposition == null)
|
|
||||||
{
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (mediaComposition != null)
|
|
||||||
{
|
|
||||||
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mediaComposition?.ChapterList != null && mediaComposition.ChapterList.Count > 0)
|
if (mediaComposition?.ChapterList != null && mediaComposition.ChapterList.Count > 0)
|
||||||
{
|
{
|
||||||
|
|||||||
@ -4,7 +4,7 @@ using System.Linq;
|
|||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api;
|
using Jellyfin.Plugin.SRFPlay.Api;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Services;
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
@ -12,24 +12,22 @@ namespace Jellyfin.Plugin.SRFPlay.Services;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Service for refreshing content from SRF API.
|
/// Service for refreshing content from SRF API.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class ContentRefreshService
|
public class ContentRefreshService : IContentRefreshService
|
||||||
{
|
{
|
||||||
private readonly ILogger<ContentRefreshService> _logger;
|
private readonly ILogger<ContentRefreshService> _logger;
|
||||||
private readonly ILoggerFactory _loggerFactory;
|
private readonly ISRFApiClientFactory _apiClientFactory;
|
||||||
private readonly MetadataCache _metadataCache;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="ContentRefreshService"/> class.
|
/// Initializes a new instance of the <see cref="ContentRefreshService"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="loggerFactory">The logger factory.</param>
|
/// <param name="loggerFactory">The logger factory.</param>
|
||||||
/// <param name="metadataCache">The metadata cache.</param>
|
/// <param name="apiClientFactory">The API client factory.</param>
|
||||||
public ContentRefreshService(
|
public ContentRefreshService(
|
||||||
ILoggerFactory loggerFactory,
|
ILoggerFactory loggerFactory,
|
||||||
MetadataCache metadataCache)
|
ISRFApiClientFactory apiClientFactory)
|
||||||
{
|
{
|
||||||
_loggerFactory = loggerFactory;
|
|
||||||
_logger = loggerFactory.CreateLogger<ContentRefreshService>();
|
_logger = loggerFactory.CreateLogger<ContentRefreshService>();
|
||||||
_metadataCache = metadataCache;
|
_apiClientFactory = apiClientFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -39,111 +37,20 @@ public class ContentRefreshService
|
|||||||
/// <returns>List of URNs for new content.</returns>
|
/// <returns>List of URNs for new content.</returns>
|
||||||
public async Task<List<string>> RefreshLatestContentAsync(CancellationToken cancellationToken)
|
public async Task<List<string>> RefreshLatestContentAsync(CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var urns = new List<string>();
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
if (config == null || !config.EnableLatestContent)
|
||||||
try
|
|
||||||
{
|
{
|
||||||
var config = Plugin.Instance?.Configuration;
|
_logger.LogDebug("Latest content refresh is disabled");
|
||||||
if (config == null || !config.EnableLatestContent)
|
return new List<string>();
|
||||||
{
|
|
||||||
_logger.LogDebug("Latest content refresh is disabled");
|
|
||||||
return urns;
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogInformation("Refreshing latest content for business unit: {BusinessUnit}", config.BusinessUnit);
|
|
||||||
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
var businessUnit = config.BusinessUnit.ToString().ToLowerInvariant();
|
|
||||||
|
|
||||||
// Get all shows from Play v3 API
|
|
||||||
var shows = await apiClient.GetAllShowsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (shows == null || shows.Count == 0)
|
|
||||||
{
|
|
||||||
_logger.LogWarning("No shows found for business unit: {BusinessUnit}", config.BusinessUnit);
|
|
||||||
return urns;
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogInformation("Found {Count} shows, fetching latest episodes from each", shows.Count);
|
|
||||||
|
|
||||||
// Get latest episodes from each show (limit to 20 shows to avoid overwhelming)
|
|
||||||
var showsToFetch = shows.Where(s => s.NumberOfEpisodes > 0)
|
|
||||||
.OrderByDescending(s => s.NumberOfEpisodes)
|
|
||||||
.Take(20)
|
|
||||||
.ToList();
|
|
||||||
|
|
||||||
foreach (var show in showsToFetch)
|
|
||||||
{
|
|
||||||
if (show.Id == null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var videos = await apiClient.GetVideosForShowAsync(businessUnit, show.Id, cancellationToken).ConfigureAwait(false);
|
|
||||||
if (videos != null && videos.Count > 0)
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Show {Show} ({ShowId}): Found {Count} videos", show.Title, show.Id, videos.Count);
|
|
||||||
|
|
||||||
// Filter to videos that are actually published (validFrom in the past)
|
|
||||||
var now = DateTime.UtcNow;
|
|
||||||
var publishedVideos = videos.Where(v =>
|
|
||||||
v.ValidFrom == null || v.ValidFrom.Value.ToUniversalTime() <= now).ToList();
|
|
||||||
|
|
||||||
_logger.LogDebug("Show {Show}: {PublishedCount} published out of {TotalCount} videos", show.Title, publishedVideos.Count, videos.Count);
|
|
||||||
|
|
||||||
if (publishedVideos.Count > 0)
|
|
||||||
{
|
|
||||||
// Take only the most recent published video from each show
|
|
||||||
var latestVideo = publishedVideos.OrderByDescending(v => v.Date).FirstOrDefault();
|
|
||||||
if (latestVideo?.Urn != null)
|
|
||||||
{
|
|
||||||
urns.Add(latestVideo.Urn);
|
|
||||||
_logger.LogInformation(
|
|
||||||
"Added latest video from show {Show}: {Title} (URN: {Urn}, Date: {Date}, ValidFrom: {ValidFrom}, ValidTo: {ValidTo})",
|
|
||||||
show.Title,
|
|
||||||
latestVideo.Title,
|
|
||||||
latestVideo.Urn,
|
|
||||||
latestVideo.Date,
|
|
||||||
latestVideo.ValidFrom,
|
|
||||||
latestVideo.ValidTo);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogWarning("Show {Show}: Latest video has null URN", show.Title);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Show {Show} has no published videos yet", show.Title);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Show {Show} ({ShowId}): No videos returned from API", show.Title, show.Id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
_logger.LogWarning(ex, "Error fetching videos for show {ShowId}", show.Id);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Respect cancellation
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogInformation("Refreshed {Count} latest content items from {ShowCount} shows", urns.Count, showsToFetch.Count);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
_logger.LogError(ex, "Error refreshing latest content");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return urns;
|
return await FetchVideosFromShowsAsync(
|
||||||
|
config.BusinessUnit.ToString().ToLowerInvariant(),
|
||||||
|
minEpisodeCount: 0,
|
||||||
|
maxShows: 20,
|
||||||
|
videosPerShow: 1,
|
||||||
|
contentType: "latest",
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -153,43 +60,62 @@ public class ContentRefreshService
|
|||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
/// <returns>List of URNs for trending content.</returns>
|
/// <returns>List of URNs for trending content.</returns>
|
||||||
public async Task<List<string>> RefreshTrendingContentAsync(CancellationToken cancellationToken)
|
public async Task<List<string>> RefreshTrendingContentAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
if (config == null || !config.EnableTrendingContent)
|
||||||
|
{
|
||||||
|
_logger.LogDebug("Trending content refresh is disabled");
|
||||||
|
return new List<string>();
|
||||||
|
}
|
||||||
|
|
||||||
|
return await FetchVideosFromShowsAsync(
|
||||||
|
config.BusinessUnit.ToString().ToLowerInvariant(),
|
||||||
|
minEpisodeCount: 10,
|
||||||
|
maxShows: 15,
|
||||||
|
videosPerShow: 2,
|
||||||
|
contentType: "trending",
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fetches videos from shows based on filter criteria.
|
||||||
|
/// </summary>
|
||||||
|
private async Task<List<string>> FetchVideosFromShowsAsync(
|
||||||
|
string businessUnit,
|
||||||
|
int minEpisodeCount,
|
||||||
|
int maxShows,
|
||||||
|
int videosPerShow,
|
||||||
|
string contentType,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var urns = new List<string>();
|
var urns = new List<string>();
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var config = Plugin.Instance?.Configuration;
|
_logger.LogInformation("Refreshing {ContentType} content for business unit: {BusinessUnit}", contentType, businessUnit);
|
||||||
if (config == null || !config.EnableTrendingContent)
|
|
||||||
{
|
|
||||||
_logger.LogDebug("Trending content refresh is disabled");
|
|
||||||
return urns;
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogInformation("Refreshing trending content for business unit: {BusinessUnit}", config.BusinessUnit);
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
|
|
||||||
using var apiClient = new SRFApiClient(_loggerFactory);
|
|
||||||
var businessUnit = config.BusinessUnit.ToString().ToLowerInvariant();
|
|
||||||
|
|
||||||
// Get all shows from Play v3 API
|
|
||||||
var shows = await apiClient.GetAllShowsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
var shows = await apiClient.GetAllShowsAsync(businessUnit, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (shows == null || shows.Count == 0)
|
if (shows == null || shows.Count == 0)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("No shows found for business unit: {BusinessUnit}", config.BusinessUnit);
|
_logger.LogWarning("No shows found for business unit: {BusinessUnit}", businessUnit);
|
||||||
return urns;
|
return urns;
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogInformation("Found {Count} shows, fetching popular content", shows.Count);
|
_logger.LogInformation("Found {Count} shows, fetching {ContentType} content", shows.Count, contentType);
|
||||||
|
|
||||||
// Get videos from popular shows (those with many episodes)
|
var filteredShows = shows
|
||||||
var popularShows = shows.Where(s => s.NumberOfEpisodes > 10)
|
.Where(s => s.NumberOfEpisodes > minEpisodeCount)
|
||||||
.OrderByDescending(s => s.NumberOfEpisodes)
|
.OrderByDescending(s => s.NumberOfEpisodes)
|
||||||
.Take(15)
|
.Take(maxShows)
|
||||||
.ToList();
|
.ToList();
|
||||||
|
|
||||||
foreach (var show in popularShows)
|
var now = DateTime.UtcNow;
|
||||||
|
|
||||||
|
foreach (var show in filteredShows)
|
||||||
{
|
{
|
||||||
if (show.Id == null)
|
if (show.Id == null || cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -197,64 +123,46 @@ public class ContentRefreshService
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var videos = await apiClient.GetVideosForShowAsync(businessUnit, show.Id, cancellationToken).ConfigureAwait(false);
|
var videos = await apiClient.GetVideosForShowAsync(businessUnit, show.Id, cancellationToken).ConfigureAwait(false);
|
||||||
if (videos != null && videos.Count > 0)
|
if (videos == null || videos.Count == 0)
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Show {Show} ({ShowId}): Found {Count} videos for trending", show.Title, show.Id, videos.Count);
|
_logger.LogDebug("Show {Show} ({ShowId}): No videos returned from API", show.Title, show.Id);
|
||||||
|
continue;
|
||||||
// Filter to videos that are actually published (validFrom in the past)
|
|
||||||
var now = DateTime.UtcNow;
|
|
||||||
var publishedVideos = videos.Where(v =>
|
|
||||||
v.ValidFrom == null || v.ValidFrom.Value.ToUniversalTime() <= now).ToList();
|
|
||||||
|
|
||||||
_logger.LogDebug("Show {Show}: {PublishedCount} published out of {TotalCount} videos for trending", show.Title, publishedVideos.Count, videos.Count);
|
|
||||||
|
|
||||||
if (publishedVideos.Count > 0)
|
|
||||||
{
|
|
||||||
// Take 2 recent published videos from each popular show
|
|
||||||
var recentVideos = publishedVideos.OrderByDescending(v => v.Date).Take(2);
|
|
||||||
foreach (var video in recentVideos)
|
|
||||||
{
|
|
||||||
if (video.Urn != null)
|
|
||||||
{
|
|
||||||
urns.Add(video.Urn);
|
|
||||||
_logger.LogInformation(
|
|
||||||
"Added trending video from show {Show}: {Title} (URN: {Urn}, Date: {Date}, ValidFrom: {ValidFrom}, ValidTo: {ValidTo})",
|
|
||||||
show.Title,
|
|
||||||
video.Title,
|
|
||||||
video.Urn,
|
|
||||||
video.Date,
|
|
||||||
video.ValidFrom,
|
|
||||||
video.ValidTo);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.LogWarning("Show {Show}: Trending video has null URN - {Title}", show.Title, video.Title);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
_logger.LogDebug("Show {Show} ({ShowId}): Found {Count} videos", show.Title, show.Id, videos.Count);
|
||||||
|
|
||||||
|
// Filter to videos that are actually published (validFrom in the past)
|
||||||
|
var publishedVideos = videos
|
||||||
|
.Where(v => v.ValidFrom == null || v.ValidFrom.Value.ToUniversalTime() <= now)
|
||||||
|
.OrderByDescending(v => v.Date)
|
||||||
|
.Take(videosPerShow)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
foreach (var video in publishedVideos)
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Show {Show} ({ShowId}): No videos returned from API for trending", show.Title, show.Id);
|
if (video.Urn != null)
|
||||||
|
{
|
||||||
|
urns.Add(video.Urn);
|
||||||
|
_logger.LogDebug(
|
||||||
|
"Added {ContentType} video from show {Show}: {Title} (URN: {Urn})",
|
||||||
|
contentType,
|
||||||
|
show.Title,
|
||||||
|
video.Title,
|
||||||
|
video.Urn);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogWarning(ex, "Error fetching videos for show {ShowId}", show.Id);
|
_logger.LogWarning(ex, "Error fetching videos for show {ShowId}", show.Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Respect cancellation
|
|
||||||
if (cancellationToken.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogInformation("Refreshed {Count} trending content items from {ShowCount} shows", urns.Count, popularShows.Count);
|
_logger.LogInformation("Refreshed {Count} {ContentType} content items from {ShowCount} shows", urns.Count, contentType, filteredShows.Count);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Error refreshing trending content");
|
_logger.LogError(ex, "Error refreshing {ContentType} content", contentType);
|
||||||
}
|
}
|
||||||
|
|
||||||
return urns;
|
return urns;
|
||||||
|
|||||||
@ -0,0 +1,48 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for managing topic/category data and filtering.
|
||||||
|
/// </summary>
|
||||||
|
public interface ICategoryService
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets all topics for a business unit.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="businessUnit">The business unit.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>List of topics.</returns>
|
||||||
|
Task<List<PlayV3Topic>> GetTopicsAsync(string businessUnit, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets a topic by ID.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="topicId">The topic ID.</param>
|
||||||
|
/// <param name="businessUnit">The business unit.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>The topic, or null if not found.</returns>
|
||||||
|
Task<PlayV3Topic?> GetTopicByIdAsync(string topicId, string businessUnit, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets shows for a specific topic, sorted by number of episodes.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="topicId">The topic ID.</param>
|
||||||
|
/// <param name="businessUnit">The business unit.</param>
|
||||||
|
/// <param name="maxResults">Maximum number of results to return.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>List of shows for the topic.</returns>
|
||||||
|
Task<List<PlayV3Show>> GetShowsByTopicAsync(
|
||||||
|
string topicId,
|
||||||
|
string businessUnit,
|
||||||
|
int maxResults = 50,
|
||||||
|
CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Clears the topics cache.
|
||||||
|
/// </summary>
|
||||||
|
void ClearCache();
|
||||||
|
}
|
||||||
@ -0,0 +1,24 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for managing content expiration.
|
||||||
|
/// </summary>
|
||||||
|
public interface IContentExpirationService
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Checks for expired content and removes it from the library.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>The number of items removed.</returns>
|
||||||
|
Task<int> CheckAndRemoveExpiredContentAsync(CancellationToken cancellationToken);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets statistics about content expiration.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>Tuple with total count, expired count, and items expiring soon.</returns>
|
||||||
|
Task<(int Total, int Expired, int ExpiringSoon)> GetExpirationStatisticsAsync(CancellationToken cancellationToken);
|
||||||
|
}
|
||||||
@ -0,0 +1,39 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for refreshing content from SRF API.
|
||||||
|
/// </summary>
|
||||||
|
public interface IContentRefreshService
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Refreshes latest content from SRF API using Play v3.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>List of URNs for new content.</returns>
|
||||||
|
Task<List<string>> RefreshLatestContentAsync(CancellationToken cancellationToken);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Refreshes trending content from SRF API using Play v3.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>List of URNs for trending content.</returns>
|
||||||
|
Task<List<string>> RefreshTrendingContentAsync(CancellationToken cancellationToken);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Refreshes all content (latest and trending).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>Tuple with counts of latest and trending items.</returns>
|
||||||
|
Task<(int LatestCount, int TrendingCount)> RefreshAllContentAsync(CancellationToken cancellationToken);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets content recommendations (combines latest and trending).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>List of recommended URNs.</returns>
|
||||||
|
Task<List<string>> GetRecommendedContentAsync(CancellationToken cancellationToken);
|
||||||
|
}
|
||||||
@ -0,0 +1,23 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for fetching media composition with caching support.
|
||||||
|
/// </summary>
|
||||||
|
public interface IMediaCompositionFetcher
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets media composition by URN, using cache if available.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="urn">The URN to fetch.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <param name="cacheDurationOverride">Optional override for cache duration (e.g., 5 min for livestreams).</param>
|
||||||
|
/// <returns>The media composition, or null if not found.</returns>
|
||||||
|
Task<MediaComposition?> GetMediaCompositionAsync(
|
||||||
|
string urn,
|
||||||
|
CancellationToken cancellationToken,
|
||||||
|
int? cacheDurationOverride = null);
|
||||||
|
}
|
||||||
@ -0,0 +1,51 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
using MediaBrowser.Model.Dto;
|
||||||
|
using MediaBrowser.Model.Entities;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Factory for creating MediaSourceInfo objects with consistent configuration.
|
||||||
|
/// </summary>
|
||||||
|
public interface IMediaSourceFactory
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a MediaSourceInfo for a chapter with proper stream authentication and proxy registration.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="chapter">The chapter containing stream resources.</param>
|
||||||
|
/// <param name="itemId">The unique item ID for proxy registration.</param>
|
||||||
|
/// <param name="urn">The URN of the content.</param>
|
||||||
|
/// <param name="qualityPreference">The preferred quality.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>A configured MediaSourceInfo, or null if stream URL cannot be resolved.</returns>
|
||||||
|
Task<MediaSourceInfo?> CreateMediaSourceAsync(
|
||||||
|
Chapter chapter,
|
||||||
|
string itemId,
|
||||||
|
string urn,
|
||||||
|
QualityPreference qualityPreference,
|
||||||
|
CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Builds a proxy URL for an item.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <returns>The full proxy URL including server address.</returns>
|
||||||
|
string BuildProxyUrl(string itemId);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the server base URL (configured public URL or smart URL).
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The server base URL without trailing slash.</returns>
|
||||||
|
string GetServerBaseUrl();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Creates MediaStream metadata based on quality preference.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="quality">The quality preference.</param>
|
||||||
|
/// <returns>List of MediaStream objects for video and audio.</returns>
|
||||||
|
IReadOnlyList<MediaStream> CreateMediaStreams(QualityPreference quality);
|
||||||
|
}
|
||||||
@ -0,0 +1,41 @@
|
|||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for caching metadata from SRF API.
|
||||||
|
/// </summary>
|
||||||
|
public interface IMetadataCache
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets cached media composition by URN.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="urn">The URN.</param>
|
||||||
|
/// <param name="cacheDurationMinutes">The cache duration in minutes.</param>
|
||||||
|
/// <returns>The cached media composition, or null if not found or expired.</returns>
|
||||||
|
MediaComposition? GetMediaComposition(string urn, int cacheDurationMinutes);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sets media composition in cache.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="urn">The URN.</param>
|
||||||
|
/// <param name="mediaComposition">The media composition to cache.</param>
|
||||||
|
void SetMediaComposition(string urn, MediaComposition mediaComposition);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Removes media composition from cache.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="urn">The URN.</param>
|
||||||
|
void RemoveMediaComposition(string urn);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Clears all cached data.
|
||||||
|
/// </summary>
|
||||||
|
void Clear();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the cache statistics.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A tuple with cache count and size estimate.</returns>
|
||||||
|
(int Count, long SizeEstimate) GetStatistics();
|
||||||
|
}
|
||||||
@ -0,0 +1,67 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for proxying SRF Play streams and managing authentication.
|
||||||
|
/// </summary>
|
||||||
|
public interface IStreamProxyService
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Registers a stream for proxying with an already-authenticated URL.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="authenticatedUrl">The authenticated stream URL.</param>
|
||||||
|
/// <param name="urn">The SRF URN for this content (used for re-fetching fresh URLs).</param>
|
||||||
|
/// <param name="isLiveStream">Whether this is a livestream (livestreams always fetch fresh URLs).</param>
|
||||||
|
void RegisterStream(string itemId, string authenticatedUrl, string? urn = null, bool isLiveStream = false);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Registers a stream for deferred authentication (authenticates on first playback request).
|
||||||
|
/// Use this when browsing to avoid wasting 30-second tokens before the user clicks play.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="unauthenticatedUrl">The unauthenticated stream URL.</param>
|
||||||
|
/// <param name="urn">The SRF URN for this content.</param>
|
||||||
|
/// <param name="isLiveStream">Whether this is a livestream.</param>
|
||||||
|
void RegisterStreamDeferred(string itemId, string unauthenticatedUrl, string? urn = null, bool isLiveStream = false);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets stream metadata for an item (URN and isLiveStream flag).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <returns>A tuple of (URN, IsLiveStream), or null if not found.</returns>
|
||||||
|
(string? Urn, bool IsLiveStream)? GetStreamMetadata(string itemId);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the authenticated URL for an item.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The authenticated URL, or null if not found or expired.</returns>
|
||||||
|
Task<string?> GetAuthenticatedUrlAsync(string itemId, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <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.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The rewritten manifest content.</returns>
|
||||||
|
Task<string?> GetRewrittenManifestAsync(string itemId, string baseProxyUrl, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <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>
|
||||||
|
Task<byte[]?> GetSegmentAsync(string itemId, string segmentPath, CancellationToken cancellationToken = default);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Cleans up old and expired stream mappings.
|
||||||
|
/// </summary>
|
||||||
|
void CleanupOldMappings();
|
||||||
|
}
|
||||||
@ -0,0 +1,42 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Interface for resolving stream URLs from media composition resources.
|
||||||
|
/// </summary>
|
||||||
|
public interface IStreamUrlResolver
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the best stream URL from a chapter based on quality preference.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="chapter">The chapter containing resources.</param>
|
||||||
|
/// <param name="qualityPreference">The quality preference.</param>
|
||||||
|
/// <returns>The stream URL, or null if no suitable stream found.</returns>
|
||||||
|
string? GetStreamUrl(Chapter chapter, QualityPreference qualityPreference);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Checks if a chapter has non-DRM playable content.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="chapter">The chapter to check.</param>
|
||||||
|
/// <returns>True if playable content is available.</returns>
|
||||||
|
bool HasPlayableContent(Chapter chapter);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Checks if content is expired based on ValidTo date.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="chapter">The chapter to check.</param>
|
||||||
|
/// <returns>True if the content is expired.</returns>
|
||||||
|
bool IsContentExpired(Chapter chapter);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Authenticates a stream URL by fetching an Akamai token.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="streamUrl">The unauthenticated stream URL.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
|
/// <returns>The authenticated stream URL with token.</returns>
|
||||||
|
Task<string> GetAuthenticatedStreamUrlAsync(string streamUrl, CancellationToken cancellationToken = default);
|
||||||
|
}
|
||||||
82
Jellyfin.Plugin.SRFPlay/Services/MediaCompositionFetcher.cs
Normal file
82
Jellyfin.Plugin.SRFPlay/Services/MediaCompositionFetcher.cs
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Service for fetching media composition with caching support.
|
||||||
|
/// This consolidates the cache-check → API-fetch → cache-store pattern.
|
||||||
|
/// </summary>
|
||||||
|
public class MediaCompositionFetcher : IMediaCompositionFetcher
|
||||||
|
{
|
||||||
|
private readonly ILogger<MediaCompositionFetcher> _logger;
|
||||||
|
private readonly IMetadataCache _metadataCache;
|
||||||
|
private readonly ISRFApiClientFactory _apiClientFactory;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="MediaCompositionFetcher"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="logger">The logger.</param>
|
||||||
|
/// <param name="metadataCache">The metadata cache.</param>
|
||||||
|
/// <param name="apiClientFactory">The API client factory.</param>
|
||||||
|
public MediaCompositionFetcher(
|
||||||
|
ILogger<MediaCompositionFetcher> logger,
|
||||||
|
IMetadataCache metadataCache,
|
||||||
|
ISRFApiClientFactory apiClientFactory)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
_metadataCache = metadataCache;
|
||||||
|
_apiClientFactory = apiClientFactory;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public async Task<MediaComposition?> GetMediaCompositionAsync(
|
||||||
|
string urn,
|
||||||
|
CancellationToken cancellationToken,
|
||||||
|
int? cacheDurationOverride = null)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(urn))
|
||||||
|
{
|
||||||
|
_logger.LogDebug("GetMediaCompositionAsync called with null/empty URN");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
if (config == null)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Plugin configuration is null, cannot fetch media composition");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var cacheDuration = cacheDurationOverride ?? config.CacheDurationMinutes;
|
||||||
|
|
||||||
|
// Try cache first
|
||||||
|
var mediaComposition = _metadataCache.GetMediaComposition(urn, cacheDuration);
|
||||||
|
if (mediaComposition != null)
|
||||||
|
{
|
||||||
|
_logger.LogDebug("Cache hit for URN: {Urn}", urn);
|
||||||
|
return mediaComposition;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch from API
|
||||||
|
_logger.LogDebug("Cache miss for URN: {Urn}, fetching from API", urn);
|
||||||
|
using var apiClient = _apiClientFactory.CreateClient();
|
||||||
|
mediaComposition = await apiClient.GetMediaCompositionByUrnAsync(urn, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (mediaComposition != null)
|
||||||
|
{
|
||||||
|
_metadataCache.SetMediaComposition(urn, mediaComposition);
|
||||||
|
_logger.LogDebug("Cached media composition for URN: {Urn}", urn);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Failed to fetch media composition for URN: {Urn}", urn);
|
||||||
|
}
|
||||||
|
|
||||||
|
return mediaComposition;
|
||||||
|
}
|
||||||
|
}
|
||||||
175
Jellyfin.Plugin.SRFPlay/Services/MediaSourceFactory.cs
Normal file
175
Jellyfin.Plugin.SRFPlay/Services/MediaSourceFactory.cs
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Constants;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using MediaBrowser.Controller;
|
||||||
|
using MediaBrowser.Model.Dto;
|
||||||
|
using MediaBrowser.Model.Entities;
|
||||||
|
using MediaBrowser.Model.MediaInfo;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Factory for creating MediaSourceInfo objects with consistent configuration.
|
||||||
|
/// Consolidates duplicated logic from SRFMediaProvider and SRFPlayChannel.
|
||||||
|
/// </summary>
|
||||||
|
public class MediaSourceFactory : IMediaSourceFactory
|
||||||
|
{
|
||||||
|
private readonly ILogger<MediaSourceFactory> _logger;
|
||||||
|
private readonly IStreamUrlResolver _streamResolver;
|
||||||
|
private readonly IStreamProxyService _proxyService;
|
||||||
|
private readonly IServerApplicationHost _appHost;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="MediaSourceFactory"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="logger">The logger.</param>
|
||||||
|
/// <param name="streamResolver">The stream URL resolver.</param>
|
||||||
|
/// <param name="proxyService">The stream proxy service.</param>
|
||||||
|
/// <param name="appHost">The server application host.</param>
|
||||||
|
public MediaSourceFactory(
|
||||||
|
ILogger<MediaSourceFactory> logger,
|
||||||
|
IStreamUrlResolver streamResolver,
|
||||||
|
IStreamProxyService proxyService,
|
||||||
|
IServerApplicationHost appHost)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
_streamResolver = streamResolver;
|
||||||
|
_proxyService = proxyService;
|
||||||
|
_appHost = appHost;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public Task<MediaSourceInfo?> CreateMediaSourceAsync(
|
||||||
|
Chapter chapter,
|
||||||
|
string itemId,
|
||||||
|
string urn,
|
||||||
|
QualityPreference qualityPreference,
|
||||||
|
CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
// Get stream URL based on quality preference (unauthenticated)
|
||||||
|
var streamUrl = _streamResolver.GetStreamUrl(chapter, qualityPreference);
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(streamUrl))
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Could not resolve stream URL for chapter: {ChapterId}", chapter.Id);
|
||||||
|
return Task.FromResult<MediaSourceInfo?>(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Detect if this is a live stream
|
||||||
|
var isLiveStream = chapter.Type == "SCHEDULED_LIVESTREAM" ||
|
||||||
|
urn.Contains("livestream", StringComparison.OrdinalIgnoreCase);
|
||||||
|
|
||||||
|
// Register stream with UNAUTHENTICATED URL - proxy will authenticate on-demand
|
||||||
|
// This avoids wasting 30-second tokens during category browsing
|
||||||
|
_proxyService.RegisterStreamDeferred(itemId, streamUrl, urn, isLiveStream);
|
||||||
|
|
||||||
|
// Build proxy URL
|
||||||
|
var proxyUrl = BuildProxyUrl(itemId);
|
||||||
|
|
||||||
|
_logger.LogDebug(
|
||||||
|
"Created media source for {Title} - ItemId: {ItemId}, IsLiveStream: {IsLiveStream}",
|
||||||
|
chapter.Title,
|
||||||
|
itemId,
|
||||||
|
isLiveStream);
|
||||||
|
|
||||||
|
// Create MediaSourceInfo with codec info so clients know they can direct play
|
||||||
|
// Provide MediaStreams with H.264+AAC so Android TV/ExoPlayer doesn't trigger transcoding
|
||||||
|
var mediaStreams = CreateMediaStreams(qualityPreference);
|
||||||
|
|
||||||
|
var mediaSource = new MediaSourceInfo
|
||||||
|
{
|
||||||
|
Id = itemId,
|
||||||
|
Name = chapter.Title,
|
||||||
|
Path = proxyUrl,
|
||||||
|
Protocol = MediaProtocol.Http,
|
||||||
|
// Use "hls" to trigger hls.js player in web client
|
||||||
|
Container = "hls",
|
||||||
|
SupportsDirectStream = true,
|
||||||
|
SupportsDirectPlay = true,
|
||||||
|
SupportsTranscoding = false,
|
||||||
|
IsRemote = true,
|
||||||
|
Type = MediaSourceType.Default,
|
||||||
|
RunTimeTicks = chapter.Duration > 0 ? TimeSpan.FromMilliseconds(chapter.Duration).Ticks : null,
|
||||||
|
VideoType = VideoType.VideoFile,
|
||||||
|
IsInfiniteStream = isLiveStream,
|
||||||
|
// Don't use RequiresOpening - it forces Jellyfin to transcode which breaks token auth
|
||||||
|
RequiresOpening = false,
|
||||||
|
RequiresClosing = false,
|
||||||
|
// Disable probing - we provide stream info directly
|
||||||
|
SupportsProbing = false,
|
||||||
|
ReadAtNativeFramerate = isLiveStream,
|
||||||
|
// Provide codec info so clients know they can direct play H.264+AAC
|
||||||
|
MediaStreams = mediaStreams.ToList(),
|
||||||
|
AnalyzeDurationMs = isLiveStream ? 1000 : 3000,
|
||||||
|
IgnoreDts = isLiveStream,
|
||||||
|
IgnoreIndex = isLiveStream,
|
||||||
|
};
|
||||||
|
|
||||||
|
return Task.FromResult<MediaSourceInfo?>(mediaSource);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public string BuildProxyUrl(string itemId)
|
||||||
|
{
|
||||||
|
return $"{GetServerBaseUrl()}{ApiEndpoints.ProxyMasterManifestPath.Replace("{0}", itemId, StringComparison.Ordinal)}";
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public string GetServerBaseUrl()
|
||||||
|
{
|
||||||
|
var config = Plugin.Instance?.Configuration;
|
||||||
|
return config != null && !string.IsNullOrWhiteSpace(config.PublicServerUrl)
|
||||||
|
? config.PublicServerUrl.TrimEnd('/')
|
||||||
|
: _appHost.GetSmartApiUrl(string.Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public IReadOnlyList<MediaStream> CreateMediaStreams(QualityPreference quality)
|
||||||
|
{
|
||||||
|
// Set resolution/bitrate based on quality preference
|
||||||
|
var (width, height, videoBitrate) = quality switch
|
||||||
|
{
|
||||||
|
QualityPreference.SD => (1280, 720, 2500000),
|
||||||
|
QualityPreference.HD => (1920, 1080, 5000000),
|
||||||
|
_ => (1280, 720, 3000000)
|
||||||
|
};
|
||||||
|
|
||||||
|
return new List<MediaStream>
|
||||||
|
{
|
||||||
|
new MediaStream
|
||||||
|
{
|
||||||
|
Type = MediaStreamType.Video,
|
||||||
|
Codec = "h264",
|
||||||
|
Profile = "high",
|
||||||
|
Level = 40,
|
||||||
|
Width = width,
|
||||||
|
Height = height,
|
||||||
|
BitRate = videoBitrate,
|
||||||
|
BitDepth = 8,
|
||||||
|
IsInterlaced = false,
|
||||||
|
IsDefault = true,
|
||||||
|
Index = 0,
|
||||||
|
IsAVC = true,
|
||||||
|
PixelFormat = "yuv420p"
|
||||||
|
},
|
||||||
|
new MediaStream
|
||||||
|
{
|
||||||
|
Type = MediaStreamType.Audio,
|
||||||
|
Codec = "aac",
|
||||||
|
Profile = "LC",
|
||||||
|
Channels = 2,
|
||||||
|
SampleRate = 48000,
|
||||||
|
BitRate = 128000,
|
||||||
|
IsDefault = true,
|
||||||
|
Index = 1
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,6 +2,7 @@ using System;
|
|||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Services;
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
@ -9,7 +10,7 @@ namespace Jellyfin.Plugin.SRFPlay.Services;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Service for caching metadata from SRF API.
|
/// Service for caching metadata from SRF API.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public sealed class MetadataCache : IDisposable
|
public sealed class MetadataCache : IMetadataCache, IDisposable
|
||||||
{
|
{
|
||||||
private readonly ILogger<MetadataCache> _logger;
|
private readonly ILogger<MetadataCache> _logger;
|
||||||
private readonly ConcurrentDictionary<string, CacheEntry<MediaComposition>> _mediaCompositionCache;
|
private readonly ConcurrentDictionary<string, CacheEntry<MediaComposition>> _mediaCompositionCache;
|
||||||
|
|||||||
@ -1,11 +1,16 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Web;
|
using System.Web;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Api;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using MediaBrowser.Common.Net;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Services;
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
@ -13,27 +18,31 @@ namespace Jellyfin.Plugin.SRFPlay.Services;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Service for proxying SRF Play streams and managing authentication.
|
/// Service for proxying SRF Play streams and managing authentication.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class StreamProxyService : IDisposable
|
public class StreamProxyService : IStreamProxyService
|
||||||
{
|
{
|
||||||
private readonly ILogger<StreamProxyService> _logger;
|
private readonly ILogger<StreamProxyService> _logger;
|
||||||
private readonly StreamUrlResolver _streamResolver;
|
private readonly IStreamUrlResolver _streamResolver;
|
||||||
private readonly HttpClient _httpClient;
|
private readonly IMediaCompositionFetcher _compositionFetcher;
|
||||||
|
private readonly IHttpClientFactory _httpClientFactory;
|
||||||
private readonly ConcurrentDictionary<string, StreamInfo> _streamMappings;
|
private readonly ConcurrentDictionary<string, StreamInfo> _streamMappings;
|
||||||
private bool _disposed;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="StreamProxyService"/> class.
|
/// Initializes a new instance of the <see cref="StreamProxyService"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="logger">The logger.</param>
|
/// <param name="logger">The logger.</param>
|
||||||
/// <param name="streamResolver">The stream URL resolver.</param>
|
/// <param name="streamResolver">The stream URL resolver.</param>
|
||||||
public StreamProxyService(ILogger<StreamProxyService> logger, StreamUrlResolver streamResolver)
|
/// <param name="compositionFetcher">The media composition fetcher.</param>
|
||||||
|
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
||||||
|
public StreamProxyService(
|
||||||
|
ILogger<StreamProxyService> logger,
|
||||||
|
IStreamUrlResolver streamResolver,
|
||||||
|
IMediaCompositionFetcher compositionFetcher,
|
||||||
|
IHttpClientFactory httpClientFactory)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_streamResolver = streamResolver;
|
_streamResolver = streamResolver;
|
||||||
_httpClient = new HttpClient
|
_compositionFetcher = compositionFetcher;
|
||||||
{
|
_httpClientFactory = httpClientFactory;
|
||||||
Timeout = TimeSpan.FromSeconds(30)
|
|
||||||
};
|
|
||||||
_streamMappings = new ConcurrentDictionary<string, StreamInfo>();
|
_streamMappings = new ConcurrentDictionary<string, StreamInfo>();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -42,7 +51,9 @@ public class StreamProxyService : IDisposable
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="itemId">The item ID.</param>
|
/// <param name="itemId">The item ID.</param>
|
||||||
/// <param name="authenticatedUrl">The authenticated stream URL.</param>
|
/// <param name="authenticatedUrl">The authenticated stream URL.</param>
|
||||||
public void RegisterStream(string itemId, string authenticatedUrl)
|
/// <param name="urn">The SRF URN for this content (used for re-fetching fresh URLs).</param>
|
||||||
|
/// <param name="isLiveStream">Whether this is a livestream (livestreams always fetch fresh URLs).</param>
|
||||||
|
public void RegisterStream(string itemId, string authenticatedUrl, string? urn = null, bool isLiveStream = false)
|
||||||
{
|
{
|
||||||
var tokenExpiry = ExtractTokenExpiry(authenticatedUrl);
|
var tokenExpiry = ExtractTokenExpiry(authenticatedUrl);
|
||||||
var unauthenticatedUrl = StripAuthenticationFromUrl(authenticatedUrl);
|
var unauthenticatedUrl = StripAuthenticationFromUrl(authenticatedUrl);
|
||||||
@ -52,36 +63,17 @@ public class StreamProxyService : IDisposable
|
|||||||
AuthenticatedUrl = authenticatedUrl,
|
AuthenticatedUrl = authenticatedUrl,
|
||||||
UnauthenticatedUrl = unauthenticatedUrl,
|
UnauthenticatedUrl = unauthenticatedUrl,
|
||||||
RegisteredAt = DateTime.UtcNow,
|
RegisteredAt = DateTime.UtcNow,
|
||||||
TokenExpiresAt = tokenExpiry
|
TokenExpiresAt = tokenExpiry,
|
||||||
|
Urn = urn,
|
||||||
|
IsLiveStream = isLiveStream,
|
||||||
|
LastLivestreamFetchAt = isLiveStream ? DateTime.UtcNow : null
|
||||||
};
|
};
|
||||||
|
|
||||||
// Register with the provided item ID
|
RegisterWithGuidFormats(itemId, streamInfo);
|
||||||
_streamMappings.AddOrUpdate(itemId, streamInfo, (key, old) => streamInfo);
|
|
||||||
|
|
||||||
// Also register with alternative GUID formats to handle Jellyfin's ID transformations
|
|
||||||
if (Guid.TryParse(itemId, out var guid))
|
|
||||||
{
|
|
||||||
var formats = new[]
|
|
||||||
{
|
|
||||||
guid.ToString("N"), // Without dashes: 00000000000000000000000000000000
|
|
||||||
guid.ToString("D"), // With dashes: 00000000-0000-0000-0000-000000000000
|
|
||||||
guid.ToString("B"), // With braces: {00000000-0000-0000-0000-000000000000}
|
|
||||||
};
|
|
||||||
|
|
||||||
foreach (var format in formats)
|
|
||||||
{
|
|
||||||
if (format != itemId) // Don't duplicate the original
|
|
||||||
{
|
|
||||||
_streamMappings.AddOrUpdate(format, streamInfo, (key, old) => streamInfo);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_logger.LogDebug("Registered stream with {Count} GUID format variations", formats.Length);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tokenExpiry.HasValue)
|
if (tokenExpiry.HasValue)
|
||||||
{
|
{
|
||||||
_logger.LogInformation(
|
_logger.LogDebug(
|
||||||
"Registered stream for item {ItemId} (token expires at {ExpiresAt} UTC): {Url}",
|
"Registered stream for item {ItemId} (token expires at {ExpiresAt} UTC): {Url}",
|
||||||
itemId,
|
itemId,
|
||||||
tokenExpiry.Value,
|
tokenExpiry.Value,
|
||||||
@ -89,23 +81,117 @@ public class StreamProxyService : IDisposable
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
_logger.LogInformation("Registered stream for item {ItemId}: {Url}", itemId, authenticatedUrl);
|
_logger.LogDebug("Registered stream for item {ItemId}: {Url}", itemId, authenticatedUrl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Registers a stream for deferred authentication (authenticates on first playback request).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="unauthenticatedUrl">The unauthenticated stream URL.</param>
|
||||||
|
/// <param name="urn">The SRF URN for this content.</param>
|
||||||
|
/// <param name="isLiveStream">Whether this is a livestream.</param>
|
||||||
|
public void RegisterStreamDeferred(string itemId, string unauthenticatedUrl, string? urn = null, bool isLiveStream = false)
|
||||||
|
{
|
||||||
|
var streamInfo = new StreamInfo
|
||||||
|
{
|
||||||
|
AuthenticatedUrl = string.Empty, // Will be populated on first access
|
||||||
|
UnauthenticatedUrl = unauthenticatedUrl,
|
||||||
|
RegisteredAt = DateTime.UtcNow,
|
||||||
|
TokenExpiresAt = null,
|
||||||
|
Urn = urn,
|
||||||
|
IsLiveStream = isLiveStream,
|
||||||
|
LastLivestreamFetchAt = null,
|
||||||
|
NeedsAuthentication = true
|
||||||
|
};
|
||||||
|
|
||||||
|
RegisterWithGuidFormats(itemId, streamInfo);
|
||||||
|
|
||||||
|
_logger.LogDebug(
|
||||||
|
"Registered deferred stream for item {ItemId} (URN: {Urn}, will authenticate on first access)",
|
||||||
|
itemId,
|
||||||
|
urn ?? "null");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets stream metadata for an item (URN and isLiveStream flag).
|
||||||
|
/// Used when propagating stream registration to transcoding sessions.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <returns>A tuple of (URN, IsLiveStream), or null if not found.</returns>
|
||||||
|
public (string? Urn, bool IsLiveStream)? GetStreamMetadata(string itemId)
|
||||||
|
{
|
||||||
|
if (_streamMappings.TryGetValue(itemId, out var streamInfo))
|
||||||
|
{
|
||||||
|
return (streamInfo.Urn, streamInfo.IsLiveStream);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try GUID normalization
|
||||||
|
var normalizedId = NormalizeGuid(itemId);
|
||||||
|
if (normalizedId != null)
|
||||||
|
{
|
||||||
|
foreach (var kvp in _streamMappings)
|
||||||
|
{
|
||||||
|
var normalizedKey = NormalizeGuid(kvp.Key);
|
||||||
|
if (normalizedKey != null && normalizedKey == normalizedId)
|
||||||
|
{
|
||||||
|
return (kvp.Value.Urn, kvp.Value.IsLiveStream);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the authenticated URL for an item.
|
/// Gets the authenticated URL for an item.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="itemId">The item ID.</param>
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="cancellationToken">Cancellation token.</param>
|
||||||
/// <returns>The authenticated URL, or null if not found or expired.</returns>
|
/// <returns>The authenticated URL, or null if not found or expired.</returns>
|
||||||
public string? GetAuthenticatedUrl(string itemId)
|
public async Task<string?> GetAuthenticatedUrlAsync(string itemId, CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
|
_logger.LogInformation("GetAuthenticatedUrlAsync called for itemId: {ItemId}", itemId);
|
||||||
|
|
||||||
// Try direct lookup first
|
// Try direct lookup first
|
||||||
if (_streamMappings.TryGetValue(itemId, out var streamInfo))
|
if (_streamMappings.TryGetValue(itemId, out var streamInfo))
|
||||||
{
|
{
|
||||||
return ValidateAndReturnStream(itemId, streamInfo);
|
// Log detailed StreamInfo state to diagnose stale alias issues
|
||||||
|
var tokenTimeLeft = streamInfo.TokenExpiresAt.HasValue
|
||||||
|
? (streamInfo.TokenExpiresAt.Value - DateTime.UtcNow).TotalSeconds
|
||||||
|
: -1;
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Found stream by direct lookup for itemId: {ItemId} - NeedsAuth={NeedsAuth}, IsLive={IsLive}, Urn={Urn}, TokenLeft={TokenLeft:F0}s, AuthUrl={HasAuth}",
|
||||||
|
itemId,
|
||||||
|
streamInfo.NeedsAuthentication,
|
||||||
|
streamInfo.IsLiveStream,
|
||||||
|
string.IsNullOrEmpty(streamInfo.Urn) ? "(empty)" : "set",
|
||||||
|
tokenTimeLeft,
|
||||||
|
!string.IsNullOrEmpty(streamInfo.AuthenticatedUrl));
|
||||||
|
|
||||||
|
// Check for stale alias: only look for fresher stream if current token is EXPIRED or EXPIRING SOON
|
||||||
|
// Don't replace a valid token (>5s left) with a new deferred registration
|
||||||
|
if (!streamInfo.NeedsAuthentication && tokenTimeLeft < 5)
|
||||||
|
{
|
||||||
|
var freshStream = FindFreshestStream();
|
||||||
|
if (freshStream != null && freshStream.Value.Value.NeedsAuthentication)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(
|
||||||
|
"Token expiring soon ({TokenLeft:F0}s), switching to fresher deferred stream {ItemId} -> {FreshKey}",
|
||||||
|
tokenTimeLeft,
|
||||||
|
itemId,
|
||||||
|
freshStream.Value.Key);
|
||||||
|
_streamMappings.AddOrUpdate(itemId, freshStream.Value.Value, (key, old) => freshStream.Value.Value);
|
||||||
|
return await ValidateAndReturnStreamAsync(itemId, freshStream.Value.Value, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return await ValidateAndReturnStreamAsync(itemId, streamInfo, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_logger.LogWarning("No direct match for itemId: {ItemId}, trying fallbacks... (Registered streams: {Count})", itemId, _streamMappings.Count);
|
||||||
|
|
||||||
// Fallback: Try to find by GUID variations (with/without dashes)
|
// Fallback: Try to find by GUID variations (with/without dashes)
|
||||||
// This handles cases where Jellyfin uses different GUID formats
|
// This handles cases where Jellyfin uses different GUID formats
|
||||||
var normalizedId = NormalizeGuid(itemId);
|
var normalizedId = NormalizeGuid(itemId);
|
||||||
@ -120,7 +206,7 @@ public class StreamProxyService : IDisposable
|
|||||||
"Found stream by GUID normalization - Requested: {RequestedId}, Registered: {RegisteredId}",
|
"Found stream by GUID normalization - Requested: {RequestedId}, Registered: {RegisteredId}",
|
||||||
itemId,
|
itemId,
|
||||||
kvp.Key);
|
kvp.Key);
|
||||||
var url = ValidateAndReturnStream(kvp.Key, kvp.Value);
|
var url = await ValidateAndReturnStreamAsync(kvp.Key, kvp.Value, cancellationToken).ConfigureAwait(false);
|
||||||
if (url != null)
|
if (url != null)
|
||||||
{
|
{
|
||||||
return url; // Found valid stream
|
return url; // Found valid stream
|
||||||
@ -133,6 +219,50 @@ public class StreamProxyService : IDisposable
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Fallback: Live TV channelId_guid format lookup
|
||||||
|
// For Live TV streams, itemId format is "channelId_urnGuid"
|
||||||
|
// Try matching by suffix (urnGuid part) or prefix (channelId part)
|
||||||
|
if (itemId.Contains('_', StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
var parts = itemId.Split('_', 2);
|
||||||
|
var prefix = parts[0]; // channelId part
|
||||||
|
var suffix = parts.Length > 1 ? parts[1] : null; // urnGuid part
|
||||||
|
|
||||||
|
foreach (var kvp in _streamMappings)
|
||||||
|
{
|
||||||
|
// Check if registered key contains the same suffix (urnGuid)
|
||||||
|
if (suffix != null && kvp.Key.Contains(suffix, StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Found stream by Live TV suffix match - Requested: {RequestedId}, Registered: {RegisteredId}",
|
||||||
|
itemId,
|
||||||
|
kvp.Key);
|
||||||
|
var url = await ValidateAndReturnStreamAsync(kvp.Key, kvp.Value, cancellationToken).ConfigureAwait(false);
|
||||||
|
if (url != null)
|
||||||
|
{
|
||||||
|
// Also register with the requested itemId for future lookups
|
||||||
|
_streamMappings.AddOrUpdate(itemId, kvp.Value, (key, old) => kvp.Value);
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if registered key starts with the same prefix (channelId)
|
||||||
|
if (kvp.Key.StartsWith(prefix + "_", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Found stream by Live TV prefix match - Requested: {RequestedId}, Registered: {RegisteredId}",
|
||||||
|
itemId,
|
||||||
|
kvp.Key);
|
||||||
|
var url = await ValidateAndReturnStreamAsync(kvp.Key, kvp.Value, cancellationToken).ConfigureAwait(false);
|
||||||
|
if (url != null)
|
||||||
|
{
|
||||||
|
_streamMappings.AddOrUpdate(itemId, kvp.Value, (key, old) => kvp.Value);
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Last resort: Use active stream fallbacks (helps when Jellyfin creates random transcoding session IDs)
|
// Last resort: Use active stream fallbacks (helps when Jellyfin creates random transcoding session IDs)
|
||||||
var activeStreams = _streamMappings.Where(kvp =>
|
var activeStreams = _streamMappings.Where(kvp =>
|
||||||
{
|
{
|
||||||
@ -146,11 +276,15 @@ public class StreamProxyService : IDisposable
|
|||||||
|
|
||||||
if (activeStreams.Count == 1)
|
if (activeStreams.Count == 1)
|
||||||
{
|
{
|
||||||
_logger.LogWarning(
|
_logger.LogInformation(
|
||||||
"No exact match for {RequestedId}, but found single active stream {RegisteredId} - using as fallback",
|
"Transcoding session detected: Aliasing {TranscodingId} -> {OriginalId} (single active stream)",
|
||||||
itemId,
|
itemId,
|
||||||
activeStreams[0].Key);
|
activeStreams[0].Key);
|
||||||
return ValidateAndReturnStream(activeStreams[0].Key, activeStreams[0].Value);
|
|
||||||
|
// Register the transcoding session ID as an alias (update if stale alias exists)
|
||||||
|
_streamMappings.AddOrUpdate(itemId, activeStreams[0].Value, (key, old) => activeStreams[0].Value);
|
||||||
|
|
||||||
|
return await ValidateAndReturnStreamAsync(activeStreams[0].Key, activeStreams[0].Value, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If multiple active streams, use the most recently registered one (likely the one being transcoded)
|
// If multiple active streams, use the most recently registered one (likely the one being transcoded)
|
||||||
@ -164,12 +298,16 @@ public class StreamProxyService : IDisposable
|
|||||||
// This indicates it's likely the stream currently being set up for transcoding
|
// This indicates it's likely the stream currently being set up for transcoding
|
||||||
if (age.TotalSeconds < 30)
|
if (age.TotalSeconds < 30)
|
||||||
{
|
{
|
||||||
_logger.LogWarning(
|
_logger.LogInformation(
|
||||||
"No exact match for {RequestedId}, but using most recently registered stream {RegisteredId} (registered {Seconds}s ago) as fallback",
|
"Transcoding session detected: Aliasing {TranscodingId} -> {OriginalId} (registered {Seconds:F1}s ago)",
|
||||||
itemId,
|
itemId,
|
||||||
mostRecent.Key,
|
mostRecent.Key,
|
||||||
age.TotalSeconds);
|
age.TotalSeconds);
|
||||||
return ValidateAndReturnStream(mostRecent.Key, mostRecent.Value);
|
|
||||||
|
// Register the transcoding session ID as an alias (update if stale alias exists)
|
||||||
|
_streamMappings.AddOrUpdate(itemId, mostRecent.Value, (key, old) => mostRecent.Value);
|
||||||
|
|
||||||
|
return await ValidateAndReturnStreamAsync(mostRecent.Key, mostRecent.Value, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -184,22 +322,90 @@ public class StreamProxyService : IDisposable
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Validates a stream and returns its URL if valid.
|
/// Validates a stream and returns its URL if valid.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private string? ValidateAndReturnStream(string itemId, StreamInfo streamInfo)
|
private async Task<string?> ValidateAndReturnStreamAsync(string itemId, StreamInfo streamInfo, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
// Check if token has expired
|
// Handle deferred authentication (first playback after browsing)
|
||||||
|
if (streamInfo.NeedsAuthentication)
|
||||||
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"First playback for item {ItemId} - authenticating stream on-demand",
|
||||||
|
itemId);
|
||||||
|
|
||||||
|
var authenticatedUrl = await AuthenticateOnDemandAsync(itemId, streamInfo, cancellationToken).ConfigureAwait(false);
|
||||||
|
if (authenticatedUrl != null)
|
||||||
|
{
|
||||||
|
return authenticatedUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogWarning("Failed to authenticate stream on-demand for item {ItemId}", itemId);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// For livestreams, use smart caching to avoid hammering the API
|
||||||
|
// Only fetch fresh if token is expiring soon or hasn't been fetched recently
|
||||||
|
if (streamInfo.IsLiveStream && !string.IsNullOrEmpty(streamInfo.Urn))
|
||||||
|
{
|
||||||
|
var now = DateTime.UtcNow;
|
||||||
|
var tokenTimeLeft = streamInfo.TokenExpiresAt.HasValue
|
||||||
|
? (streamInfo.TokenExpiresAt.Value - now).TotalSeconds
|
||||||
|
: 30; // Assume 30s if no expiry
|
||||||
|
|
||||||
|
var timeSinceLastFetch = streamInfo.LastLivestreamFetchAt.HasValue
|
||||||
|
? (now - streamInfo.LastLivestreamFetchAt.Value).TotalSeconds
|
||||||
|
: double.MaxValue;
|
||||||
|
|
||||||
|
// Use cached URL if: token has >10s left AND we fetched within last 15 seconds
|
||||||
|
if (tokenTimeLeft > 10 && timeSinceLastFetch < 15)
|
||||||
|
{
|
||||||
|
_logger.LogDebug(
|
||||||
|
"Livestream {ItemId}: Using cached URL (token expires in {TokenTimeLeft:F0}s, last fetch {TimeSinceFetch:F0}s ago)",
|
||||||
|
itemId,
|
||||||
|
tokenTimeLeft,
|
||||||
|
timeSinceLastFetch);
|
||||||
|
return streamInfo.AuthenticatedUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Livestream {ItemId}: Fetching fresh URL (token expires in {TokenTimeLeft:F0}s, last fetch {TimeSinceFetch:F0}s ago)",
|
||||||
|
itemId,
|
||||||
|
tokenTimeLeft,
|
||||||
|
timeSinceLastFetch);
|
||||||
|
|
||||||
|
var freshUrl = await FetchFreshStreamUrlAsync(itemId, streamInfo, cancellationToken).ConfigureAwait(false);
|
||||||
|
if (freshUrl != null)
|
||||||
|
{
|
||||||
|
return freshUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogWarning("Failed to fetch fresh URL for livestream {ItemId}, falling back to cached URL", itemId);
|
||||||
|
// Fall through to use cached URL as fallback
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if token has expired or is about to expire
|
||||||
if (streamInfo.TokenExpiresAt.HasValue)
|
if (streamInfo.TokenExpiresAt.HasValue)
|
||||||
{
|
{
|
||||||
var now = DateTime.UtcNow;
|
var now = DateTime.UtcNow;
|
||||||
if (now >= streamInfo.TokenExpiresAt.Value)
|
var timeUntilExpiry = streamInfo.TokenExpiresAt.Value - now;
|
||||||
|
|
||||||
|
// Proactive refresh: refresh if token has expired OR will expire within 5 seconds
|
||||||
|
// This prevents race conditions during rapid segment fetching in transcoding
|
||||||
|
var shouldRefresh = now >= streamInfo.TokenExpiresAt.Value || timeUntilExpiry.TotalSeconds <= 5;
|
||||||
|
|
||||||
|
if (shouldRefresh)
|
||||||
{
|
{
|
||||||
|
var reason = now >= streamInfo.TokenExpiresAt.Value
|
||||||
|
? "expired"
|
||||||
|
: $"expiring in {timeUntilExpiry.TotalSeconds:F1}s";
|
||||||
|
|
||||||
_logger.LogWarning(
|
_logger.LogWarning(
|
||||||
"Token expired for item {ItemId} (expired at {ExpiresAt}, now is {Now}) - attempting to refresh",
|
"Token {Reason} for item {ItemId} (expires at {ExpiresAt}, now is {Now}) - attempting to refresh",
|
||||||
|
reason,
|
||||||
itemId,
|
itemId,
|
||||||
streamInfo.TokenExpiresAt.Value,
|
streamInfo.TokenExpiresAt.Value,
|
||||||
now);
|
now);
|
||||||
|
|
||||||
// Try to refresh the token
|
// Try to refresh the token
|
||||||
var refreshedUrl = RefreshToken(itemId, streamInfo);
|
var refreshedUrl = await RefreshTokenAsync(itemId, streamInfo, cancellationToken).ConfigureAwait(false);
|
||||||
if (refreshedUrl != null)
|
if (refreshedUrl != null)
|
||||||
{
|
{
|
||||||
_logger.LogInformation("Successfully refreshed token for item {ItemId}", itemId);
|
_logger.LogInformation("Successfully refreshed token for item {ItemId}", itemId);
|
||||||
@ -212,19 +418,75 @@ public class StreamProxyService : IDisposable
|
|||||||
}
|
}
|
||||||
|
|
||||||
_logger.LogDebug(
|
_logger.LogDebug(
|
||||||
"Token valid for item {ItemId} (expires at {ExpiresAt}, {TimeLeft} remaining)",
|
"Token valid for item {ItemId} (expires at {ExpiresAt}, {TimeLeft:F1}s remaining)",
|
||||||
itemId,
|
itemId,
|
||||||
streamInfo.TokenExpiresAt.Value,
|
streamInfo.TokenExpiresAt.Value,
|
||||||
streamInfo.TokenExpiresAt.Value - now);
|
timeUntilExpiry.TotalSeconds);
|
||||||
}
|
}
|
||||||
|
|
||||||
return streamInfo.AuthenticatedUrl;
|
return streamInfo.AuthenticatedUrl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fetches a fresh stream URL from the SRF API for livestreams.
|
||||||
|
/// </summary>
|
||||||
|
private async Task<string?> FetchFreshStreamUrlAsync(string itemId, StreamInfo streamInfo, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(streamInfo.Urn))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Use short cache duration (5 min) for livestreams
|
||||||
|
var mediaComposition = await _compositionFetcher.GetMediaCompositionAsync(streamInfo.Urn, cancellationToken, 5).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (mediaComposition?.ChapterList == null || mediaComposition.ChapterList.Count == 0)
|
||||||
|
{
|
||||||
|
_logger.LogWarning("No chapters found when refreshing livestream URL for URN: {Urn}", streamInfo.Urn);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var chapter = mediaComposition.ChapterList[0];
|
||||||
|
var config = Plugin.Instance?.Configuration ?? new PluginConfiguration();
|
||||||
|
var streamUrl = _streamResolver.GetStreamUrl(chapter, config.QualityPreference);
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(streamUrl))
|
||||||
|
{
|
||||||
|
_logger.LogWarning("No stream URL found when refreshing livestream for URN: {Urn}", streamInfo.Urn);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Authenticate the fresh URL
|
||||||
|
var authenticatedUrl = await _streamResolver.GetAuthenticatedStreamUrlAsync(streamUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
// Update the stored stream info with the fresh data
|
||||||
|
var newTokenExpiry = ExtractTokenExpiry(authenticatedUrl);
|
||||||
|
streamInfo.AuthenticatedUrl = authenticatedUrl;
|
||||||
|
streamInfo.UnauthenticatedUrl = StripAuthenticationFromUrl(authenticatedUrl);
|
||||||
|
streamInfo.TokenExpiresAt = newTokenExpiry;
|
||||||
|
streamInfo.LastLivestreamFetchAt = DateTime.UtcNow;
|
||||||
|
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Fetched fresh livestream URL for item {ItemId} (URN: {Urn}, new expiry: {Expiry})",
|
||||||
|
itemId,
|
||||||
|
streamInfo.Urn,
|
||||||
|
newTokenExpiry);
|
||||||
|
|
||||||
|
return authenticatedUrl;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error fetching fresh stream URL for livestream {ItemId} (URN: {Urn})", itemId, streamInfo.Urn);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Attempts to refresh an expired token.
|
/// Attempts to refresh an expired token.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private string? RefreshToken(string itemId, StreamInfo streamInfo)
|
private async Task<string?> RefreshTokenAsync(string itemId, StreamInfo streamInfo, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(streamInfo.UnauthenticatedUrl))
|
if (string.IsNullOrEmpty(streamInfo.UnauthenticatedUrl))
|
||||||
{
|
{
|
||||||
@ -234,10 +496,10 @@ public class StreamProxyService : IDisposable
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// Re-authenticate the stream URL synchronously (blocking call)
|
// Re-authenticate the stream URL
|
||||||
var newAuthenticatedUrl = _streamResolver.GetAuthenticatedStreamUrlAsync(
|
var newAuthenticatedUrl = await _streamResolver.GetAuthenticatedStreamUrlAsync(
|
||||||
streamInfo.UnauthenticatedUrl,
|
streamInfo.UnauthenticatedUrl,
|
||||||
CancellationToken.None).GetAwaiter().GetResult();
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(newAuthenticatedUrl))
|
if (string.IsNullOrEmpty(newAuthenticatedUrl))
|
||||||
{
|
{
|
||||||
@ -263,6 +525,54 @@ public class StreamProxyService : IDisposable
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Authenticates a stream on-demand (first playback after browsing).
|
||||||
|
/// </summary>
|
||||||
|
private async Task<string?> AuthenticateOnDemandAsync(string itemId, StreamInfo streamInfo, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(streamInfo.UnauthenticatedUrl))
|
||||||
|
{
|
||||||
|
_logger.LogWarning("Cannot authenticate on-demand for {ItemId} - no unauthenticated URL stored", itemId);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Authenticate the stream URL
|
||||||
|
var authenticatedUrl = await _streamResolver.GetAuthenticatedStreamUrlAsync(
|
||||||
|
streamInfo.UnauthenticatedUrl,
|
||||||
|
cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(authenticatedUrl))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the stream info - no longer needs authentication
|
||||||
|
var tokenExpiry = ExtractTokenExpiry(authenticatedUrl);
|
||||||
|
streamInfo.AuthenticatedUrl = authenticatedUrl;
|
||||||
|
streamInfo.TokenExpiresAt = tokenExpiry;
|
||||||
|
streamInfo.NeedsAuthentication = false;
|
||||||
|
|
||||||
|
if (streamInfo.IsLiveStream)
|
||||||
|
{
|
||||||
|
streamInfo.LastLivestreamFetchAt = DateTime.UtcNow;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Authenticated stream on-demand for item {ItemId} (expires at {ExpiresAt} UTC)",
|
||||||
|
itemId,
|
||||||
|
tokenExpiry);
|
||||||
|
|
||||||
|
return authenticatedUrl;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Error authenticating stream on-demand for item {ItemId}", itemId);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Strips authentication parameters from a URL to get the base unauthenticated URL.
|
/// Strips authentication parameters from a URL to get the base unauthenticated URL.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -312,6 +622,41 @@ public class StreamProxyService : IDisposable
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Finds the freshest (most recently registered) stream that needs authentication or has a valid token.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The freshest stream entry, or null if none found.</returns>
|
||||||
|
private KeyValuePair<string, StreamInfo>? FindFreshestStream()
|
||||||
|
{
|
||||||
|
var now = DateTime.UtcNow;
|
||||||
|
|
||||||
|
// Find streams that either need authentication (fresh deferred registration)
|
||||||
|
// or have tokens that aren't expired yet
|
||||||
|
var candidates = _streamMappings.Where(kvp =>
|
||||||
|
{
|
||||||
|
if (kvp.Value.NeedsAuthentication)
|
||||||
|
{
|
||||||
|
return true; // Fresh deferred registration
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!kvp.Value.TokenExpiresAt.HasValue)
|
||||||
|
{
|
||||||
|
return true; // No expiry
|
||||||
|
}
|
||||||
|
|
||||||
|
// Token not expired yet
|
||||||
|
return now < kvp.Value.TokenExpiresAt.Value;
|
||||||
|
}).ToList();
|
||||||
|
|
||||||
|
if (candidates.Count == 0)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prefer the most recently registered stream
|
||||||
|
return candidates.OrderByDescending(kvp => kvp.Value.RegisteredAt).First();
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Fetches and rewrites an HLS manifest to use proxy URLs.
|
/// Fetches and rewrites an HLS manifest to use proxy URLs.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -324,7 +669,7 @@ public class StreamProxyService : IDisposable
|
|||||||
string baseProxyUrl,
|
string baseProxyUrl,
|
||||||
CancellationToken cancellationToken = default)
|
CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
var authenticatedUrl = GetAuthenticatedUrl(itemId);
|
var authenticatedUrl = await GetAuthenticatedUrlAsync(itemId, cancellationToken).ConfigureAwait(false);
|
||||||
if (authenticatedUrl == null)
|
if (authenticatedUrl == null)
|
||||||
{
|
{
|
||||||
return null;
|
return null;
|
||||||
@ -332,13 +677,16 @@ public class StreamProxyService : IDisposable
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
_logger.LogDebug("Fetching manifest from: {Url}", authenticatedUrl);
|
_logger.LogInformation("Fetching manifest from: {Url}", authenticatedUrl);
|
||||||
var manifestContent = await _httpClient.GetStringAsync(authenticatedUrl, cancellationToken).ConfigureAwait(false);
|
using var httpClient = _httpClientFactory.CreateClient(NamedClient.Default);
|
||||||
|
var manifestContent = await httpClient.GetStringAsync(authenticatedUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
_logger.LogDebug("Original manifest ({Length} bytes):\n{Content}", manifestContent.Length, manifestContent);
|
||||||
|
|
||||||
// Rewrite the manifest to replace Akamai URLs with proxy URLs
|
// Rewrite the manifest to replace Akamai URLs with proxy URLs
|
||||||
var rewrittenContent = RewriteManifestUrls(manifestContent, authenticatedUrl, baseProxyUrl);
|
var rewrittenContent = RewriteManifestUrls(manifestContent, authenticatedUrl, baseProxyUrl);
|
||||||
|
|
||||||
_logger.LogDebug("Successfully rewrote manifest for item {ItemId}", itemId);
|
_logger.LogDebug("Rewritten manifest for item {ItemId} ({Length} bytes):\n{Content}", itemId, rewrittenContent.Length, rewrittenContent);
|
||||||
return rewrittenContent;
|
return rewrittenContent;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@ -360,7 +708,7 @@ public class StreamProxyService : IDisposable
|
|||||||
string segmentPath,
|
string segmentPath,
|
||||||
CancellationToken cancellationToken = default)
|
CancellationToken cancellationToken = default)
|
||||||
{
|
{
|
||||||
var authenticatedUrl = GetAuthenticatedUrl(itemId);
|
var authenticatedUrl = await GetAuthenticatedUrlAsync(itemId, cancellationToken).ConfigureAwait(false);
|
||||||
if (authenticatedUrl == null)
|
if (authenticatedUrl == null)
|
||||||
{
|
{
|
||||||
return null;
|
return null;
|
||||||
@ -378,8 +726,14 @@ public class StreamProxyService : IDisposable
|
|||||||
// Build full segment URL
|
// Build full segment URL
|
||||||
var segmentUrl = $"{baseUrl}/{segmentPath}{queryParams}";
|
var segmentUrl = $"{baseUrl}/{segmentPath}{queryParams}";
|
||||||
|
|
||||||
_logger.LogDebug("Fetching segment: {SegmentUrl}", segmentUrl);
|
_logger.LogDebug(
|
||||||
var segmentData = await _httpClient.GetByteArrayAsync(segmentUrl, cancellationToken).ConfigureAwait(false);
|
"Fetching segment - BaseUri: {BaseUri}, BaseUrl: {BaseUrl}, SegmentPath: {SegmentPath}, FullUrl: {FullUrl}",
|
||||||
|
authenticatedUrl,
|
||||||
|
baseUrl,
|
||||||
|
segmentPath,
|
||||||
|
segmentUrl);
|
||||||
|
using var httpClient = _httpClientFactory.CreateClient(NamedClient.Default);
|
||||||
|
var segmentData = await httpClient.GetByteArrayAsync(segmentUrl, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
_logger.LogDebug("Successfully fetched segment {SegmentPath} ({Size} bytes)", segmentPath, segmentData.Length);
|
_logger.LogDebug("Successfully fetched segment {SegmentPath} ({Size} bytes)", segmentPath, segmentData.Length);
|
||||||
return segmentData;
|
return segmentData;
|
||||||
@ -403,24 +757,61 @@ public class StreamProxyService : IDisposable
|
|||||||
var baseUri = new Uri(originalBaseUrl);
|
var baseUri = new Uri(originalBaseUrl);
|
||||||
var baseUrl = $"{baseUri.Scheme}://{baseUri.Host}{string.Join('/', baseUri.AbsolutePath.Split('/')[..^1])}";
|
var baseUrl = $"{baseUri.Scheme}://{baseUri.Host}{string.Join('/', baseUri.AbsolutePath.Split('/')[..^1])}";
|
||||||
|
|
||||||
// Pattern to match .m3u8 and .ts/.mp4 segment references
|
// Extract query parameters from proxyBaseUrl to propagate them
|
||||||
var pattern = @"(?:^|\n)([^#\n][^\n]*\.(?:m3u8|ts|mp4|m4s|aac)[^\n]*)";
|
var queryParams = string.Empty;
|
||||||
|
var queryStart = proxyBaseUrl.IndexOf('?', StringComparison.Ordinal);
|
||||||
var rewritten = Regex.Replace(manifestContent, pattern, match =>
|
if (queryStart >= 0)
|
||||||
{
|
{
|
||||||
var url = match.Groups[1].Value.Trim();
|
queryParams = proxyBaseUrl[queryStart..];
|
||||||
|
proxyBaseUrl = proxyBaseUrl[..queryStart]; // Remove query from base URL
|
||||||
|
_logger.LogDebug("Extracted query parameters from proxy URL: {QueryParams}", queryParams);
|
||||||
|
}
|
||||||
|
|
||||||
// Skip if it's already an absolute URL
|
// Helper function to rewrite a URL to proxy
|
||||||
if (url.StartsWith("http://", StringComparison.OrdinalIgnoreCase) ||
|
string RewriteUrl(string url)
|
||||||
url.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
|
{
|
||||||
|
// Try to parse as absolute URL
|
||||||
|
if (Uri.TryCreate(url, UriKind.Absolute, out var absoluteUri))
|
||||||
{
|
{
|
||||||
// Rewrite absolute URLs to proxy
|
// Check if it's from the same CDN host
|
||||||
var relativePath = url.Replace(baseUrl + "/", string.Empty, StringComparison.Ordinal);
|
if (!absoluteUri.Host.Equals(baseUri.Host, StringComparison.OrdinalIgnoreCase))
|
||||||
return $"\n{proxyBaseUrl}/{relativePath}";
|
{
|
||||||
|
// External URL (e.g., subtitles from different domain) - leave as-is
|
||||||
|
_logger.LogDebug("Leaving external URL unchanged: {Url}", url);
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Same host - extract just the filename (last path segment)
|
||||||
|
var segments = absoluteUri.AbsolutePath.Split('/');
|
||||||
|
var filename = segments[^1];
|
||||||
|
return $"{proxyBaseUrl}/{filename}{queryParams}";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Relative URL - rewrite to proxy
|
// Relative URL - extract just the path without query params
|
||||||
return $"\n{proxyBaseUrl}/{url}";
|
var path = url;
|
||||||
|
var queryIndex = path.IndexOf('?', StringComparison.Ordinal);
|
||||||
|
if (queryIndex >= 0)
|
||||||
|
{
|
||||||
|
path = path[..queryIndex];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $"{proxyBaseUrl}/{path}{queryParams}";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pattern 1: Standalone URL lines (non-# lines ending with media extensions)
|
||||||
|
var pattern1 = @"(?:^|\n)([^#\n][^\n]*\.(?:m3u8|ts|mp4|m4s|aac)[^\n]*)";
|
||||||
|
var rewritten = Regex.Replace(manifestContent, pattern1, match =>
|
||||||
|
{
|
||||||
|
var url = match.Groups[1].Value.Trim();
|
||||||
|
return $"\n{RewriteUrl(url)}";
|
||||||
|
});
|
||||||
|
|
||||||
|
// Pattern 2: URI="..." attributes in HLS tags (e.g., #EXT-X-MEDIA, #EXT-X-I-FRAME-STREAM-INF)
|
||||||
|
var pattern2 = @"URI=""([^""]+)""";
|
||||||
|
rewritten = Regex.Replace(rewritten, pattern2, match =>
|
||||||
|
{
|
||||||
|
var url = match.Groups[1].Value;
|
||||||
|
return $"URI=\"{RewriteUrl(url)}\"";
|
||||||
});
|
});
|
||||||
|
|
||||||
return rewritten;
|
return rewritten;
|
||||||
@ -503,31 +894,33 @@ public class StreamProxyService : IDisposable
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Disposes the service.
|
/// Registers a stream with multiple GUID format variations to handle Jellyfin's ID transformations.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public void Dispose()
|
/// <param name="itemId">The item ID.</param>
|
||||||
|
/// <param name="streamInfo">The stream information to register.</param>
|
||||||
|
private void RegisterWithGuidFormats(string itemId, StreamInfo streamInfo)
|
||||||
{
|
{
|
||||||
Dispose(true);
|
_streamMappings.AddOrUpdate(itemId, streamInfo, (key, old) => streamInfo);
|
||||||
GC.SuppressFinalize(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
if (Guid.TryParse(itemId, out var guid))
|
||||||
/// Disposes the service.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="disposing">True if disposing.</param>
|
|
||||||
protected virtual void Dispose(bool disposing)
|
|
||||||
{
|
|
||||||
if (_disposed)
|
|
||||||
{
|
{
|
||||||
return;
|
var formats = new[]
|
||||||
}
|
{
|
||||||
|
guid.ToString("N"), // Without dashes: 00000000000000000000000000000000
|
||||||
|
guid.ToString("D"), // With dashes: 00000000-0000-0000-0000-000000000000
|
||||||
|
guid.ToString("B"), // With braces: {00000000-0000-0000-0000-000000000000}
|
||||||
|
};
|
||||||
|
|
||||||
if (disposing)
|
foreach (var format in formats)
|
||||||
{
|
{
|
||||||
_httpClient?.Dispose();
|
if (format != itemId) // Don't duplicate the original
|
||||||
}
|
{
|
||||||
|
_streamMappings.AddOrUpdate(format, streamInfo, (key, old) => streamInfo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_disposed = true;
|
_logger.LogDebug("Registered stream with {Count} GUID format variations", formats.Length);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -542,5 +935,28 @@ public class StreamProxyService : IDisposable
|
|||||||
public DateTime RegisteredAt { get; set; }
|
public DateTime RegisteredAt { get; set; }
|
||||||
|
|
||||||
public DateTime? TokenExpiresAt { get; set; }
|
public DateTime? TokenExpiresAt { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the SRF URN for this stream (used for re-fetching fresh URLs).
|
||||||
|
/// </summary>
|
||||||
|
public string? Urn { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a value indicating whether this is a livestream.
|
||||||
|
/// Livestreams always fetch fresh URLs from the API to avoid stale CDN paths.
|
||||||
|
/// </summary>
|
||||||
|
public bool IsLiveStream { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets when this livestream URL was last fetched from the API.
|
||||||
|
/// Used to prevent rapid-fire API calls from clients like Android TV.
|
||||||
|
/// </summary>
|
||||||
|
public DateTime? LastLivestreamFetchAt { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets a value indicating whether this stream needs authentication on first access.
|
||||||
|
/// True when registered via RegisterStreamDeferred (authentication deferred until playback).
|
||||||
|
/// </summary>
|
||||||
|
public bool NeedsAuthentication { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -6,6 +6,9 @@ using System.Threading;
|
|||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
using Jellyfin.Plugin.SRFPlay.Api.Models;
|
||||||
using Jellyfin.Plugin.SRFPlay.Configuration;
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Constants;
|
||||||
|
using Jellyfin.Plugin.SRFPlay.Services.Interfaces;
|
||||||
|
using MediaBrowser.Common.Net;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
namespace Jellyfin.Plugin.SRFPlay.Services;
|
namespace Jellyfin.Plugin.SRFPlay.Services;
|
||||||
@ -13,20 +16,20 @@ namespace Jellyfin.Plugin.SRFPlay.Services;
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Service for resolving stream URLs from media composition resources.
|
/// Service for resolving stream URLs from media composition resources.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class StreamUrlResolver : IDisposable
|
public class StreamUrlResolver : IStreamUrlResolver
|
||||||
{
|
{
|
||||||
private readonly ILogger<StreamUrlResolver> _logger;
|
private readonly ILogger<StreamUrlResolver> _logger;
|
||||||
private readonly HttpClient _httpClient;
|
private readonly IHttpClientFactory _httpClientFactory;
|
||||||
private bool _disposed;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="StreamUrlResolver"/> class.
|
/// Initializes a new instance of the <see cref="StreamUrlResolver"/> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="logger">The logger instance.</param>
|
/// <param name="logger">The logger instance.</param>
|
||||||
public StreamUrlResolver(ILogger<StreamUrlResolver> logger)
|
/// <param name="httpClientFactory">The HTTP client factory.</param>
|
||||||
|
public StreamUrlResolver(ILogger<StreamUrlResolver> logger, IHttpClientFactory httpClientFactory)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_httpClient = new HttpClient();
|
_httpClientFactory = httpClientFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -87,6 +90,17 @@ public class StreamUrlResolver : IDisposable
|
|||||||
chapter.Id,
|
chapter.Id,
|
||||||
hlsResources.Count);
|
hlsResources.Count);
|
||||||
|
|
||||||
|
// Log all HLS resources with their quality info to help debug quality selection
|
||||||
|
foreach (var resource in hlsResources)
|
||||||
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Available HLS resource - Quality={Quality}, Protocol={Protocol}, Streaming={Streaming}, URL={Url}",
|
||||||
|
resource.Quality ?? "NULL",
|
||||||
|
resource.Protocol ?? "NULL",
|
||||||
|
resource.Streaming ?? "NULL",
|
||||||
|
resource.Url);
|
||||||
|
}
|
||||||
|
|
||||||
if (hlsResources.Count == 0)
|
if (hlsResources.Count == 0)
|
||||||
{
|
{
|
||||||
_logger.LogWarning("No HLS resources found for chapter: {ChapterId}", chapter.Id);
|
_logger.LogWarning("No HLS resources found for chapter: {ChapterId}", chapter.Id);
|
||||||
@ -114,6 +128,10 @@ public class StreamUrlResolver : IDisposable
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Select based on quality preference
|
// Select based on quality preference
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Selecting stream with quality preference: {QualityPreference}",
|
||||||
|
qualityPreference);
|
||||||
|
|
||||||
Resource? selectedResource = qualityPreference switch
|
Resource? selectedResource = qualityPreference switch
|
||||||
{
|
{
|
||||||
QualityPreference.HD => SelectHDResource(hlsResources) ?? SelectBestAvailableResource(hlsResources),
|
QualityPreference.HD => SelectHDResource(hlsResources) ?? SelectBestAvailableResource(hlsResources),
|
||||||
@ -127,7 +145,7 @@ public class StreamUrlResolver : IDisposable
|
|||||||
_logger.LogDebug(
|
_logger.LogDebug(
|
||||||
"Selected stream for chapter {ChapterId}: Quality={Quality}, Protocol={Protocol}, URL={Url}",
|
"Selected stream for chapter {ChapterId}: Quality={Quality}, Protocol={Protocol}, URL={Url}",
|
||||||
chapter.Id,
|
chapter.Id,
|
||||||
selectedResource.Quality,
|
selectedResource.Quality ?? "NULL",
|
||||||
selectedResource.Protocol,
|
selectedResource.Protocol,
|
||||||
selectedResource.Url);
|
selectedResource.Url);
|
||||||
return selectedResource.Url;
|
return selectedResource.Url;
|
||||||
@ -226,11 +244,12 @@ public class StreamUrlResolver : IDisposable
|
|||||||
|
|
||||||
// Build ACL path: /{segment1}/{segment2}/*
|
// Build ACL path: /{segment1}/{segment2}/*
|
||||||
var aclPath = $"/{pathSegments[0]}/{pathSegments[1]}/*";
|
var aclPath = $"/{pathSegments[0]}/{pathSegments[1]}/*";
|
||||||
var tokenUrl = $"https://tp.srgssr.ch/akahd/token?acl={Uri.EscapeDataString(aclPath)}";
|
var tokenUrl = $"{ApiEndpoints.AkamaiTokenEndpoint}?acl={Uri.EscapeDataString(aclPath)}";
|
||||||
|
|
||||||
_logger.LogDebug("Fetching auth token from: {TokenUrl}", tokenUrl);
|
_logger.LogDebug("Fetching auth token from: {TokenUrl}", tokenUrl);
|
||||||
|
|
||||||
var response = await _httpClient.GetAsync(tokenUrl, cancellationToken).ConfigureAwait(false);
|
using var httpClient = _httpClientFactory.CreateClient(NamedClient.Default);
|
||||||
|
var response = await httpClient.GetAsync(tokenUrl, cancellationToken).ConfigureAwait(false);
|
||||||
response.EnsureSuccessStatusCode();
|
response.EnsureSuccessStatusCode();
|
||||||
|
|
||||||
var jsonContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
var jsonContent = await response.Content.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
|
||||||
@ -261,28 +280,4 @@ public class StreamUrlResolver : IDisposable
|
|||||||
return streamUrl; // Return original URL as fallback
|
return streamUrl; // Return original URL as fallback
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc/>
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
Dispose(true);
|
|
||||||
GC.SuppressFinalize(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Releases the unmanaged resources and optionally releases the managed resources.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="disposing">True to release both managed and unmanaged resources; false to release only unmanaged resources.</param>
|
|
||||||
protected virtual void Dispose(bool disposing)
|
|
||||||
{
|
|
||||||
if (!_disposed)
|
|
||||||
{
|
|
||||||
if (disposing)
|
|
||||||
{
|
|
||||||
_httpClient?.Dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
_disposed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
41
Jellyfin.Plugin.SRFPlay/Utilities/Extensions.cs
Normal file
41
Jellyfin.Plugin.SRFPlay/Utilities/Extensions.cs
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
using Jellyfin.Plugin.SRFPlay.Configuration;
|
||||||
|
using MediaBrowser.Controller.Entities;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Extension methods for common plugin operations.
|
||||||
|
/// </summary>
|
||||||
|
public static class Extensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the SRF URN from the item's provider IDs.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="item">The base item.</param>
|
||||||
|
/// <returns>The SRF URN, or null if not found or empty.</returns>
|
||||||
|
public static string? GetSrfUrn(this BaseItem item)
|
||||||
|
{
|
||||||
|
return item.ProviderIds.TryGetValue("SRF", out var urn) && !string.IsNullOrEmpty(urn)
|
||||||
|
? urn
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Converts the BusinessUnit enum to its lowercase string representation.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="businessUnit">The business unit.</param>
|
||||||
|
/// <returns>The lowercase string representation.</returns>
|
||||||
|
public static string ToLowerString(this BusinessUnit businessUnit)
|
||||||
|
{
|
||||||
|
return businessUnit.ToString().ToLowerInvariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the plugin configuration safely, returning null if not available.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The plugin configuration, or null if not available.</returns>
|
||||||
|
public static PluginConfiguration? GetPluginConfig()
|
||||||
|
{
|
||||||
|
return Plugin.Instance?.Configuration;
|
||||||
|
}
|
||||||
|
}
|
||||||
74
Jellyfin.Plugin.SRFPlay/Utilities/MimeTypeHelper.cs
Normal file
74
Jellyfin.Plugin.SRFPlay/Utilities/MimeTypeHelper.cs
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Helper class for determining MIME content types.
|
||||||
|
/// </summary>
|
||||||
|
public static class MimeTypeHelper
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the MIME content type for an image based on URL or file extension.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="url">The image URL.</param>
|
||||||
|
/// <returns>The MIME content type, defaulting to "image/jpeg" for SRF images.</returns>
|
||||||
|
public static string GetImageContentType(string? url)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrEmpty(url))
|
||||||
|
{
|
||||||
|
return "image/jpeg";
|
||||||
|
}
|
||||||
|
|
||||||
|
var uri = new Uri(url);
|
||||||
|
var path = uri.AbsolutePath.ToLowerInvariant();
|
||||||
|
|
||||||
|
if (path.EndsWith(".png", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
return "image/png";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.EndsWith(".gif", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
return "image/gif";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.EndsWith(".webp", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
return "image/webp";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.EndsWith(".jpg", StringComparison.Ordinal) || path.EndsWith(".jpeg", StringComparison.Ordinal))
|
||||||
|
{
|
||||||
|
return "image/jpeg";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default to JPEG for SRF images (most common)
|
||||||
|
return "image/jpeg";
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the MIME content type for a media segment based on file extension.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="path">The segment path or filename.</param>
|
||||||
|
/// <returns>The MIME content type.</returns>
|
||||||
|
public static string GetSegmentContentType(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";
|
||||||
|
}
|
||||||
|
}
|
||||||
141
Jellyfin.Plugin.SRFPlay/Utilities/PlaceholderImageGenerator.cs
Normal file
141
Jellyfin.Plugin.SRFPlay/Utilities/PlaceholderImageGenerator.cs
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
using System.IO;
|
||||||
|
using SkiaSharp;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Generates placeholder images for content without thumbnails.
|
||||||
|
/// </summary>
|
||||||
|
public static class PlaceholderImageGenerator
|
||||||
|
{
|
||||||
|
private const int Width = 640;
|
||||||
|
private const int Height = 360; // 16:9 aspect ratio
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Generates a placeholder image with the given text centered.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="text">The text to display (typically channel/show name).</param>
|
||||||
|
/// <returns>A memory stream containing the PNG image.</returns>
|
||||||
|
public static MemoryStream GeneratePlaceholder(string text)
|
||||||
|
{
|
||||||
|
using var surface = SKSurface.Create(new SKImageInfo(Width, Height));
|
||||||
|
var canvas = surface.Canvas;
|
||||||
|
|
||||||
|
// Dark gradient background
|
||||||
|
using var backgroundPaint = new SKPaint();
|
||||||
|
using var shader = SKShader.CreateLinearGradient(
|
||||||
|
new SKPoint(0, 0),
|
||||||
|
new SKPoint(Width, Height),
|
||||||
|
new[] { new SKColor(45, 45, 48), new SKColor(28, 28, 30) },
|
||||||
|
null,
|
||||||
|
SKShaderTileMode.Clamp);
|
||||||
|
backgroundPaint.Shader = shader;
|
||||||
|
canvas.DrawRect(0, 0, Width, Height, backgroundPaint);
|
||||||
|
|
||||||
|
// Text
|
||||||
|
using var textPaint = new SKPaint
|
||||||
|
{
|
||||||
|
Color = SKColors.White,
|
||||||
|
IsAntialias = true,
|
||||||
|
TextAlign = SKTextAlign.Center,
|
||||||
|
Typeface = SKTypeface.FromFamilyName("Arial", SKFontStyle.Bold)
|
||||||
|
};
|
||||||
|
|
||||||
|
// Calculate font size and wrap text if needed
|
||||||
|
float fontSize = 48;
|
||||||
|
textPaint.TextSize = fontSize;
|
||||||
|
var maxWidth = Width * 0.85f;
|
||||||
|
var maxHeight = Height * 0.8f;
|
||||||
|
|
||||||
|
var lines = WrapText(text, textPaint, maxWidth, maxHeight, ref fontSize);
|
||||||
|
|
||||||
|
// Draw each line centered
|
||||||
|
var lineHeight = fontSize * 1.2f;
|
||||||
|
var totalHeight = lines.Count * lineHeight;
|
||||||
|
var startY = ((Height - totalHeight) / 2) + fontSize;
|
||||||
|
|
||||||
|
foreach (var line in lines)
|
||||||
|
{
|
||||||
|
canvas.DrawText(line, Width / 2, startY, textPaint);
|
||||||
|
startY += lineHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Encode to PNG
|
||||||
|
using var image = surface.Snapshot();
|
||||||
|
using var data = image.Encode(SKEncodedImageFormat.Png, 100);
|
||||||
|
|
||||||
|
var stream = new MemoryStream();
|
||||||
|
data.SaveTo(stream);
|
||||||
|
stream.Position = 0;
|
||||||
|
return stream;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Wraps text to fit within the specified width and height constraints.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="text">The text to wrap.</param>
|
||||||
|
/// <param name="paint">The paint to use for measuring.</param>
|
||||||
|
/// <param name="maxWidth">Maximum width for each line.</param>
|
||||||
|
/// <param name="maxHeight">Maximum total height.</param>
|
||||||
|
/// <param name="fontSize">Font size (will be adjusted if needed).</param>
|
||||||
|
/// <returns>List of text lines.</returns>
|
||||||
|
private static System.Collections.Generic.List<string> WrapText(
|
||||||
|
string text,
|
||||||
|
SKPaint paint,
|
||||||
|
float maxWidth,
|
||||||
|
float maxHeight,
|
||||||
|
ref float fontSize)
|
||||||
|
{
|
||||||
|
const float minFontSize = 20;
|
||||||
|
var words = text.Split(' ');
|
||||||
|
var lines = new System.Collections.Generic.List<string>();
|
||||||
|
|
||||||
|
while (fontSize >= minFontSize)
|
||||||
|
{
|
||||||
|
paint.TextSize = fontSize;
|
||||||
|
lines.Clear();
|
||||||
|
|
||||||
|
var currentLine = string.Empty;
|
||||||
|
foreach (var word in words)
|
||||||
|
{
|
||||||
|
var testLine = string.IsNullOrEmpty(currentLine) ? word : $"{currentLine} {word}";
|
||||||
|
var testWidth = paint.MeasureText(testLine);
|
||||||
|
|
||||||
|
if (testWidth > maxWidth && !string.IsNullOrEmpty(currentLine))
|
||||||
|
{
|
||||||
|
lines.Add(currentLine);
|
||||||
|
currentLine = word;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
currentLine = testLine;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(currentLine))
|
||||||
|
{
|
||||||
|
lines.Add(currentLine);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if total height fits
|
||||||
|
var lineHeight = fontSize * 1.2f;
|
||||||
|
var totalHeight = lines.Count * lineHeight;
|
||||||
|
|
||||||
|
if (totalHeight <= maxHeight)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reduce font size and try again
|
||||||
|
fontSize -= 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If still doesn't fit, just return what we have
|
||||||
|
if (lines.Count == 0)
|
||||||
|
{
|
||||||
|
lines.Add(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
return lines;
|
||||||
|
}
|
||||||
|
}
|
||||||
27
Jellyfin.Plugin.SRFPlay/Utilities/UrnHelper.cs
Normal file
27
Jellyfin.Plugin.SRFPlay/Utilities/UrnHelper.cs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
using System;
|
||||||
|
using System.Security.Cryptography;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
namespace Jellyfin.Plugin.SRFPlay.Utilities;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Helper class for URN-related operations.
|
||||||
|
/// </summary>
|
||||||
|
public static class UrnHelper
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Generates a deterministic GUID from a URN.
|
||||||
|
/// This ensures the same URN always produces the same GUID.
|
||||||
|
/// MD5 is used for non-cryptographic purposes only (generating stable IDs).
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="urn">The URN to convert.</param>
|
||||||
|
/// <returns>A deterministic GUID string.</returns>
|
||||||
|
#pragma warning disable CA5351 // MD5 is used for non-cryptographic purposes (ID generation)
|
||||||
|
public static string ToGuid(string urn)
|
||||||
|
{
|
||||||
|
var hash = MD5.HashData(Encoding.UTF8.GetBytes(urn));
|
||||||
|
var guid = new Guid(hash);
|
||||||
|
return guid.ToString();
|
||||||
|
}
|
||||||
|
#pragma warning restore CA5351
|
||||||
|
}
|
||||||
141
README.md
141
README.md
@ -1,6 +1,20 @@
|
|||||||
# Jellyfin SRF Play Plugin
|
# Jellyfin SRF Play Plugin
|
||||||
|
|
||||||
A Jellyfin plugin for accessing SRF Play (Swiss Radio and Television) video-on-demand content.
|
A Jellyfin plugin for accessing SRF Play (Swiss Radio and Television) video-on-demand content and live sports streaming.
|
||||||
|
|
||||||
|
## Status
|
||||||
|
|
||||||
|
**Beta/Alpha** - This plugin has been tested on two Jellyfin instances and is working. Some clients may experience issues with hardware decoding, which appears to be client-specific behavior.
|
||||||
|
|
||||||
|
## Quick Install
|
||||||
|
|
||||||
|
Add this repository URL in Jellyfin (Dashboard → Plugins → Repositories):
|
||||||
|
|
||||||
|
```
|
||||||
|
https://gitea.tourolle.paris/dtourolle/jellyfin-srfPlay/raw/branch/master/manifest.json
|
||||||
|
```
|
||||||
|
|
||||||
|
Then install "SRF Play" from the plugin catalog.
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
@ -25,95 +39,13 @@ The main channel interface showing the content folders.
|
|||||||
|
|
||||||
Video playback with HLS streaming support and quality selection.
|
Video playback with HLS streaming support and quality selection.
|
||||||
|
|
||||||
## Project Status
|
## Testing
|
||||||
|
|
||||||
### ✅ Completed Components
|
The plugin includes comprehensive testing:
|
||||||
|
|
||||||
#### Phase 1: Project Setup
|
- **Unit tests** (xUnit framework) for core services
|
||||||
- ✅ Renamed from Template to SRFPlay
|
- **API spec validation tests** for all business units (SRF, RTS, RSI, RTR, SWI)
|
||||||
- ✅ Updated all namespaces and identifiers
|
- **Integration testing** - Tested on multiple Jellyfin instances with VOD and live sports streaming
|
||||||
- ✅ Configured plugin metadata (ID, name)
|
|
||||||
|
|
||||||
#### Phase 2: Core API Infrastructure
|
|
||||||
- ✅ API Models (MediaComposition, Chapter, Resource, Show, Episode)
|
|
||||||
- ✅ SRF API Client with HTTP client wrapper
|
|
||||||
- ✅ JSON deserialization support
|
|
||||||
- ✅ Error handling and logging
|
|
||||||
|
|
||||||
#### Phase 3: Configuration
|
|
||||||
- ✅ Business unit selection (SRF/RTS/RSI/RTR/SWI)
|
|
||||||
- ✅ Quality preferences (Auto/SD/HD)
|
|
||||||
- ✅ Content refresh intervals
|
|
||||||
- ✅ Expiration check settings
|
|
||||||
- ✅ Cache duration configuration
|
|
||||||
- ✅ HTML configuration page
|
|
||||||
|
|
||||||
#### Phase 4: Services
|
|
||||||
- ✅ Stream URL Resolver
|
|
||||||
- HLS stream selection
|
|
||||||
- Quality-based filtering
|
|
||||||
- DRM content filtering
|
|
||||||
- Content expiration checking
|
|
||||||
- Akamai token authentication
|
|
||||||
- Upcoming livestream detection
|
|
||||||
- ✅ Metadata Cache Service
|
|
||||||
- Efficient caching with configurable duration
|
|
||||||
- Thread-safe with ReaderWriterLockSlim
|
|
||||||
- IDisposable implementation
|
|
||||||
- Dynamic TTL for scheduled livestreams
|
|
||||||
- ✅ Content Expiration Service
|
|
||||||
- Automatic expiration checking
|
|
||||||
- Library cleanup of expired content
|
|
||||||
- Statistics and monitoring
|
|
||||||
- ✅ Content Refresh Service
|
|
||||||
- Latest and trending content discovery
|
|
||||||
- Scheduled sports livestreams
|
|
||||||
- Automatic cache population
|
|
||||||
- Recommendations system
|
|
||||||
|
|
||||||
#### Phase 5: Content Providers
|
|
||||||
- ✅ Series Provider (for show metadata)
|
|
||||||
- ✅ Episode Provider (for episode metadata)
|
|
||||||
- ✅ Image Provider (for thumbnails and artwork)
|
|
||||||
- ✅ Media Provider (for playback URLs and HLS streams)
|
|
||||||
|
|
||||||
#### Phase 6: Scheduled Tasks
|
|
||||||
- ✅ Content Refresh Task
|
|
||||||
- Periodic discovery of new content
|
|
||||||
- Configurable refresh intervals
|
|
||||||
- ✅ Expiration Check Task
|
|
||||||
- Automatic cleanup of expired content
|
|
||||||
- Configurable check intervals
|
|
||||||
|
|
||||||
#### Phase 7: Dependency Injection & Integration
|
|
||||||
- ✅ Service registration (ServiceRegistrator)
|
|
||||||
- ✅ Jellyfin provider interfaces implementation
|
|
||||||
- ✅ Plugin initialization and configuration
|
|
||||||
|
|
||||||
#### Phase 8: Live Sports Streaming
|
|
||||||
- ✅ Play v3 API integration for scheduled livestreams
|
|
||||||
- ✅ Sports events folder in channel
|
|
||||||
- ✅ Upcoming event detection and display
|
|
||||||
- ✅ Akamai token authentication for streams
|
|
||||||
- ✅ Dynamic cache refresh for live events
|
|
||||||
- ✅ Event scheduling with ValidFrom/ValidTo handling
|
|
||||||
- ✅ Automatic stream URL resolution when events go live
|
|
||||||
|
|
||||||
### ✅ Build Status
|
|
||||||
**Successfully compiling!** All code analysis warnings resolved.
|
|
||||||
|
|
||||||
### 🧪 Testing Status
|
|
||||||
- [x] Unit tests (xUnit framework)
|
|
||||||
- StreamUrlResolver tests
|
|
||||||
- MetadataCache tests
|
|
||||||
- More to be added
|
|
||||||
- [x] API spec validation tests (nightly automated runs)
|
|
||||||
- All business units (SRF, RTS, RSI, RTR, SWI)
|
|
||||||
- Response schema validation
|
|
||||||
- Performance monitoring
|
|
||||||
- [ ] Integration testing with Jellyfin instance
|
|
||||||
- [ ] End-to-end playback testing for VOD content
|
|
||||||
- [ ] Live sports streaming validation
|
|
||||||
|
|
||||||
**Run tests:**
|
**Run tests:**
|
||||||
```bash
|
```bash
|
||||||
@ -132,13 +64,6 @@ dotnet test --collect:"XPlat Code Coverage"
|
|||||||
|
|
||||||
See [Test Documentation](Jellyfin.Plugin.SRFPlay.Tests/README.md) for more details.
|
See [Test Documentation](Jellyfin.Plugin.SRFPlay.Tests/README.md) for more details.
|
||||||
|
|
||||||
### 📝 Next Steps
|
|
||||||
1. Test live sports streaming when events are scheduled
|
|
||||||
2. Verify Akamai token authentication
|
|
||||||
3. Test with different business units (RTS, RSI, RTR)
|
|
||||||
4. Add more unit tests for remaining services
|
|
||||||
5. Performance optimization if needed
|
|
||||||
|
|
||||||
## API Information
|
## API Information
|
||||||
|
|
||||||
**Base URL:** `https://il.srgssr.ch/integrationlayer/2.0/`
|
**Base URL:** `https://il.srgssr.ch/integrationlayer/2.0/`
|
||||||
@ -187,7 +112,7 @@ dotnet build
|
|||||||
|
|
||||||
The compiled plugin will be in `bin/Debug/net8.0/`
|
The compiled plugin will be in `bin/Debug/net8.0/`
|
||||||
|
|
||||||
## Installation
|
## Manual Installation
|
||||||
|
|
||||||
1. Build the plugin (see above)
|
1. Build the plugin (see above)
|
||||||
2. Copy the compiled DLL to your Jellyfin plugins directory
|
2. Copy the compiled DLL to your Jellyfin plugins directory
|
||||||
@ -297,9 +222,7 @@ Currently focused on SRF but easily extensible.
|
|||||||
|
|
||||||
## Development
|
## Development
|
||||||
|
|
||||||
### Current Status
|
The plugin includes:
|
||||||
|
|
||||||
All core functionality is implemented and compiling successfully! The plugin includes:
|
|
||||||
- Complete API integration with SRF Play (Integration Layer v2.0 and Play v3)
|
- Complete API integration with SRF Play (Integration Layer v2.0 and Play v3)
|
||||||
- **Live sports streaming** with scheduled event detection
|
- **Live sports streaming** with scheduled event detection
|
||||||
- Channel with Latest, Trending, and Live Sports folders
|
- Channel with Latest, Trending, and Live Sports folders
|
||||||
@ -310,25 +233,15 @@ All core functionality is implemented and compiling successfully! The plugin inc
|
|||||||
- Scheduled tasks for content refresh
|
- Scheduled tasks for content refresh
|
||||||
- Smart caching with dynamic TTL for upcoming livestreams
|
- Smart caching with dynamic TTL for upcoming livestreams
|
||||||
|
|
||||||
### Testing
|
### Known Issues
|
||||||
|
|
||||||
To test the plugin:
|
- Some clients may experience issues with hardware decoding (appears to be client-specific)
|
||||||
|
- Some edge cases may need additional handling
|
||||||
1. Build the plugin: `dotnet build`
|
- Performance optimization may be needed for very large content catalogs
|
||||||
2. Copy `bin/Debug/net8.0/Jellyfin.Plugin.SRFPlay.dll` to your Jellyfin plugins directory
|
|
||||||
3. Restart Jellyfin
|
|
||||||
4. Configure the plugin in Dashboard → Plugins → SRF Play
|
|
||||||
5. Add a library with the SRF Play content provider
|
|
||||||
|
|
||||||
### Known Limitations
|
|
||||||
|
|
||||||
- This is a first version that has not been tested with a live Jellyfin instance yet
|
|
||||||
- Some edge cases may need handling
|
|
||||||
- Performance optimization may be needed for large content catalogs
|
|
||||||
|
|
||||||
### Contributing
|
### Contributing
|
||||||
|
|
||||||
This plugin is in active development. Contributions welcome!
|
Contributions welcome!
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
|||||||
BIN
assests/main logo.png
Normal file
BIN
assests/main logo.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 20 KiB |
37
manifest.json
Normal file
37
manifest.json
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"guid": "a4b12f86-8c3d-4e9a-b7f2-1d5e6c8a9b4f",
|
||||||
|
"name": "SRF Play",
|
||||||
|
"description": "SRF Play plugin for Jellyfin enables streaming of content from SRF (Swiss Radio and Television). Access live TV channels and on-demand content from Switzerland's German-language public broadcaster directly within Jellyfin.",
|
||||||
|
"overview": "Stream SRF content in Jellyfin",
|
||||||
|
"owner": "dtourolle",
|
||||||
|
"category": "Live TV",
|
||||||
|
"imageUrl": "https://gitea.tourolle.paris/dtourolle/jellyfin-srfPlay/raw/branch/master/assests/main%20logo.png",
|
||||||
|
"versions": [
|
||||||
|
{
|
||||||
|
"version": "1.0.14",
|
||||||
|
"changelog": "Release 1.0.14",
|
||||||
|
"targetAbi": "10.9.0.0",
|
||||||
|
"sourceUrl": "https://gitea.tourolle.paris/dtourolle/jellyfin-srfPlay/releases/download/v1.0.14/srfplay_1.0.14.0.zip",
|
||||||
|
"checksum": "49e6afe16f7abf95c099ecc1d016e725",
|
||||||
|
"timestamp": "2025-12-30T12:38:41Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.0.13",
|
||||||
|
"changelog": "Release 1.0.13",
|
||||||
|
"targetAbi": "10.9.0.0",
|
||||||
|
"sourceUrl": "https://gitea.tourolle.paris/dtourolle/jellyfin-srfPlay/releases/download/v1.0.13/srfplay_1.0.13.0.zip",
|
||||||
|
"checksum": "ac7c1e33c926c21f8e4319da91ef079c",
|
||||||
|
"timestamp": "2025-12-21T13:02:37Z"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "1.0.12",
|
||||||
|
"changelog": "Release 1.0.12",
|
||||||
|
"targetAbi": "10.9.0.0",
|
||||||
|
"sourceUrl": "https://gitea.tourolle.paris/dtourolle/jellyfin-srfPlay/releases/download/v1.0.12/srfplay_1.0.12.0.zip",
|
||||||
|
"checksum": "47baa02ade413253db94fe3ba0763f69",
|
||||||
|
"timestamp": "2025-12-20T13:35:38Z"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
Loading…
x
Reference in New Issue
Block a user