Net Core集成Exceptionless分布式日志功能以及全局异常过滤

# Net Core集成Exceptionless分布式日志功能以及全局异常过滤

相信很多朋友都看过我的上篇关于Exceptionless的简单入门教程[asp.Net Core免费开源分布式异常日志收集框架Exceptionless安装配置以及简单使用图文教程][https://www.cnblogs.com/yilezhu/p/9193723.html] 上篇文章只是简单的介绍了Exceptionless是什么?能做什么呢?以及怎么进行本地部署和异常提交的简单用法,而这篇文章将带你探讨一下Exceptionless的异常收集高级用法以及你熟悉的类似NLog的日志用法。

这篇文章有一部分内容翻译自官方文档,[点我阅读][https://github.com/exceptionless/Exceptionless.Net/wiki/Sending-Events] 英语好的可以自行阅读 。当然中间很多代码我都进行了重构,还有参考周旭龙的代码,进行了简单地封装,同时加入了为webapi加入异常全局过滤器进行异常日志的记录。希望对大家有所帮助。

本文地址:https://www.cnblogs.com/yilezhu/p/9339017.html
作者:依乐祝

手动发送错误

上篇文章介绍了,导入命名空间后,并使用如下代码就可以简单地提交异常日志:

try {
    throw new ApplicationException(Guid.NewGuid().ToString());
} catch (Exception ex) {
    ex.ToExceptionless().Submit();
}

发送附加信息

当然你还可以为发送的事件添加额外的标记信息,比如坐标,标签,以及其他的用户相关的信息等等

try {
    throw new ApplicationException("Unable to create order from quote.");
} catch (Exception ex) {
    ex.ToExceptionless()
        // 设置一个ReferenceId方便查找
        .SetReferenceId(Guid.NewGuid().ToString("N"))
        // 添加一个不包含CreditCardNumber属性的对象信息
        .AddObject(order, "Order", excludedPropertyNames: new [] { "CreditCardNumber" }, maxDepth: 2)
        // 设置一个名为"Quote"的编号
        .SetProperty("Quote", 123)
        // 添加一个名为“Order”的标签
        .AddTags("Order")
        //  标记为关键异常
        .MarkAsCritical()
        // 设置一个位置坐标
        .SetGeo(43.595089, -88.444602)
        // 在你的系统中设置userid并提供一个有好的名字,俗称昵称
        .SetUserIdentity(user.Id, user.FullName)
        // 为异常信息添加一些用户描述信息.
        .SetUserDescription(user.EmailAddress, "I tried creating an order from my saved quote.")
        // 提交.
        .Submit();
}

统一修改未处理的异常报告

你可以在通过SubmittingEvent 事件设置全局的忽略异常信息添加一些自定义信息等等


 #region Exceptionless配置
            ExceptionlessClient.Default.Configuration.ApiKey = exceptionlessOptions.Value.ApiKey;
            ExceptionlessClient.Default.Configuration.ServerUrl = exceptionlessOptions.Value.ServerUrl;
            ExceptionlessClient.Default.SubmittingEvent += OnSubmittingEvent;
            app.UseExceptionless();
#endregion

 /// <summary>
        /// 全局配置Exceptionless
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void OnSubmittingEvent(object sender, EventSubmittingEventArgs e)
        {
            // 只处理未处理的异常
            if (!e.IsUnhandledError)
                return;

            // 忽略404错误
            if (e.Event.IsNotFound())
            {
                e.Cancel = true;
                return;
            }

            // 忽略没有错误体的错误
            var error = e.Event.GetError();
            if (error == null)
                return;
            // 忽略 401 (Unauthorized) 和 请求验证的错误.
            if (error.Code == "401" || error.Type == "System.Web.HttpRequestValidationException")
            {
                e.Cancel = true;
                return;
            }
            // Ignore any exceptions that were not thrown by our code.
            var handledNamespaces = new List<string> { "Exceptionless" };
            if (!error.StackTrace.Select(s => s.DeclaringNamespace).Distinct().Any(ns => handledNamespaces.Any(ns.Contains)))
            {
                e.Cancel = true;
                return;
            }
            // 添加附加信息.
            //e.Event.AddObject(order, "Order", excludedPropertyNames: new[] { "CreditCardNumber" }, maxDepth: 2);
            e.Event.Tags.Add("MunicipalPublicCenter.BusinessApi");
            e.Event.MarkAsCritical();
            //e.Event.SetUserIdentity();
        }

配合使用 NLog 或 Log4Net

有时候,程序中需要对日志信息做非常详细的记录,比如在开发阶段。这个时候可以配合 log4net 或者 nlog 来联合使用 exceptionless,详细可以查看这个官方的 [示例][https://github.com/exceptionless/Exceptionless.Net/tree/master/samples/Exceptionless.SampleConsole]

如果你的程序中有在短时间内生成大量日志的情况,比如一分钟产生上千的日志。这个时候你需要使用内存存储(in-memory store)事件,这样客户端就不会将事件系列化的磁盘,所以会快很多。这样就可以使用Log4net 或者 Nlog来将一些事件存储到磁盘,另外 Exceptionless 事件存储到内存当中。

Exceptionless 日志记录的封装

  1. 首先简单地封装一个ILoggerHelper接口

    /// <summary>
        /// lzhu
        /// 2018.7.19
        /// 日志接口
        /// </summary>
        public interface ILoggerHelper
        {
            /// <summary>
            /// 记录trace日志
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            void Trace(string source, string message, params string[] args);
            /// <summary>
            /// 记录debug信息
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            void Debug(string source, string message, params string[] args);
            /// <summary>
            /// 记录信息
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            void Info(string source, string message, params string[] args);
            /// <summary>
            /// 记录警告日志
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            void Warn(string source, string message, params string[] args);
            /// <summary>
            /// 记录错误日志
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            void Error(string source, string message, params string[] args);
        }
    
  1. 既然有了接口,那么当然得实现它了

    /// <summary>
        /// lzhu
        /// 2018.7.19
        /// Exceptionless日志实现
        /// </summary>
        public class ExceptionlessLogger : ILoggerHelper
        {
            /// <summary>
            /// 记录trace日志
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">添加标记</param>
            public void Trace(string source,string message, params string[] args)
            {
                if (args != null && args.Length > 0)
                {
                    ExceptionlessClient.Default.CreateLog(source, message, LogLevel.Trace).AddTags(args).Submit();
    
                }
                else
                {
                    ExceptionlessClient.Default.SubmitLog(source, message, LogLevel.Trace);
                }
            }
            /// <summary>
            /// 记录debug信息
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            public void Debug(string source, string message, params string[] args)
            {
                if (args != null && args.Length > 0)
                {
                    ExceptionlessClient.Default.CreateLog(source, message, LogLevel.Debug).AddTags(args).Submit();
                }
                else
                {
                    ExceptionlessClient.Default.SubmitLog(source, message, LogLevel.Debug);
                }
            }
            /// <summary>
            /// 记录信息
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            public void Info(string source, string message, params string[] args)
            {
                if (args != null && args.Length > 0)
                {
                    ExceptionlessClient.Default.CreateLog(source, message, LogLevel.Info).AddTags(args).Submit();
                }
                else
                {
                    ExceptionlessClient.Default.SubmitLog(source, message, LogLevel.Info);
                }
            }
            /// <summary>
            /// 记录警告日志
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            public void Warn(string source, string message, params string[] args)
            {
                if (args != null && args.Length > 0)
                {
                    ExceptionlessClient.Default.CreateLog(source, message, LogLevel.Warn).AddTags(args).Submit();
                }
                else
                {
                    ExceptionlessClient.Default.SubmitLog(source, message, LogLevel.Warn);
                }
            }
            /// <summary>
            /// 记录错误日志
            /// </summary>
            /// <param name="source">信息来源</param>
            /// <param name="message">日志内容</param>
            /// <param name="args">标记</param>
            public void Error(string source, string message, params string[] args)
            {
                if (args != null && args.Length > 0)
                {
                    ExceptionlessClient.Default.CreateLog(source, message, LogLevel.Error).AddTags(args).Submit();
                }
                else
                {
                    ExceptionlessClient.Default.SubmitLog(source, message, LogLevel.Error);
                }
            }
        }
    
  2. 当然实现好了,可别忘了依赖注入哦

 //注入ExceptionlessLogger服务
            services.AddSingleton<ILoggerHelper, ExceptionlessLogger>();
  1. 这时候该写一个全局异常过滤器了

    /// <summary>
        /// lzhu
        /// 2018.7.19
        /// 定义全局过滤器
        /// </summary>
        public class GlobalExceptionFilter : IExceptionFilter
        {
            
            private readonly ILoggerHelper _loggerHelper;
            //构造函数注入ILoggerHelper
            public GlobalExceptionFilter(ILoggerHelper loggerHelper)
            {
                _loggerHelper = loggerHelper;
            }
    
            public void OnException(ExceptionContext filterContext)
            {
                _loggerHelper.Error(filterContext.Exception.TargetSite.GetType().FullName, filterContext.Exception.ToString(), MpcKeys.GlobalExceptionCommonTags, filterContext.Exception.GetType().FullName);
                var result = new BaseResult()
                {
                    errcode = ResultCodeAddMsgKeys.CommonExceptionCode,//系统异常代码
                    errmsg= ResultCodeAddMsgKeys.CommonExceptionMsg,//系统异常信息
                };
                filterContext.Result = new ObjectResult(result);
                filterContext.HttpContext.Response.StatusCode = (int)HttpStatusCode.InternalServerError;
                filterContext.ExceptionHandled = true;
            }
        }
    
  2. 全局过滤器写好了,怎么让它生效呢,客观别急啊,上正菜

      //添加验证
                services.AddMvc(options=> {
                    options.Filters.Add<GlobalExceptionFilter>();
                }).AddFluentValidation();
    
  3. 哈哈,没什么说的了,代码都已经写好了,剩下的就是上代码测试结果了。我这里只是简单地api测试下,万能的ValuesController登场:

     // GET api/values/5
            [HttpGet("{id}")]
            public ActionResult<string> Get(int id)
            {
                //try
                //{
                throw new Exception($"测试抛出的异常{id}");
                //}
                //catch (Exception ex)
                //{
                //    ex.ToExceptionless().Submit();
                //}
                //return "Unknown Error!";
    
            }
    

    这里是直接抛出异常,不进行trycatch,这时候异常会被全局过滤器捕获,然后放到Exceptionless的Log里面,别问我为什么会在log里面,因为我全局过滤器代码里面已经写明了,不明白的回去看代码,然后看接口调用的实现方法。下面上结果:

    1532012181451

点进去,看看详细信息:

1532012249948
  1. 再测试下使用try catch捕获的异常处理,这时候异常信息会被提交到Exception这个里面。直接上代码吧

    // GET api/values/5
            [HttpGet("{id}")]
            public ActionResult<string> Get(int id)
            {
                try
                {
                    throw new Exception($"测试抛出的异常{id}");
                }
                catch (Exception ex)
                {
                    ex.ToExceptionless().Submit();
                }
                return "Unknown Error!";
    
            }
    

    到exceptionless里面看看不活的异常吧。打字很累直接上图吧

    1532012444443

    点进去看看详细信息,有三个tab,下面之粘贴一个图片了:

1532012508157

最后,源码就不上了,因为上面代码很清楚了

总结

本文没有对Exceptionless进行过多地介绍,因为博主的[上篇文章][https://www.cnblogs.com/yilezhu/p/9193723.html] 已经进行了详细的介绍。直接切入正题,先对官方高级用法进行了简单地翻译。然后对Exceptionless Log这个eventtype进行了简单地封装,让你可以像使用NLog一样很爽的使用Exceptionless。最后通过一个asp.net core web api的项目进行了演示,在全局过滤器中利用封装的Log方法进行全局异常的捕获。希望对大家使用Exceptionless有所帮助。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,701评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,649评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,037评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,994评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,018评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,796评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,481评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,370评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,868评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,014评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,153评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,832评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,494评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,039评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,156评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,437评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,131评论 2 356

推荐阅读更多精彩内容