223 lines
8.0 KiB
C#
223 lines
8.0 KiB
C#
using New_College.Common.LogHelper;
|
||
using New_College.Hubs;
|
||
using Castle.DynamicProxy;
|
||
using Microsoft.AspNetCore.Http;
|
||
using Microsoft.AspNetCore.SignalR;
|
||
using Newtonsoft.Json;
|
||
using StackExchange.Profiling;
|
||
using System;
|
||
using System.Linq;
|
||
using System.Reflection;
|
||
using System.Threading.Tasks;
|
||
|
||
namespace New_College.AOP
|
||
{
|
||
/// <summary>
|
||
/// 拦截器BlogLogAOP 继承IInterceptor接口
|
||
/// </summary>
|
||
public class BlogLogAOP : IInterceptor
|
||
{
|
||
private readonly IHubContext<ChatHub> _hubContext;
|
||
private readonly IHttpContextAccessor _accessor;
|
||
|
||
public BlogLogAOP(IHubContext<ChatHub> hubContext, IHttpContextAccessor accessor)
|
||
{
|
||
_hubContext = hubContext;
|
||
_accessor = accessor;
|
||
}
|
||
|
||
|
||
/// <summary>
|
||
/// 实例化IInterceptor唯一方法
|
||
/// </summary>
|
||
/// <param name="invocation">包含被拦截方法的信息</param>
|
||
public void Intercept(IInvocation invocation)
|
||
{
|
||
string NickName = _accessor.HttpContext?.User?.Identity?.Name;
|
||
|
||
//记录被拦截方法信息的日志信息
|
||
var dataIntercept = "" +
|
||
$"【当前操作用户】:{ NickName} \r\n" +
|
||
$"【当前执行方法】:{ invocation.Method.Name} \r\n" +
|
||
$"【携带的参数有】: {string.Join(", ", invocation.Arguments.Select(a => (a ?? "").ToString()).ToArray())} \r\n";
|
||
|
||
try
|
||
{
|
||
MiniProfiler.Current.Step($"执行Service方法:{invocation.Method.Name}() -> ");
|
||
//在被拦截的方法执行完毕后 继续执行当前方法,注意是被拦截的是异步的
|
||
invocation.Proceed();
|
||
|
||
|
||
// 异步获取异常,先执行
|
||
if (IsAsyncMethod(invocation.Method))
|
||
{
|
||
|
||
#region 方案一
|
||
//Wait task execution and modify return value
|
||
if (invocation.Method.ReturnType == typeof(Task))
|
||
{
|
||
invocation.ReturnValue = InternalAsyncHelper.AwaitTaskWithPostActionAndFinally(
|
||
(Task)invocation.ReturnValue,
|
||
async () => await SuccessAction(invocation, dataIntercept),/*成功时执行*/
|
||
ex =>
|
||
{
|
||
LogEx(ex, dataIntercept);
|
||
});
|
||
}
|
||
//Task<TResult>
|
||
else
|
||
{
|
||
invocation.ReturnValue = InternalAsyncHelper.CallAwaitTaskWithPostActionAndFinallyAndGetResult(
|
||
invocation.Method.ReturnType.GenericTypeArguments[0],
|
||
invocation.ReturnValue,
|
||
//async () => await SuccessAction(invocation, dataIntercept),/*成功时执行*/
|
||
async (o) => await SuccessAction(invocation, dataIntercept, o),/*成功时执行*/
|
||
ex =>
|
||
{
|
||
LogEx(ex, dataIntercept);
|
||
});
|
||
}
|
||
#endregion
|
||
|
||
|
||
// 如果方案一不行,试试这个方案
|
||
#region 方案二
|
||
|
||
//var type = invocation.Method.ReturnType;
|
||
//var resultProperty = type.GetProperty("Result");
|
||
//dataIntercept += ($"【执行完成结果】:{JsonConvert.SerializeObject(resultProperty.GetValue(invocation.ReturnValue))}");
|
||
|
||
//Parallel.For(0, 1, e =>
|
||
//{
|
||
// LogLock.OutSql2Log("AOPLog", new string[] { dataIntercept });
|
||
//});
|
||
|
||
#endregion
|
||
}
|
||
else
|
||
{// 同步1
|
||
|
||
dataIntercept += ($"【执行完成结果】:{invocation.ReturnValue}");
|
||
Parallel.For(0, 1, e =>
|
||
{
|
||
LogLock.OutSql2Log("AOPLog", new string[] { dataIntercept });
|
||
});
|
||
}
|
||
}
|
||
catch (Exception ex)// 同步2
|
||
{
|
||
LogEx(ex, dataIntercept);
|
||
|
||
}
|
||
|
||
_hubContext.Clients.All.SendAsync("ReceiveUpdate", LogLock.GetLogData()).Wait();
|
||
}
|
||
|
||
private async Task SuccessAction(IInvocation invocation, string dataIntercept, object o = null)
|
||
{
|
||
//invocation.ReturnValue = o;
|
||
//var type = invocation.Method.ReturnType;
|
||
//if (typeof(Task).IsAssignableFrom(type))
|
||
//{
|
||
// //var resultProperty = type.GetProperty("Result");
|
||
// //类型错误 都可以不要invocation参数,直接将o系列化保存到日记中
|
||
// dataIntercept += ($"【执行完成结果】:{JsonConvert.SerializeObject(invocation.ReturnValue)}");
|
||
//}
|
||
//else
|
||
//{
|
||
// dataIntercept += ($"【执行完成结果】:{invocation.ReturnValue}");
|
||
//}
|
||
|
||
dataIntercept += ($"【执行完成结果】:{JsonConvert.SerializeObject(o)}");
|
||
|
||
|
||
await Task.Run(() =>
|
||
{
|
||
Parallel.For(0, 1, e =>
|
||
{
|
||
LogLock.OutSql2Log("AOPLog", new string[] { dataIntercept });
|
||
});
|
||
});
|
||
}
|
||
|
||
private void LogEx(Exception ex, string dataIntercept)
|
||
{
|
||
if (ex != null)
|
||
{
|
||
//执行的 service 中,收录异常
|
||
MiniProfiler.Current.CustomTiming("Errors:", ex.Message);
|
||
//执行的 service 中,捕获异常
|
||
dataIntercept += ($"【执行完成结果】:方法中出现异常:{ex.Message + ex.InnerException}\r\n");
|
||
|
||
// 异常日志里有详细的堆栈信息
|
||
Parallel.For(0, 1, e =>
|
||
{
|
||
LogLock.OutSql2Log("AOPLog", new string[] { dataIntercept });
|
||
});
|
||
}
|
||
}
|
||
|
||
|
||
public static bool IsAsyncMethod(MethodInfo method)
|
||
{
|
||
return (
|
||
method.ReturnType == typeof(Task) ||
|
||
(method.ReturnType.IsGenericType && method.ReturnType.GetGenericTypeDefinition() == typeof(Task<>))
|
||
);
|
||
}
|
||
|
||
}
|
||
|
||
|
||
internal static class InternalAsyncHelper
|
||
{
|
||
public static async Task AwaitTaskWithPostActionAndFinally(Task actualReturnValue, Func<Task> postAction, Action<Exception> finalAction)
|
||
{
|
||
Exception exception = null;
|
||
|
||
try
|
||
{
|
||
await actualReturnValue;
|
||
await postAction();
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
exception = ex;
|
||
}
|
||
finally
|
||
{
|
||
finalAction(exception);
|
||
}
|
||
}
|
||
|
||
public static async Task<T> AwaitTaskWithPostActionAndFinallyAndGetResult<T>(Task<T> actualReturnValue, Func<object, Task> postAction, Action<Exception> finalAction)
|
||
{
|
||
Exception exception = null;
|
||
try
|
||
{
|
||
var result = await actualReturnValue;
|
||
await postAction(result);
|
||
return result;
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
exception = ex;
|
||
throw;
|
||
}
|
||
finally
|
||
{
|
||
finalAction(exception);
|
||
}
|
||
}
|
||
|
||
public static object CallAwaitTaskWithPostActionAndFinallyAndGetResult(Type taskReturnType, object actualReturnValue, Func<object, Task> action, Action<Exception> finalAction)
|
||
{
|
||
return typeof(InternalAsyncHelper)
|
||
.GetMethod("AwaitTaskWithPostActionAndFinallyAndGetResult", BindingFlags.Public | BindingFlags.Static)
|
||
.MakeGenericMethod(taskReturnType)
|
||
.Invoke(null, new object[] { actualReturnValue, action, finalAction });
|
||
}
|
||
}
|
||
|
||
}
|