Refactored MIME handling to be static typed

This commit is contained in:
Thorsten Sommer 2026-01-05 15:56:10 +01:00
parent 495b6bab14
commit b781a5ab07
Signed by: tsommer
GPG Key ID: 371BBA77A02C0108
15 changed files with 450 additions and 27 deletions

View File

@ -1,3 +1,4 @@
using AIStudio.Tools.MIME;
using AIStudio.Tools.PluginSystem;
namespace AIStudio.Chat;
@ -6,7 +7,7 @@ public static class IImageSourceExtensions
{
private static string TB(string fallbackEN) => I18N.I.T(fallbackEN, typeof(IImageSourceExtensions).Namespace, nameof(IImageSourceExtensions));
public static string DetermineMimeType(this IImageSource image)
public static MIMEType DetermineMimeType(this IImageSource image)
{
switch (image.SourceType)
{
@ -18,13 +19,11 @@ public static class IImageSourceExtensions
{
var mimeEnd = base64Data.IndexOf(';');
if (mimeEnd > 5)
{
return base64Data[5..mimeEnd];
}
return Builder.FromTextRepresentation(base64Data[5..mimeEnd]);
}
// Fallback:
return "application/octet-stream";
return Builder.Create().UseApplication().UseSubtype(ApplicationSubtype.OCTET_STREAM).Build();
}
case ContentImageSource.URL:
@ -32,38 +31,36 @@ public static class IImageSourceExtensions
// Try to detect the mime type from the URL extension:
var uri = new Uri(image.Source);
var extension = Path.GetExtension(uri.AbsolutePath).ToLowerInvariant();
return extension switch
{
".png" => "image/png",
".jpg" or ".jpeg" => "image/jpeg",
".gif" => "image/gif",
".bmp" => "image/bmp",
".webp" => "image/webp",
_ => "application/octet-stream"
};
return DeriveMIMETypeFromExtension(extension);
}
case ContentImageSource.LOCAL_PATH:
{
var extension = Path.GetExtension(image.Source).ToLowerInvariant();
return extension switch
{
".png" => "image/png",
".jpg" or ".jpeg" => "image/jpeg",
".gif" => "image/gif",
".bmp" => "image/bmp",
".webp" => "image/webp",
_ => "application/octet-stream"
};
return DeriveMIMETypeFromExtension(extension);
}
default:
return "application/octet-stream";
return Builder.Create().UseApplication().UseSubtype(ApplicationSubtype.OCTET_STREAM).Build();
}
}
private static MIMEType DeriveMIMETypeFromExtension(string extension)
{
var imageBuilder = Builder.Create().UseImage();
return extension switch
{
".png" => imageBuilder.UseSubtype(ImageSubtype.PNG).Build(),
".jpg" or ".jpeg" => imageBuilder.UseSubtype(ImageSubtype.JPEG).Build(),
".gif" => imageBuilder.UseSubtype(ImageSubtype.GIF).Build(),
".webp" => imageBuilder.UseSubtype(ImageSubtype.WEBP).Build(),
".tiff" or ".tif" => imageBuilder.UseSubtype(ImageSubtype.TIFF).Build(),
".heic" or ".heif" => imageBuilder.UseSubtype(ImageSubtype.HEIC).Build(),
_ => Builder.Create().UseApplication().UseSubtype(ApplicationSubtype.OCTET_STREAM).Build()
};
}
/// <summary>
/// Read the image content as a base64 string.
/// </summary>

View File

@ -0,0 +1,63 @@
namespace AIStudio.Tools.MIME;
public class ApplicationBuilder : Builder, ISubtype
{
private ApplicationBuilder()
{
}
private ApplicationSubtype subtype;
public ApplicationBuilder UseSubtype(string subType)
{
this.subtype = subType.ToLowerInvariant() switch
{
"vnd.ms-excel" => ApplicationSubtype.EXCEL_OLD,
"vnd.ms-word" => ApplicationSubtype.WORD_OLD,
"vnd.ms-powerpoint" => ApplicationSubtype.POWERPOINT_OLD,
"vnd.openxmlformats-officedocument.spreadsheetml.sheet" => ApplicationSubtype.EXCEL,
"vnd.openxmlformats-officedocument.wordprocessingml.document" => ApplicationSubtype.WORD,
"vnd.openxmlformats-officedocument.presentationml.presentation" => ApplicationSubtype.POWERPOINT,
"octet-stream" => ApplicationSubtype.OCTET_STREAM,
"json" => ApplicationSubtype.JSON,
"xml" => ApplicationSubtype.XML,
"pdf" => ApplicationSubtype.PDF,
"zip" => ApplicationSubtype.ZIP,
"x-www-form-urlencoded" => ApplicationSubtype.X_WWW_FORM_URLENCODED,
_ => throw new ArgumentOutOfRangeException(nameof(subType), "Unsupported MIME application subtype.")
};
return this;
}
public ApplicationBuilder UseSubtype(ApplicationSubtype subType)
{
this.subtype = subType;
return this;
}
#region Implementation of IMIMESubtype
public MIMEType Build() => new()
{
Type = this,
TextRepresentation = this.subtype switch
{
ApplicationSubtype.EXCEL_OLD => $"{this.baseType}/vnd.ms-excel".ToLowerInvariant(),
ApplicationSubtype.WORD_OLD => $"{this.baseType}/vnd.ms-word".ToLowerInvariant(),
ApplicationSubtype.POWERPOINT_OLD => $"{this.baseType}/vnd.ms-powerpoint".ToLowerInvariant(),
ApplicationSubtype.EXCEL => $"{this.baseType}/vnd.openxmlformats-officedocument.spreadsheetml.sheet".ToLowerInvariant(),
ApplicationSubtype.WORD => $"{this.baseType}/vnd.openxmlformats-officedocument.wordprocessingml.document".ToLowerInvariant(),
ApplicationSubtype.POWERPOINT => $"{this.baseType}/vnd.openxmlformats-officedocument.presentationml.presentation".ToLowerInvariant(),
_ => $"{this.baseType}/{this.subtype}".ToLowerInvariant()
}
};
#endregion
}

