Use a dictionary to look up mimetypes

This commit is contained in:
Bond-009 2019-02-13 16:37:18 +01:00 committed by Bond_009
parent a8014b3942
commit 8b1bd7ac6b
4 changed files with 123 additions and 285 deletions

View File

@ -181,19 +181,6 @@ namespace Emby.Dlna.Didl
writer.WriteFullEndElement(); writer.WriteFullEndElement();
} }
private string GetMimeType(string input)
{
var mime = MimeTypes.GetMimeType(input);
// TODO: Instead of being hard-coded here, this should probably be moved into all of the existing profiles
if (string.Equals(mime, "video/mp2t", StringComparison.OrdinalIgnoreCase))
{
mime = "video/mpeg";
}
return mime;
}
private void AddVideoResource(DlnaOptions options, XmlWriter writer, BaseItem video, string deviceId, Filter filter, StreamInfo streamInfo = null) private void AddVideoResource(DlnaOptions options, XmlWriter writer, BaseItem video, string deviceId, Filter filter, StreamInfo streamInfo = null)
{ {
if (streamInfo == null) if (streamInfo == null)
@ -384,7 +371,7 @@ namespace Emby.Dlna.Didl
var filename = url.Substring(0, url.IndexOf('?')); var filename = url.Substring(0, url.IndexOf('?'));
var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType) var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
? GetMimeType(filename) ? MimeTypes.GetMimeType(filename)
: mediaProfile.MimeType; : mediaProfile.MimeType;
writer.WriteAttributeString("protocolInfo", string.Format( writer.WriteAttributeString("protocolInfo", string.Format(
@ -520,7 +507,7 @@ namespace Emby.Dlna.Didl
var filename = url.Substring(0, url.IndexOf('?')); var filename = url.Substring(0, url.IndexOf('?'));
var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType) var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
? GetMimeType(filename) ? MimeTypes.GetMimeType(filename)
: mediaProfile.MimeType; : mediaProfile.MimeType;
var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container, var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container,
@ -545,17 +532,10 @@ namespace Emby.Dlna.Didl
} }
public static bool IsIdRoot(string id) public static bool IsIdRoot(string id)
{ => string.IsNullOrWhiteSpace(id)
if (string.IsNullOrWhiteSpace(id)
|| string.Equals(id, "0", StringComparison.OrdinalIgnoreCase) || string.Equals(id, "0", StringComparison.OrdinalIgnoreCase)
// Samsung sometimes uses 1 as root // Samsung sometimes uses 1 as root
|| string.Equals(id, "1", StringComparison.OrdinalIgnoreCase)) || string.Equals(id, "1", StringComparison.OrdinalIgnoreCase);
{
return true;
}
return false;
}
public void WriteFolderElement(XmlWriter writer, BaseItem folder, StubType? stubType, BaseItem context, int childCount, Filter filter, string requestedId = null) public void WriteFolderElement(XmlWriter writer, BaseItem folder, StubType? stubType, BaseItem context, int childCount, Filter filter, string requestedId = null)
{ {
@ -971,7 +951,7 @@ namespace Emby.Dlna.Didl
writer.WriteAttributeString("protocolInfo", string.Format( writer.WriteAttributeString("protocolInfo", string.Format(
"http-get:*:{0}:{1}", "http-get:*:{0}:{1}",
GetMimeType("file." + format), MimeTypes.GetMimeType("file." + format),
contentFeatures contentFeatures
)); ));

View File

@ -9,7 +9,7 @@ namespace Emby.Dlna.Profiles
{ {
Name = "Dish Hopper-Joey"; Name = "Dish Hopper-Joey";
ProtocolInfo = "http-get:*:video/mp2t:*,http-get:*:video/MP1S:*,http-get:*:video/mpeg2:*,http-get:*:video/mp4:*,http-get:*:video/x-matroska:*,http-get:*:audio/mpeg:*,http-get:*:audio/mpeg3:*,http-get:*:audio/mp3:*,http-get:*:audio/mp4:*,http-get:*:audio/mp4a-latm:*,http-get:*:image/jpeg:*"; ProtocolInfo = "http-get:*:video/mp2t:*,http-get:*:video/mpeg:*,http-get:*:video/MP1S:*,http-get:*:video/mpeg2:*,http-get:*:video/mp4:*,http-get:*:video/x-matroska:*,http-get:*:audio/mpeg:*,http-get:*:audio/mpeg3:*,http-get:*:audio/mp3:*,http-get:*:audio/mp4:*,http-get:*:audio/mp4a-latm:*,http-get:*:image/jpeg:*";
Identification = new DeviceIdentification Identification = new DeviceIdentification
{ {

View File

@ -28,7 +28,7 @@
<MaxStaticBitrate>140000000</MaxStaticBitrate> <MaxStaticBitrate>140000000</MaxStaticBitrate>
<MusicStreamingTranscodingBitrate>192000</MusicStreamingTranscodingBitrate> <MusicStreamingTranscodingBitrate>192000</MusicStreamingTranscodingBitrate>
<MaxStaticMusicBitrate xsi:nil="true" /> <MaxStaticMusicBitrate xsi:nil="true" />
<ProtocolInfo>http-get:*:video/mp2t:*,http-get:*:video/MP1S:*,http-get:*:video/mpeg2:*,http-get:*:video/mp4:*,http-get:*:video/x-matroska:*,http-get:*:audio/mpeg:*,http-get:*:audio/mpeg3:*,http-get:*:audio/mp3:*,http-get:*:audio/mp4:*,http-get:*:audio/mp4a-latm:*,http-get:*:image/jpeg:*</ProtocolInfo> <ProtocolInfo>http-get:*:video/mp2t:http-get:*:video/mpeg:*,http-get:*:video/MP1S:*,http-get:*:video/mpeg2:*,http-get:*:video/mp4:*,http-get:*:video/x-matroska:*,http-get:*:audio/mpeg:*,http-get:*:audio/mpeg3:*,http-get:*:audio/mp3:*,http-get:*:audio/mp4:*,http-get:*:audio/mp4a-latm:*,http-get:*:image/jpeg:*</ProtocolInfo>
<TimelineOffsetSeconds>0</TimelineOffsetSeconds> <TimelineOffsetSeconds>0</TimelineOffsetSeconds>
<RequiresPlainVideoItems>false</RequiresPlainVideoItems> <RequiresPlainVideoItems>false</RequiresPlainVideoItems>
<RequiresPlainFolders>false</RequiresPlainFolders> <RequiresPlainFolders>false</RequiresPlainFolders>

View File

@ -12,113 +12,133 @@ namespace MediaBrowser.Model.Net
public static class MimeTypes public static class MimeTypes
{ {
/// <summary> /// <summary>
/// Any extension in this list is considered a video file - can be added to at runtime for extensibility /// Any extension in this list is considered a video file
/// </summary> /// </summary>
private static readonly string[] VideoFileExtensions = new string[] private static readonly HashSet<string> _videoFileExtensions = new HashSet<string>(StringComparer.OrdinalIgnoreCase)
{
".mkv",
".m2t",
".m2ts",
".img",
".iso",
".mk3d",
".ts",
".rmvb",
".mov",
".avi",
".mpg",
".mpeg",
".wmv",
".mp4",
".divx",
".dvr-ms",
".wtv",
".ogm",
".ogv",
".asf",
".m4v",
".flv",
".f4v",
".3gp",
".webm",
".mts",
".m2v",
".rec"
};
private static Dictionary<string, string> GetVideoFileExtensionsDictionary()
{ {
var dict = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase); ".mkv",
".m2t",
foreach (string ext in VideoFileExtensions) ".m2ts",
{ ".img",
dict[ext] = ext; ".iso",
} ".mk3d",
".ts",
return dict; ".rmvb",
} ".mov",
".avi",
private static readonly Dictionary<string, string> VideoFileExtensionsDictionary = GetVideoFileExtensionsDictionary(); ".mpg",
".mpeg",
".wmv",
".mp4",
".divx",
".dvr-ms",
".wtv",
".ogm",
".ogv",
".asf",
".m4v",
".flv",
".f4v",
".3gp",
".webm",
".mts",
".m2v",
".rec"
};
// http://en.wikipedia.org/wiki/Internet_media_type // http://en.wikipedia.org/wiki/Internet_media_type
// Add more as needed // Add more as needed
private static readonly Dictionary<string, string> _mimeTypeLookup = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
private static Dictionary<string, string> GetMimeTypeLookup()
{ {
var dict = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase); // Type application
{ ".cbz", "application/x-cbz" },
{ ".cbr", "application/epub+zip" },
{ ".eot", "application/vnd.ms-fontobject" },
{ ".epub", "application/epub+zip" },
{ ".js", "application/x-javascript" },
{ ".json", "application/json" },
{ ".map", "application/x-javascript" },
{ ".pdf", "application/pdf" },
{ ".ttml", "application/ttml+xml" },
{ ".m3u8", "application/x-mpegURL" },
{ ".mobi", "application/x-mobipocket-ebook" },
{ ".xml", "application/xml" },
dict.Add(".jpg", "image/jpeg"); // Type image
dict.Add(".jpeg", "image/jpeg"); { ".jpg", "image/jpeg" },
dict.Add(".tbn", "image/jpeg"); { ".jpeg", "image/jpeg" },
dict.Add(".png", "image/png"); { ".tbn", "image/jpeg" },
dict.Add(".gif", "image/gif"); { ".png", "image/png" },
dict.Add(".tiff", "image/tiff"); { ".gif", "image/gif" },
dict.Add(".webp", "image/webp"); { ".tiff", "image/tiff" },
dict.Add(".ico", "image/vnd.microsoft.icon"); { ".webp", "image/webp" },
dict.Add(".mpg", "video/mpeg"); { ".ico", "image/vnd.microsoft.icon" },
dict.Add(".mpeg", "video/mpeg"); { ".svg", "image/svg+xml" },
dict.Add(".ogv", "video/ogg"); { ".svgz", "image/svg+xml" },
dict.Add(".mov", "video/quicktime");
dict.Add(".webm", "video/webm");
dict.Add(".mkv", "video/x-matroska");
dict.Add(".wmv", "video/x-ms-wmv");
dict.Add(".flv", "video/x-flv");
dict.Add(".avi", "video/x-msvideo");
dict.Add(".asf", "video/x-ms-asf");
dict.Add(".m4v", "video/x-m4v");
dict.Add(".m4s", "video/mp4");
dict.Add(".cbz", "application/x-cbz");
dict.Add(".cbr", "application/epub+zip");
dict.Add(".epub", "application/epub+zip");
dict.Add(".pdf", "application/pdf");
dict.Add(".mobi", "application/x-mobipocket-ebook");
dict.Add(".ass", "text/x-ssa"); // Type font
dict.Add(".ssa", "text/x-ssa"); { ".ttf" , "font/ttf" },
{ ".woff" , "font/woff" },
return dict; // Type text
} { ".ass", "text/x-ssa" },
{ ".ssa", "text/x-ssa" },
{ ".css", "text/css" },
{ ".csv", "text/csv" },
{ ".txt", "text/plain" },
{ ".vtt", "text/vtt" },
private static readonly Dictionary<string, string> MimeTypeLookup = GetMimeTypeLookup(); // Type video
{ ".mpg", "video/mpeg" },
{ ".ogv", "video/ogg" },
{ ".mov", "video/quicktime" },
{ ".webm", "video/webm" },
{ ".mkv", "video/x-matroska" },
{ ".wmv", "video/x-ms-wmv" },
{ ".flv", "video/x-flv" },
{ ".avi", "video/x-msvideo" },
{ ".asf", "video/x-ms-asf" },
{ ".m4v", "video/x-m4v" },
{ ".m4s", "video/mp4" },
{ ".3gp", "video/3gpp" },
{ ".3g2", "video/3gpp2" },
{ ".mpd", "video/vnd.mpeg.dash.mpd" },
{ ".ts", "video/mp2t" },
private static readonly Dictionary<string, string> ExtensionLookup = CreateExtensionLookup(); // Type audio
{ ".mp3", "audio/mpeg" },
{ ".m4a", "audio/mp4" },
{ ".aac", "audio/mp4" },
{ ".webma", "audio/webm" },
{ ".wav", "audio/wav" },
{ ".wma", "audio/x-ms-wma" },
{ ".ogg", "audio/ogg" },
{ ".oga", "audio/ogg" },
{ ".opus", "audio/ogg" },
{ ".ac3", "audio/ac3" },
{ ".dsf", "audio/dsf" },
{ ".m4b", "audio/m4b" },
{ ".xsp", "audio/xsp" },
{ ".dsp", "audio/dsp" },
};
private static readonly Dictionary<string, string> _extensionLookup = CreateExtensionLookup();
private static Dictionary<string, string> CreateExtensionLookup() private static Dictionary<string, string> CreateExtensionLookup()
{ {
var dict = MimeTypeLookup var dict = _mimeTypeLookup
.GroupBy(i => i.Value) .GroupBy(i => i.Value)
.ToDictionary(x => x.Key, x => x.First().Key, StringComparer.OrdinalIgnoreCase); .ToDictionary(x => x.Key, x => x.First().Key, StringComparer.OrdinalIgnoreCase);
dict["image/jpg"] = ".jpg"; dict["image/jpg"] = ".jpg";
dict["image/x-png"] = ".png"; dict["image/x-png"] = ".png";
dict["audio/x-aac"] = ".aac";
return dict; return dict;
} }
public static string GetMimeType(string path) public static string GetMimeType(string path) => GetMimeType(path, true);
{
return GetMimeType(path, true);
}
/// <summary> /// <summary>
/// Gets the type of the MIME. /// Gets the type of the MIME.
@ -130,138 +150,31 @@ namespace MediaBrowser.Model.Net
throw new ArgumentNullException(nameof(path)); throw new ArgumentNullException(nameof(path));
} }
var ext = Path.GetExtension(path) ?? string.Empty; var ext = Path.GetExtension(path);
if (MimeTypeLookup.TryGetValue(ext, out string result)) if (_mimeTypeLookup.TryGetValue(ext, out string result))
{ {
return result; return result;
} }
// Type video
if (StringHelper.EqualsIgnoreCase(ext, ".3gp"))
{
return "video/3gpp";
}
if (StringHelper.EqualsIgnoreCase(ext, ".3g2"))
{
return "video/3gpp2";
}
if (StringHelper.EqualsIgnoreCase(ext, ".ts"))
{
return "video/mp2t";
}
if (StringHelper.EqualsIgnoreCase(ext, ".mpd"))
{
return "video/vnd.mpeg.dash.mpd";
}
// Catch-all for all video types that don't require specific mime types // Catch-all for all video types that don't require specific mime types
if (VideoFileExtensionsDictionary.ContainsKey(ext)) if (_videoFileExtensions.Contains(ext))
{ {
return "video/" + ext.TrimStart('.').ToLowerInvariant(); return "video/" + ext.Substring(1);
} }
// Type text // Type text
if (StringHelper.EqualsIgnoreCase(ext, ".css")) if (StringHelper.EqualsIgnoreCase(ext, ".html")
{ || StringHelper.EqualsIgnoreCase(ext, ".htm"))
return "text/css";
}
if (StringHelper.EqualsIgnoreCase(ext, ".csv"))
{
return "text/csv";
}
if (StringHelper.EqualsIgnoreCase(ext, ".html"))
{ {
return "text/html; charset=UTF-8"; return "text/html; charset=UTF-8";
} }
if (StringHelper.EqualsIgnoreCase(ext, ".htm"))
{ if (StringHelper.EqualsIgnoreCase(ext, ".log")
return "text/html; charset=UTF-8"; || StringHelper.EqualsIgnoreCase(ext, ".srt"))
}
if (StringHelper.EqualsIgnoreCase(ext, ".txt"))
{ {
return "text/plain"; return "text/plain";
} }
if (StringHelper.EqualsIgnoreCase(ext, ".log"))
{
return "text/plain";
}
if (StringHelper.EqualsIgnoreCase(ext, ".xml"))
{
return "application/xml";
}
// Type audio
if (StringHelper.EqualsIgnoreCase(ext, ".mp3"))
{
return "audio/mpeg";
}
if (StringHelper.EqualsIgnoreCase(ext, ".m4a"))
{
return "audio/mp4";
}
if (StringHelper.EqualsIgnoreCase(ext, ".aac"))
{
return "audio/mp4";
}
if (StringHelper.EqualsIgnoreCase(ext, ".webma"))
{
return "audio/webm";
}
if (StringHelper.EqualsIgnoreCase(ext, ".wav"))
{
return "audio/wav";
}
if (StringHelper.EqualsIgnoreCase(ext, ".wma"))
{
return "audio/x-ms-wma";
}
if (StringHelper.EqualsIgnoreCase(ext, ".flac"))
{
return "audio/flac";
}
if (StringHelper.EqualsIgnoreCase(ext, ".aac"))
{
return "audio/x-aac";
}
if (StringHelper.EqualsIgnoreCase(ext, ".ogg"))
{
return "audio/ogg";
}
if (StringHelper.EqualsIgnoreCase(ext, ".oga"))
{
return "audio/ogg";
}
if (StringHelper.EqualsIgnoreCase(ext, ".opus"))
{
return "audio/ogg";
}
if (StringHelper.EqualsIgnoreCase(ext, ".ac3"))
{
return "audio/ac3";
}
if (StringHelper.EqualsIgnoreCase(ext, ".dsf"))
{
return "audio/dsf";
}
if (StringHelper.EqualsIgnoreCase(ext, ".m4b"))
{
return "audio/m4b";
}
if (StringHelper.EqualsIgnoreCase(ext, ".xsp"))
{
return "audio/xsp";
}
if (StringHelper.EqualsIgnoreCase(ext, ".dsp"))
{
return "audio/dsp";
}
// Playlists
if (StringHelper.EqualsIgnoreCase(ext, ".m3u8"))
{
return "application/x-mpegURL";
}
// Misc // Misc
if (StringHelper.EqualsIgnoreCase(ext, ".dll")) if (StringHelper.EqualsIgnoreCase(ext, ".dll"))
@ -269,63 +182,7 @@ namespace MediaBrowser.Model.Net
return "application/octet-stream"; return "application/octet-stream";
} }
// Web return enableStreamDefault ? "application/octet-stream" : null;
if (StringHelper.EqualsIgnoreCase(ext, ".js"))
{
return "application/x-javascript";
}
if (StringHelper.EqualsIgnoreCase(ext, ".json"))
{
return "application/json";
}
if (StringHelper.EqualsIgnoreCase(ext, ".map"))
{
return "application/x-javascript";
}
if (StringHelper.EqualsIgnoreCase(ext, ".woff"))
{
return "font/woff";
}
if (StringHelper.EqualsIgnoreCase(ext, ".ttf"))
{
return "font/ttf";
}
if (StringHelper.EqualsIgnoreCase(ext, ".eot"))
{
return "application/vnd.ms-fontobject";
}
if (StringHelper.EqualsIgnoreCase(ext, ".svg"))
{
return "image/svg+xml";
}
if (StringHelper.EqualsIgnoreCase(ext, ".svgz"))
{
return "image/svg+xml";
}
if (StringHelper.EqualsIgnoreCase(ext, ".srt"))
{
return "text/plain";
}
if (StringHelper.EqualsIgnoreCase(ext, ".vtt"))
{
return "text/vtt";
}
if (StringHelper.EqualsIgnoreCase(ext, ".ttml"))
{
return "application/ttml+xml";
}
if (enableStreamDefault)
{
return "application/octet-stream";
}
return null;
} }
public static string ToExtension(string mimeType) public static string ToExtension(string mimeType)
@ -338,10 +195,11 @@ namespace MediaBrowser.Model.Net
// handle text/html; charset=UTF-8 // handle text/html; charset=UTF-8
mimeType = mimeType.Split(';')[0]; mimeType = mimeType.Split(';')[0];
if (ExtensionLookup.TryGetValue(mimeType, out string result)) if (_extensionLookup.TryGetValue(mimeType, out string result))
{ {
return result; return result;
} }
return null; return null;
} }
} }