WebAPI+NLog实现接口调用日志输出

在生产环境下,经常需要监测查看WebAPI接口的调用情况,日志就成了必不可少的手段之一,本文采用NLog实现WebAPI的日志记录。

环境 版本

操作系统 Windows 10 prefessional

编译器 Visual Studio 2015 update3

创建WebAPI

创建WebAPI项目,选择ASP.NET 4.5.2模板中的WebAPI模板。

引用NLog类库

打开项目的NuGet包管理器,搜索NLog,为项目添加程序包引用。

修改项目配置文件

在webAPI项目的Web.config中进行NLog的配置。首先在节点configuration>configSections下添加节点:

此处name必需为nlog,否则配置信息将不能被读取。

然后在configuration节点下添加节点nlog:

这里定义了日志文件的保存路径、命名格式以及日志记录类型和监听级别。

创建日志跟踪相关类

创建日志跟踪类AppLog,继承于System.Web.Http.Tracing下的跟踪编写器接口ITraceWriter,用于日志生成和写入:

public sealed class AppLog : ITraceWriter

{

//日志写入

private static readonly Logger AppLogger = LogManager.GetCurrentClassLogger();

private static readonly Lazy<Dictionary<TraceLevel, Action<string>>> LoggingMap = new Lazy<Dictionary<TraceLevel, Action<string>>>(() => new Dictionary<TraceLevel, Action<string>>

{

{TraceLevel.Info,AppLogger.Info },

{TraceLevel.Debug,AppLogger.Debug },

{TraceLevel.Error,AppLogger.Error },

{TraceLevel.Fatal,AppLogger.Fatal },

{TraceLevel.Warn,AppLogger.Warn }

});

private Dictionary<TraceLevel, Action<string>> Logger

{

get { return LoggingMap.Value; }

}

/// <summary>

/// 跟踪编写器接口实现

/// </summary>

/// <param name="request"></param>

/// <param name="category"></param>

/// <param name="level"></param>

/// <param name="traceAction"></param>

public void Trace(HttpRequestMessage request, string category, TraceLevel level, Action<TraceRecord> traceAction)

{

if (level != TraceLevel.Off)//未禁用日志跟踪

{

if (traceAction != null && traceAction.Target != null)

{

category = category + Environment.NewLine + "Action Parameters : " + JsonConvert.SerializeObject(traceAction.Target);

}

var record = new TraceRecord(request, category, level);

if (traceAction != null)

{

traceAction(record);

}

// traceAction?.Invoke(record);

Log(record);

}

//throw new NotImplementedException();

}

/// <summary>

/// 日志写入

/// </summary>

/// <param name="record"></param>

private void Log(TraceRecord record)

{

var message = new StringBuilder();

/**************************运行日志****************************/

if (!string.IsNullOrWhiteSpace(record.Message))

{

message.Append("").Append(record.Message + Environment.NewLine);

}

if (record.Request != null)

{

if (record.Request.Method != null)

{

message.Append("Method : " + record.Request.Method + Environment.NewLine);

}

if (record.Request.RequestUri != null)

{

message.Append("").Append("URL : " + record.Request.RequestUri + Environment.NewLine);

}

if (record.Request.Headers != null && record.Request.Headers.Contains("Token") && record.Request.Headers.GetValues("Token") != null && record.Request.Headers.GetValues("Token").FirstOrDefault() != null)

{

message.Append("").Append("Token : " + record.Request.Headers.GetValues("Token").FirstOrDefault() + Environment.NewLine);

}

}

if (!string.IsNullOrWhiteSpace(record.Category))

{

message.Append("").Append(record.Category);

}

if (!string.IsNullOrWhiteSpace(record.Operator))

{

message.Append(" ").Append(record.Operator).Append(" ").Append(record.Operation);

}

//***************************异常日志***********************************//

if (record.Exception != null && !string.IsNullOrWhiteSpace(record.Exception.GetBaseException().Message))

{

var exceptionType = record.Exception.GetType();

message.Append(Environment.NewLine);

message.Append("").Append("Error : " + record.Exception.GetBaseException().Message + Environment.NewLine);

}

//日志写入本地文件

Logger[record.Level](Convert.ToString(message) + Environment.NewLine);

}

}

