add support for REST-ful API building

- allow parameters within url
- allow different methods other then just POST
-> still needs to be tested!
This commit is contained in:
Pilzinsel64
2024-11-28 09:03:48 +01:00
parent 74ebbbca7b
commit 9dcaa7e507
9 changed files with 239 additions and 53 deletions

View File

@@ -16,42 +16,62 @@ public class ApiClient(string apiUrl) : IApiClient
public virtual Task<ApiResponse> SendRequest(string route) public virtual Task<ApiResponse> SendRequest(string route)
{ {
return SendRequest(route, null); return SendRequest(route, HttpMethod.Post);
}
public virtual Task<ApiResponse> SendRequest(string route, HttpMethod method)
{
return SendRequest(route, method, null);
} }
public virtual Task<ApiResponse> SendRequest(string route, ApiMessage? message) public virtual Task<ApiResponse> SendRequest(string route, ApiMessage? message)
{ {
return SendRequest(route, message, null); return SendRequest(route, HttpMethod.Post, message);
} }
public virtual async Task<ApiResponse> SendRequest(string route, ApiMessage? message, IApiMessageSerializer? serializer) public virtual Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message)
{
return SendRequest(route, method, message, null);
}
public virtual async Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer)
{ {
serializer ??= Serializer; serializer ??= Serializer;
Log.InfoFormat("Send message to {0}", route); Log.InfoFormat("Send message to {0}", route);
var res = await Send(route, message, serializer); var res = await Send(route, method, message, serializer);
return new(res.StatusCode); return new(res.StatusCode);
} }
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage
{ {
return SendRequest<TResponse>(route, null); return SendRequest<TResponse>(route, HttpMethod.Post);
}
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method) where TResponse : ApiMessage
{
return SendRequest<TResponse>(route, method, null);
} }
public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage
{ {
return SendRequest<TResponse>(route, message, null); return SendRequest<TResponse>(route, HttpMethod.Post, message);
} }
public virtual async Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage public virtual Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message) where TResponse : ApiMessage
{
return SendRequest<TResponse>(route, method, message, null);
}
public virtual async Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage
{ {
serializer ??= Serializer; serializer ??= Serializer;
Log.InfoFormat("Send request to {0}", route); Log.InfoFormat("Send request to {0}", route);
var res = await Send(route, message, serializer); var res = await Send(route, method, message, serializer);
TResponse? result = null; TResponse? result = null;
if (res.IsSuccessStatusCode) if (res.IsSuccessStatusCode)
@@ -60,7 +80,7 @@ public class ApiClient(string apiUrl) : IApiClient
return new(res.StatusCode, result); return new(res.StatusCode, result);
} }
protected virtual async Task<HttpResponseMessage> Send(string route, ApiMessage? message, IApiMessageSerializer serializer) protected virtual async Task<HttpResponseMessage> Send(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer serializer)
{ {
var url = ApiUrl + route; var url = ApiUrl + route;
HttpContent content; HttpContent content;
@@ -80,7 +100,12 @@ public class ApiClient(string apiUrl) : IApiClient
Log.Debug("Sending request"); Log.Debug("Sending request");
return await httpClient.PostAsync(url, content); var httpmsg = new HttpRequestMessage(method, url)
{
Content = content
};
return await httpClient.SendAsync(httpmsg);
} }
protected virtual string? EncodeAuthKey() protected virtual string? EncodeAuthKey()

View File

@@ -1,9 +1,21 @@
namespace Pilz.Net.Api; namespace Pilz.Net.Api;
[AttributeUsage(AttributeTargets.Method)] [AttributeUsage(AttributeTargets.Method)]
public class ApiMessageHandlerAttribute(string route) : Attribute public class ApiMessageHandlerAttribute(string route, params string[] methods) : Attribute
{ {
public string Route { get; set; } = route; public ApiMessageHandlerAttribute(string route, params HttpMethod[] methods)
: this(route, methods.Select(m => m.Method).ToArray())
{
}
public ApiMessageHandlerAttribute(string route)
: this(route, HttpMethod.Post)
{
}
public string Route { get; } = route;
public string[] Methods { get; } = methods;
public Type? Serializer { get; set; } public Type? Serializer { get; set; }
public bool RequiesAuth { get; set; } public bool RequiesAuth { get; set; }
} }

View File

@@ -6,4 +6,6 @@ public record class ApiRequestInfo(
ApiMessage? Message, ApiMessage? Message,
[property: MemberNotNullWhen(true, "AuthKey")] [property: MemberNotNullWhen(true, "AuthKey")]
bool IsAuthenticated, bool IsAuthenticated,
string? AuthKey); string? AuthKey,
string Url,
string Method);

