webapi框架搭建系列博客

前言

  本篇讲怎么在前几篇已经创建好的项目里加上日志处理机制,我们采用Log4net技术。跟多的log4net技术的细节请查阅log4net的官网。

log4net官网:http://logging.apache.org/log4net/

步骤

引用log4net包

在nuget里引入包

此包没有任何的依赖项

编写日志处理代码

  现在项目已经用到了autofac的依赖注入技术(查看webapi框架搭建-依赖注入之autofac),如果不用依赖注入,log4net的用法类似下面的代码using Systemusing System.Web.Http;

  1. using log4net;
  2.  
  3. /// <summary>
  4. /// 日志处理测试接口,使用log4net
  5. /// </summary>
  6. namespace webapi.example
  7. {
  8. public class LogTestController : ApiController
  9. {
  10. public IHttpActionResult Get()
  11. {
            // 通过LogManager的静态方法GetLogger生成一个Ilog对象
  12. ILog log = LogManager.GetLogger(typeof(LogTestController));
           // 下面是日志处理
  13. log.Debug("测试debug", new Exception("debug异常"));
  14. log.Info("测试Info", new Exception("Info异常"));
  15. log.Warn("测试Warn", new Exception("Warn异常"));
  16. log.Error("测试Error", new Exception("Error异常"));
  17. log.Fatal("测试Fatal", new Exception("Fatal异常"));
  18. return Ok("已经写入日志");
  19. }
  20. }
  21. }

  

现在我们用autofac的方式去解耦Ilog对象的创建。

编辑日志测试控制器

  1. using System;
  2. using System.Web.Http;
  3. using log4net;
  4.  
  5. /// <summary>
  6. /// 日志处理测试接口,使用log4net
  7. /// </summary>
  8. namespace webapi.example
  9. {
  10. public class LogTestController : ApiController
  11. {
  12. private ILog _log;
  13.  
  14. public LogTestController(ILog log)
  15. {
  16. _log = log;
  17. }
  18. public IHttpActionResult Get()
  19. {
  20. _log.Debug("测试debug",new Exception("debug异常"));
  21. _log.Info("测试Info", new Exception("Info异常"));
  22. _log.Warn("测试Warn", new Exception("Warn异常"));
  23. _log.Error("测试Error", new Exception("Error异常"));
  24. _log.Fatal("测试Fatal", new Exception("Fatal异常"));
  25. return Ok("已经写入日志");
  26. }
  27. }
  28. }

autofac注册ILog组件

我们用autofac的Module方式去注册log4net组件

1、编写autofac 的module代码

  1. using System.Linq;
  2. using System.Reflection;
  3. using Autofac.Core;
  4. using log4net;
  5. using Module = Autofac.Module;
  6.  
  7. namespace webapi.AutoFac.Modules
  8. {
  9. public class LoggingModule:Module
  10. {
  11. private static void InjectLoggerProperties(object instance)
  12. {
  13. var instanceType = instance.GetType();
  14.  
  15. // Get all the injectable properties to set.
  16. // If you wanted to ensure the properties were only UNSET properties,
  17. // here's where you'd do it.
  18. var properties = instanceType
  19. .GetProperties(BindingFlags.Public | BindingFlags.Instance)
  20. .Where(p => p.PropertyType == typeof(ILog) && p.CanWrite && p.GetIndexParameters().Length == 0);
  21.  
  22. // Set the properties located.
  23. foreach (var propToSet in properties)
  24. {
  25. propToSet.SetValue(instance, LogManager.GetLogger(instanceType), null);
  26. }
  27. }
  28.  
  29. private static void OnComponentPreparing(object sender, PreparingEventArgs e)
  30. {
  31. e.Parameters = e.Parameters.Union(
  32. new[]
  33. {
  34. new ResolvedParameter(
  35. (p, i) => p.ParameterType == typeof(ILog),
  36. (p, i) => LogManager.GetLogger(p.Member.DeclaringType)
  37. ),
  38. });
  39. }
  40.  
  41. protected override void AttachToComponentRegistration(IComponentRegistry componentRegistry, IComponentRegistration registration)
  42. {
  43. // Handle constructor parameters.
  44. registration.Preparing += OnComponentPreparing;
  45.  
  46. // Handle properties.
  47. registration.Activated += (sender, e) => InjectLoggerProperties(e.Instance);
  48. }
  49. }
  50. }

  这段代码为autofac官网里提供的,参考地址:http://autofaccn.readthedocs.io/en/latest/examples/log4net.html?highlight=module

