mirror of
https://gitee.com/langsisi_admin/serein-flow
synced 2026-04-02 22:36:35 +08:00
438 lines
15 KiB
C#
438 lines
15 KiB
C#
|
|
using Serein.Library;
|
|||
|
|
using Serein.Library.Api;
|
|||
|
|
using Serein.Library.Utils;
|
|||
|
|
using System;
|
|||
|
|
using System.Collections;
|
|||
|
|
using System.Collections.Concurrent;
|
|||
|
|
using System.Collections.Generic;
|
|||
|
|
using System.IO;
|
|||
|
|
using System.Linq;
|
|||
|
|
using System.Net;
|
|||
|
|
using System.Reflection;
|
|||
|
|
using System.Text;
|
|||
|
|
using System.Threading.Tasks;
|
|||
|
|
using static Serein.Proto.HttpApi.PathRouter;
|
|||
|
|
using Enum = System.Enum;
|
|||
|
|
using Type = System.Type;
|
|||
|
|
|
|||
|
|
namespace Serein.Proto.HttpApi
|
|||
|
|
{
|
|||
|
|
/// <summary>
|
|||
|
|
/// 路由接口
|
|||
|
|
/// </summary>
|
|||
|
|
public interface IPathRouter
|
|||
|
|
{
|
|||
|
|
/// <summary>
|
|||
|
|
/// 添加处理模块
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="controllerType"></param>
|
|||
|
|
void AddHandle(Type controllerType);
|
|||
|
|
/// <summary>
|
|||
|
|
/// 获取路由信息
|
|||
|
|
/// </summary>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
List<RouterInfo> GetRouterInfos();
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 路由解析开始处理
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="context"></param>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
Task<InvokeResult> HandleAsync(HttpListenerContext context);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
public enum HandleState
|
|||
|
|
{
|
|||
|
|
/// <summary>
|
|||
|
|
/// 默认值
|
|||
|
|
/// </summary>
|
|||
|
|
None ,
|
|||
|
|
/// <summary>
|
|||
|
|
/// 没有异常
|
|||
|
|
/// </summary>
|
|||
|
|
Ok,
|
|||
|
|
/// <summary>
|
|||
|
|
/// 没有对应的控制器
|
|||
|
|
/// </summary>
|
|||
|
|
NotHanleController,
|
|||
|
|
/// <summary>
|
|||
|
|
/// 没有对应的Http请求类型
|
|||
|
|
/// </summary>
|
|||
|
|
NotHttpApiRequestType,
|
|||
|
|
/// <summary>
|
|||
|
|
/// 没有处理配置
|
|||
|
|
/// </summary>
|
|||
|
|
NotHandleConfig,
|
|||
|
|
/// <summary>
|
|||
|
|
/// 无法实例化控制器
|
|||
|
|
/// </summary>
|
|||
|
|
HanleControllerIsNull,
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 调用参数获取错误
|
|||
|
|
/// </summary>
|
|||
|
|
InvokeArgError,
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 调用发生异常
|
|||
|
|
/// </summary>
|
|||
|
|
InvokeErrored,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 调用结果
|
|||
|
|
/// </summary>
|
|||
|
|
public class InvokeResult
|
|||
|
|
{
|
|||
|
|
/// <summary>
|
|||
|
|
/// 调用状态
|
|||
|
|
/// </summary>
|
|||
|
|
public HandleState State { get; set; }
|
|||
|
|
/// <summary>
|
|||
|
|
/// 调用正常时这里会有数据
|
|||
|
|
/// </summary>
|
|||
|
|
public object? Data{ get; set; }
|
|||
|
|
/// <summary>
|
|||
|
|
/// 调用失败时这里可能会有异常数据
|
|||
|
|
/// </summary>
|
|||
|
|
public Exception? Exception { get; set; }
|
|||
|
|
|
|||
|
|
public static InvokeResult Ok(object? data) => new InvokeResult
|
|||
|
|
{
|
|||
|
|
Data = data,
|
|||
|
|
State = HandleState.Ok,
|
|||
|
|
};
|
|||
|
|
public static InvokeResult Fail(HandleState state, Exception? ex = null) => new InvokeResult
|
|||
|
|
{
|
|||
|
|
State = state,
|
|||
|
|
Exception = ex,
|
|||
|
|
};
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 路由信息
|
|||
|
|
/// </summary>
|
|||
|
|
public class RouterInfo
|
|||
|
|
{
|
|||
|
|
#if NET6_0_OR_GREATER
|
|||
|
|
/// <summary>
|
|||
|
|
/// 接口类型
|
|||
|
|
/// </summary>
|
|||
|
|
public ApiType ApiType { get; set; }
|
|||
|
|
/// <summary>
|
|||
|
|
/// 接口URL
|
|||
|
|
/// </summary>
|
|||
|
|
public required string Url { get; set; }
|
|||
|
|
/// <summary>
|
|||
|
|
/// 对应的处理方法
|
|||
|
|
/// </summary>
|
|||
|
|
public required MethodInfo MethodInfo { get; set; }
|
|||
|
|
#else
|
|||
|
|
/// <summary>
|
|||
|
|
/// 接口类型
|
|||
|
|
/// </summary>
|
|||
|
|
public ApiType ApiType { get; set; }
|
|||
|
|
/// <summary>
|
|||
|
|
/// 接口URL
|
|||
|
|
/// </summary>
|
|||
|
|
public string Url { get; set; }
|
|||
|
|
/// <summary>
|
|||
|
|
/// 对应的处理方法
|
|||
|
|
/// </summary>
|
|||
|
|
public MethodInfo MethodInfo { get; set; }
|
|||
|
|
#endif
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 路由注册与解析
|
|||
|
|
/// </summary>
|
|||
|
|
public class PathRouter : IPathRouter
|
|||
|
|
{
|
|||
|
|
private readonly ISereinIOC SereinIOC; // 用于存储路由信息
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 控制器实例对象的类型,每次调用都会重新实例化,[Url - ControllerType]
|
|||
|
|
/// </summary>
|
|||
|
|
private readonly ConcurrentDictionary<string, Type> _controllerTypes = new ConcurrentDictionary<string, Type>();
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 用于存储路由信息,[GET|POST - [Url - ApiHandleConfig]]
|
|||
|
|
/// </summary>
|
|||
|
|
private readonly ConcurrentDictionary<string, ConcurrentDictionary<string, ApiHandleConfig>> HandleModels = new ConcurrentDictionary<string, ConcurrentDictionary<string, ApiHandleConfig>>();
|
|||
|
|
|
|||
|
|
public PathRouter(ISereinIOC SereinIOC)
|
|||
|
|
{
|
|||
|
|
this.SereinIOC = SereinIOC;
|
|||
|
|
foreach (ApiType method in Enum.GetValues(typeof(ApiType))) // 遍历 HTTP 枚举类型的所有值
|
|||
|
|
{
|
|||
|
|
HandleModels.TryAdd(method.ToString(), new ConcurrentDictionary<string, ApiHandleConfig>()); // 初始化每种 HTTP 方法对应的路由字典
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 获取路由信息
|
|||
|
|
/// </summary>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
public List<RouterInfo> GetRouterInfos()
|
|||
|
|
{
|
|||
|
|
var t = HandleModels.Select(kvp => kvp.Value.Select(kvp2 => kvp2.Value.RouterInfo));
|
|||
|
|
return t.SelectMany(x => x).ToList();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 添加处理模块
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="controllerType"></param>
|
|||
|
|
public void AddHandle(Type controllerType)
|
|||
|
|
{
|
|||
|
|
if (!controllerType.IsClass || controllerType.IsAbstract) return; // 如果不是类或者是抽象类,则直接返回
|
|||
|
|
|
|||
|
|
var autoHostingAttribute = controllerType.GetCustomAttribute<WebApiControllerAttribute>();
|
|||
|
|
if(autoHostingAttribute is null)
|
|||
|
|
{
|
|||
|
|
SereinEnv.WriteLine(InfoType.WARN, $"类型 \"{controllerType}\" 需要实现 \"WebApiControllerAttribute\" 才可以作为路由控制器");
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
var methods = controllerType.GetMethods().Where(m => m.GetCustomAttribute<WebApiAttribute>() != null).ToArray();
|
|||
|
|
|
|||
|
|
|
|||
|
|
foreach (var method in methods) // 遍历控制器类型的所有方法
|
|||
|
|
{
|
|||
|
|
var routeAttribute = method.GetCustomAttribute<WebApiAttribute>(); // 获取方法上的 WebAPIAttribute 自定义属性
|
|||
|
|
if (routeAttribute is null) // 如果存在 WebAPIAttribute 属性
|
|||
|
|
{
|
|||
|
|
continue;
|
|||
|
|
}
|
|||
|
|
var url = GetRoutesUrl(autoHostingAttribute, routeAttribute, controllerType, method);
|
|||
|
|
if (url is null) continue;
|
|||
|
|
|
|||
|
|
SereinEnv.WriteLine(InfoType.INFO, url);
|
|||
|
|
var apiType = routeAttribute.ApiType.ToString();
|
|||
|
|
|
|||
|
|
var routerInfo = new RouterInfo
|
|||
|
|
{
|
|||
|
|
ApiType = routeAttribute.ApiType,
|
|||
|
|
Url = url,
|
|||
|
|
MethodInfo = method,
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
var config = new ApiHandleConfig(routerInfo);
|
|||
|
|
if(!HandleModels.TryGetValue(apiType, out var configs))
|
|||
|
|
{
|
|||
|
|
configs = new ConcurrentDictionary<string, ApiHandleConfig>();
|
|||
|
|
HandleModels[apiType] = configs;
|
|||
|
|
}
|
|||
|
|
configs.TryAdd(url, config);
|
|||
|
|
_controllerTypes.TryAdd(url,controllerType);
|
|||
|
|
}
|
|||
|
|
return;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 在外部调用API后,解析路由,调用对应的方法
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="context"></param>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
public async Task<InvokeResult> HandleAsync(HttpListenerContext context)
|
|||
|
|
{
|
|||
|
|
var request = context.Request; // 获取请求对象
|
|||
|
|
var httpMethod = request.HttpMethod.ToUpper(); // 获取请求的 HTTP 方法
|
|||
|
|
if (!HandleModels.TryGetValue(httpMethod, out var modules) || request.Url is null)
|
|||
|
|
{
|
|||
|
|
return InvokeResult.Fail(HandleState.NotHttpApiRequestType); // 没有对应HTTP请求方法的处理
|
|||
|
|
}
|
|||
|
|
var template = request.Url.AbsolutePath.ToLower() ;
|
|||
|
|
if (!_controllerTypes.TryGetValue(template, out var controllerType))
|
|||
|
|
{
|
|||
|
|
return InvokeResult.Fail(HandleState.NotHanleController); // 没有对应的处理器
|
|||
|
|
}
|
|||
|
|
if (!modules.TryGetValue(template, out var config))
|
|||
|
|
{
|
|||
|
|
return InvokeResult.Fail(HandleState.NotHandleConfig); // 没有对应的处理配置
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
ControllerBase controllerInstance;
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
controllerInstance = (ControllerBase)SereinIOC.CreateObject(controllerType);
|
|||
|
|
}
|
|||
|
|
catch
|
|||
|
|
{
|
|||
|
|
return InvokeResult.Fail(HandleState.HanleControllerIsNull); // 未找到控制器实例
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
var url = request.Url; // 获取请求的完整URL
|
|||
|
|
var routeValues = GetUrlData(url); // 解析 URL 获取路由参数
|
|||
|
|
controllerInstance.Url = url.AbsolutePath;
|
|||
|
|
|
|||
|
|
object?[] args;
|
|||
|
|
switch (httpMethod)
|
|||
|
|
{
|
|||
|
|
case "GET":
|
|||
|
|
args = config.GetArgsOfGet(routeValues); // Get请求
|
|||
|
|
break;
|
|||
|
|
case "POST":
|
|||
|
|
var requestBody = await ReadRequestBodyAsync(request); // 读取请求体内容
|
|||
|
|
controllerInstance.Body = requestBody;
|
|||
|
|
if (!JsonHelper.TryParse(requestBody, out var requestJObject))
|
|||
|
|
{
|
|||
|
|
var exTips = $"body 无法转换为 json 数据, body: {requestBody}";
|
|||
|
|
return InvokeResult.Fail(HandleState.InvokeArgError, new Exception(exTips));
|
|||
|
|
}
|
|||
|
|
(var isPass, var index, var type, var ex) = config.TryGetArgsOfPost(routeValues, requestJObject, out args);
|
|||
|
|
if (!isPass)
|
|||
|
|
{
|
|||
|
|
var exTips = $"尝试转换第{index}个入参参数时,类型 {type.FullName} 参数获取失败:{ex?.Message}";
|
|||
|
|
return InvokeResult.Fail(HandleState.InvokeArgError, new Exception(exTips));
|
|||
|
|
}
|
|||
|
|
break;
|
|||
|
|
default:
|
|||
|
|
return InvokeResult.Fail(HandleState.NotHttpApiRequestType);
|
|||
|
|
}
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
var invokeResult = await config.HandleAsync(controllerInstance, args);
|
|||
|
|
|
|||
|
|
return InvokeResult.Ok(invokeResult);
|
|||
|
|
}
|
|||
|
|
catch (Exception ex)
|
|||
|
|
{
|
|||
|
|
return InvokeResult.Fail(HandleState.InvokeErrored, ex);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 读取Body中的消息
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="request"></param>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
private async Task<string> ReadRequestBodyAsync(HttpListenerRequest request)
|
|||
|
|
{
|
|||
|
|
using (Stream stream = request.InputStream)
|
|||
|
|
{
|
|||
|
|
using (StreamReader reader = new StreamReader(stream, Encoding.UTF8))
|
|||
|
|
{
|
|||
|
|
return await reader.ReadToEndAsync();
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 从方法中收集路由信息,返回方法对应的url
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="autoHostingAttribute">类的特性</param>
|
|||
|
|
/// <param name="webAttribute">方法的特性</param>
|
|||
|
|
/// <param name="controllerType">控制器类型</param>
|
|||
|
|
/// <param name="method">方法信息</param>
|
|||
|
|
/// <returns>方法对应的urk</returns>
|
|||
|
|
private static string GetRoutesUrl(WebApiControllerAttribute autoHostingAttribute, WebApiAttribute webAttribute, Type controllerType, MethodInfo method)
|
|||
|
|
{
|
|||
|
|
string controllerName;
|
|||
|
|
if (string.IsNullOrWhiteSpace(autoHostingAttribute.Url))
|
|||
|
|
{
|
|||
|
|
controllerName = controllerType.Name.Replace("Controller", "").ToLower(); // 获取控制器名称并转换为小写
|
|||
|
|
}
|
|||
|
|
else
|
|||
|
|
{
|
|||
|
|
controllerName = autoHostingAttribute.Url;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var httpMethod = webAttribute.ApiType; // 获取 HTTP 方法
|
|||
|
|
var customUrl = webAttribute.Url; // 获取自定义 URL
|
|||
|
|
|
|||
|
|
string url;
|
|||
|
|
controllerName = CleanUrl(controllerName);
|
|||
|
|
if (string.IsNullOrWhiteSpace(customUrl))
|
|||
|
|
{
|
|||
|
|
url = $"/{controllerName}/{method.Name}".ToLower();// 清理自定义 URL,并构建新的 URL
|
|||
|
|
}
|
|||
|
|
else
|
|||
|
|
{
|
|||
|
|
if(customUrl == "/")
|
|||
|
|
{
|
|||
|
|
url = $"/{controllerName}".ToLower(); // 使用控制器
|
|||
|
|
}
|
|||
|
|
else
|
|||
|
|
{
|
|||
|
|
|
|||
|
|
customUrl = CleanUrl(customUrl);
|
|||
|
|
url = $"/{controllerName}/{customUrl}".ToLower();// 清理自定义 URL,并构建新的 URL
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
return url;
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 修正方法特性中的URL格式
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="url"></param>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
private static string CleanUrl(string url)
|
|||
|
|
{
|
|||
|
|
|
|||
|
|
while (url.Length > 0 && url[0] == '/') // 去除开头的斜杠
|
|||
|
|
{
|
|||
|
|
url = url.Substring(1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
while (url.Length > 0 && url[url.Length - 1] == '/') // 去除末尾的斜杠
|
|||
|
|
{
|
|||
|
|
url = url.Substring(0, url.Length - 1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
for (int i = 0; i < url.Length - 1; i++) // 去除连续的斜杠
|
|||
|
|
{
|
|||
|
|
if (url[i] == '/' && url[i + 1] == '/')
|
|||
|
|
{
|
|||
|
|
url = url.Remove(i, 1);
|
|||
|
|
i--;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return url; // 返回清理后的 URL
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// <summary>
|
|||
|
|
/// 方法声明,用于解析 URL 获取路由参数
|
|||
|
|
/// </summary>
|
|||
|
|
/// <param name="uri"></param>
|
|||
|
|
/// <returns></returns>
|
|||
|
|
private Dictionary<string, string> GetUrlData(Uri uri)
|
|||
|
|
{
|
|||
|
|
Dictionary<string, string> routeValues = new Dictionary<string, string>();
|
|||
|
|
|
|||
|
|
var pathParts = uri.ToString().Split('?'); // 拆分 URL,获取路径部分
|
|||
|
|
|
|||
|
|
if (pathParts.Length > 1) // 如果包含查询字符串
|
|||
|
|
{
|
|||
|
|
//var queryParams = HttpUtility.ParseQueryString(pathParts[1]); // 解析查询字符串
|
|||
|
|
//foreach (string key in queryParams) // 遍历查询字符串的键值对
|
|||
|
|
//{
|
|||
|
|
// if (key == null) continue;
|
|||
|
|
// routeValues[key] = queryParams[key]; // 将键值对添加到路由参数字典中
|
|||
|
|
//}
|
|||
|
|
var parsedQuery = QueryStringParser.ParseQueryString(pathParts[1]);
|
|||
|
|
foreach (var kvp in parsedQuery)
|
|||
|
|
{
|
|||
|
|
//Console.WriteLine($"{kvp.Key}: {kvp.Value}");
|
|||
|
|
routeValues[kvp.Key.ToLower()] = kvp.Value; // 将键值对添加到路由参数字典中
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
return routeValues; // 返回路由参数字典
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|