View File

@@ -1,14 +1,20 @@
using Castle.Core.Logging; using Castle.Core.Logging;
using Newtonsoft.Json.Linq;
using Pilz.Extensions.Reflection; using Pilz.Extensions.Reflection;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Net; using System.Net;
using System.Reflection; using System.Reflection;
using System.Text.Encodings.Web;
using System.Text.RegularExpressions;
using System.Web;
using static Pilz.Net.Api.IApiServer; using static Pilz.Net.Api.IApiServer;
namespace Pilz.Net.Api; namespace Pilz.Net.Api;
public class ApiServer(string apiUrl) : IApiServer public class ApiServer(string apiUrl) : IApiServer
{ {
protected readonly Dictionary<string, Delegate> handlers = []; protected readonly List<PrivateMessageHandler> handlers = [];
protected readonly Dictionary<Type, IApiMessageSerializer> serializers = []; protected readonly Dictionary<Type, IApiMessageSerializer> serializers = [];
protected readonly HttpListener httpListener = new(); protected readonly HttpListener httpListener = new();
@@ -16,6 +22,10 @@ public class ApiServer(string apiUrl) : IApiServer
public event OnCheckContextEventHandler? OnCheckContext; public event OnCheckContextEventHandler? OnCheckContext;
public event OnCheckContextEventHandler? OnCheckContextCompleted; public event OnCheckContextEventHandler? OnCheckContextCompleted;
protected record PrivateParameterInfo(string Name, int Index);
protected record PrivateMessageHandler(string Url, bool UseRegEx, Delegate Handler, PrivateParameterInfo[] Parameters, ApiMessageHandlerAttribute Attribute);
protected record PrivateApiResult(ApiResult Original, string? ResultJson); protected record PrivateApiResult(ApiResult Original, string? ResultJson);
public string ApiUrl { get; } = apiUrl; public string ApiUrl { get; } = apiUrl;
@@ -72,10 +82,33 @@ public class ApiServer(string apiUrl) : IApiServer
return; return;
} }
// Resolves parameters
var url = attribute.Route;
var useRegEx = false;
var nextBreacket = url.IndexOf('{');
var parameters = new List<PrivateParameterInfo>();
while (nextBreacket != -1)
{
var endBreacket = url.IndexOf('}', nextBreacket + 1);
if (endBreacket == -1)
{
var name = url.Substring(nextBreacket + 1, endBreacket - nextBreacket - 1);
const string regex = "[A-Za-z0-9%]+";
url = url.Replace(url.Substring(nextBreacket, endBreacket + 1), regex);
var index = url.Substring(0, nextBreacket + 1).Split('/').Length;
parameters.Add(new(name, index));
useRegEx = true;
nextBreacket = url.IndexOf('{', endBreacket + 1);
}
}
if (useRegEx)
url = url.Replace(".", "\\."); // Escape special characters
// Add handler // Add handler
var fullUrl = attribute.Route; Log.InfoFormat("Added handler for {0}", attribute.Route);
Log.InfoFormat("Added handler for {0}", fullUrl); handlers.Add(new(url, useRegEx, handler, [.. parameters], attribute));
handlers.Add(fullUrl, handler);
} }
protected void Receive() protected void Receive()
@@ -143,16 +176,6 @@ public class ApiServer(string apiUrl) : IApiServer
context.Response.OutputStream.Close(); context.Response.OutputStream.Close();
} }
Log.Debug("Sanity checks");
if (context.Request.HttpMethod != HttpMethod.Post.Method
|| context.Request.ContentType is not string contentType
|| !contentType.Contains("application/json"))
{
Log.Info("Request has no json content");
close();
return;
}
// Parse url // Parse url
Log.Debug("Parse url"); Log.Debug("Parse url");
var path = context.Request.Url?.PathAndQuery.Replace(ApiUrl, string.Empty); var path = context.Request.Url?.PathAndQuery.Replace(ApiUrl, string.Empty);
@@ -163,25 +186,51 @@ public class ApiServer(string apiUrl) : IApiServer
return; return;
} }
// Read input content // Find handler
Log.Debug("Read input content"); Log.Debug("Find handler");
string? contentJson; if (!TryGetHandler(path, context.Request.HttpMethod, out var handler))
if (context.Request.ContentLength64 > 0)
{ {
using StreamReader input = new(context.Request.InputStream); Log.Info("Request handler couldn't be found");
contentJson = input.ReadToEnd(); close();
return;
} }
else
contentJson = null;
// Get auth key // Get auth key
Log.Debug("Get auth key"); Log.Debug("Get auth key");
if (context.Request.Headers.Get("API-AUTH-KEY") is not string authKey) if (context.Request.Headers.Get("API-AUTH-KEY") is not string authKey)
authKey = null!; authKey = null!;
// Read input content
Log.Debug("Read input content");
string? contentJson;
if (context.Request.ContentType is string contentType
&& contentType.Contains("application/json")
&& context.Request.ContentLength64 > 0)
{
try
{
using StreamReader input = new(context.Request.InputStream);
contentJson = input.ReadToEnd();
}
catch (OutOfMemoryException)
{
Log.Error("Error reading remote data due to missing memory");
close();
return;
}
catch (Exception ex)
{
Log.Error("Error reading remote data", ex);
close();
return;
}
}
else
contentJson = null;
// Handle message // Handle message
Log.Debug("Handle mssage"); Log.Debug("Handle mssage");
if (HandleMessage(path, contentJson, authKey) is not PrivateApiResult result) if (HandleMessage(path, context.Request.HttpMethod, handler, contentJson, authKey) is not PrivateApiResult result)
{ {
Log.Warn("Request couldn't be handled"); Log.Warn("Request couldn't be handled");
close(); close();
@@ -206,28 +255,22 @@ public class ApiServer(string apiUrl) : IApiServer
close(); close();
} }
protected virtual PrivateApiResult? HandleMessage(string url, string? json, string? authKey) protected virtual PrivateApiResult? HandleMessage(string url, string method, PrivateMessageHandler handler, string? json, string? authKey)
{ {
// Get handler
Log.Debug("Find handler");
if (!handlers.TryGetValue(url, out var handler)
|| handler.Method.GetCustomAttribute<ApiMessageHandlerAttribute>() is not ApiMessageHandlerAttribute attribute)
return null;
// Check authentication // Check authentication
Log.Debug("Check authentication"); Log.Debug("Check authentication");
var isAuthenticated = false; var isAuthenticated = false;
if (!string.IsNullOrWhiteSpace(authKey) && DecodeAuthKey(authKey) is string authKeyDecoded) if (!string.IsNullOrWhiteSpace(authKey) && DecodeAuthKey(authKey) is string authKeyDecoded)
isAuthenticated = CheckAuthentication(authKeyDecoded, handler); isAuthenticated = CheckAuthentication(authKeyDecoded, handler.Handler);
else else
authKeyDecoded = null!; authKeyDecoded = null!;
if (attribute.RequiesAuth && !isAuthenticated) if (handler.Attribute.RequiesAuth && !isAuthenticated)
return new(ApiResult.Unauthorized(), null); return new(ApiResult.Unauthorized(), null);
// Get required infos // Get required infos
Log.Debug("Identify message parameter type and serializer"); Log.Debug("Identify message parameter type and serializer");
var targetType = handler.Method.GetParameters().FirstOrDefault(p => p.ParameterType.IsAssignableTo(typeof(ApiMessage)))?.ParameterType; var targetType = handler.Handler.Method.GetParameters().FirstOrDefault(p => p.ParameterType.IsAssignableTo(typeof(ApiMessage)))?.ParameterType;
var serializer = GetSerializer(attribute.Serializer); var serializer = GetSerializer(handler.Attribute.Serializer);
// Deserialize // Deserialize
Log.Debug("Deserialize message"); Log.Debug("Deserialize message");
@@ -239,8 +282,8 @@ public class ApiServer(string apiUrl) : IApiServer
// Invoke handler // Invoke handler
Log.Debug("Invoke handler"); Log.Debug("Invoke handler");
var parameters = BuildParameters(handler, () => message, () => new(message, isAuthenticated, authKeyDecoded)); var parameters = BuildParameters(url, handler, () => message, () => new(message, isAuthenticated, authKeyDecoded, url, method));
if (handler.DynamicInvoke(parameters) is not ApiResult result) if (handler.Handler.DynamicInvoke(parameters) is not ApiResult result)
return new(ApiResult.InternalServerError(), null); return new(ApiResult.InternalServerError(), null);
// Return result without message // Return result without message
@@ -258,9 +301,25 @@ public class ApiServer(string apiUrl) : IApiServer
return new(result, resultStr); return new(result, resultStr);
} }
protected virtual object?[]? BuildParameters(Delegate handler, Func<ApiMessage?> getMessage, Func<ApiRequestInfo> getRequestInfo) protected virtual bool TryGetHandler(string url, string method, [NotNullWhen(true)] out PrivateMessageHandler? handler)
{ {
var infos = handler.Method.GetParameters(); handler = handlers.FirstOrDefault(handler =>
{
if (!handler.Attribute.Methods.Contains(method))
return false;
if (handler.UseRegEx)
return Regex.IsMatch(url, handler.Url, RegexOptions.IgnoreCase);
return handler.Url.Equals(url, StringComparison.InvariantCultureIgnoreCase);
});
return handler != null;
}
protected virtual object?[]? BuildParameters(string url, PrivateMessageHandler handler, Func<ApiMessage?> getMessage, Func<ApiRequestInfo> getRequestInfo)
{
var infos = handler.Handler.Method.GetParameters();
var objs = new List<object?>(); var objs = new List<object?>();
foreach (var info in infos) foreach (var info in infos)
@@ -269,6 +328,9 @@ public class ApiServer(string apiUrl) : IApiServer
objs.Add(getMessage()); objs.Add(getMessage());
else if (info.ParameterType.IsAssignableTo(typeof(ApiRequestInfo))) else if (info.ParameterType.IsAssignableTo(typeof(ApiRequestInfo)))
objs.Add(getRequestInfo()); objs.Add(getRequestInfo());
else if (handler.Parameters.FirstOrDefault(p => p.Name.Equals(info.Name, StringComparison.InvariantCultureIgnoreCase)) is PrivateParameterInfo parameterInfo
&& url.Split('/').ElementAtOrDefault(parameterInfo.Index) is string parameterValue)
objs.Add(Convert.ChangeType(HttpUtility.UrlDecode(parameterValue), info.ParameterType)); // or Uri.UnescapeDataString(); maybe run this line twice?
else else
objs.Add(null); objs.Add(null);
} }

