Remove Webserver
Will be re-added as submodule
This commit is contained in:
@@ -1,90 +0,0 @@
|
|||||||
using Microsoft.Extensions.Hosting;
|
|
||||||
using RazorLight;
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Dynamic;
|
|
||||||
using System.Net;
|
|
||||||
using System.Text;
|
|
||||||
using System.Text.RegularExpressions;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using WebServer.Models;
|
|
||||||
|
|
||||||
namespace WebServer {
|
|
||||||
public abstract class AreaBase { // Formally `HttpEndpointHandler`
|
|
||||||
// Provides razor pages, resources and relevent db tables and entities
|
|
||||||
public readonly HttpServer Server;
|
|
||||||
public readonly RazorLightEngine RazorEngine;
|
|
||||||
public readonly string Hostname;
|
|
||||||
public readonly string AreaPath;
|
|
||||||
public readonly AreaBase? ParentArea;
|
|
||||||
|
|
||||||
public AreaBase(HttpServer server, string host) : this(server, host, string.Empty) { }
|
|
||||||
|
|
||||||
public AreaBase(HttpServer server, string host, string areaPath) {
|
|
||||||
if (server == null) throw new ArgumentNullException(nameof(server));
|
|
||||||
if (string.IsNullOrEmpty(host)) throw new ArgumentNullException(nameof(host));
|
|
||||||
Server = server;
|
|
||||||
RazorEngine = server.GetOrCreateRazorEngine(host);
|
|
||||||
Hostname = host;
|
|
||||||
AreaPath = areaPath ?? string.Empty;
|
|
||||||
}
|
|
||||||
|
|
||||||
public AreaBase(AreaBase parentArea, string relativeAreaPath) { // So you can build on top of other areas
|
|
||||||
ParentArea = parentArea;
|
|
||||||
Server = parentArea.Server;
|
|
||||||
RazorEngine = parentArea.RazorEngine;
|
|
||||||
Hostname = parentArea.Hostname;
|
|
||||||
AreaPath = $"{parentArea.AreaPath}/{relativeAreaPath.Trim(' ', '/', '\\')}";
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Uri BuildUri(HttpListenerRequest request, string path) {
|
|
||||||
string uriString = $"{(request.IsSecureConnection ? "https" : "http")}://{request.UserHostName}/";
|
|
||||||
if (string.IsNullOrEmpty(path))
|
|
||||||
return new Uri(uriString);
|
|
||||||
path = path.Replace('\\', '/');
|
|
||||||
if (path[0] == '/')
|
|
||||||
path = path.Substring(1);
|
|
||||||
uriString += path;
|
|
||||||
return new Uri(uriString);
|
|
||||||
}
|
|
||||||
|
|
||||||
#region Razor Stuff
|
|
||||||
public async Task<HttpResponse> GetGenericStatusPageAsync(StatusPageModel model, ExpandoObject? viewBag = null) {
|
|
||||||
return await Server.GetGenericStatusPageAsync(model, Hostname, viewBag);
|
|
||||||
}
|
|
||||||
public async Task<HttpResponse> GetGenericStatusPageAsync(HttpStatusCode statusCode, ExpandoObject? viewBag = null)
|
|
||||||
=> await GetGenericStatusPageAsync(new StatusPageModel(statusCode), viewBag);
|
|
||||||
|
|
||||||
/// <summary>Renders a View with a T model</summary>
|
|
||||||
/// <typeparam name="T">Type of the model</typeparam>
|
|
||||||
/// <param name="viewPath">The view path relative to the AreaName, keep in mind that prepending a '/' (or '\') will go to root of the top-most area</param>
|
|
||||||
/// <param name="model">The model supplied to the view</param>
|
|
||||||
/// <returns>A string of the compiled view</returns>
|
|
||||||
public async Task<string> RenderAsync<T>(string viewPath, T model, ExpandoObject viewBag = null) { // Formally Try/GetTemplate
|
|
||||||
viewPath = viewPath.Trim().Replace('\\', '/');
|
|
||||||
string root = viewPath[0] == '/' ? "" : AreaPath;
|
|
||||||
return await RazorEngine.CompileRenderAsync<T>($"{root}{viewPath}", model, viewBag);
|
|
||||||
}
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Event Binds
|
|
||||||
public Regex GetRegex(string regex) => new Regex(regex, RegexOptions.IgnoreCase | RegexOptions.Singleline | RegexOptions.CultureInvariant);
|
|
||||||
|
|
||||||
public bool TryAddEventCallback(Regex regex, HttpServer.Callback callback)
|
|
||||||
=> Server.TryAddEventCallback(Hostname, regex, callback);
|
|
||||||
|
|
||||||
public void AddEventCallback(Regex regex, HttpServer.Callback callback) {
|
|
||||||
if (!TryAddEventCallback(regex, callback))
|
|
||||||
throw new Exception($"Error in Area '{AreaPath}'! Failed to bind event callback to `{callback.Method.Name}`");
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryAddEventCallback(string regexPattern, HttpServer.Callback callback)
|
|
||||||
=> TryAddEventCallback(GetRegex(regexPattern), callback);
|
|
||||||
|
|
||||||
public void AddEventCallback(string regexPattern, HttpServer.Callback callback) {
|
|
||||||
if (!TryAddEventCallback(regexPattern, callback))
|
|
||||||
throw new Exception($"Error in Area '{AreaPath}'! Failed to bind event callback to `{callback.Method.Name}`");
|
|
||||||
}
|
|
||||||
#endregion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Dynamic;
|
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
namespace WebServer.Extensions {
|
|
||||||
public static class ExpandoObjectExt {
|
|
||||||
public static object GetPropertyOrDefault(this ExpandoObject expando, string propertyName, object defaultValue) {
|
|
||||||
if (expando is IDictionary<string, object> dict) {
|
|
||||||
if (!dict.TryGetValue(propertyName, out var result))
|
|
||||||
dict.Add(propertyName, result = defaultValue);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
return defaultValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static bool TryGetProperty(this ExpandoObject expando, string propertyName, out object value) {
|
|
||||||
value = null;
|
|
||||||
return expando is IDictionary<string, object> dict
|
|
||||||
&& dict.TryGetValue(propertyName, out value);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
namespace WebServer {
|
|
||||||
public class HttpConfiguration {
|
|
||||||
public bool AutoStart = false;
|
|
||||||
public ushort Port = 80;
|
|
||||||
public string DefaultDomain = "localhost"; // The domain that the server will fallback on if the
|
|
||||||
public bool ShowExceptionOnErrorPages = true; // On InternalServerError(500), should it show the exception?
|
|
||||||
public ushort MaxConcurrentRequests = 100;
|
|
||||||
public List<string> UriFillers = new List<string>() { // Todo: Needs revision,
|
|
||||||
".html",
|
|
||||||
".htm",
|
|
||||||
".txt",
|
|
||||||
"./index.html",
|
|
||||||
"./index.htm",
|
|
||||||
"./index.txt",
|
|
||||||
"./default.webp",
|
|
||||||
"./default.png",
|
|
||||||
"../default.webp",
|
|
||||||
"../default.png"
|
|
||||||
};
|
|
||||||
public Dictionary<string, string?> GenericHeaders = new Dictionary<string, string?>() {
|
|
||||||
{ "x-content-type-options: nosniff", null },
|
|
||||||
{ "x-xss-protection:1; mode=block", null },
|
|
||||||
{ "x-frame-options:DENY", null }
|
|
||||||
};
|
|
||||||
public ushort ResponseTimeout = 10;
|
|
||||||
public uint MaxCacheAge = 604800;
|
|
||||||
public bool DebugMode = false; // If true, it bypasses cache and also shows exceptions on error page (where applicable)
|
|
||||||
|
|
||||||
public virtual HttpServer CreateServer(IHttpLogger? logger = null) => new HttpServer(this, logger);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Net;
|
|
||||||
using System.Text;
|
|
||||||
using WebServer.Utils;
|
|
||||||
|
|
||||||
namespace WebServer {
|
|
||||||
public class HttpResponse {
|
|
||||||
public HttpStatusCode StatusCode = HttpStatusCode.NoContent;
|
|
||||||
public bool IsSuccessStatusCode => (int)StatusCode >= 200 && (int)StatusCode <= 299;
|
|
||||||
public string ContentType = MimeTypeMap.GetMimeType(".txt");
|
|
||||||
public byte[] Content = Array.Empty<byte>();
|
|
||||||
public string ContentString {
|
|
||||||
get => Encoding.UTF8.GetString(Content);
|
|
||||||
set => Content = Encoding.UTF8.GetBytes(value);
|
|
||||||
}
|
|
||||||
public Dictionary<string, string?> Headers = new Dictionary<string, string?>();
|
|
||||||
|
|
||||||
public HttpResponse() { }
|
|
||||||
|
|
||||||
public HttpResponse(HttpStatusCode statusCode, byte[] content, string contentType = "text/plain") {
|
|
||||||
StatusCode = statusCode;
|
|
||||||
Content = content;
|
|
||||||
ContentType = contentType;
|
|
||||||
}
|
|
||||||
|
|
||||||
public HttpResponse(HttpStatusCode statusCode, string content, string contentType = "text/plain") {
|
|
||||||
StatusCode = statusCode;
|
|
||||||
ContentString = content;
|
|
||||||
ContentType = contentType;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,371 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Dynamic;
|
|
||||||
using System.IO;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Net;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text;
|
|
||||||
using System.Text.RegularExpressions;
|
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using RazorLight;
|
|
||||||
using RazorLight.Razor;
|
|
||||||
using WebServer.Models;
|
|
||||||
using WebServer.Utils;
|
|
||||||
|
|
||||||
namespace WebServer {
|
|
||||||
public class HttpServer {
|
|
||||||
public readonly HttpConfiguration Config;
|
|
||||||
public readonly DirectoryInfo ViewsDirectory;
|
|
||||||
public ushort ActivePort { get; protected set; }
|
|
||||||
public HttpListener HttpListener { get; protected set; }
|
|
||||||
public IHttpLogger? Logger;
|
|
||||||
public delegate Task<HttpResponse?> Callback(HttpListenerContext context, CachedResponse? cache);
|
|
||||||
|
|
||||||
Dictionary<string, Dictionary<Regex, Callback>> HttpCallbacks = new Dictionary<string, Dictionary<Regex, Callback>>();
|
|
||||||
public List<AreaBase> RegisteredAreas = new List<AreaBase>();
|
|
||||||
Dictionary<string, RazorLightEngine> RazorEngines = new Dictionary<string, RazorLightEngine>();
|
|
||||||
public readonly RazorLightEngine DefaultRazorEngine;
|
|
||||||
/* Instead of three directories (from previous render engine)
|
|
||||||
* - Public/PublicTemplates (was public static stuff, also allowed views)
|
|
||||||
* - Static (was used for builds)
|
|
||||||
* - Views/PrivateTemplates (was used for private views)
|
|
||||||
* There will be only two:
|
|
||||||
* - Static - For builds, there should also be a build process for this
|
|
||||||
* - Views - Site source, including public/private views (views are able to be processed)
|
|
||||||
*/
|
|
||||||
|
|
||||||
public HttpServer(HttpConfiguration config, IHttpLogger? logger = null) {
|
|
||||||
Config = config;
|
|
||||||
Logger = logger;
|
|
||||||
ViewsDirectory = new DirectoryInfo(Path.Combine(Environment.CurrentDirectory, "Views"));
|
|
||||||
if (!ViewsDirectory.Exists)
|
|
||||||
ViewsDirectory.Create();
|
|
||||||
if (!string.IsNullOrEmpty(config.DefaultDomain))
|
|
||||||
ViewsDirectory.CreateSubdirectory(config.DefaultDomain);
|
|
||||||
if (config.AutoStart)
|
|
||||||
_ = StartAsync();
|
|
||||||
DefaultRazorEngine = GetOrCreateRazorEngine(config.DefaultDomain);
|
|
||||||
}
|
|
||||||
|
|
||||||
public RazorLightEngine GetOrCreateRazorEngine(string hostname) {
|
|
||||||
hostname = hostname.Trim(' ', '/', '\\');
|
|
||||||
if (!RazorEngines.TryGetValue(hostname, out RazorLightEngine engine)) {
|
|
||||||
RazorEngines.Add(hostname, engine = new RazorLightEngineBuilder()
|
|
||||||
.UseOptions(new RazorLightOptions() { // TODO: make this part of the config
|
|
||||||
EnableDebugMode = true
|
|
||||||
})
|
|
||||||
.UseProject(new FileSystemRazorProject(Path.Combine(ViewsDirectory.FullName, hostname), ".cshtml"))
|
|
||||||
.UseMemoryCachingProvider()
|
|
||||||
.Build()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return engine;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task StartAsync() {
|
|
||||||
if (HttpListener?.IsListening ?? false)
|
|
||||||
await StopAsync();
|
|
||||||
HttpListener = new HttpListener() {
|
|
||||||
IgnoreWriteExceptions = true // Used to crash the server, don't think it is needed anymore
|
|
||||||
};
|
|
||||||
HttpListener.Prefixes.Add($"http://+:{Config.Port}/");
|
|
||||||
|
|
||||||
HttpListener.Start();
|
|
||||||
ActivePort = Config.Port;
|
|
||||||
Logger?.Log($"HTTP Service started on port '{ActivePort}'");
|
|
||||||
|
|
||||||
while (HttpListener.IsListening)
|
|
||||||
await ListenAsync((ListenToken = new CancellationTokenSource()).Token);
|
|
||||||
}
|
|
||||||
|
|
||||||
CancellationTokenSource ListenToken;
|
|
||||||
public int TasksCount => Tasks.Count;
|
|
||||||
HashSet<Task> Tasks = new HashSet<Task>();
|
|
||||||
async Task ListenAsync(CancellationToken token) {
|
|
||||||
Tasks = new HashSet<Task>(128);
|
|
||||||
for (int i = 0; i < 64; i++) // Create 64 tasks
|
|
||||||
Tasks.Add(HttpListener.GetContextAsync());
|
|
||||||
Logger?.Log($"Listening with {Tasks.Count} worker(s)");
|
|
||||||
while (!token.IsCancellationRequested) {
|
|
||||||
Task t = await Task.WhenAny(Tasks);
|
|
||||||
Tasks.Remove(t);
|
|
||||||
|
|
||||||
if (t is Task<HttpListenerContext> context) {
|
|
||||||
if (Tasks.Count < Config.MaxConcurrentRequests)
|
|
||||||
Tasks.Add(HttpListener.GetContextAsync());
|
|
||||||
Tasks.Add(ProcessRequestAsync(context.Result, token)); // Should I really be adding this to tasks?
|
|
||||||
}
|
|
||||||
//ProcessRequestAsync triggers this:
|
|
||||||
// else Logger?.Log($"Got an unexpected task of type '{t.GetType().FullName}'");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task StopAsync() {
|
|
||||||
HttpListener.Stop();
|
|
||||||
ListenToken.Cancel();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task ProcessRequestAsync(HttpListenerContext context, CancellationToken token) {
|
|
||||||
List<MethodBase> methodsUsed = new List<MethodBase>() { MethodBase.GetCurrentMethod() };
|
|
||||||
try {
|
|
||||||
// Add generic headers
|
|
||||||
foreach (var kvp in Config.GenericHeaders) {
|
|
||||||
if (kvp.Value is null)
|
|
||||||
context.Response.Headers.Add(kvp.Key);
|
|
||||||
else context.Response.Headers.Add(kvp.Key, kvp.Value);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get domain and path
|
|
||||||
string host = (context.Request.Url?.Host ?? Config.DefaultDomain).Trim('/', ' ').ToLowerInvariant(),
|
|
||||||
callbackKey = FormatCallbackKey(context.Request.Url?.LocalPath ?? string.Empty),
|
|
||||||
path = callbackKey.Any() ? $"/{callbackKey}" : "";
|
|
||||||
string cacheName = $"{host}{path}";
|
|
||||||
CachedResponse? cache = CachedResponse.Get(this, cacheName);
|
|
||||||
HttpResponse? response = null;
|
|
||||||
if (cache != null) {
|
|
||||||
if (!cache.NeedsUpdate)
|
|
||||||
response = cache;
|
|
||||||
else if (cache.UpdateMethod != null)
|
|
||||||
response = await cache.UpdateMethod(context, cache);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create reponse timeout logic, this will return a string to the client but an exception on the server
|
|
||||||
CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
|
|
||||||
Task cancelationTask = Task.Delay(Timeout.Infinite, cancellationTokenSource.Token);
|
|
||||||
if (Config.ResponseTimeout > 0)
|
|
||||||
cancellationTokenSource.CancelAfter(1000 * Config.ResponseTimeout);
|
|
||||||
|
|
||||||
StatusPageModel? statusPageModel = null;
|
|
||||||
try {
|
|
||||||
#region Event Callbacks
|
|
||||||
var hostnames = new[] { host ?? Config.DefaultDomain, Config.DefaultDomain }.Distinct();
|
|
||||||
|
|
||||||
if (response == null) {
|
|
||||||
var domainCallbackMatches = hostnames.Where(name => HttpCallbacks.ContainsKey(name));
|
|
||||||
var regexCallbacks = domainCallbackMatches.SelectMany(name => HttpCallbacks[name]);
|
|
||||||
/*var regexCallbacks = new[] { host, DefaultDomain }.Distinct()
|
|
||||||
.Where(domain => HttpCallbacks.ContainsKey(domain))
|
|
||||||
.SelectMany(domain => HttpCallbacks[domain]);*/
|
|
||||||
foreach (var kvp in regexCallbacks) {
|
|
||||||
bool match = kvp.Key.IsMatch(path);
|
|
||||||
if (!match) continue;
|
|
||||||
methodsUsed.Add(kvp.Value.Method);
|
|
||||||
var callback = kvp.Value(context, cache);
|
|
||||||
var completedTask = await Task.WhenAny(callback, cancelationTask);
|
|
||||||
// Check that the callback didn't throw any errors (hence it didn't complete)
|
|
||||||
if (callback.IsFaulted) throw callback.Exception;
|
|
||||||
// So no faults, check if the completed task is the callback
|
|
||||||
if (completedTask != callback)
|
|
||||||
throw new ResponseTimedOutException(cacheName, methodsUsed, cancellationTokenSource.Token);
|
|
||||||
response = callback.Result;
|
|
||||||
|
|
||||||
if (response != null) break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Try get Razor File
|
|
||||||
string fileName = Path.GetFileNameWithoutExtension(path);
|
|
||||||
bool isPrivateView = fileName.Length > 0 && fileName.StartsWith('_');
|
|
||||||
if (response == null && !isPrivateView) {
|
|
||||||
foreach (var hostname in hostnames) {
|
|
||||||
// If no razor engine exists, continue to next hostname
|
|
||||||
if (!RazorEngines.TryGetValue(hostname, out var razorEngine))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
string cwd = Path.Combine(ViewsDirectory.FullName, hostname).Replace('\\', '/');
|
|
||||||
string[] razorPaths = new[] { $"{path}", $"{path}.cshtml", $"{path}/index.cshtml" };
|
|
||||||
foreach (var razorPath in razorPaths) {
|
|
||||||
string fullpath = cwd + razorPath;
|
|
||||||
if (Path.GetExtension(fullpath).ToLower() != ".cshtml") continue;
|
|
||||||
bool fileExists = File.Exists(fullpath);
|
|
||||||
if (!fileExists) continue;
|
|
||||||
|
|
||||||
cache ??= new CachedResponse(this, null);
|
|
||||||
cache.StatusCode = HttpStatusCode.OK;
|
|
||||||
cache.ContentString = await razorEngine.CompileRenderAsync<object?>(razorPath, null);
|
|
||||||
cache.ContentType = "text/html";
|
|
||||||
response = cache;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Try Static File
|
|
||||||
// Finally get a static file
|
|
||||||
if (response == null) {
|
|
||||||
Callback getStaticFile = GetStaticFile;
|
|
||||||
methodsUsed.Add(getStaticFile.Method);
|
|
||||||
response = await getStaticFile(context, cache);
|
|
||||||
}
|
|
||||||
#endregion
|
|
||||||
}
|
|
||||||
catch (AggregateException ex) { // Also catches OperationCanceledException and ResponseTimedOutException
|
|
||||||
statusPageModel = new StatusPageModel(HttpStatusCode.ServiceUnavailable) {
|
|
||||||
Exception = ex
|
|
||||||
};
|
|
||||||
}
|
|
||||||
catch (Exception ex) {
|
|
||||||
statusPageModel = new StatusPageModel(HttpStatusCode.InternalServerError) {
|
|
||||||
Exception = ex
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
response ??= await GetGenericStatusPageAsync(statusPageModel ?? new StatusPageModel(HttpStatusCode.NotFound));
|
|
||||||
|
|
||||||
#region Ship Response
|
|
||||||
context.Response.StatusCode = (int)response.StatusCode;
|
|
||||||
if (response.StatusCode == HttpStatusCode.Redirect) {
|
|
||||||
context.Response.Redirect(response.ContentString);
|
|
||||||
}
|
|
||||||
foreach (var kvp in response.Headers) {
|
|
||||||
if (kvp.Value is null)
|
|
||||||
context.Response.Headers.Add(kvp.Key);
|
|
||||||
else context.Response.Headers.Add(kvp.Key, kvp.Value);
|
|
||||||
}
|
|
||||||
bool omitBody = new[] { "HEAD", "PUT", "DELETE" }.Contains(context.Request.HttpMethod.ToUpper()) ||
|
|
||||||
(100 <= (int)response.StatusCode && (int)response.StatusCode <= 199) ||
|
|
||||||
response.StatusCode == HttpStatusCode.NoContent ||
|
|
||||||
response.StatusCode == HttpStatusCode.NotModified;
|
|
||||||
if (!omitBody) {
|
|
||||||
context.Response.Headers["Content-Type"] = response.ContentType;
|
|
||||||
context.Response.ContentEncoding = Encoding.UTF8;
|
|
||||||
context.Response.ContentLength64 = response.Content.Length;
|
|
||||||
await context.Response.OutputStream.WriteAsync(response.Content, 0, response.Content.Length);
|
|
||||||
}
|
|
||||||
Logger?.LogRequest(context, response, methodsUsed);
|
|
||||||
context.Response.Close();
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
// And finally, if the response allows cache (by being a cachedResponse)
|
|
||||||
// override the path that it is on, so it can be reused
|
|
||||||
if (response is CachedResponse cachedResponse) {
|
|
||||||
cachedResponse.Path = $"{host}{path}"; // Rewrite the path, so it can be located when a new request is received
|
|
||||||
if (Config.DebugMode) cachedResponse.RaiseUpdateFlag();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex) {
|
|
||||||
Logger?.LogError(context, methodsUsed, ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static string FormatCallbackKey(string key)
|
|
||||||
=> string.IsNullOrEmpty(key) ? string.Empty
|
|
||||||
: key.ToLower().Replace('\\', '/').Replace("//", "/").Trim(' ', '/');
|
|
||||||
|
|
||||||
public async Task<HttpResponse> GetStaticFile(HttpListenerContext context, CachedResponse? cache) => await GetStaticFile(context.Request.Url?.Host, context.Request.Url?.LocalPath, cache);
|
|
||||||
|
|
||||||
public async Task<HttpResponse> GetStaticFile(string? targetDomain, string? localPath, CachedResponse? cache) {
|
|
||||||
if (!cache?.NeedsUpdate ?? false) return cache;
|
|
||||||
string? fileName = Path.GetFileName(localPath);
|
|
||||||
if (fileName != null && fileName.StartsWith('_') && fileName.EndsWith(".cshtml")) // Is the file a private cshtml file?
|
|
||||||
localPath = localPath?.Substring(0, localPath.Length - fileName.Length);
|
|
||||||
DirectoryInfo directory = ViewsDirectory; // Might be changed later
|
|
||||||
// Works on windows, but on linux, the domain folder will need to be lowercase
|
|
||||||
targetDomain = targetDomain?.ToLower() ?? Config.DefaultDomain;
|
|
||||||
string basePath = Path.Combine(directory.FullName, targetDomain);
|
|
||||||
bool usingFallbackDomain = !Directory.Exists(basePath);
|
|
||||||
if (usingFallbackDomain) { // Only fallback to default if domain folder doesn't exist
|
|
||||||
targetDomain = Config.DefaultDomain;
|
|
||||||
basePath = Path.Combine(directory.FullName, Config.DefaultDomain);
|
|
||||||
}
|
|
||||||
string resourceIdentifier = FormatCallbackKey(localPath ?? string.Empty);
|
|
||||||
CachedResponse resource = cache ?? new CachedResponse(this, null);
|
|
||||||
string filePath = Path.Combine(basePath, resourceIdentifier);
|
|
||||||
if (File.Exists(filePath)) {
|
|
||||||
resource.StatusCode = HttpStatusCode.OK;
|
|
||||||
resource.ContentType = MimeTypeMap.GetMimeType(Path.GetExtension(filePath).ToLower());
|
|
||||||
resource.Content = File.ReadAllBytes(filePath);
|
|
||||||
resource.Headers["cache-control"] = Config.DebugMode ? "no-store, no-cache, must-revalidate"
|
|
||||||
: "max-age=360000, s-max-age=900, stale-while-revalidate=120, stale-if-error=86400";
|
|
||||||
resource.ClearFlag();
|
|
||||||
return resource;
|
|
||||||
}
|
|
||||||
string? hitPath = Config.UriFillers.Select(filler => filePath + filler)
|
|
||||||
.FirstOrDefault(path => path.Contains(basePath) && File.Exists(path));
|
|
||||||
if (hitPath != null) {
|
|
||||||
resource.StatusCode = HttpStatusCode.OK;
|
|
||||||
resource.ContentType = MimeTypeMap.GetMimeType(Path.GetExtension(hitPath).ToLower());
|
|
||||||
resource.Content = File.ReadAllBytes(hitPath);
|
|
||||||
resource.Headers["cache-control"] = Config.DebugMode ? "no-store, no-cache, must-revalidate"
|
|
||||||
: "max-age=360000, s-max-age=900, stale-while-revalidate=120, stale-if-error=86400";
|
|
||||||
resource.ClearFlag();
|
|
||||||
return resource;
|
|
||||||
}
|
|
||||||
return await GetGenericStatusPageAsync(new StatusPageModel(Directory.Exists(filePath) ? HttpStatusCode.Forbidden : HttpStatusCode.NotFound), host: targetDomain);
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<HttpResponse> GetGenericStatusPageAsync(StatusPageModel pageModel, string? host = null, ExpandoObject? viewBag = null) {
|
|
||||||
//try { pageModel.Exception ??= throw new Exception("test"); }
|
|
||||||
//catch (Exception e) { pageModel.Exception = e; }
|
|
||||||
var potentialHosts = new string[] { host ?? Config.DefaultDomain, Config.DefaultDomain }.Distinct()
|
|
||||||
.Select(h => h.Trim(' ', '/', '\\'));
|
|
||||||
const string viewName = "_StatusPage.cshtml";
|
|
||||||
foreach (string hostname in potentialHosts) {
|
|
||||||
if (!File.Exists(Path.Combine(ViewsDirectory.FullName, hostname, viewName))) continue;
|
|
||||||
try {
|
|
||||||
return new HttpResponse(
|
|
||||||
pageModel.StatusCode,
|
|
||||||
await GetOrCreateRazorEngine(hostname).CompileRenderAsync(viewName, pageModel, viewBag),
|
|
||||||
MimeTypeMap.GetMimeType(".cshtml")
|
|
||||||
);
|
|
||||||
} catch (Exception ex) {
|
|
||||||
pageModel.Exception = pageModel.Exception != null
|
|
||||||
? new AggregateException(pageModel.Exception, ex)
|
|
||||||
: ex;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If no template was found, fallback to plain text
|
|
||||||
var bckResponse = new HttpResponse() {
|
|
||||||
StatusCode = pageModel.StatusCode,
|
|
||||||
ContentType = MimeTypeMap.GetMimeType(".txt"),
|
|
||||||
ContentString = $"{(int)pageModel.StatusCode} {pageModel.Header} - {pageModel.Details}\n{pageModel.Exception?.ToString() ?? ""}"
|
|
||||||
};
|
|
||||||
return bckResponse;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool ContainsEventCallback(Task<HttpResponse?> callback) {
|
|
||||||
if (callback == null) return false;
|
|
||||||
return HttpCallbacks.Any(domainKvp => domainKvp.Value.Values.Any(v => v.Equals(callback)));
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryAddEventCallback(string host, Regex regex, Callback callback) {
|
|
||||||
if (string.IsNullOrEmpty(host) || regex == null || callback == null) return false;
|
|
||||||
host = FormatCallbackKey(host);
|
|
||||||
if (!HttpCallbacks.TryGetValue(host, out var domainCallbacks))
|
|
||||||
HttpCallbacks.Add(host, domainCallbacks = new Dictionary<Regex, Callback>());
|
|
||||||
domainCallbacks[regex] = callback;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryRemoveEventCallback(Callback method) {
|
|
||||||
if (method == null) return false;
|
|
||||||
|
|
||||||
ushort removeCount = 0;
|
|
||||||
foreach (var callbackDict in HttpCallbacks.Values) {
|
|
||||||
foreach (var kvp in callbackDict.Where(kvp => kvp.Value == method)) {
|
|
||||||
callbackDict.Remove(kvp.Key);
|
|
||||||
removeCount++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return removeCount > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryRegisterArea<T>(Func<T>? areaInitilizer, out T area) where T : AreaBase {
|
|
||||||
area = areaInitilizer?.Invoke() ?? default;
|
|
||||||
if (area == null) return false;
|
|
||||||
RegisteredAreas.Add(area);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void RegisterArea<T>(Func<T>? areaInitializer, out T area) where T : AreaBase {
|
|
||||||
if (!TryRegisterArea(areaInitializer, out area))
|
|
||||||
throw new Exception($"Failed to bind {typeof(T).FullName}! Is your initializer returning a valid area?");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Net;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
namespace WebServer {
|
|
||||||
public interface IHttpLogger {
|
|
||||||
void Log(string message);
|
|
||||||
void LogRequest(HttpListenerContext context, HttpResponse response, List<MethodBase> methodsUsed);
|
|
||||||
void LogError(HttpListenerContext context, List<MethodBase> methodsUsed, Exception exception);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Dynamic;
|
|
||||||
using System.Net;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
|
|
||||||
namespace WebServer.Models {
|
|
||||||
public class StatusPageModel {
|
|
||||||
public HttpStatusCode StatusCode;
|
|
||||||
public ushort StatusCodeAsUShort => (ushort)StatusCode;
|
|
||||||
public string Header;
|
|
||||||
public string Details;
|
|
||||||
public Exception? Exception;
|
|
||||||
|
|
||||||
static readonly Dictionary<HttpStatusCode, (string header, string details)> GenericStatusContent = new Dictionary<HttpStatusCode, (string header, string details)>() {
|
|
||||||
{ HttpStatusCode.OK, ("OK", "The request succeeded") },
|
|
||||||
{ HttpStatusCode.BadRequest, ("Bad Request", "The server cannot or will not process the request due to something that is perceived to be a client error") },
|
|
||||||
{ HttpStatusCode.Unauthorized, ("Unauthorized", "Authorization Required") },
|
|
||||||
{ HttpStatusCode.Forbidden, ("Forbidden", "Access to this resource has been revoked") },
|
|
||||||
{ HttpStatusCode.NotFound, ("Not Found", "Object or Resource not Found") },
|
|
||||||
{ HttpStatusCode.PreconditionFailed, ("Precondition Failed!", "The server has indicated preconditions which the client does not meet") },
|
|
||||||
{ HttpStatusCode.InternalServerError, ("Internal Server Error", "The server has encountered a situation it does not know how to handle") },
|
|
||||||
{ HttpStatusCode.NotImplemented, ("Not Implemented", "he server has encountered a situation it does not know how to handle.") },
|
|
||||||
{ HttpStatusCode.ServiceUnavailable, ("Service Unavailable", "The server is not ready to handle the request") }
|
|
||||||
};
|
|
||||||
|
|
||||||
public StatusPageModel(string title, string subtitle) : this(HttpStatusCode.OK, title, subtitle) {}
|
|
||||||
|
|
||||||
public StatusPageModel(HttpStatusCode statusCode, string? title = null, string? subtitle = null) {
|
|
||||||
StatusCode = statusCode;
|
|
||||||
if (!GenericStatusContent.TryGetValue(statusCode, out var tuple))
|
|
||||||
tuple = (statusCode.ToString(), "Great! Something happened, not sure what though");
|
|
||||||
Header = title ?? tuple.header;
|
|
||||||
Details = subtitle ?? tuple.details;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Net;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace WebServer {
|
|
||||||
public class ResponseTimedOutException : OperationCanceledException {
|
|
||||||
public string FullPath;
|
|
||||||
public List<MethodBase> MethodsUsed;
|
|
||||||
public ResponseTimedOutException(string fullPath, List<MethodBase> methodsUsed, CancellationToken token)
|
|
||||||
: base("The request took too long to fulfil", token) {
|
|
||||||
FullPath = fullPath;
|
|
||||||
MethodsUsed = methodsUsed;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
return base.ToString()
|
|
||||||
+ $"\nFullPath: '{FullPath}';"
|
|
||||||
+ $"\nMethods Used: [\n{string.Join(",\n", MethodsUsed.Select(m => $" '{m.ReflectedType.FullName}.{m.Name}'"))} \n];";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,48 +0,0 @@
|
|||||||
using System.Collections.Generic;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Linq;
|
|
||||||
|
|
||||||
namespace WebServer.Utils {
|
|
||||||
public class CachedResponse : HttpResponse {
|
|
||||||
#region Static Methods, etc
|
|
||||||
public static bool BypassCache = false;
|
|
||||||
public static List<CachedResponse> Instances { get; private set; } = new List<CachedResponse>();
|
|
||||||
public static ushort RaiseAllUpdateFlags() {
|
|
||||||
ushort flagsRaised = 0;
|
|
||||||
foreach (CachedResponse resource in Instances) {
|
|
||||||
if (resource.NeedsUpdate) continue;
|
|
||||||
flagsRaised++;
|
|
||||||
resource.RaiseUpdateFlag();
|
|
||||||
//Logger.LogDebug($"Raised Update Flag for '{resource.Name}'");
|
|
||||||
}
|
|
||||||
return flagsRaised;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static CachedResponse? Get(HttpServer server, string path) {
|
|
||||||
path = path.Trim();
|
|
||||||
return Instances.FirstOrDefault(x => x.Server == server && x.Path == path);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static bool TryGet(HttpServer server, string path, out CachedResponse? resource)
|
|
||||||
=> (resource = Get(server, path)) != null;
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
public readonly HttpServer Server;
|
|
||||||
public string Path;
|
|
||||||
bool UpdateFlagRaised;
|
|
||||||
Stopwatch TimeSinceLastUpdate;
|
|
||||||
public HttpServer.Callback? UpdateMethod; // If null, HttpServer will attempt to find one or GetStaticFile
|
|
||||||
|
|
||||||
public bool NeedsUpdate => UpdateFlagRaised || BypassCache || Server.Config.MaxCacheAge < TimeSinceLastUpdate.Elapsed.TotalSeconds;
|
|
||||||
|
|
||||||
public CachedResponse(HttpServer parentServer, HttpServer.Callback? updateMethod) {
|
|
||||||
Instances.Add(this);
|
|
||||||
Server = parentServer;
|
|
||||||
TimeSinceLastUpdate = Stopwatch.StartNew();
|
|
||||||
UpdateMethod = updateMethod;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void RaiseUpdateFlag() => UpdateFlagRaised = true;
|
|
||||||
public void ClearFlag() => UpdateFlagRaised = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,829 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
|
|
||||||
namespace WebServer.Utils
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Class MimeTypeMap
|
|
||||||
/// Copied from: https://github.com/samuelneff/MimeTypeMap/blob/master/MimeTypeMap.cs; Commit Id: ec28e2e; Commit Date: May 25, 2022;
|
|
||||||
/// Referer: https://stackoverflow.com/a/3393525
|
|
||||||
/// </summary>
|
|
||||||
public static class MimeTypeMap
|
|
||||||
{
|
|
||||||
private const string Dot = ".";
|
|
||||||
private const string QuestionMark = "?";
|
|
||||||
private const string DefaultMimeType = "text/plain";
|
|
||||||
private static readonly Lazy<IDictionary<string, string>> _mappings = new Lazy<IDictionary<string, string>>(BuildMappings);
|
|
||||||
public static string HtmlDocument = GetMimeType(".html");
|
|
||||||
|
|
||||||
private static IDictionary<string, string> BuildMappings()
|
|
||||||
{
|
|
||||||
var mappings = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase) {
|
|
||||||
|
|
||||||
#region Big freaking list of mime types
|
|
||||||
|
|
||||||
// maps both ways,
|
|
||||||
// extension -> mime type
|
|
||||||
// and
|
|
||||||
// mime type -> extension
|
|
||||||
//
|
|
||||||
// any mime types on left side not pre-loaded on right side, are added automatically
|
|
||||||
// some mime types can map to multiple extensions, so to get a deterministic mapping,
|
|
||||||
// add those to the dictionary specifically
|
|
||||||
//
|
|
||||||
// combination of values from Windows 7 Registry and
|
|
||||||
// from C:\Windows\System32\inetsrv\config\applicationHost.config
|
|
||||||
// some added, including .7z and .dat
|
|
||||||
//
|
|
||||||
// Some added based on http://www.iana.org/assignments/media-types/media-types.xhtml
|
|
||||||
// which lists mime types, but not extensions
|
|
||||||
//
|
|
||||||
{".323", "text/h323"},
|
|
||||||
{".3g2", "video/3gpp2"},
|
|
||||||
{".3gp", "video/3gpp"},
|
|
||||||
{".3gp2", "video/3gpp2"},
|
|
||||||
{".3gpp", "video/3gpp"},
|
|
||||||
{".7z", "application/x-7z-compressed"},
|
|
||||||
{".aa", "audio/audible"},
|
|
||||||
{".AAC", "audio/aac"},
|
|
||||||
{".aaf", "application/octet-stream"},
|
|
||||||
{".aax", "audio/vnd.audible.aax"},
|
|
||||||
{".ac3", "audio/ac3"},
|
|
||||||
{".aca", "application/octet-stream"},
|
|
||||||
{".accda", "application/msaccess.addin"},
|
|
||||||
{".accdb", "application/msaccess"},
|
|
||||||
{".accdc", "application/msaccess.cab"},
|
|
||||||
{".accde", "application/msaccess"},
|
|
||||||
{".accdr", "application/msaccess.runtime"},
|
|
||||||
{".accdt", "application/msaccess"},
|
|
||||||
{".accdw", "application/msaccess.webapplication"},
|
|
||||||
{".accft", "application/msaccess.ftemplate"},
|
|
||||||
{".acx", "application/internet-property-stream"},
|
|
||||||
{".AddIn", "text/xml"},
|
|
||||||
{".ade", "application/msaccess"},
|
|
||||||
{".adobebridge", "application/x-bridge-url"},
|
|
||||||
{".adp", "application/msaccess"},
|
|
||||||
{".ADT", "audio/vnd.dlna.adts"},
|
|
||||||
{".ADTS", "audio/aac"},
|
|
||||||
{".afm", "application/octet-stream"},
|
|
||||||
{".ai", "application/postscript"},
|
|
||||||
{".aif", "audio/aiff"},
|
|
||||||
{".aifc", "audio/aiff"},
|
|
||||||
{".aiff", "audio/aiff"},
|
|
||||||
{".air", "application/vnd.adobe.air-application-installer-package+zip"},
|
|
||||||
{".amc", "application/mpeg"},
|
|
||||||
{".anx", "application/annodex"},
|
|
||||||
{".apk", "application/vnd.android.package-archive"},
|
|
||||||
{".apng", "image/apng"},
|
|
||||||
{".application", "application/x-ms-application"},
|
|
||||||
{".art", "image/x-jg"},
|
|
||||||
{".asa", "application/xml"},
|
|
||||||
{".asax", "application/xml"},
|
|
||||||
{".ascx", "application/xml"},
|
|
||||||
{".asd", "application/octet-stream"},
|
|
||||||
{".asf", "video/x-ms-asf"},
|
|
||||||
{".ashx", "application/xml"},
|
|
||||||
{".asi", "application/octet-stream"},
|
|
||||||
{".asm", "text/plain"},
|
|
||||||
{".asmx", "application/xml"},
|
|
||||||
{".aspx", "application/xml"},
|
|
||||||
{".asr", "video/x-ms-asf"},
|
|
||||||
{".asx", "video/x-ms-asf"},
|
|
||||||
{".atom", "application/atom+xml"},
|
|
||||||
{".au", "audio/basic"},
|
|
||||||
{".avci", "image/avci"},
|
|
||||||
{".avcs", "image/avcs"},
|
|
||||||
{".avi", "video/x-msvideo"},
|
|
||||||
{".avif", "image/avif"},
|
|
||||||
{".avifs", "image/avif-sequence"},
|
|
||||||
{".axa", "audio/annodex"},
|
|
||||||
{".axs", "application/olescript"},
|
|
||||||
{".axv", "video/annodex"},
|
|
||||||
{".bas", "text/plain"},
|
|
||||||
{".bcpio", "application/x-bcpio"},
|
|
||||||
{".bin", "application/octet-stream"},
|
|
||||||
{".bmp", "image/bmp"},
|
|
||||||
{".c", "text/plain"},
|
|
||||||
{".cab", "application/octet-stream"},
|
|
||||||
{".caf", "audio/x-caf"},
|
|
||||||
{".calx", "application/vnd.ms-office.calx"},
|
|
||||||
{".cat", "application/vnd.ms-pki.seccat"},
|
|
||||||
{".cc", "text/plain"},
|
|
||||||
{".cd", "text/plain"},
|
|
||||||
{".cdda", "audio/aiff"},
|
|
||||||
{".cdf", "application/x-cdf"},
|
|
||||||
{".cer", "application/x-x509-ca-cert"},
|
|
||||||
{".cfg", "text/plain"},
|
|
||||||
{".chm", "application/octet-stream"},
|
|
||||||
{".class", "application/x-java-applet"},
|
|
||||||
{".clp", "application/x-msclip"},
|
|
||||||
{".cmd", "text/plain"},
|
|
||||||
{".cmx", "image/x-cmx"},
|
|
||||||
{".cnf", "text/plain"},
|
|
||||||
{".cod", "image/cis-cod"},
|
|
||||||
{".config", "application/xml"},
|
|
||||||
{".contact", "text/x-ms-contact"},
|
|
||||||
{".coverage", "application/xml"},
|
|
||||||
{".cpio", "application/x-cpio"},
|
|
||||||
{".cpp", "text/plain"},
|
|
||||||
{".crd", "application/x-mscardfile"},
|
|
||||||
{".crl", "application/pkix-crl"},
|
|
||||||
{".crt", "application/x-x509-ca-cert"},
|
|
||||||
{".cs", "text/plain"},
|
|
||||||
{".csdproj", "text/plain"},
|
|
||||||
{".csh", "application/x-csh"},
|
|
||||||
{".csproj", "text/plain"},
|
|
||||||
{".css", "text/css"},
|
|
||||||
{".csv", "text/csv"},
|
|
||||||
{".cur", "application/octet-stream"},
|
|
||||||
{".czx", "application/x-czx"},
|
|
||||||
{".cxx", "text/plain"},
|
|
||||||
{".dat", "application/octet-stream"},
|
|
||||||
{".datasource", "application/xml"},
|
|
||||||
{".dbproj", "text/plain"},
|
|
||||||
{".dcr", "application/x-director"},
|
|
||||||
{".def", "text/plain"},
|
|
||||||
{".deploy", "application/octet-stream"},
|
|
||||||
{".der", "application/x-x509-ca-cert"},
|
|
||||||
{".dgml", "application/xml"},
|
|
||||||
{".dib", "image/bmp"},
|
|
||||||
{".dif", "video/x-dv"},
|
|
||||||
{".dir", "application/x-director"},
|
|
||||||
{".disco", "text/xml"},
|
|
||||||
{".divx", "video/divx"},
|
|
||||||
{".dll", "application/x-msdownload"},
|
|
||||||
{".dll.config", "text/xml"},
|
|
||||||
{".dlm", "text/dlm"},
|
|
||||||
{".doc", "application/msword"},
|
|
||||||
{".docm", "application/vnd.ms-word.document.macroEnabled.12"},
|
|
||||||
{".docx", "application/vnd.openxmlformats-officedocument.wordprocessingml.document"},
|
|
||||||
{".dot", "application/msword"},
|
|
||||||
{".dotm", "application/vnd.ms-word.template.macroEnabled.12"},
|
|
||||||
{".dotx", "application/vnd.openxmlformats-officedocument.wordprocessingml.template"},
|
|
||||||
{".dsp", "application/octet-stream"},
|
|
||||||
{".dsw", "text/plain"},
|
|
||||||
{".dtd", "text/xml"},
|
|
||||||
{".dtsConfig", "text/xml"},
|
|
||||||
{".dv", "video/x-dv"},
|
|
||||||
{".dvi", "application/x-dvi"},
|
|
||||||
{".dwf", "drawing/x-dwf"},
|
|
||||||
{".dwg", "application/acad"},
|
|
||||||
{".dwp", "application/octet-stream"},
|
|
||||||
{".dxf", "application/x-dxf"},
|
|
||||||
{".dxr", "application/x-director"},
|
|
||||||
{".eml", "message/rfc822"},
|
|
||||||
{".emf", "image/emf"},
|
|
||||||
{".emz", "application/octet-stream"},
|
|
||||||
{".eot", "application/vnd.ms-fontobject"},
|
|
||||||
{".eps", "application/postscript"},
|
|
||||||
{".es", "application/ecmascript"},
|
|
||||||
{".etl", "application/etl"},
|
|
||||||
{".etx", "text/x-setext"},
|
|
||||||
{".evy", "application/envoy"},
|
|
||||||
{".exe", "application/vnd.microsoft.portable-executable"},
|
|
||||||
{".exe.config", "text/xml"},
|
|
||||||
{".f4v", "video/mp4"},
|
|
||||||
{".fdf", "application/vnd.fdf"},
|
|
||||||
{".fif", "application/fractals"},
|
|
||||||
{".filters", "application/xml"},
|
|
||||||
{".fla", "application/octet-stream"},
|
|
||||||
{".flac", "audio/flac"},
|
|
||||||
{".flr", "x-world/x-vrml"},
|
|
||||||
{".flv", "video/x-flv"},
|
|
||||||
{".fsscript", "application/fsharp-script"},
|
|
||||||
{".fsx", "application/fsharp-script"},
|
|
||||||
{".generictest", "application/xml"},
|
|
||||||
{".geojson", "application/geo+json"},
|
|
||||||
{".gif", "image/gif"},
|
|
||||||
{".gpx", "application/gpx+xml"},
|
|
||||||
{".group", "text/x-ms-group"},
|
|
||||||
{".gsm", "audio/x-gsm"},
|
|
||||||
{".gtar", "application/x-gtar"},
|
|
||||||
{".gz", "application/x-gzip"},
|
|
||||||
{".h", "text/plain"},
|
|
||||||
{".hdf", "application/x-hdf"},
|
|
||||||
{".hdml", "text/x-hdml"},
|
|
||||||
{".heic", "image/heic"},
|
|
||||||
{".heics", "image/heic-sequence"},
|
|
||||||
{".heif", "image/heif"},
|
|
||||||
{".heifs", "image/heif-sequence"},
|
|
||||||
{".hhc", "application/x-oleobject"},
|
|
||||||
{".hhk", "application/octet-stream"},
|
|
||||||
{".hhp", "application/octet-stream"},
|
|
||||||
{".hlp", "application/winhlp"},
|
|
||||||
{".hpp", "text/plain"},
|
|
||||||
{".hqx", "application/mac-binhex40"},
|
|
||||||
{".hta", "application/hta"},
|
|
||||||
{".htc", "text/x-component"},
|
|
||||||
{".htm", "text/html"},
|
|
||||||
{".html", "text/html"},
|
|
||||||
{".cshtml", "text/html"},
|
|
||||||
{".htt", "text/webviewhtml"},
|
|
||||||
{".hxa", "application/xml"},
|
|
||||||
{".hxc", "application/xml"},
|
|
||||||
{".hxd", "application/octet-stream"},
|
|
||||||
{".hxe", "application/xml"},
|
|
||||||
{".hxf", "application/xml"},
|
|
||||||
{".hxh", "application/octet-stream"},
|
|
||||||
{".hxi", "application/octet-stream"},
|
|
||||||
{".hxk", "application/xml"},
|
|
||||||
{".hxq", "application/octet-stream"},
|
|
||||||
{".hxr", "application/octet-stream"},
|
|
||||||
{".hxs", "application/octet-stream"},
|
|
||||||
{".hxt", "text/html"},
|
|
||||||
{".hxv", "application/xml"},
|
|
||||||
{".hxw", "application/octet-stream"},
|
|
||||||
{".hxx", "text/plain"},
|
|
||||||
{".i", "text/plain"},
|
|
||||||
{".ical", "text/calendar"},
|
|
||||||
{".icalendar", "text/calendar"},
|
|
||||||
{".ico", "image/x-icon"},
|
|
||||||
{".ics", "text/calendar"},
|
|
||||||
{".idl", "text/plain"},
|
|
||||||
{".ief", "image/ief"},
|
|
||||||
{".ifb", "text/calendar"},
|
|
||||||
{".iii", "application/x-iphone"},
|
|
||||||
{".inc", "text/plain"},
|
|
||||||
{".inf", "application/octet-stream"},
|
|
||||||
{".ini", "text/plain"},
|
|
||||||
{".inl", "text/plain"},
|
|
||||||
{".ins", "application/x-internet-signup"},
|
|
||||||
{".ipa", "application/x-itunes-ipa"},
|
|
||||||
{".ipg", "application/x-itunes-ipg"},
|
|
||||||
{".ipproj", "text/plain"},
|
|
||||||
{".ipsw", "application/x-itunes-ipsw"},
|
|
||||||
{".iqy", "text/x-ms-iqy"},
|
|
||||||
{".isp", "application/x-internet-signup"},
|
|
||||||
{".isma", "application/octet-stream"},
|
|
||||||
{".ismv", "application/octet-stream"},
|
|
||||||
{".ite", "application/x-itunes-ite"},
|
|
||||||
{".itlp", "application/x-itunes-itlp"},
|
|
||||||
{".itms", "application/x-itunes-itms"},
|
|
||||||
{".itpc", "application/x-itunes-itpc"},
|
|
||||||
{".IVF", "video/x-ivf"},
|
|
||||||
{".jar", "application/java-archive"},
|
|
||||||
{".java", "application/octet-stream"},
|
|
||||||
{".jck", "application/liquidmotion"},
|
|
||||||
{".jcz", "application/liquidmotion"},
|
|
||||||
{".jfif", "image/pjpeg"},
|
|
||||||
{".jnlp", "application/x-java-jnlp-file"},
|
|
||||||
{".jpb", "application/octet-stream"},
|
|
||||||
{".jpe", "image/jpeg"},
|
|
||||||
{".jpeg", "image/jpeg"},
|
|
||||||
{".jpg", "image/jpeg"},
|
|
||||||
{".js", "application/javascript"},
|
|
||||||
{".json", "application/json"},
|
|
||||||
{".jsx", "text/jscript"},
|
|
||||||
{".jsxbin", "text/plain"},
|
|
||||||
{".key", "application/vnd.apple.keynote"},
|
|
||||||
{".latex", "application/x-latex"},
|
|
||||||
{".library-ms", "application/windows-library+xml"},
|
|
||||||
{".lit", "application/x-ms-reader"},
|
|
||||||
{".loadtest", "application/xml"},
|
|
||||||
{".lpk", "application/octet-stream"},
|
|
||||||
{".lsf", "video/x-la-asf"},
|
|
||||||
{".lst", "text/plain"},
|
|
||||||
{".lsx", "video/x-la-asf"},
|
|
||||||
{".lzh", "application/octet-stream"},
|
|
||||||
{".m13", "application/x-msmediaview"},
|
|
||||||
{".m14", "application/x-msmediaview"},
|
|
||||||
{".m1v", "video/mpeg"},
|
|
||||||
{".m2t", "video/vnd.dlna.mpeg-tts"},
|
|
||||||
{".m2ts", "video/vnd.dlna.mpeg-tts"},
|
|
||||||
{".m2v", "video/mpeg"},
|
|
||||||
{".m3u", "audio/x-mpegurl"},
|
|
||||||
{".m3u8", "audio/x-mpegurl"},
|
|
||||||
{".m4a", "audio/m4a"},
|
|
||||||
{".m4b", "audio/m4b"},
|
|
||||||
{".m4p", "audio/m4p"},
|
|
||||||
{".m4r", "audio/x-m4r"},
|
|
||||||
{".m4v", "video/x-m4v"},
|
|
||||||
{".mac", "image/x-macpaint"},
|
|
||||||
{".mak", "text/plain"},
|
|
||||||
{".man", "application/x-troff-man"},
|
|
||||||
{".manifest", "application/x-ms-manifest"},
|
|
||||||
{".map", "text/plain"},
|
|
||||||
{".master", "application/xml"},
|
|
||||||
{".mbox", "application/mbox"},
|
|
||||||
{".mda", "application/msaccess"},
|
|
||||||
{".mdb", "application/x-msaccess"},
|
|
||||||
{".mde", "application/msaccess"},
|
|
||||||
{".mdp", "application/octet-stream"},
|
|
||||||
{".me", "application/x-troff-me"},
|
|
||||||
{".mfp", "application/x-shockwave-flash"},
|
|
||||||
{".mht", "message/rfc822"},
|
|
||||||
{".mhtml", "message/rfc822"},
|
|
||||||
{".mid", "audio/mid"},
|
|
||||||
{".midi", "audio/mid"},
|
|
||||||
{".mix", "application/octet-stream"},
|
|
||||||
{".mk", "text/plain"},
|
|
||||||
{".mk3d", "video/x-matroska-3d"},
|
|
||||||
{".mka", "audio/x-matroska"},
|
|
||||||
{".mkv", "video/x-matroska"},
|
|
||||||
{".mmf", "application/x-smaf"},
|
|
||||||
{".mno", "text/xml"},
|
|
||||||
{".mny", "application/x-msmoney"},
|
|
||||||
{".mod", "video/mpeg"},
|
|
||||||
{".mov", "video/quicktime"},
|
|
||||||
{".movie", "video/x-sgi-movie"},
|
|
||||||
{".mp2", "video/mpeg"},
|
|
||||||
{".mp2v", "video/mpeg"},
|
|
||||||
{".mp3", "audio/mpeg"},
|
|
||||||
{".mp4", "video/mp4"},
|
|
||||||
{".mp4v", "video/mp4"},
|
|
||||||
{".mpa", "video/mpeg"},
|
|
||||||
{".mpe", "video/mpeg"},
|
|
||||||
{".mpeg", "video/mpeg"},
|
|
||||||
{".mpf", "application/vnd.ms-mediapackage"},
|
|
||||||
{".mpg", "video/mpeg"},
|
|
||||||
{".mpp", "application/vnd.ms-project"},
|
|
||||||
{".mpv2", "video/mpeg"},
|
|
||||||
{".mqv", "video/quicktime"},
|
|
||||||
{".ms", "application/x-troff-ms"},
|
|
||||||
{".msg", "application/vnd.ms-outlook"},
|
|
||||||
{".msi", "application/octet-stream"},
|
|
||||||
{".mso", "application/octet-stream"},
|
|
||||||
{".mts", "video/vnd.dlna.mpeg-tts"},
|
|
||||||
{".mtx", "application/xml"},
|
|
||||||
{".mvb", "application/x-msmediaview"},
|
|
||||||
{".mvc", "application/x-miva-compiled"},
|
|
||||||
{".mxf", "application/mxf"},
|
|
||||||
{".mxp", "application/x-mmxp"},
|
|
||||||
{".nc", "application/x-netcdf"},
|
|
||||||
{".nsc", "video/x-ms-asf"},
|
|
||||||
{".numbers", "application/vnd.apple.numbers"},
|
|
||||||
{".nws", "message/rfc822"},
|
|
||||||
{".ocx", "application/octet-stream"},
|
|
||||||
{".oda", "application/oda"},
|
|
||||||
{".odb", "application/vnd.oasis.opendocument.database"},
|
|
||||||
{".odc", "application/vnd.oasis.opendocument.chart"},
|
|
||||||
{".odf", "application/vnd.oasis.opendocument.formula"},
|
|
||||||
{".odg", "application/vnd.oasis.opendocument.graphics"},
|
|
||||||
{".odh", "text/plain"},
|
|
||||||
{".odi", "application/vnd.oasis.opendocument.image"},
|
|
||||||
{".odl", "text/plain"},
|
|
||||||
{".odm", "application/vnd.oasis.opendocument.text-master"},
|
|
||||||
{".odp", "application/vnd.oasis.opendocument.presentation"},
|
|
||||||
{".ods", "application/vnd.oasis.opendocument.spreadsheet"},
|
|
||||||
{".odt", "application/vnd.oasis.opendocument.text"},
|
|
||||||
{".oga", "audio/ogg"},
|
|
||||||
{".ogg", "audio/ogg"},
|
|
||||||
{".ogv", "video/ogg"},
|
|
||||||
{".ogx", "application/ogg"},
|
|
||||||
{".one", "application/onenote"},
|
|
||||||
{".onea", "application/onenote"},
|
|
||||||
{".onepkg", "application/onenote"},
|
|
||||||
{".onetmp", "application/onenote"},
|
|
||||||
{".onetoc", "application/onenote"},
|
|
||||||
{".onetoc2", "application/onenote"},
|
|
||||||
{".opus", "audio/ogg"},
|
|
||||||
{".orderedtest", "application/xml"},
|
|
||||||
{".osdx", "application/opensearchdescription+xml"},
|
|
||||||
{".otf", "application/font-sfnt"},
|
|
||||||
{".otg", "application/vnd.oasis.opendocument.graphics-template"},
|
|
||||||
{".oth", "application/vnd.oasis.opendocument.text-web"},
|
|
||||||
{".otp", "application/vnd.oasis.opendocument.presentation-template"},
|
|
||||||
{".ots", "application/vnd.oasis.opendocument.spreadsheet-template"},
|
|
||||||
{".ott", "application/vnd.oasis.opendocument.text-template"},
|
|
||||||
{".oxps", "application/oxps"},
|
|
||||||
{".oxt", "application/vnd.openofficeorg.extension"},
|
|
||||||
{".p10", "application/pkcs10"},
|
|
||||||
{".p12", "application/x-pkcs12"},
|
|
||||||
{".p7b", "application/x-pkcs7-certificates"},
|
|
||||||
{".p7c", "application/pkcs7-mime"},
|
|
||||||
{".p7m", "application/pkcs7-mime"},
|
|
||||||
{".p7r", "application/x-pkcs7-certreqresp"},
|
|
||||||
{".p7s", "application/pkcs7-signature"},
|
|
||||||
{".pages", "application/vnd.apple.pages"},
|
|
||||||
{".pbm", "image/x-portable-bitmap"},
|
|
||||||
{".pcast", "application/x-podcast"},
|
|
||||||
{".pct", "image/pict"},
|
|
||||||
{".pcx", "application/octet-stream"},
|
|
||||||
{".pcz", "application/octet-stream"},
|
|
||||||
{".pdf", "application/pdf"},
|
|
||||||
{".pfb", "application/octet-stream"},
|
|
||||||
{".pfm", "application/octet-stream"},
|
|
||||||
{".pfx", "application/x-pkcs12"},
|
|
||||||
{".pgm", "image/x-portable-graymap"},
|
|
||||||
{".pic", "image/pict"},
|
|
||||||
{".pict", "image/pict"},
|
|
||||||
{".pkgdef", "text/plain"},
|
|
||||||
{".pkgundef", "text/plain"},
|
|
||||||
{".pko", "application/vnd.ms-pki.pko"},
|
|
||||||
{".pls", "audio/scpls"},
|
|
||||||
{".pma", "application/x-perfmon"},
|
|
||||||
{".pmc", "application/x-perfmon"},
|
|
||||||
{".pml", "application/x-perfmon"},
|
|
||||||
{".pmr", "application/x-perfmon"},
|
|
||||||
{".pmw", "application/x-perfmon"},
|
|
||||||
{".png", "image/png"},
|
|
||||||
{".pnm", "image/x-portable-anymap"},
|
|
||||||
{".pnt", "image/x-macpaint"},
|
|
||||||
{".pntg", "image/x-macpaint"},
|
|
||||||
{".pnz", "image/png"},
|
|
||||||
{".pot", "application/vnd.ms-powerpoint"},
|
|
||||||
{".potm", "application/vnd.ms-powerpoint.template.macroEnabled.12"},
|
|
||||||
{".potx", "application/vnd.openxmlformats-officedocument.presentationml.template"},
|
|
||||||
{".ppa", "application/vnd.ms-powerpoint"},
|
|
||||||
{".ppam", "application/vnd.ms-powerpoint.addin.macroEnabled.12"},
|
|
||||||
{".ppm", "image/x-portable-pixmap"},
|
|
||||||
{".pps", "application/vnd.ms-powerpoint"},
|
|
||||||
{".ppsm", "application/vnd.ms-powerpoint.slideshow.macroEnabled.12"},
|
|
||||||
{".ppsx", "application/vnd.openxmlformats-officedocument.presentationml.slideshow"},
|
|
||||||
{".ppt", "application/vnd.ms-powerpoint"},
|
|
||||||
{".pptm", "application/vnd.ms-powerpoint.presentation.macroEnabled.12"},
|
|
||||||
{".pptx", "application/vnd.openxmlformats-officedocument.presentationml.presentation"},
|
|
||||||
{".prf", "application/pics-rules"},
|
|
||||||
{".prm", "application/octet-stream"},
|
|
||||||
{".prx", "application/octet-stream"},
|
|
||||||
{".ps", "application/postscript"},
|
|
||||||
{".psc1", "application/PowerShell"},
|
|
||||||
{".psd", "application/octet-stream"},
|
|
||||||
{".psess", "application/xml"},
|
|
||||||
{".psm", "application/octet-stream"},
|
|
||||||
{".psp", "application/octet-stream"},
|
|
||||||
{".pst", "application/vnd.ms-outlook"},
|
|
||||||
{".pub", "application/x-mspublisher"},
|
|
||||||
{".pwz", "application/vnd.ms-powerpoint"},
|
|
||||||
{".qht", "text/x-html-insertion"},
|
|
||||||
{".qhtm", "text/x-html-insertion"},
|
|
||||||
{".qt", "video/quicktime"},
|
|
||||||
{".qti", "image/x-quicktime"},
|
|
||||||
{".qtif", "image/x-quicktime"},
|
|
||||||
{".qtl", "application/x-quicktimeplayer"},
|
|
||||||
{".qxd", "application/octet-stream"},
|
|
||||||
{".ra", "audio/x-pn-realaudio"},
|
|
||||||
{".ram", "audio/x-pn-realaudio"},
|
|
||||||
{".rar", "application/x-rar-compressed"},
|
|
||||||
{".ras", "image/x-cmu-raster"},
|
|
||||||
{".rat", "application/rat-file"},
|
|
||||||
{".rc", "text/plain"},
|
|
||||||
{".rc2", "text/plain"},
|
|
||||||
{".rct", "text/plain"},
|
|
||||||
{".rdlc", "application/xml"},
|
|
||||||
{".reg", "text/plain"},
|
|
||||||
{".resx", "application/xml"},
|
|
||||||
{".rf", "image/vnd.rn-realflash"},
|
|
||||||
{".rgb", "image/x-rgb"},
|
|
||||||
{".rgs", "text/plain"},
|
|
||||||
{".rm", "application/vnd.rn-realmedia"},
|
|
||||||
{".rmi", "audio/mid"},
|
|
||||||
{".rmp", "application/vnd.rn-rn_music_package"},
|
|
||||||
{".rmvb", "application/vnd.rn-realmedia-vbr"},
|
|
||||||
{".roff", "application/x-troff"},
|
|
||||||
{".rpm", "audio/x-pn-realaudio-plugin"},
|
|
||||||
{".rqy", "text/x-ms-rqy"},
|
|
||||||
{".rtf", "application/rtf"},
|
|
||||||
{".rtx", "text/richtext"},
|
|
||||||
{".rvt", "application/octet-stream"},
|
|
||||||
{".ruleset", "application/xml"},
|
|
||||||
{".s", "text/plain"},
|
|
||||||
{".safariextz", "application/x-safari-safariextz"},
|
|
||||||
{".scd", "application/x-msschedule"},
|
|
||||||
{".scr", "text/plain"},
|
|
||||||
{".sct", "text/scriptlet"},
|
|
||||||
{".sd2", "audio/x-sd2"},
|
|
||||||
{".sdp", "application/sdp"},
|
|
||||||
{".sea", "application/octet-stream"},
|
|
||||||
{".searchConnector-ms", "application/windows-search-connector+xml"},
|
|
||||||
{".setpay", "application/set-payment-initiation"},
|
|
||||||
{".setreg", "application/set-registration-initiation"},
|
|
||||||
{".settings", "application/xml"},
|
|
||||||
{".sgimb", "application/x-sgimb"},
|
|
||||||
{".sgml", "text/sgml"},
|
|
||||||
{".sh", "application/x-sh"},
|
|
||||||
{".shar", "application/x-shar"},
|
|
||||||
{".shtml", "text/html"},
|
|
||||||
{".sit", "application/x-stuffit"},
|
|
||||||
{".sitemap", "application/xml"},
|
|
||||||
{".skin", "application/xml"},
|
|
||||||
{".skp", "application/x-koan"},
|
|
||||||
{".sldm", "application/vnd.ms-powerpoint.slide.macroEnabled.12"},
|
|
||||||
{".sldx", "application/vnd.openxmlformats-officedocument.presentationml.slide"},
|
|
||||||
{".slk", "application/vnd.ms-excel"},
|
|
||||||
{".sln", "text/plain"},
|
|
||||||
{".slupkg-ms", "application/x-ms-license"},
|
|
||||||
{".smd", "audio/x-smd"},
|
|
||||||
{".smi", "application/octet-stream"},
|
|
||||||
{".smx", "audio/x-smd"},
|
|
||||||
{".smz", "audio/x-smd"},
|
|
||||||
{".snd", "audio/basic"},
|
|
||||||
{".snippet", "application/xml"},
|
|
||||||
{".snp", "application/octet-stream"},
|
|
||||||
{".sql", "application/sql"},
|
|
||||||
{".sol", "text/plain"},
|
|
||||||
{".sor", "text/plain"},
|
|
||||||
{".spc", "application/x-pkcs7-certificates"},
|
|
||||||
{".spl", "application/futuresplash"},
|
|
||||||
{".spx", "audio/ogg"},
|
|
||||||
{".src", "application/x-wais-source"},
|
|
||||||
{".srf", "text/plain"},
|
|
||||||
{".SSISDeploymentManifest", "text/xml"},
|
|
||||||
{".ssm", "application/streamingmedia"},
|
|
||||||
{".sst", "application/vnd.ms-pki.certstore"},
|
|
||||||
{".stl", "application/vnd.ms-pki.stl"},
|
|
||||||
{".sv4cpio", "application/x-sv4cpio"},
|
|
||||||
{".sv4crc", "application/x-sv4crc"},
|
|
||||||
{".svc", "application/xml"},
|
|
||||||
{".svg", "image/svg+xml"},
|
|
||||||
{".swf", "application/x-shockwave-flash"},
|
|
||||||
{".step", "application/step"},
|
|
||||||
{".stp", "application/step"},
|
|
||||||
{".t", "application/x-troff"},
|
|
||||||
{".tar", "application/x-tar"},
|
|
||||||
{".tcl", "application/x-tcl"},
|
|
||||||
{".testrunconfig", "application/xml"},
|
|
||||||
{".testsettings", "application/xml"},
|
|
||||||
{".tex", "application/x-tex"},
|
|
||||||
{".texi", "application/x-texinfo"},
|
|
||||||
{".texinfo", "application/x-texinfo"},
|
|
||||||
{".tgz", "application/x-compressed"},
|
|
||||||
{".thmx", "application/vnd.ms-officetheme"},
|
|
||||||
{".thn", "application/octet-stream"},
|
|
||||||
{".tif", "image/tiff"},
|
|
||||||
{".tiff", "image/tiff"},
|
|
||||||
{".tlh", "text/plain"},
|
|
||||||
{".tli", "text/plain"},
|
|
||||||
{".toc", "application/octet-stream"},
|
|
||||||
{".tr", "application/x-troff"},
|
|
||||||
{".trm", "application/x-msterminal"},
|
|
||||||
{".trx", "application/xml"},
|
|
||||||
{".ts", "video/vnd.dlna.mpeg-tts"},
|
|
||||||
{".tsv", "text/tab-separated-values"},
|
|
||||||
{".ttf", "application/font-sfnt"},
|
|
||||||
{".tts", "video/vnd.dlna.mpeg-tts"},
|
|
||||||
{".txt", "text/plain"},
|
|
||||||
{".u32", "application/octet-stream"},
|
|
||||||
{".uls", "text/iuls"},
|
|
||||||
{".user", "text/plain"},
|
|
||||||
{".ustar", "application/x-ustar"},
|
|
||||||
{".vb", "text/plain"},
|
|
||||||
{".vbdproj", "text/plain"},
|
|
||||||
{".vbk", "video/mpeg"},
|
|
||||||
{".vbproj", "text/plain"},
|
|
||||||
{".vbs", "text/vbscript"},
|
|
||||||
{".vcf", "text/x-vcard"},
|
|
||||||
{".vcproj", "application/xml"},
|
|
||||||
{".vcs", "text/plain"},
|
|
||||||
{".vcxproj", "application/xml"},
|
|
||||||
{".vddproj", "text/plain"},
|
|
||||||
{".vdp", "text/plain"},
|
|
||||||
{".vdproj", "text/plain"},
|
|
||||||
{".vdx", "application/vnd.ms-visio.viewer"},
|
|
||||||
{".vml", "text/xml"},
|
|
||||||
{".vscontent", "application/xml"},
|
|
||||||
{".vsct", "text/xml"},
|
|
||||||
{".vsd", "application/vnd.visio"},
|
|
||||||
{".vsi", "application/ms-vsi"},
|
|
||||||
{".vsix", "application/vsix"},
|
|
||||||
{".vsixlangpack", "text/xml"},
|
|
||||||
{".vsixmanifest", "text/xml"},
|
|
||||||
{".vsmdi", "application/xml"},
|
|
||||||
{".vspscc", "text/plain"},
|
|
||||||
{".vss", "application/vnd.visio"},
|
|
||||||
{".vsscc", "text/plain"},
|
|
||||||
{".vssettings", "text/xml"},
|
|
||||||
{".vssscc", "text/plain"},
|
|
||||||
{".vst", "application/vnd.visio"},
|
|
||||||
{".vstemplate", "text/xml"},
|
|
||||||
{".vsto", "application/x-ms-vsto"},
|
|
||||||
{".vsw", "application/vnd.visio"},
|
|
||||||
{".vsx", "application/vnd.visio"},
|
|
||||||
{".vtt", "text/vtt"},
|
|
||||||
{".vtx", "application/vnd.visio"},
|
|
||||||
{".wasm", "application/wasm"},
|
|
||||||
{".wav", "audio/wav"},
|
|
||||||
{".wave", "audio/wav"},
|
|
||||||
{".wax", "audio/x-ms-wax"},
|
|
||||||
{".wbk", "application/msword"},
|
|
||||||
{".wbmp", "image/vnd.wap.wbmp"},
|
|
||||||
{".wcm", "application/vnd.ms-works"},
|
|
||||||
{".wdb", "application/vnd.ms-works"},
|
|
||||||
{".wdp", "image/vnd.ms-photo"},
|
|
||||||
{".webarchive", "application/x-safari-webarchive"},
|
|
||||||
{".webm", "video/webm"},
|
|
||||||
{".webp", "image/webp"}, /* https://en.wikipedia.org/wiki/WebP */
|
|
||||||
{".webtest", "application/xml"},
|
|
||||||
{".wiq", "application/xml"},
|
|
||||||
{".wiz", "application/msword"},
|
|
||||||
{".wks", "application/vnd.ms-works"},
|
|
||||||
{".WLMP", "application/wlmoviemaker"},
|
|
||||||
{".wlpginstall", "application/x-wlpg-detect"},
|
|
||||||
{".wlpginstall3", "application/x-wlpg3-detect"},
|
|
||||||
{".wm", "video/x-ms-wm"},
|
|
||||||
{".wma", "audio/x-ms-wma"},
|
|
||||||
{".wmd", "application/x-ms-wmd"},
|
|
||||||
{".wmf", "application/x-msmetafile"},
|
|
||||||
{".wml", "text/vnd.wap.wml"},
|
|
||||||
{".wmlc", "application/vnd.wap.wmlc"},
|
|
||||||
{".wmls", "text/vnd.wap.wmlscript"},
|
|
||||||
{".wmlsc", "application/vnd.wap.wmlscriptc"},
|
|
||||||
{".wmp", "video/x-ms-wmp"},
|
|
||||||
{".wmv", "video/x-ms-wmv"},
|
|
||||||
{".wmx", "video/x-ms-wmx"},
|
|
||||||
{".wmz", "application/x-ms-wmz"},
|
|
||||||
{".woff", "application/font-woff"},
|
|
||||||
{".woff2", "application/font-woff2"},
|
|
||||||
{".wpl", "application/vnd.ms-wpl"},
|
|
||||||
{".wps", "application/vnd.ms-works"},
|
|
||||||
{".wri", "application/x-mswrite"},
|
|
||||||
{".wrl", "x-world/x-vrml"},
|
|
||||||
{".wrz", "x-world/x-vrml"},
|
|
||||||
{".wsc", "text/scriptlet"},
|
|
||||||
{".wsdl", "text/xml"},
|
|
||||||
{".wvx", "video/x-ms-wvx"},
|
|
||||||
{".x", "application/directx"},
|
|
||||||
{".xaf", "x-world/x-vrml"},
|
|
||||||
{".xaml", "application/xaml+xml"},
|
|
||||||
{".xap", "application/x-silverlight-app"},
|
|
||||||
{".xbap", "application/x-ms-xbap"},
|
|
||||||
{".xbm", "image/x-xbitmap"},
|
|
||||||
{".xdr", "text/plain"},
|
|
||||||
{".xht", "application/xhtml+xml"},
|
|
||||||
{".xhtml", "application/xhtml+xml"},
|
|
||||||
{".xla", "application/vnd.ms-excel"},
|
|
||||||
{".xlam", "application/vnd.ms-excel.addin.macroEnabled.12"},
|
|
||||||
{".xlc", "application/vnd.ms-excel"},
|
|
||||||
{".xld", "application/vnd.ms-excel"},
|
|
||||||
{".xlk", "application/vnd.ms-excel"},
|
|
||||||
{".xll", "application/vnd.ms-excel"},
|
|
||||||
{".xlm", "application/vnd.ms-excel"},
|
|
||||||
{".xls", "application/vnd.ms-excel"},
|
|
||||||
{".xlsb", "application/vnd.ms-excel.sheet.binary.macroEnabled.12"},
|
|
||||||
{".xlsm", "application/vnd.ms-excel.sheet.macroEnabled.12"},
|
|
||||||
{".xlsx", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"},
|
|
||||||
{".xlt", "application/vnd.ms-excel"},
|
|
||||||
{".xltm", "application/vnd.ms-excel.template.macroEnabled.12"},
|
|
||||||
{".xltx", "application/vnd.openxmlformats-officedocument.spreadsheetml.template"},
|
|
||||||
{".xlw", "application/vnd.ms-excel"},
|
|
||||||
{".xml", "text/xml"},
|
|
||||||
{".xmp", "application/octet-stream"},
|
|
||||||
{".xmta", "application/xml"},
|
|
||||||
{".xof", "x-world/x-vrml"},
|
|
||||||
{".XOML", "text/plain"},
|
|
||||||
{".xpm", "image/x-xpixmap"},
|
|
||||||
{".xps", "application/vnd.ms-xpsdocument"},
|
|
||||||
{".xrm-ms", "text/xml"},
|
|
||||||
{".xsc", "application/xml"},
|
|
||||||
{".xsd", "text/xml"},
|
|
||||||
{".xsf", "text/xml"},
|
|
||||||
{".xsl", "text/xml"},
|
|
||||||
{".xslt", "text/xml"},
|
|
||||||
{".xsn", "application/octet-stream"},
|
|
||||||
{".xss", "application/xml"},
|
|
||||||
{".xspf", "application/xspf+xml"},
|
|
||||||
{".xtp", "application/octet-stream"},
|
|
||||||
{".xwd", "image/x-xwindowdump"},
|
|
||||||
{".z", "application/x-compress"},
|
|
||||||
{".zip", "application/zip"},
|
|
||||||
|
|
||||||
{"application/fsharp-script", ".fsx"},
|
|
||||||
{"application/msaccess", ".adp"},
|
|
||||||
{"application/msword", ".doc"},
|
|
||||||
{"application/octet-stream", ".bin"},
|
|
||||||
{"application/onenote", ".one"},
|
|
||||||
{"application/postscript", ".eps"},
|
|
||||||
{"application/step", ".step"},
|
|
||||||
{"application/vnd.apple.keynote", ".key"},
|
|
||||||
{"application/vnd.apple.numbers", ".numbers"},
|
|
||||||
{"application/vnd.apple.pages", ".pages"},
|
|
||||||
{"application/vnd.ms-excel", ".xls"},
|
|
||||||
{"application/vnd.ms-powerpoint", ".ppt"},
|
|
||||||
{"application/vnd.ms-works", ".wks"},
|
|
||||||
{"application/vnd.visio", ".vsd"},
|
|
||||||
{"application/x-director", ".dir"},
|
|
||||||
{"application/x-msdos-program", ".exe"},
|
|
||||||
{"application/x-shockwave-flash", ".swf"},
|
|
||||||
{"application/x-x509-ca-cert", ".cer"},
|
|
||||||
{"application/x-zip-compressed", ".zip"},
|
|
||||||
{"application/xhtml+xml", ".xhtml"},
|
|
||||||
{"application/x-iwork-keynote-sffkey", ".key"},
|
|
||||||
{"application/x-iwork-numbers-sffnumbers", ".numbers"},
|
|
||||||
{"application/x-iwork-pages-sffpages", ".pages"},
|
|
||||||
{"application/xml", ".xml"}, // anomaly, .xml -> text/xml, but application/xml -> many things, but all are xml, so safest is .xml
|
|
||||||
{"audio/aac", ".AAC"},
|
|
||||||
{"audio/aiff", ".aiff"},
|
|
||||||
{"audio/basic", ".snd"},
|
|
||||||
{"audio/mid", ".midi"},
|
|
||||||
{"audio/mp4", ".m4a"}, // one way mapping only, mime -> ext
|
|
||||||
{"audio/ogg", ".ogg"},
|
|
||||||
{"audio/ogg; codecs=opus", ".opus"},
|
|
||||||
{"audio/wav", ".wav"},
|
|
||||||
{"audio/x-m4a", ".m4a"},
|
|
||||||
{"audio/x-mpegurl", ".m3u"},
|
|
||||||
{"audio/x-pn-realaudio", ".ra"},
|
|
||||||
{"audio/x-smd", ".smd"},
|
|
||||||
{"image/bmp", ".bmp"},
|
|
||||||
{"image/heic", ".heic"},
|
|
||||||
{"image/heif", ".heif"},
|
|
||||||
{"image/jpeg", ".jpg"},
|
|
||||||
{"image/pict", ".pic"},
|
|
||||||
{"image/png", ".png"}, // Defined in [RFC-2045], [RFC-2048]
|
|
||||||
{"image/x-png", ".png"}, // See https://www.w3.org/TR/PNG/#A-Media-type :"It is recommended that implementations also recognize the media type "image/x-png"."
|
|
||||||
{"image/tiff", ".tiff"},
|
|
||||||
{"image/x-macpaint", ".mac"},
|
|
||||||
{"image/x-quicktime", ".qti"},
|
|
||||||
{"message/rfc822", ".eml"},
|
|
||||||
{"text/calendar", ".ics"},
|
|
||||||
{"text/html", ".html"},
|
|
||||||
{"text/plain", ".txt"},
|
|
||||||
{"text/scriptlet", ".wsc"},
|
|
||||||
{"text/xml", ".xml"},
|
|
||||||
{"video/3gpp", ".3gp"},
|
|
||||||
{"video/3gpp2", ".3gp2"},
|
|
||||||
{"video/mp4", ".mp4"},
|
|
||||||
{"video/mpeg", ".mpg"},
|
|
||||||
{"video/quicktime", ".mov"},
|
|
||||||
{"video/vnd.dlna.mpeg-tts", ".m2t"},
|
|
||||||
{"video/x-dv", ".dv"},
|
|
||||||
{"video/x-la-asf", ".lsf"},
|
|
||||||
{"video/x-ms-asf", ".asf"},
|
|
||||||
{"x-world/x-vrml", ".xof"},
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
var cache = mappings.ToList(); // need ToList() to avoid modifying while still enumerating
|
|
||||||
|
|
||||||
foreach (var mapping in cache)
|
|
||||||
{
|
|
||||||
if (!mappings.ContainsKey(mapping.Value))
|
|
||||||
{
|
|
||||||
mappings.Add(mapping.Value, mapping.Key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return mappings;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Tries to get the type of the MIME from the provided string.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="str">The filename or extension.</param>
|
|
||||||
/// <param name="mimeType">The variable to store the MIME type.</param>
|
|
||||||
/// <returns>The MIME type.</returns>
|
|
||||||
/// <exception cref="ArgumentNullException" />
|
|
||||||
public static bool TryGetMimeType(string str, out string mimeType)
|
|
||||||
{
|
|
||||||
if (str == null) throw new ArgumentNullException(nameof(str));
|
|
||||||
|
|
||||||
var indexQuestionMark = str.IndexOf(QuestionMark, StringComparison.Ordinal);
|
|
||||||
if (indexQuestionMark != -1)
|
|
||||||
{
|
|
||||||
str = str.Remove(indexQuestionMark);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (!str.StartsWith(Dot))
|
|
||||||
{
|
|
||||||
var index = str.LastIndexOf(Dot);
|
|
||||||
if (index != -1 && str.Length > index + 1)
|
|
||||||
{
|
|
||||||
str = str.Substring(index + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
str = Dot + str;
|
|
||||||
}
|
|
||||||
|
|
||||||
return _mappings.Value.TryGetValue(str, out mimeType);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets the type of the MIME from the provided string.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="str">The filename or extension.</param>
|
|
||||||
/// <returns>The MIME type.</returns>
|
|
||||||
/// <exception cref="ArgumentNullException" />
|
|
||||||
public static string GetMimeType(string str)
|
|
||||||
{
|
|
||||||
return TryGetMimeType(str, out var result) ? result : DefaultMimeType;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets the extension from the provided MINE type.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="mimeType">Type of the MIME.</param>
|
|
||||||
/// <param name="throwErrorIfNotFound">if set to <c>true</c>, throws error if extension's not found.</param>
|
|
||||||
/// <returns>The extension.</returns>
|
|
||||||
/// <exception cref="ArgumentNullException" />
|
|
||||||
/// <exception cref="ArgumentException" />
|
|
||||||
public static string GetExtension(string mimeType, bool throwErrorIfNotFound = true)
|
|
||||||
{
|
|
||||||
if (mimeType == null)
|
|
||||||
throw new ArgumentNullException(nameof(mimeType));
|
|
||||||
|
|
||||||
if (mimeType.StartsWith(Dot))
|
|
||||||
throw new ArgumentException("Requested mime type is not valid: " + mimeType);
|
|
||||||
|
|
||||||
if (_mappings.Value.TryGetValue(mimeType, out string extension))
|
|
||||||
return extension;
|
|
||||||
|
|
||||||
if (throwErrorIfNotFound)
|
|
||||||
throw new ArgumentException("Requested mime type is not registered: " + mimeType);
|
|
||||||
|
|
||||||
return string.Empty;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
|
||||||
|
|
||||||
<PropertyGroup>
|
|
||||||
<TargetFramework>netstandard2.1</TargetFramework>
|
|
||||||
<Nullable>enable</Nullable>
|
|
||||||
<PreserveCompilationContext>true</PreserveCompilationContext>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="RazorLight" Version="2.3.1" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
</Project>
|
|
||||||
Reference in New Issue
Block a user