2、注册module

  1. using System.Reflection;
  2. using Autofac;
  3. using Autofac.Integration.WebApi;
  4. using webapi.AutoFac.Modules;
  5. using webapi.example;
  6.  
  7. namespace webapi.AutoFac
  8. {
  9. public static class ContainerBuilerCommon
  10. {
  11. public static IContainer GetWebApiContainer()
  12. {
  13. var builder = new ContainerBuilder();
  14. // 注册当前程序集里的所有webapi控制器
  15. builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
  16.  
  17. #region 注册modules
  18. builder.RegisterModule<LoggingModule>();
  19. #endregion
  20.  
  21. #region 注册组件,如果项目比较大可以从此方法里单独移出
  22. //这里写注册组件的代码
  23. #region 测试
  24. builder.RegisterType<Chinese>().As<People>();
  25. #endregion
  26. #endregion
  27.  
  28. return builder.Build();
  29. }
  30. }
  31. }

  上面的红色代码即是增加的注册module的代码:builder.RegisterModule<LoggingModule>(),Startup.cs文件的代码不变,附上前几篇Startup.cs的代码如下

  1. using Microsoft.Owin;
  2. using Owin;
  3. using System.Threading.Tasks;
  4. using System.Web.Http;
  5. using Autofac.Integration.WebApi;
  6. using webapi.AutoFac;
  7. using webapi.Configs;
  8.  
  9. // 标识webapiOwin.Startup类为owin的启动类,也可写在AssemblyInfo.cs文件里
  10. [assembly: OwinStartup(typeof(webapi.Owin.Startup))]
  11.  
  12. namespace webapi.Owin
  13. {
  14. public class Startup
  15. {
  16. /// <summary>
  17. /// owin的http请求管道配置函数
  18. /// </summary>
  19. /// <param name="app"></param>
  20. public void Configuration(IAppBuilder app)
  21. {
  22.  
  23. #region 写在前面的配置
  24. // 获取webapi的配置
  25. var config = WebApiConfig.OwinWebApiConfiguration(new HttpConfiguration());
  26. // 获取webapi的依赖注入容器
  27. var container = ContainerBuilerCommon.GetWebApiContainer();
  28. // 配置webapi的依赖注入
  29. config.DependencyResolver = new AutofacWebApiDependencyResolver(container);
  30. #endregion
  31.  
  32. #region owin组件注册(要注意顺序)
  33.  
  34. app.UseAutofacMiddleware(container);// 先注册autofac组件,需要依赖注入功能的组件在此后注册
  35. app.UseAutofacWebApi(config);//注册AutofacWebApi组件后再注册WebApi组件
  36. app.UseWebApi(config);
  37. #endregion
  38. }
  39. }
  40. }

  

  现在编译程序后用postman工具的get方法访问接口:http://localhost:101/api/LogTest,程序是运行正常的 。但现在日志即没有写入到某个文件、数据库或是发送到邮件里,也没有输出到控制台上。这就是log4net的设计的好处,在程序里你只管做日志的处理,如调用ILog的Debug()、Info()、Warn()、Error()、Fatal(),至于日志是由什么机制去处理(如写入文件,写入数据库等)是由另一个流程来控制,即log4net的配置文件。如果程序里没有log4net的配置文件,程序也能正常运行。下面说怎么去配置log4net。

配置Log4net

新建配置文件