View File

@@ -14,13 +14,21 @@ public interface IApiClient
Task<ApiResponse> SendRequest(string route); Task<ApiResponse> SendRequest(string route);
Task<ApiResponse> SendRequest(string route, HttpMethod method);
Task<ApiResponse> SendRequest(string route, ApiMessage? message); Task<ApiResponse> SendRequest(string route, ApiMessage? message);
Task<ApiResponse> SendRequest(string route, ApiMessage? message, IApiMessageSerializer? serializer); Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message);
Task<ApiResponse> SendRequest(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer);
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage; Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route) where TResponse : ApiMessage;
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method) where TResponse : ApiMessage;
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage; Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message) where TResponse : ApiMessage;
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage; Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message) where TResponse : ApiMessage;
Task<ApiResponse<TResponse>> SendRequest<TResponse>(string route, HttpMethod method, ApiMessage? message, IApiMessageSerializer? serializer) where TResponse : ApiMessage;
} }

View File

@@ -0,0 +1,10 @@
namespace Pilz.UI.Controls.ConfigurationManager;
public class ConfigurationEntry(string name, string title, ConfigurationValueListener Listener)
{
public event EventHandler? OnValueChanged;
public string Name { get; } = name;
public string Title { get; } = title;
public ConfigurationValueListener Listener { get; } = Listener;
}