创建日志筛选器类LogFilterAttribute,继承于System.Web.Http.Filters下的筛选器特性基类,用于定义日志内容:

public class LogFilterAttribute : ActionFilterAttribute

{

public override void OnActionExecuting(HttpActionContext actionContext)

{

GlobalConfiguration.Configuration.Services.Replace(typeof(ITraceWriter), new AppLog());

var trace = GlobalConfiguration.Configuration.Services.GetTraceWriter();

trace.Info(actionContext.Request, "Controller : " + actionContext.ControllerContext.ControllerDescriptor.ControllerType.FullName + Environment.NewLine + "Action : " + actionContext.ActionDescriptor.ActionName, "JSON", actionContext.ActionArguments);

//base.OnActionExecuting(actionContext);

}

}

创建异常筛选器类AbnormalFilterAttribute,继承于System.Web.Http.Filters下的异常筛选器类,用于异常信息的跟踪筛选:

public class AbnormalFilterAttribute: ExceptionFilterAttribute

{

public override void OnException(HttpActionExecutedContext actionExecutedContext)

{

GlobalConfiguration.Configuration.Services.Replace(typeof(ITraceWriter), new AppLog());

var trace = GlobalConfiguration.Configuration.Services.GetTraceWriter();

trace.Error(actionExecutedContext.Request, "Controller : " + actionExecutedContext.ActionContext.ControllerContext.ControllerDescriptor.ControllerType.FullName + Environment.NewLine + "Action : " + actionExecutedContext.ActionContext.ActionDescriptor.ActionName, actionExecutedContext.Exception);

var exceptionType = actionExecutedContext.Exception.GetType();

if (exceptionType == typeof(ValidationException))

{

var resp = new HttpResponseMessage(HttpStatusCode.BadRequest) { Content = new StringContent(actionExecutedContext.Exception.Message), ReasonPhrase = "ValidationException" };

throw new HttpResponseException(resp);

}

else if (exceptionType == typeof(UnauthorizedAccessException))

{

throw new HttpResponseException(actionExecutedContext.Request.CreateResponse(HttpStatusCode.Unauthorized));

}

else

{

throw new HttpResponseException(actionExecutedContext.Request.CreateResponse(HttpStatusCode.InternalServerError));

}

//base.OnException(actionExecutedContext);

}

}

应用NLog配置

最后需要将上述配置应用到项目中去,在pp_Data>WebApiConfig中添加如下配置:

将上述步骤完成后,运行项目,调用API接口进行测试,调用完成后就可以在站点目录下找到日志文件,能够为API接口的日常运行维护提供很大的便利。

原文地址:https://www.cnblogs.com/nayilvyangguang/p/10115206.html

时间: 2024-11-06 03:04:11

WebAPI+NLog实现接口调用日志输出的相关文章

retrofit工具增加http调用日志输出

Retrofit 是一个很好用的http调用组件.内置的其实也是okhttp.把okhttp封装了一下,让日常的业务交互对http调用的感知变小. 一个http调用只需要一行代码就可以了.具体的Retrofit的用法不在这里阐述.自行百度. 这里只讲解如何让Retrofit能够打印http调用的相关参数日志.开讲: Retrofit每次调用的时候都会用到OKhttpcilent这个客户端,它默认的client是没有打印日志功能的,查看Retrofit的源码: public final class

Log4j将不同Package的日志输出到不同的文件

转自:http://www.crazyant.net/1931.html 随着项目规模的越来越大,会不断的引入新的模块,不同的模块都会打印自己的日志,最后就造成日志根本没法查看,比如我自己的项目中,就存在以下这些日志: 接收外界消息的日志.对外发送消息的日志: 后台常驻线程的处理日志: 外部接口访问的参数.返回结果等接口日志: Service访问数据库产生的SQL日志: 这其中,消息日志和后台线程的日志数据量非常庞大,如果所有日志打印在一个文件中,使用tail -f log.log文件,会发现日

工作经验:Java 系统记录调用日志,并且记录错误堆栈