View File

@ -0,0 +1,21 @@
namespace AIStudio.Tools.MIME;
public enum ApplicationSubtype
{
OCTET_STREAM,
JSON,
XML,
PDF,
ZIP,
X_WWW_FORM_URLENCODED,
WORD_OLD,
WORD,
EXCEL_OLD,
EXCEL,
POWERPOINT_OLD,
POWERPOINT,
}

View File

@ -0,0 +1,47 @@
namespace AIStudio.Tools.MIME;
public class AudioBuilder : Builder, ISubtype
{
private AudioBuilder()
{
}
private AudioSubtype subtype;
public AudioBuilder UseSubtype(string subType)
{
this.subtype = subType.ToLowerInvariant() switch
{
"mpeg" => AudioSubtype.MPEG,
"wav" => AudioSubtype.WAV,
"ogg" => AudioSubtype.OGG,
"aac" => AudioSubtype.AAC,
"flac" => AudioSubtype.FLAC,
"webm" => AudioSubtype.WEBM,
"mp4" => AudioSubtype.MP4,
"mp3" => AudioSubtype.MP3,
"m4a" => AudioSubtype.M4A,
"aiff" => AudioSubtype.AIFF,
_ => throw new ArgumentException("Unsupported MIME audio subtype.", nameof(subType))
};
return this;
}
public AudioBuilder UseSubtype(AudioSubtype subType)
{
this.subtype = subType;
return this;
}
#region Implementation of IMIMESubtype
public MIMEType Build() => new()
{
Type = this,
TextRepresentation = $"{this.baseType}/{this.subtype}".ToLowerInvariant()
};
#endregion
}

View File

@ -0,0 +1,16 @@
namespace AIStudio.Tools.MIME;
public enum AudioSubtype
{
WAV,
MP3,
OGG,
AAC,
FLAC,
// ReSharper disable once InconsistentNaming
M4A,
MPEG,
MP4,
WEBM,
AIFF
}

View File

@ -0,0 +1,10 @@
namespace AIStudio.Tools.MIME;
public enum BaseType
{
APPLICATION,
AUDIO,
IMAGE,
VIDEO,
TEXT,
}

View File

@ -0,0 +1,80 @@
namespace AIStudio.Tools.MIME;
public class Builder
{
protected Builder()
{
}
protected BaseType baseType;
public static Builder Create() => new();
public static MIMEType FromTextRepresentation(string textRepresentation)
{
var parts = textRepresentation.Split('/');
if (parts.Length != 2)
throw new ArgumentException("Invalid MIME type format.", nameof(textRepresentation));
var baseType = parts[0].ToLowerInvariant();
var subType = parts[1].ToLowerInvariant();
var builder = Create();
switch (baseType)
{
case "application":
var appBuilder = builder.UseApplication();
return appBuilder.UseSubtype(subType).Build();
case "text":
var textBuilder = builder.UseText();
return textBuilder.UseSubtype(subType).Build();
case "audio":
var audioBuilder = builder.UseAudio();
return audioBuilder.UseSubtype(subType).Build();
case "image":
var imageBuilder = builder.UseImage();
return imageBuilder.UseSubtype(subType).Build();
case "video":
var videoBuilder = builder.UseVideo();
return videoBuilder.UseSubtype(subType).Build();
default:
throw new ArgumentException("Unsupported base type.", nameof(textRepresentation));
}
}
public ApplicationBuilder UseApplication()
{
this.baseType = BaseType.APPLICATION;
return (ApplicationBuilder)this;
}
public TextBuilder UseText()
{
this.baseType = BaseType.TEXT;
return (TextBuilder)this;
}
public AudioBuilder UseAudio()
{
this.baseType = BaseType.AUDIO;
return (AudioBuilder)this;
}
public ImageBuilder UseImage()
{
this.baseType = BaseType.IMAGE;
return (ImageBuilder)this;
}
public VideoBuilder UseVideo()
{
this.baseType = BaseType.VIDEO;
return (VideoBuilder)this;
}
}