View File

@@ -0,0 +1,20 @@
namespace Pilz.UI.Controls.ConfigurationManager;
public class ConfigurationManager
{
private readonly List<ConfigurationPanel> panels = [];
public void Register(ConfigurationPanel panel)
{
panels.Add(panel);
}
public IEnumerable<Control> Build()
{
foreach (var panel in panels)
{
panel.Build(this);
yield return panel;
}
}
}

View File

@@ -0,0 +1,32 @@
namespace Pilz.UI.Controls.ConfigurationManager;
public class ConfigurationPanel : TableLayoutPanel
{
private readonly List<ConfigurationEntry> entries = [];
public ConfigurationEntry CreateEntry(string name, string title)
{
return CreateEntry(name, title, null);
}
public ConfigurationEntry CreateEntry(string name, string title, Action? create)
{
return CreateEntry(new(name, title, ));
}
public ConfigurationEntry CreateEntry(ConfigurationEntry entry)
{
entries.Add(entry);
return entry;
}
internal protected void Build(ConfigurationManager manager)
{
foreach (var entry in entries)
{
var control = ;
entry.Listener.Initialize();
// ...
}
}
}

View File

@@ -0,0 +1,15 @@
namespace Pilz.UI.Controls.ConfigurationManager;
public abstract class ConfigurationValueListener : IDisposable
{
public event EventHandler? ValueChanged;
protected virtual void OnValueChanged(object sender, EventArgs e)
{
ValueChanged?.Invoke(sender, e);
}
internal protected abstract void Initialize(Control control);
public abstract void Dispose();
}