配置文件为Log4net.config,代码如下

  1. <log4net>
  2. <!--记录所有的完整日志-->
  3. <appender name="AllLogFileAppender" type="log4net.Appender.RollingFileAppender">
  4. <!--或者是文件名或是文件夹(没有后缀)Gets or sets the path to the file that logging will be written to.,-->
  5. <file value="log/all/log_" />
  6. <!--是否总是写在一个文件里Gets or sets a value indicating whether to always log to the same file.-->
  7. <staticLogFileName value="false" />
  8. <!--Gets or sets a flag that indicates whether the file should be appended to or overwritten.-->
  9. <appendToFile value="true" />
  10. <!--可设置为Size、Date,即大小/日期超出一定范围后就新建一个日志文件-->
  11. <rollingStyle value="Date" />
  12. <!--一天最多保存多少Gets or sets the maximum number of backup files that are kept before the oldest is erased.-->
  13. <maxSizeRollBackups value="10" />
  14. <!--每个文件最大大小,单位可是MB,KBGets or sets the maximum size that the output file is allowed to reach before being rolled over to backup files.-->
  15. <maximumFileSize value="5MB" />
  16. <!--设置用来生产文件的日期格式Gets or sets the date pattern to be used for generating file names when rolling over on date.-->
  17. <datePattern value="yyyy-MM-dd'.log'"/>
  18. <!--日志输入的通用格式(日志的内容格式)-->
  19. <layout type="log4net.Layout.PatternLayout">
  20. <conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline" />
  21. </layout>
  22. <filter type="log4net.Filter.LevelRangeFilter,log4net">
  23. <levelMin value="DEBUG" />
  24. <levelMax value="FATAL" />
  25. </filter>
  26. </appender>
  27.  
  28. <!--记录错误日志,这些错误往往是一个程序bug或是要注意的-->
  29. <appender name="ErrorLogFileAppender" type="log4net.Appender.RollingFileAppender">
  30. <!--或者是文件名或是文件夹(没有后缀)Gets or sets the path to the file that logging will be written to.,-->
  31. <file value="log/error/error_" />
  32. <!--是否总是写在一个文件里Gets or sets a value indicating whether to always log to the same file.-->
  33. <staticLogFileName value="false" />
  34. <!--Gets or sets a flag that indicates whether the file should be appended to or overwritten.-->
  35. <appendToFile value="true" />
  36. <!--可设置为Size、Date,即大小/日期超出一定范围后就新建一个日志文件-->
  37. <rollingStyle value="Date" />
  38. <!--一天最多保存多少Gets or sets the maximum number of backup files that are kept before the oldest is erased.-->
  39. <maxSizeRollBackups value="10" />
  40. <!--每个文件最大大小,单位可是MB,KBGets or sets the maximum size that the output file is allowed to reach before being rolled over to backup files.-->
  41. <maximumFileSize value="5MB" />
  42. <!--设置用来生产文件的日期格式Gets or sets the date pattern to be used for generating file names when rolling over on date.-->
  43. <datePattern value="yyyy-MM-dd'.log'"/>
  44. <!--日志输入的通用格式(日志的内容格式)-->
  45. <layout type="log4net.Layout.PatternLayout">
  46. <conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline" />
  47. </layout>
  48. <filter type="log4net.Filter.LevelRangeFilter,log4net">
  49. <levelMin value="ERROR" />
  50. <levelMax value="FATAL" />
  51. </filter>
  52. </appender>
  53. <!--Set root logger level to DEBUG and its only appender to A1-->
  54. <root>
  55. <!--控制级别,由低到高: ALL|DEBUG|INFO|WARN|ERROR|FATAL|OFF-->
  56. <level value="ALL" />
  57. <appender-ref ref="AllLogFileAppender" />
  58. <appender-ref ref="ErrorLogFileAppender" />
  59. </root>
  60. </log4net>

  代码里已经做了简单的配置描述,log4net的配置教程后续会补充,博友们可参考官网:http://logging.apache.org/log4net/release/manual/configuration.html,或是自行百度查找它的用法。我这里只是做了简单的配置,所有的日志都用RollingFileAppender去写入到网站的子目录log/all文件里,并将程序错误级别的日志单独存储在log/error里,方便程序出错时快速定位。

代码配置

  上面已经有了Log4net的配置文件,但程序里如何去读这个文件呢?有很多的方法,在官网这篇文章里已经说的很明白了:http://logging.apache.org/log4net/release/manual/configuration.html。

我采用程序集特性的方式去配置,在项目的AssemblyInfo.cs里加入如下代码

  1. // log4net的配置文件,参考:http://logging.apache.org/log4net/release/manual/configuration.html
  2. [assembly: XmlConfigurator(Watch = true, ConfigFile = "Log4Net.config")]

  ConfigFile的值为我们Log4net配置文件的文件名,Watch为true时会实时监听Log4Net.config文件的变化并应用(程序不须再编译)

最终测试结果

用postman工具的get方法访问测试接口http://localhost:101/api/LogTest,log4net会在项目主目录下生成日志文件

log_2018-01-12.log和error_2018-01-12.log文件里的内容如下:

webapi框架搭建-日志管理log4net的更多相关文章

  1. webapi框架搭建系列博客

    webapi框架搭建系列博客 webapi框架搭建-创建项目(一) webapi框架搭建-创建项目(二)-以iis为部署环境的配置 webapi框架搭建-创建项目(三)-webapi owin web ...

  2. webapi框架搭建-webapi异常处理

    webapi框架搭建系列博客 前言 上一篇我们已经完成了项目的日志管理,在项目开发中日志会经常记录程序中的异常,供后续问题排查使用.本篇讲如何在webapi里加入异常处理机制. 目的和原则 1.程序任 ...

  3. webapi框架搭建-创建项目(二)-以iis为部署环境的配置

    上篇:webapi快速框架搭建-创建项目(一) 在"创建项目(一)"这一篇里已经创建了一个空的项目,但项目上什么都没有,本篇描述如何将webapi配置成部署在iis上. 步骤 用n ...

  4. webapi框架搭建-安全机制(四)-可配置的基于角色的权限控制

    webapi框架搭建系列博客 在上一篇的webapi框架搭建-安全机制(三)-简单的基于角色的权限控制,某个角色拥有哪些接口的权限是用硬编码的方式写在接口上的,如RBAuthorize(Roles = ...

  5. webapi框架搭建-安全机制(三)-简单的基于角色的权限控制

    webapi框架搭建系列博客 上一篇已经完成了“身份验证”,如果只是想简单的实现基于角色的权限管理,我们基本上不用写代码,微软已经提供了authorize特性,直接用就行. Authorize特性的使 ...

  6. webapi框架搭建-数据访问ef code first

    webapi框架搭建系列博客 为什么用ef? 我相信很多博友和我一样都有这种“选择困难症”,我曾经有,现在也有,这是技术人的一个通病——总想用“更完美”的方式去实现,导致在技术选择上犹豫不决,或总是推 ...

  7. webapi框架搭建-依赖注入之autofac

    前言 c#的依赖注入框架有unity.autofac,两个博主都用过,感觉unity比较简单而autofac的功能相对更丰富(自然也更复杂一点),本篇将基于前几篇已经创建好的webapi项目,引入au ...

  8. webapi框架搭建-创建项目(三)-webapi owin

    上一篇:创建项目(二) 在上一篇里,我们已经创建好了webapi应用,并已经部署到iis里,本篇讲如何用owin自宿主或是iis宿主来部署webapi应用. owin介绍 传统的asp.net网站只能 ...

  9. webapi框架搭建-安全机制(二)-身份验证

    webapi框架搭建系列博客 身份验证(authentication)的责任是识别出http请求者的身份,除此之外尽量不要管其它的事.webapi的authentication我用authentica ...

随机推荐

  1. LeetCode :My solution N-Queens

    N-Queens Total Accepted: 15603 Total Submissions: 60198My Submissions The n-queens puzzle is the pro ...

  2. Android-Dalvik指令集

    转载请标明出处:http://blog.csdn.net/goldenfish1919/article/details/40821777 原文:http://pallergabor.uw.hu/and ...

  3. UVA - 11396 Claw Decomposition(二分图染色)

    题目大意:给你一张无向图,每一个点的度数都是3. 你的任务是推断是否能把它分解成若干个爪(每条边仅仅能属于一个爪) 解题思路:二分图染色裸题.能够得出:爪的中心点和旁边的三个点的颜色是不一样的 #in ...

  4. 利用reverse索引优化like语句的方法详解

    在有一些情况下,开发同学经常使用like去实现一些业务需求,当使用like时,我们都知道使用like 前%(like '%111')这种情况是无法使用索引的,那么如何优化此类的SQL呢,下面是一个案例 ...

  5. 以List为例浅谈C#的学习方法

    前言:关于学习方法的讨论其实是个比较模糊的概念,对于List的介绍的资料其实已经很多了,但是一般是介绍List本身,我打算分享的是,以温故List为例,来获取新知识的这么一个过程.这里的新知识也不是什 ...

  6. CentOS 7 学习(四)Git配置(一)

    CentOS 7 学习(四)Git配置(一) 1.对于版本管理系统,目前常用的是Subverion和Git,Subversion是集中式版本管理系统中最好的,所有人的代码都要提交到服务器上,如果要知道 ...

  7. Node: 如何控制子进程的输出

    大家知道,在一个node程序中,如果当前进程想要生成一个子进程,它可以调用child_process模块的spawn方法.spawn方法签名如下: child_process.spawn(comman ...

  8. Libevent源码分析 (1) hello-world

    Libevent源码分析 (1) hello-world ⑨月份接触了久闻大名的libevent,当时想读读源码,可是由于事情比较多一直没有时间,现在手头的东西基本告一段落了,我准备读读libeven ...

  9. xml文件解析(使用解析器)

    一.Xml解析,解析xml并封装到list中的javabean中 OM是用与平台和语言无关的方式表示XML文档的官方W3C标准.DOM是以层次结构组织的节点或信息片断的集合.这个层次结构允许开发人员在 ...

  10. useradd 命令 及 相关文件解读

    翻译自 man useradd 名称: 创建新用户或更新默认的新用户信息 快照:useradd -D 描述: 在不加-D参数调用时,useradd命令使用指定的参数和系统的默认值来创建新账户. 取决于 ...