parent
41ac5f8d76
commit
2351eeba56
@ -1,74 +1,94 @@
|
||||
#nullable disable
|
||||
#pragma warning disable CS1591
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Xml.Serialization;
|
||||
using Jellyfin.Extensions;
|
||||
using MediaBrowser.Model.Extensions;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna
|
||||
/// <summary>
|
||||
/// Defines the <see cref="CodecProfile"/>.
|
||||
/// </summary>
|
||||
public class CodecProfile
|
||||
{
|
||||
public class CodecProfile
|
||||
/// <summary>
|
||||
/// Initializes a new instance of the <see cref="CodecProfile"/> class.
|
||||
/// </summary>
|
||||
public CodecProfile()
|
||||
{
|
||||
public CodecProfile()
|
||||
{
|
||||
Conditions = Array.Empty<ProfileCondition>();
|
||||
ApplyConditions = Array.Empty<ProfileCondition>();
|
||||
}
|
||||
|
||||
[XmlAttribute("type")]
|
||||
public CodecType Type { get; set; }
|
||||
|
||||
public ProfileCondition[] Conditions { get; set; }
|
||||
|
||||
public ProfileCondition[] ApplyConditions { get; set; }
|
||||
|
||||
[XmlAttribute("codec")]
|
||||
public string Codec { get; set; }
|
||||
Conditions = [];
|
||||
ApplyConditions = [];
|
||||
}
|
||||
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the <see cref="CodecType"/> which this container must meet.
|
||||
/// </summary>
|
||||
[XmlAttribute("type")]
|
||||
public CodecType Type { get; set; }
|
||||
|
||||
[XmlAttribute("subcontainer")]
|
||||
public string SubContainer { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the list of <see cref="ProfileCondition"/> which this profile must meet.
|
||||
/// </summary>
|
||||
public ProfileCondition[] Conditions { get; set; }
|
||||
|
||||
public string[] GetCodecs()
|
||||
{
|
||||
return ContainerProfile.SplitValue(Codec);
|
||||
}
|
||||
/// <summary>
|
||||
/// Gets or sets the list of <see cref="ProfileCondition"/> to apply if this profile is met.
|
||||
/// </summary>
|
||||
public ProfileCondition[] ApplyConditions { get; set; }
|
||||
|
||||
private bool ContainsContainer(string container, bool useSubContainer = false)
|
||||
{
|
||||
var containerToCheck = useSubContainer && string.Equals(Container, "hls", StringComparison.OrdinalIgnoreCase) ? SubContainer : Container;
|
||||
return ContainerProfile.ContainsContainer(containerToCheck, container);
|
||||
}
|
||||
/// <summary>
|
||||
/// Gets or sets the codec(s) that this profile applies to.
|
||||
/// </summary>
|
||||
[XmlAttribute("codec")]
|
||||
public string? Codec { get; set; }
|
||||
|
||||
public bool ContainsAnyCodec(string codec, string container, bool useSubContainer = false)
|
||||
{
|
||||
return ContainsAnyCodec(ContainerProfile.SplitValue(codec), container, useSubContainer);
|
||||
}
|
||||
/// <summary>
|
||||
/// Gets or sets the container(s) which this profile will be applied to.
|
||||
/// </summary>
|
||||
[XmlAttribute("container")]
|
||||
public string? Container { get; set; }
|
||||
|
||||
public bool ContainsAnyCodec(string[] codec, string container, bool useSubContainer = false)
|
||||
{
|
||||
if (!ContainsContainer(container, useSubContainer))
|
||||
{
|
||||
return false;
|
||||
}
|
||||
/// <summary>
|
||||
/// Gets or sets the sub-container(s) which this profile will be applied to.
|
||||
/// </summary>
|
||||
[XmlAttribute("subcontainer")]
|
||||
public string? SubContainer { get; set; }
|
||||
|
||||
var codecs = GetCodecs();
|
||||
if (codecs.Length == 0)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
/// <summary>
|
||||
/// Checks to see whether the codecs and containers contain the given parameters.
|
||||
/// </summary>
|
||||
/// <param name="codecs">The codecs to match.</param>
|
||||
/// <param name="container">The container to match.</param>
|
||||
/// <param name="useSubContainer">Consider sub-containers.</param>
|
||||
/// <returns>True if both conditions are met.</returns>
|
||||
public bool ContainsAnyCodec(IReadOnlyList<string> codecs, string? container, bool useSubContainer = false)
|
||||
{
|
||||
var containerToCheck = useSubContainer && string.Equals(Container, "hls", StringComparison.OrdinalIgnoreCase) ? SubContainer : Container;
|
||||
return ContainerHelper.ContainsContainer(containerToCheck, container) && codecs.Any(c => ContainerHelper.ContainsContainer(Codec, false, c));
|
||||
}
|
||||
|
||||
foreach (var val in codec)
|
||||
{
|
||||
if (codecs.Contains(val, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
return true;
|
||||
}
|
||||
}
|
||||
/// <summary>
|
||||
/// Checks to see whether the codecs and containers contain the given parameters.
|
||||
/// </summary>
|
||||
/// <param name="codec">The codec to match.</param>
|
||||
/// <param name="container">The container to match.</param>
|
||||
/// <param name="useSubContainer">Consider sub-containers.</param>
|
||||
/// <returns>True if both conditions are met.</returns>
|
||||
public bool ContainsAnyCodec(string? codec, string? container, bool useSubContainer = false)
|
||||
{
|
||||
return ContainsAnyCodec(codec.AsSpan(), container, useSubContainer);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
/// <summary>
|
||||
/// Checks to see whether the codecs and containers contain the given parameters.
|
||||
/// </summary>
|
||||
/// <param name="codec">The codec to match.</param>
|
||||
/// <param name="container">The container to match.</param>
|
||||
/// <param name="useSubContainer">Consider sub-containers.</param>
|
||||
/// <returns>True if both conditions are met.</returns>
|
||||
public bool ContainsAnyCodec(ReadOnlySpan<char> codec, string? container, bool useSubContainer = false)
|
||||
{
|
||||
var containerToCheck = useSubContainer && string.Equals(Container, "hls", StringComparison.OrdinalIgnoreCase) ? SubContainer : Container;
|
||||
return ContainerHelper.ContainsContainer(containerToCheck, container) && ContainerHelper.ContainsContainer(Codec, false, codec);
|
||||
}
|
||||
}
|
||||
|
@ -1,74 +1,49 @@
|
||||
#pragma warning disable CS1591
|
||||
#pragma warning disable CA1819 // Properties should not return arrays
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Xml.Serialization;
|
||||
using Jellyfin.Extensions;
|
||||
using MediaBrowser.Model.Extensions;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna
|
||||
namespace MediaBrowser.Model.Dlna;
|
||||
|
||||
/// <summary>
|
||||
/// Defines the <see cref="ContainerProfile"/>.
|
||||
/// </summary>
|
||||
public class ContainerProfile
|
||||
{
|
||||
public class ContainerProfile
|
||||
/// <summary>
|
||||
/// Gets or sets the <see cref="DlnaProfileType"/> which this container must meet.
|
||||
/// </summary>
|
||||
[XmlAttribute("type")]
|
||||
public DlnaProfileType Type { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the list of <see cref="ProfileCondition"/> which this container will be applied to.
|
||||
/// </summary>
|
||||
public ProfileCondition[] Conditions { get; set; } = [];
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the container(s) which this container must meet.
|
||||
/// </summary>
|
||||
[XmlAttribute("container")]
|
||||
public string? Container { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the sub container(s) which this container must meet.
|
||||
/// </summary>
|
||||
[XmlAttribute("subcontainer")]
|
||||
public string? SubContainer { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Returns true if an item in <paramref name="container"/> appears in the <see cref="Container"/> property.
|
||||
/// </summary>
|
||||
/// <param name="container">The item to match.</param>
|
||||
/// <param name="useSubContainer">Consider subcontainers.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public bool ContainsContainer(ReadOnlySpan<char> container, bool useSubContainer = false)
|
||||
{
|
||||
[XmlAttribute("type")]
|
||||
public DlnaProfileType Type { get; set; }
|
||||
|
||||
public ProfileCondition[] Conditions { get; set; } = Array.Empty<ProfileCondition>();
|
||||
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; } = string.Empty;
|
||||
|
||||
public static string[] SplitValue(string? value)
|
||||
{
|
||||
if (string.IsNullOrEmpty(value))
|
||||
{
|
||||
return Array.Empty<string>();
|
||||
}
|
||||
|
||||
return value.Split(',', StringSplitOptions.RemoveEmptyEntries);
|
||||
}
|
||||
|
||||
public bool ContainsContainer(string? container)
|
||||
{
|
||||
var containers = SplitValue(Container);
|
||||
|
||||
return ContainsContainer(containers, container);
|
||||
}
|
||||
|
||||
public static bool ContainsContainer(string? profileContainers, string? inputContainer)
|
||||
{
|
||||
var isNegativeList = false;
|
||||
if (profileContainers is not null && profileContainers.StartsWith('-'))
|
||||
{
|
||||
isNegativeList = true;
|
||||
profileContainers = profileContainers.Substring(1);
|
||||
}
|
||||
|
||||
return ContainsContainer(SplitValue(profileContainers), isNegativeList, inputContainer);
|
||||
}
|
||||
|
||||
public static bool ContainsContainer(string[]? profileContainers, string? inputContainer)
|
||||
{
|
||||
return ContainsContainer(profileContainers, false, inputContainer);
|
||||
}
|
||||
|
||||
public static bool ContainsContainer(string[]? profileContainers, bool isNegativeList, string? inputContainer)
|
||||
{
|
||||
if (profileContainers is null || profileContainers.Length == 0)
|
||||
{
|
||||
// Empty profiles always support all containers/codecs
|
||||
return true;
|
||||
}
|
||||
|
||||
var allInputContainers = SplitValue(inputContainer);
|
||||
|
||||
foreach (var container in allInputContainers)
|
||||
{
|
||||
if (profileContainers.Contains(container, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
return !isNegativeList;
|
||||
}
|
||||
}
|
||||
|
||||
return isNegativeList;
|
||||
}
|
||||
var containerToCheck = useSubContainer && string.Equals(Container, "hls", StringComparison.OrdinalIgnoreCase) ? SubContainer : Container;
|
||||
return ContainerHelper.ContainsContainer(containerToCheck, container);
|
||||
}
|
||||
}
|
||||
|
@ -1,74 +1,71 @@
|
||||
#pragma warning disable CA1819 // Properties should not return arrays
|
||||
|
||||
using System;
|
||||
using System.Xml.Serialization;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna
|
||||
namespace MediaBrowser.Model.Dlna;
|
||||
|
||||
/// <summary>
|
||||
/// A <see cref="DeviceProfile" /> represents a set of metadata which determines which content a certain device is able to play.
|
||||
/// <br/>
|
||||
/// Specifically, it defines the supported <see cref="ContainerProfiles">containers</see> and
|
||||
/// <see cref="CodecProfiles">codecs</see> (video and/or audio, including codec profiles and levels)
|
||||
/// the device is able to direct play (without transcoding or remuxing),
|
||||
/// as well as which <see cref="TranscodingProfiles">containers/codecs to transcode to</see> in case it isn't.
|
||||
/// </summary>
|
||||
public class DeviceProfile
|
||||
{
|
||||
/// <summary>
|
||||
/// A <see cref="DeviceProfile" /> represents a set of metadata which determines which content a certain device is able to play.
|
||||
/// <br/>
|
||||
/// Specifically, it defines the supported <see cref="ContainerProfiles">containers</see> and
|
||||
/// <see cref="CodecProfiles">codecs</see> (video and/or audio, including codec profiles and levels)
|
||||
/// the device is able to direct play (without transcoding or remuxing),
|
||||
/// as well as which <see cref="TranscodingProfiles">containers/codecs to transcode to</see> in case it isn't.
|
||||
/// Gets or sets the name of this device profile. User profiles must have a unique name.
|
||||
/// </summary>
|
||||
public class DeviceProfile
|
||||
{
|
||||
/// <summary>
|
||||
/// Gets or sets the name of this device profile.
|
||||
/// </summary>
|
||||
public string? Name { get; set; }
|
||||
public string? Name { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the Id.
|
||||
/// </summary>
|
||||
[XmlIgnore]
|
||||
public string? Id { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the unique internal identifier.
|
||||
/// </summary>
|
||||
public Guid Id { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for all streamed content.
|
||||
/// </summary>
|
||||
public int? MaxStreamingBitrate { get; set; } = 8000000;
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for all streamed content.
|
||||
/// </summary>
|
||||
public int? MaxStreamingBitrate { get; set; } = 8000000;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for statically streamed content (= direct played files).
|
||||
/// </summary>
|
||||
public int? MaxStaticBitrate { get; set; } = 8000000;
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for statically streamed content (= direct played files).
|
||||
/// </summary>
|
||||
public int? MaxStaticBitrate { get; set; } = 8000000;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for transcoded music streams.
|
||||
/// </summary>
|
||||
public int? MusicStreamingTranscodingBitrate { get; set; } = 128000;
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for transcoded music streams.
|
||||
/// </summary>
|
||||
public int? MusicStreamingTranscodingBitrate { get; set; } = 128000;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for statically streamed (= direct played) music files.
|
||||
/// </summary>
|
||||
public int? MaxStaticMusicBitrate { get; set; } = 8000000;
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum allowed bitrate for statically streamed (= direct played) music files.
|
||||
/// </summary>
|
||||
public int? MaxStaticMusicBitrate { get; set; } = 8000000;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the direct play profiles.
|
||||
/// </summary>
|
||||
public DirectPlayProfile[] DirectPlayProfiles { get; set; } = Array.Empty<DirectPlayProfile>();
|
||||
/// <summary>
|
||||
/// Gets or sets the direct play profiles.
|
||||
/// </summary>
|
||||
public DirectPlayProfile[] DirectPlayProfiles { get; set; } = [];
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the transcoding profiles.
|
||||
/// </summary>
|
||||
public TranscodingProfile[] TranscodingProfiles { get; set; } = Array.Empty<TranscodingProfile>();
|
||||
/// <summary>
|
||||
/// Gets or sets the transcoding profiles.
|
||||
/// </summary>
|
||||
public TranscodingProfile[] TranscodingProfiles { get; set; } = [];
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the container profiles.
|
||||
/// </summary>
|
||||
public ContainerProfile[] ContainerProfiles { get; set; } = Array.Empty<ContainerProfile>();
|
||||
/// <summary>
|
||||
/// Gets or sets the container profiles. Failing to meet these optional conditions causes transcoding to occur.
|
||||
/// </summary>
|
||||
public ContainerProfile[] ContainerProfiles { get; set; } = [];
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the codec profiles.
|
||||
/// </summary>
|
||||
public CodecProfile[] CodecProfiles { get; set; } = Array.Empty<CodecProfile>();
|
||||
/// <summary>
|
||||
/// Gets or sets the codec profiles.
|
||||
/// </summary>
|
||||
public CodecProfile[] CodecProfiles { get; set; } = [];
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the subtitle profiles.
|
||||
/// </summary>
|
||||
public SubtitleProfile[] SubtitleProfiles { get; set; } = Array.Empty<SubtitleProfile>();
|
||||
}
|
||||
/// <summary>
|
||||
/// Gets or sets the subtitle profiles.
|
||||
/// </summary>
|
||||
public SubtitleProfile[] SubtitleProfiles { get; set; } = [];
|
||||
}
|
||||
|
@ -1,36 +1,65 @@
|
||||
#pragma warning disable CS1591
|
||||
|
||||
using System.Xml.Serialization;
|
||||
using MediaBrowser.Model.Extensions;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna
|
||||
/// <summary>
|
||||
/// Defines the <see cref="DirectPlayProfile"/>.
|
||||
/// </summary>
|
||||
public class DirectPlayProfile
|
||||
{
|
||||
public class DirectPlayProfile
|
||||
{
|
||||
[XmlAttribute("container")]
|
||||
public string? Container { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the container.
|
||||
/// </summary>
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; } = string.Empty;
|
||||
|
||||
[XmlAttribute("audioCodec")]
|
||||
public string? AudioCodec { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the audio codec.
|
||||
/// </summary>
|
||||
[XmlAttribute("audioCodec")]
|
||||
public string? AudioCodec { get; set; }
|
||||
|
||||
[XmlAttribute("videoCodec")]
|
||||
public string? VideoCodec { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the video codec.
|
||||
/// </summary>
|
||||
[XmlAttribute("videoCodec")]
|
||||
public string? VideoCodec { get; set; }
|
||||
|
||||
[XmlAttribute("type")]
|
||||
public DlnaProfileType Type { get; set; }
|
||||
/// <summary>
|
||||
/// Gets or sets the Dlna profile type.
|
||||
/// </summary>
|
||||
[XmlAttribute("type")]
|
||||
public DlnaProfileType Type { get; set; }
|
||||
|
||||
public bool SupportsContainer(string? container)
|
||||
{
|
||||
return ContainerProfile.ContainsContainer(Container, container);
|
||||
}
|
||||
/// <summary>
|
||||
/// Returns whether the <see cref="Container"/> supports the <paramref name="container"/>.
|
||||
/// </summary>
|
||||
/// <param name="container">The container to match against.</param>
|
||||
/// <returns>True if supported.</returns>
|
||||
public bool SupportsContainer(string? container)
|
||||
{
|
||||
return ContainerHelper.ContainsContainer(Container, container);
|
||||
}
|
||||
|
||||
public bool SupportsVideoCodec(string? codec)
|
||||
{
|
||||
return Type == DlnaProfileType.Video && ContainerProfile.ContainsContainer(VideoCodec, codec);
|
||||
}
|
||||
/// <summary>
|
||||
/// Returns whether the <see cref="VideoCodec"/> supports the <paramref name="codec"/>.
|
||||
/// </summary>
|
||||
/// <param name="codec">The codec to match against.</param>
|
||||
/// <returns>True if supported.</returns>
|
||||
public bool SupportsVideoCodec(string? codec)
|
||||
{
|
||||
return Type == DlnaProfileType.Video && ContainerHelper.ContainsContainer(VideoCodec, codec);
|
||||
}
|
||||
|
||||
public bool SupportsAudioCodec(string? codec)
|
||||
{
|
||||
return (Type == DlnaProfileType.Audio || Type == DlnaProfileType.Video) && ContainerProfile.ContainsContainer(AudioCodec, codec);
|
||||
}
|
||||
/// <summary>
|
||||
/// Returns whether the <see cref="AudioCodec"/> supports the <paramref name="codec"/>.
|
||||
/// </summary>
|
||||
/// <param name="codec">The codec to match against.</param>
|
||||
/// <returns>True if supported.</returns>
|
||||
public bool SupportsAudioCodec(string? codec)
|
||||
{
|
||||
// Video profiles can have audio codec restrictions too, therefore incude Video as valid type.
|
||||
return (Type == DlnaProfileType.Audio || Type == DlnaProfileType.Video) && ContainerHelper.ContainsContainer(AudioCodec, codec);
|
||||
}
|
||||
}
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -1,48 +1,62 @@
|
||||
#nullable disable
|
||||
#pragma warning disable CS1591
|
||||
|
||||
using System;
|
||||
using System.Xml.Serialization;
|
||||
using Jellyfin.Extensions;
|
||||
using MediaBrowser.Model.Extensions;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna
|
||||
namespace MediaBrowser.Model.Dlna;
|
||||
|
||||
/// <summary>
|
||||
/// A class for subtitle profile information.
|
||||
/// </summary>
|
||||
public class SubtitleProfile
|
||||
{
|
||||
public class SubtitleProfile
|
||||
/// <summary>
|
||||
/// Gets or sets the format.
|
||||
/// </summary>
|
||||
[XmlAttribute("format")]
|
||||
public string Format { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the delivery method.
|
||||
/// </summary>
|
||||
[XmlAttribute("method")]
|
||||
public SubtitleDeliveryMethod Method { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the DIDL mode.
|
||||
/// </summary>
|
||||
[XmlAttribute("didlMode")]
|
||||
public string DidlMode { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the language.
|
||||
/// </summary>
|
||||
[XmlAttribute("language")]
|
||||
public string Language { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the container.
|
||||
/// </summary>
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Checks if a language is supported.
|
||||
/// </summary>
|
||||
/// <param name="subLanguage">The language to check for support.</param>
|
||||
/// <returns><c>true</c> if supported.</returns>
|
||||
public bool SupportsLanguage(string subLanguage)
|
||||
{
|
||||
[XmlAttribute("format")]
|
||||
public string Format { get; set; }
|
||||
|
||||
[XmlAttribute("method")]
|
||||
public SubtitleDeliveryMethod Method { get; set; }
|
||||
|
||||
[XmlAttribute("didlMode")]
|
||||
public string DidlMode { get; set; }
|
||||
|
||||
[XmlAttribute("language")]
|
||||
public string Language { get; set; }
|
||||
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; }
|
||||
|
||||
public string[] GetLanguages()
|
||||
if (string.IsNullOrEmpty(Language))
|
||||
{
|
||||
return ContainerProfile.SplitValue(Language);
|
||||
return true;
|
||||
}
|
||||
|
||||
public bool SupportsLanguage(string subLanguage)
|
||||
if (string.IsNullOrEmpty(subLanguage))
|
||||
{
|
||||
if (string.IsNullOrEmpty(Language))
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
if (string.IsNullOrEmpty(subLanguage))
|
||||
{
|
||||
subLanguage = "und";
|
||||
}
|
||||
|
||||
var languages = GetLanguages();
|
||||
return languages.Length == 0 || languages.Contains(subLanguage, StringComparison.OrdinalIgnoreCase);
|
||||
subLanguage = "und";
|
||||
}
|
||||
|
||||
return ContainerHelper.ContainsContainer(Language, subLanguage);
|
||||
}
|
||||
}
|
||||
|
@ -1,82 +1,130 @@
|
||||
#pragma warning disable CS1591
|
||||
|
||||
using System;
|
||||
using System.ComponentModel;
|
||||
using System.Xml.Serialization;
|
||||
using Jellyfin.Data.Enums;
|
||||
|
||||
namespace MediaBrowser.Model.Dlna
|
||||
namespace MediaBrowser.Model.Dlna;
|
||||
|
||||
/// <summary>
|
||||
/// A class for transcoding profile information.
|
||||
/// </summary>
|
||||
public class TranscodingProfile
|
||||
{
|
||||
public class TranscodingProfile
|
||||
/// <summary>
|
||||
/// Initializes a new instance of the <see cref="TranscodingProfile" /> class.
|
||||
/// </summary>
|
||||
public TranscodingProfile()
|
||||
{
|
||||
public TranscodingProfile()
|
||||
{
|
||||
Conditions = Array.Empty<ProfileCondition>();
|
||||
}
|
||||
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; } = string.Empty;
|
||||
|
||||
[XmlAttribute("type")]
|
||||
public DlnaProfileType Type { get; set; }
|
||||
|
||||
[XmlAttribute("videoCodec")]
|
||||
public string VideoCodec { get; set; } = string.Empty;
|
||||
|
||||
[XmlAttribute("audioCodec")]
|
||||
public string AudioCodec { get; set; } = string.Empty;
|
||||
|
||||
[XmlAttribute("protocol")]
|
||||
public MediaStreamProtocol Protocol { get; set; } = MediaStreamProtocol.http;
|
||||
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("estimateContentLength")]
|
||||
public bool EstimateContentLength { get; set; }
|
||||
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("enableMpegtsM2TsMode")]
|
||||
public bool EnableMpegtsM2TsMode { get; set; }
|
||||
|
||||
[DefaultValue(TranscodeSeekInfo.Auto)]
|
||||
[XmlAttribute("transcodeSeekInfo")]
|
||||
public TranscodeSeekInfo TranscodeSeekInfo { get; set; }
|
||||
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("copyTimestamps")]
|
||||
public bool CopyTimestamps { get; set; }
|
||||
|
||||
[DefaultValue(EncodingContext.Streaming)]
|
||||
[XmlAttribute("context")]
|
||||
public EncodingContext Context { get; set; }
|
||||
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("enableSubtitlesInManifest")]
|
||||
public bool EnableSubtitlesInManifest { get; set; }
|
||||
|
||||
[XmlAttribute("maxAudioChannels")]
|
||||
public string? MaxAudioChannels { get; set; }
|
||||
|
||||
[DefaultValue(0)]
|
||||
[XmlAttribute("minSegments")]
|
||||
public int MinSegments { get; set; }
|
||||
|
||||
[DefaultValue(0)]
|
||||
[XmlAttribute("segmentLength")]
|
||||
public int SegmentLength { get; set; }
|
||||
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("breakOnNonKeyFrames")]
|
||||
public bool BreakOnNonKeyFrames { get; set; }
|
||||
|
||||
public ProfileCondition[] Conditions { get; set; }
|
||||
|
||||
[DefaultValue(true)]
|
||||
[XmlAttribute("enableAudioVbrEncoding")]
|
||||
public bool EnableAudioVbrEncoding { get; set; } = true;
|
||||
|
||||
public string[] GetAudioCodecs()
|
||||
{
|
||||
return ContainerProfile.SplitValue(AudioCodec);
|
||||
}
|
||||
Conditions = [];
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the container.
|
||||
/// </summary>
|
||||
[XmlAttribute("container")]
|
||||
public string Container { get; set; } = string.Empty;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the DLNA profile type.
|
||||
/// </summary>
|
||||
[XmlAttribute("type")]
|
||||
public DlnaProfileType Type { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the video codec.
|
||||
/// </summary>
|
||||
[XmlAttribute("videoCodec")]
|
||||
public string VideoCodec { get; set; } = string.Empty;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the audio codec.
|
||||
/// </summary>
|
||||
[XmlAttribute("audioCodec")]
|
||||
public string AudioCodec { get; set; } = string.Empty;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the protocol.
|
||||
/// </summary>
|
||||
[XmlAttribute("protocol")]
|
||||
public MediaStreamProtocol Protocol { get; set; } = MediaStreamProtocol.http;
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a value indicating whether the content length should be estimated.
|
||||
/// </summary>
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("estimateContentLength")]
|
||||
public bool EstimateContentLength { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a value indicating whether M2TS mode is enabled.
|
||||
/// </summary>
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("enableMpegtsM2TsMode")]
|
||||
public bool EnableMpegtsM2TsMode { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the transcoding seek info mode.
|
||||
/// </summary>
|
||||
[DefaultValue(TranscodeSeekInfo.Auto)]
|
||||
[XmlAttribute("transcodeSeekInfo")]
|
||||
public TranscodeSeekInfo TranscodeSeekInfo { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a value indicating whether timestamps should be copied.
|
||||
/// </summary>
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("copyTimestamps")]
|
||||
public bool CopyTimestamps { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the encoding context.
|
||||
/// </summary>
|
||||
[DefaultValue(EncodingContext.Streaming)]
|
||||
[XmlAttribute("context")]
|
||||
public EncodingContext Context { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a value indicating whether subtitles are allowed in the manifest.
|
||||
/// </summary>
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("enableSubtitlesInManifest")]
|
||||
public bool EnableSubtitlesInManifest { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the maximum audio channels.
|
||||
/// </summary>
|
||||
[XmlAttribute("maxAudioChannels")]
|
||||
public string? MaxAudioChannels { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the minimum amount of segments.
|
||||
/// </summary>
|
||||
[DefaultValue(0)]
|
||||
[XmlAttribute("minSegments")]
|
||||
public int MinSegments { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the segment length.
|
||||
/// </summary>
|
||||
[DefaultValue(0)]
|
||||
[XmlAttribute("segmentLength")]
|
||||
public int SegmentLength { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a value indicating whether breaking the video stream on non-keyframes is supported.
|
||||
/// </summary>
|
||||
[DefaultValue(false)]
|
||||
[XmlAttribute("breakOnNonKeyFrames")]
|
||||
public bool BreakOnNonKeyFrames { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the profile conditions.
|
||||
/// </summary>
|
||||
public ProfileCondition[] Conditions { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a value indicating whether variable bitrate encoding is supported.
|
||||
/// </summary>
|
||||
[DefaultValue(true)]
|
||||
[XmlAttribute("enableAudioVbrEncoding")]
|
||||
public bool EnableAudioVbrEncoding { get; set; } = true;
|
||||
}
|
||||
|
@ -0,0 +1,145 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using Jellyfin.Extensions;
|
||||
|
||||
namespace MediaBrowser.Model.Extensions;
|
||||
|
||||
/// <summary>
|
||||
/// Defines the <see cref="ContainerHelper"/> class.
|
||||
/// </summary>
|
||||
public static class ContainerHelper
|
||||
{
|
||||
/// <summary>
|
||||
/// Compares two containers, returning true if an item in <paramref name="inputContainer"/> exists
|
||||
/// in <paramref name="profileContainers"/>.
|
||||
/// </summary>
|
||||
/// <param name="profileContainers">The comma-delimited string being searched.
|
||||
/// If the parameter begins with the <c>-</c> character, the operation is reversed.</param>
|
||||
/// <param name="inputContainer">The comma-delimited string being matched.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public static bool ContainsContainer(string? profileContainers, string? inputContainer)
|
||||
{
|
||||
var isNegativeList = false;
|
||||
if (profileContainers != null && profileContainers.StartsWith('-'))
|
||||
{
|
||||
isNegativeList = true;
|
||||
profileContainers = profileContainers[1..];
|
||||
}
|
||||
|
||||
return ContainsContainer(profileContainers, isNegativeList, inputContainer);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Compares two containers, returning true if an item in <paramref name="inputContainer"/> exists
|
||||
/// in <paramref name="profileContainers"/>.
|
||||
/// </summary>
|
||||
/// <param name="profileContainers">The comma-delimited string being searched.
|
||||
/// If the parameter begins with the <c>-</c> character, the operation is reversed.</param>
|
||||
/// <param name="inputContainer">The comma-delimited string being matched.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public static bool ContainsContainer(string? profileContainers, ReadOnlySpan<char> inputContainer)
|
||||
{
|
||||
var isNegativeList = false;
|
||||
if (profileContainers != null && profileContainers.StartsWith('-'))
|
||||
{
|
||||
isNegativeList = true;
|
||||
profileContainers = profileContainers[1..];
|
||||
}
|
||||
|
||||
return ContainsContainer(profileContainers, isNegativeList, inputContainer);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Compares two containers, returning <paramref name="isNegativeList"/> if an item in <paramref name="inputContainer"/>
|
||||
/// does not exist in <paramref name="profileContainers"/>.
|
||||
/// </summary>
|
||||
/// <param name="profileContainers">The comma-delimited string being searched.</param>
|
||||
/// <param name="isNegativeList">The boolean result to return if a match is not found.</param>
|
||||
/// <param name="inputContainer">The comma-delimited string being matched.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public static bool ContainsContainer(string? profileContainers, bool isNegativeList, string? inputContainer)
|
||||
{
|
||||
if (string.IsNullOrEmpty(inputContainer))
|
||||
{
|
||||
return isNegativeList;
|
||||
}
|
||||
|
||||
return ContainsContainer(profileContainers, isNegativeList, inputContainer.AsSpan());
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Compares two containers, returning <paramref name="isNegativeList"/> if an item in <paramref name="inputContainer"/>
|
||||
/// does not exist in <paramref name="profileContainers"/>.
|
||||
/// </summary>
|
||||
/// <param name="profileContainers">The comma-delimited string being searched.</param>
|
||||
/// <param name="isNegativeList">The boolean result to return if a match is not found.</param>
|
||||
/// <param name="inputContainer">The comma-delimited string being matched.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public static bool ContainsContainer(string? profileContainers, bool isNegativeList, ReadOnlySpan<char> inputContainer)
|
||||
{
|
||||
if (string.IsNullOrEmpty(profileContainers))
|
||||
{
|
||||
// Empty profiles always support all containers/codecs.
|
||||
return true;
|
||||
}
|
||||
|
||||
var allInputContainers = inputContainer.Split(',');
|
||||
var allProfileContainers = profileContainers.SpanSplit(',');
|
||||
foreach (var container in allInputContainers)
|
||||
{
|
||||
if (!container.IsEmpty)
|
||||
{
|
||||
foreach (var profile in allProfileContainers)
|
||||
{
|
||||
if (container.Equals(profile, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
return !isNegativeList;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return isNegativeList;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Compares two containers, returning <paramref name="isNegativeList"/> if an item in <paramref name="inputContainer"/>
|
||||
/// does not exist in <paramref name="profileContainers"/>.
|
||||
/// </summary>
|
||||
/// <param name="profileContainers">The profile containers being matched searched.</param>
|
||||
/// <param name="isNegativeList">The boolean result to return if a match is not found.</param>
|
||||
/// <param name="inputContainer">The comma-delimited string being matched.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public static bool ContainsContainer(IReadOnlyList<string>? profileContainers, bool isNegativeList, string inputContainer)
|
||||
{
|
||||
if (profileContainers is null)
|
||||
{
|
||||
// Empty profiles always support all containers/codecs.
|
||||
return true;
|
||||
}
|
||||
|
||||
var allInputContainers = inputContainer.Split(',');
|
||||
foreach (var container in allInputContainers)
|
||||
{
|
||||
foreach (var profile in profileContainers)
|
||||
{
|
||||
if (string.Equals(profile, container, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
return !isNegativeList;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return isNegativeList;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Splits and input string.
|
||||
/// </summary>
|
||||
/// <param name="input">The input string.</param>
|
||||
/// <returns>The result of the operation.</returns>
|
||||
public static string[] Split(string? input)
|
||||
{
|
||||
return input?.Split(',', StringSplitOptions.RemoveEmptyEntries) ?? [];
|
||||
}
|
||||
}
|
@ -0,0 +1,54 @@
|
||||
using System;
|
||||
using MediaBrowser.Model.Dlna;
|
||||
using MediaBrowser.Model.Extensions;
|
||||
using Xunit;
|
||||
|
||||
namespace Jellyfin.Model.Tests.Dlna;
|
||||
|
||||
public class ContainerHelperTests
|
||||
{
|
||||
private readonly ContainerProfile _emptyContainerProfile = new ContainerProfile();
|
||||
|
||||
[Theory]
|
||||
[InlineData(null)]
|
||||
[InlineData("")]
|
||||
[InlineData("mp4")]
|
||||
public void ContainsContainer_EmptyContainerProfile_ReturnsTrue(string? containers)
|
||||
{
|
||||
Assert.True(_emptyContainerProfile.ContainsContainer(containers));
|
||||
}
|
||||
|
||||
[Theory]
|
||||
[InlineData("mp3,mpeg", "mp3")]
|
||||
[InlineData("mp3,mpeg,avi", "mp3,avi")]
|
||||
[InlineData("-mp3,mpeg", "avi")]
|
||||
[InlineData("-mp3,mpeg,avi", "mp4,jpg")]
|
||||
public void ContainsContainer_InList_ReturnsTrue(string container, string? extension)
|
||||
{
|
||||
Assert.True(ContainerHelper.ContainsContainer(container, extension));
|
||||
}
|
||||
|
||||
[Theory]
|
||||
[InlineData("mp3,mpeg", "avi")]
|
||||
[InlineData("mp3,mpeg,avi", "mp4,jpg")]
|
||||
[InlineData("mp3,mpeg", null)]
|
||||
[InlineData("mp3,mpeg", "")]
|
||||
[InlineData("-mp3,mpeg", "mp3")]
|
||||
[InlineData("-mp3,mpeg,avi", "mpeg,avi")]
|
||||
[InlineData(",mp3,", ",avi,")] // Empty values should be discarded
|
||||
[InlineData("-,mp3,", ",mp3,")] // Empty values should be discarded
|
||||
public void ContainsContainer_NotInList_ReturnsFalse(string container, string? extension)
|
||||
{
|
||||
Assert.False(ContainerHelper.ContainsContainer(container, extension));
|
||||
}
|
||||
|
||||
[Theory]
|
||||
[InlineData("mp3,mpeg", "mp3")]
|
||||
[InlineData("mp3,mpeg,avi", "mp3,avi")]
|
||||
[InlineData("-mp3,mpeg", "avi")]
|
||||
[InlineData("-mp3,mpeg,avi", "mp4,jpg")]
|
||||
public void ContainsContainer_InList_ReturnsTrue_SpanVersion(string container, string? extension)
|
||||
{
|
||||
Assert.True(ContainerHelper.ContainsContainer(container, extension.AsSpan()));
|
||||
}
|
||||
}
|
@ -1,19 +0,0 @@
|
||||
using MediaBrowser.Model.Dlna;
|
||||
using Xunit;
|
||||
|
||||
namespace Jellyfin.Model.Tests.Dlna
|
||||
{
|
||||
public class ContainerProfileTests
|
||||
{
|
||||
private readonly ContainerProfile _emptyContainerProfile = new ContainerProfile();
|
||||
|
||||
[Theory]
|
||||
[InlineData(null)]
|
||||
[InlineData("")]
|
||||
[InlineData("mp4")]
|
||||
public void ContainsContainer_EmptyContainerProfile_True(string? containers)
|
||||
{
|
||||
Assert.True(_emptyContainerProfile.ContainsContainer(containers));
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in new issue