View File

@ -0,0 +1,6 @@
namespace AIStudio.Tools.MIME;
public interface ISubtype
{
public MIMEType Build();
}

View File

@ -0,0 +1,44 @@
namespace AIStudio.Tools.MIME;
public class ImageBuilder : Builder, ISubtype
{
private ImageBuilder()
{
}
private ImageSubtype subtype;
public ImageBuilder UseSubtype(string subType)
{
this.subtype = subType.ToLowerInvariant() switch
{
"jpeg" or "jpg" => ImageSubtype.JPEG,
"png" => ImageSubtype.PNG,
"gif" => ImageSubtype.GIF,
"webp" => ImageSubtype.WEBP,
"tiff" or "tif" => ImageSubtype.TIFF,
"svg+xml" or "svg" => ImageSubtype.SVG,
"heic" => ImageSubtype.HEIC,
_ => throw new ArgumentException("Unsupported MIME image subtype.", nameof(subType))
};
return this;
}
public ImageBuilder UseSubtype(ImageSubtype subType)
{
this.subtype = subType;
return this;
}
#region Implementation of IMIMESubtype
public MIMEType Build() => new()
{
Type = this,
TextRepresentation = $"{this.baseType}/{this.subtype}".ToLowerInvariant()
};
#endregion
}

View File

@ -0,0 +1,12 @@
namespace AIStudio.Tools.MIME;
public enum ImageSubtype
{
JPEG,
PNG,
GIF,
TIFF,
WEBP,
SVG,
HEIC,
}

View File

@ -0,0 +1,16 @@
namespace AIStudio.Tools.MIME;
public record MIMEType
{
public required ISubtype Type { get; init; }
public required string TextRepresentation { get; init; }
#region Overrides of Object
public override string ToString() => this.TextRepresentation;
#endregion
public static implicit operator string(MIMEType mimeType) => mimeType.TextRepresentation;
}

View File

@ -0,0 +1,45 @@
namespace AIStudio.Tools.MIME;
public class TextBuilder : Builder, ISubtype
{
private TextBuilder()
{
}
private TextSubtype subtype;
public TextBuilder UseSubtype(string subType)
{
this.subtype = subType.ToLowerInvariant() switch
{
"plain" => TextSubtype.PLAIN,
"html" => TextSubtype.HTML,
"css" => TextSubtype.CSS,
"csv" => TextSubtype.CSV,
"javascript" => TextSubtype.JAVASCRIPT,
"xml" => TextSubtype.XML,
"markdown" => TextSubtype.MARKDOWN,
"json" => TextSubtype.JSON,
_ => throw new ArgumentException("Unsupported MIME text subtype.", nameof(subType))
};
return this;
}
public TextBuilder UseSubtype(TextSubtype subType)
{
this.subtype = subType;
return this;
}
#region Implementation of IMIMESubtype
public MIMEType Build() => new()
{
Type = this,
TextRepresentation = $"{this.baseType}/{this.subtype}".ToLowerInvariant()
};
#endregion
}

View File

@ -0,0 +1,13 @@
namespace AIStudio.Tools.MIME;
public enum TextSubtype
{
PLAIN,
HTML,
CSS,
CSV,
JAVASCRIPT,
XML,
JSON,
MARKDOWN,
}

View File

@ -0,0 +1,42 @@
namespace AIStudio.Tools.MIME;
public class VideoBuilder : Builder, ISubtype
{
private VideoBuilder()
{
}
private VideoSubtype subtype;
public VideoBuilder UseSubtype(string subType)
{
this.subtype = subType.ToLowerInvariant() switch
{
"mp4" => VideoSubtype.MP4,
"webm" => VideoSubtype.WEBM,
"avi" => VideoSubtype.AVI,
"mov" => VideoSubtype.MOV,
"mkv" => VideoSubtype.MKV,
_ => throw new ArgumentException("Unsupported MIME video subtype.", nameof(subType))
};
return this;
}
public VideoBuilder UseSubtype(VideoSubtype subType)
{
this.subtype = subType;
return this;
}
#region Implementation of IMIMESubtype
public MIMEType Build() => new()
{
Type = this,
TextRepresentation = $"{this.baseType}/{this.subtype}".ToLowerInvariant()
};
#endregion
}

View File

@ -0,0 +1,11 @@
namespace AIStudio.Tools.MIME;
public enum VideoSubtype
{
MP4,
AVI,
MOV,
MKV,
WEBM,
MPEG,
}