前言:现在有一个系统,主要是为了给其他系统提供数据查询接口的,这个系统上线不会轻易更新,更不会跟随业务系统的更新而更新(这也是有一个数据查询接口系统的原因,解耦).这时,这个系统就需要有一定的方便的线上查错方式,我便想到了记录每一次的调用日志,而且需要记录错误堆栈,同时被白名单过滤的也要记录下来. 想法 这个日志记录,需要在每一次访问接口时记录一下,在有异常时将异常的堆栈信息记录在每次访问记录里.这里由于要使用数据库信息,所以选择了 spring 的拦截器. 在拦截器抛放心之后,运行业务代码,如

ASP.NET Core 2.1 : 十二.内置日志、使用Nlog将日志输出到文件

应用离不开日志,虽然现在使用VS有强大的调试功能,开发过程中不复杂的情况懒得输出日志了(想起print和echo的有木有),但在一些复杂的过程中以及应用日常运行中的日志还是非常有用. ASP.NET Core提供了内置的日志,但没弄明白这么把它输出到文件, 只能在VS的输出中查看, 谁知道怎么弄告诉我一下.(ASP.NET Core 系列目录) 本例 GitHub 一.内置日志的使用 上一篇:如何在后台运行一个任务  中使用到了内置的日志,直接在构造中注入一下,然后直接使用即可, 非常方便 pu

添加webservice调用日志

之前想用spring的AOP给webservice添加切面的,但是使用around切面后,居然调用端得不到webservice的返回结果,而且报文的详细情况也不得而知,很是尴尬,所以偷了个懒.但是该做的还是要做,不要以后要求查看调用日志的时候,什么都拿不出,不是一个尴尬能搞定的. 我使用的是基于cxf的webservice,所以添加调用日志的方法也是基于cxf的,其次是配合sping开发webservice.基本的webservice的实现,这里就不再说明. 一.使用LoggingInInter

.NET Core的日志[2]:将日志输出到控制台

对于一个控制台应用,比如采用控制台应用作为宿主的ASP.NET Core应用,我们可以将记录的日志直接输出到控制台上.针对控制台的Logger是一个类型为ConsoleLogger的对象,ConsoleLogger对应的LoggerProvider类型为ConsoleLoggerProvider,这两个类型都定义在 NuGet包"Microsoft.Extensions.Logging.Console"之中. 本文已经同步到<ASP.NET Core框架揭秘>之中] 目录一

Mybatis下log4j日志输出不正常的解决办法 ,很实用哦 !!!!

使用Mybatis的时候,有些时候能输出(主要是指sql,参数,结果)日志.有些时候就不能. 无法输出日志的时候,无论怎么配置log4j,不管是properties的还是xml的,都不起作用. 有些时候,我们没做什么配置就能输出日志.... 这是一个让无数人烦躁的问题.其实解决问题很容易(我过了这么久才解决,以前都用拦截器输出). 这是一个普大喜奔的日子,让我们一起来看看如何解决mybatis的日志问题. 为什么说这个问题很容易解决呢?因为mybatis的文档写的很清楚. 为什么我们都没找到解决

支持windows linux下将指定内存段转为16进制与ascii码的日志输出类

来源:http://blog.csdn.net/lezhiyong 1. 简介 将指定内存段转为16进制与asci码的输出到日志文件的类. 2. 功能介绍 1) 支持window与linus双系统. 2) 可指定输出目录. 3) 日志能输出的时间精确到毫秒,日志能输出线程号. 4) 提供字符串输出. 5) 提供指定内存转换为16进制输出. 6) 提供指定内存转换为16进制和ascii码同时输出. 7) 可调整16进制输出间距. 3. 原理和算法 1) 模块是基于C++语言编写: 2) 通过条件定

在chroot环境下将MySQL日志输出到syslog

好久没写博客了,这几个月一直在学习nodejs,angularjs,做一些前端开发,目前还是学习阶段,等有一些体会再来分享. 这两天碰到的一个问题是,我们的产品给客户后,客户要统一管理日志,MySQL要将日志输出到syslog,就需要在my.cnf的[mysqld_safe]段配置syslog.但是修改了以后发现服务启动失败,而且没有日志,经过几番调查,还需要做如下的事情: 1. chroot MySQL的时候需要将/bin/logger拷贝到chroot jail中,因为logger是sysl