ServiceStack - 如何禁用默认异常日志记录

时间:2013-09-18 09:22:21

标签: c# logging log4net servicestack

the ServiceStack documentation一致,我们有一个全局服务异常处理程序。文档说这个处理程序应该记录异常然后调用DtoUtils.HandleException,如下所示:

private object LogServiceException(object request, Exception exception)
{
    var message = string.Format("Here we make a custom message...");
    _logger.Error(message, exception);
    return DtoUtils.HandleException(this, request, exception);
 }

这会导致错误被记录两次,因为DTOUtils.HandleException also logs it是一种不太自定义的格式。是的,我更喜欢这个DTOUtils日志记录,不想只使用它。

如何在保留其余功能的同时关闭DTOUtils日志记录?没有人喜欢收到两倍的错误电子邮件。

2 个答案:

答案 0 :(得分:2)

我希望以下代码能解决您的问题。

基于文档New API, Custom Hooks, ServiceRunner

Fine grain error handling using the New API's ServiceRunner

AppHost.Configure中的

   LogManager.LogFactory = new ServiceStack.Logging.Support.Logging.ConsoleLogFactory();   

然后在AppHost类

         public override IServiceRunner<TRequest> CreateServiceRunner<TRequest>(ActionContext      actionContext)
          {
               return new MyServiceRunner<TRequest>(this, actionContext);
            }  

在ServiceRunner类

       public class MyServiceRunner<T> : ServiceRunner<T>
       {

            public override object HandleException(IRequestContext requestContext, T request, Exception ex)
              {
                  if ( isYourCondition ) 
                  {
                        ResponseStatus rs = new ResponseStatus("error1", "your_message");
                        // optionally you can add custom response errors
                             rs.Errors = new List<ResponseError>();
                             rs.Errors.Add(new ResponseError());
                             rs.Errors[0].ErrorCode = "more details 2";

                            // create an ErrorResponse with the ResponseStatus as parameter
                            var errorResponse = DtoUtils.CreateErrorResponse(request, ex, rs);
                             // log the error
                             Log.Error("your_message", ex);
                             return errorResponse;

                   }
                   else
                        return base.HandleException(requestContext, request, ex);
               }

            }

如果返回base.HandleException,则会在内部调用DtoUtils.HandleException。 您将在控制台中看到,只有一个日志错误。

在客户端中,如果您处理自定义错误的WebServiceException。

            catch (WebServiceException err)
            {
                if ( err.ResponseStatus.Errors != null)
               { // do something with err.ResponseStatus.Errors[0].ErrorCode; 
               }
            }

答案 1 :(得分:1)

请勿在记录错误时调用DtoUtils.HandleException。也不要拨打ServiceRunner.HandleException,它会调用DtoUtils.HandleException

致电DtoUtils.CreateErrorResponse进行回复(由DtoUtils.HandleException使用)。 ToResponseStatus帮助程序也位于DtoUtils

我的AppServiceRunner现在是这样的:

public class AppServiceRunner<T> : ServiceRunner<T>
{
    public AppServiceRunner(AppHost appHost, ActionContext actionContext)
        : base(appHost, actionContext)
    {
    }

    public override object HandleException(IRequestContext requestContext,
        T request, Exception ex)
    {   
        LogException(requestContext, request, ex);

        var responseStatus = ex.ToResponseStatus();
        return DtoUtils.CreateErrorResponse(request, ex, responseStatus);
    }

    private void LogException(IRequestContext requestContext, T request, Exception ex)
    {
        // since AppHost.CreateServiceRunner can be called before AppHost.Configure
        // don't get the logger in the constructor, only make it when it is needed
        var logger = MakeLogger();
        var requestType = typeof(T);
        var message = string.Format("Exception at URI:'{0}' on service {1} : {2}",
            requestContext.AbsoluteUri, requestType.Name, request.ToJson());

        logger.Error(message, ex);
    }

    private static ILog MakeLogger()
    {
        return LogManager.GetLogger(typeof(AppServiceRunner<T>));
    }
}

现在我收到的唯一服务错误是由此代码生成的错误。