Служба WCF 4 с настраиваемым обработчиком ошибок для json возвращает 202 принято

У меня есть служба REST WCF 4, настроенная на использование json. Я хочу перехватывать исключения и возвращать код состояния HTTP 400 с сообщением об исключении в виде объекта json. Я следил за примерами в Интернете, чтобы реализовать свой собственный интерфейс IErrorHandler и IService для этого.

Например:

http://zamd.net/2008/07/08/error-handling-with-webhttpbinding-for-ajaxjson/

Возвращение сведений об ошибке из службы WCF с поддержкой AJAX

http://social.msdn.microsoft.com/Forums/en/wcf/thread/fb906fa1-8ce9-412e-a16a-5d4a2a0c2ac5

Однако, как и в этом посте

успешный обратный вызов jQuery вызывается с пустым ответ, когда метод WCF выдает исключение

Я получаю ответ 202 Accepted без данных из-за ошибки сериализации, когда я пытаюсь создать свою ошибку. Это регистрируется моей службой следующим образом:

2012-01-31 00:37:19,229 [8] DEBUG JsonWebScriptServiceHostFactory: creating service host
2012-01-31 00:37:19,292 [8] DEBUG JsonErrorHandler.ApplyDispatchBehavior: adding error handler
2012-01-31 00:43:06,995 [10] DEBUG ForemanSvc.GetSessionID
2012-01-31 00:43:39,292 [10] DEBUG ForemanSvc.GetProjects
2012-01-31 00:43:39,448 [10] DEBUG JsonErrorHandler.ProvideFault: creating fault
2012-01-31 00:43:39,635 [10] ERROR ForemanSvc exeption
Type: System.ServiceModel.CommunicationException
Message: Server returned an invalid SOAP Fault.  Please see InnerException for more details.
Source: System.ServiceModel
StackTrace:    
at System.ServiceModel.Channels.MessageFault.CreateFault(Message message, Int32 maxBufferSize)
at System.ServiceModel.Description.WebScriptEnablingBehavior.JsonErrorHandler.ProvideFault(Exception error, MessageVersion version, Message& fault)
at System.ServiceModel.Dispatcher.ErrorBehavior.ProvideFault(Exception e, FaultConverter faultConverter, ErrorHandlerFaultInfo& faultInfo)
at System.ServiceModel.Dispatcher.ErrorBehavior.ProvideMessageFaultCore(MessageRpc& rpc)
at System.ServiceModel.Dispatcher.ImmutableDispatchRuntime.ProcessMessage8(MessageRpc& rpc)
Type: System.Xml.XmlException
Message: Start element 'Fault' from namespace 'http://schemas.microsoft.com/ws/2005/05/envelope/none' expected. Found element 'root' from namespace ''.
Source: System.Runtime.Serialization
StackTrace:    
at System.Xml.XmlExceptionHelper.ThrowXmlException(XmlDictionaryReader reader, String res, String arg1, String arg2, String arg3)
at System.Xml.XmlExceptionHelper.ThrowStartElementExpected(XmlDictionaryReader reader, String localName, String ns)
at System.Xml.XmlDictionaryReader.ReadStartElement(XmlDictionaryString localName, XmlDictionaryString namespaceUri)
at System.ServiceModel.Channels.ReceivedFault.CreateFault12Driver(XmlDictionaryReader reader, Int32 maxBufferSize, EnvelopeVersion version)
at System.ServiceModel.Channels.MessageFault.CreateFault(Message message, Int32 maxBufferSize)

Из того поста непонятно, как это исправить. Я пробовал все виды - используя атрибут, используя поведение конечной точки, пробуя простой CreateMessage без форматирования json или дополнительной информации - похоже, ничего не работает. Кто-нибудь может помочь?

Вот несколько фрагментов кода - обработчик ошибок

public class JsonErrorHandler : IServiceBehavior, IErrorHandler
{
  private static readonly ILog log =
    LogManager.GetLogger(System.Reflection.MethodInfo.GetCurrentMethod().DeclaringType);


  public void Validate(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
  {
    //Dont do anything
  }

  public void AddBindingParameters(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase,
                                 Collection<ServiceEndpoint> endpoints, BindingParameterCollection bindingParameters)
  {
    //dont do anything
  }

  public void ApplyDispatchBehavior(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
  {
    log.IfDebug("JsonErrorHandler.ApplyDispatchBehavior: adding error handler");
    foreach (ChannelDispatcherBase dispatcherBase in serviceHostBase.ChannelDispatchers)
    {
      ChannelDispatcher channelDispatcher = dispatcherBase as ChannelDispatcher; 
      if (channelDispatcher != null)
      {
        channelDispatcher.ErrorHandlers.Add(this);
      } 
    }
  }


  public bool HandleError(Exception error)
  {
    log.IfError("ForemanSvc exeption", error);
    //Tell the system that we handle all errors here.
    return true;
  }

  public void ProvideFault(Exception error, MessageVersion version, ref Message fault)
  {
    log.IfDebug("JsonErrorHandler.ProvideFault: creating fault");

    JsonError msErrObject =
    new JsonError
    {
      Message = error.Message,
      Source = error.Source,
      Detail = error.InnerException != null ? error.InnerException.Message : null
    };


    //The fault to be returned
    fault = Message.CreateMessage(version, "", msErrObject, new DataContractJsonSerializer(msErrObject.GetType()));

    // tell WCF to use JSON encoding rather than default XML
    WebBodyFormatMessageProperty wbf = new WebBodyFormatMessageProperty(WebContentFormat.Json);

    // Add the formatter to the fault
    fault.Properties.Add(WebBodyFormatMessageProperty.Name, wbf);

    //Modify response
    HttpResponseMessageProperty rmp = new HttpResponseMessageProperty();

    if (error is SecurityException &&
      (error.Message == "Session expired" || error.Message == "Authentication ticket expired"))
    {
      rmp.StatusCode = HttpStatusCode.Unauthorized;
      rmp.StatusDescription = "Unauthorized";
    }
    else
    {
      // return custom error code, 400.
      rmp.StatusCode = HttpStatusCode.BadRequest;
      rmp.StatusDescription = "Bad request";
    }

    //Mark the jsonerror and json content
    rmp.Headers[HttpResponseHeader.ContentType] = "application/json";
    rmp.Headers["jsonerror"] = "true";

    //Add to fault
    fault.Properties.Add(HttpResponseMessageProperty.Name, rmp);

  } 
}

и где я добавляю собственный обработчик ошибок для службы

public class JsonWebScriptServiceHostFactory : WebScriptServiceHostFactory
{
  private static readonly ILog log =
   LogManager.GetLogger(System.Reflection.MethodInfo.GetCurrentMethod().DeclaringType);

  protected override ServiceHost CreateServiceHost(Type serviceType, Uri[] baseAddresses)
  {
    log.IfDebug("JsonWebScriptServiceHostFactory: creating service host");
    ServiceHost host = base.CreateServiceHost(serviceType, baseAddresses);
    host.Description.Behaviors.Add(new JsonErrorHandler());
    return host;
  } 
}

и настраиваемая ошибка

[DataContract(Namespace = "VSS.Nighthawk.Foreman", Name = "JsonError")]
public class JsonError
{
  [DataMember]
  public string Message { get; set; }

  [DataMember]
  public string Source { get; set; }

  [DataMember]
  public string Detail { get; set; }
}

person Elspeth Jeet    schedule 03.02.2012    source источник
comment
Вы нашли решение?   -  person Asher    schedule 13.08.2013


Ответы (1)


Какие привязки и кодировщики вы используете и какие настройки для них настроили? Кроме того, какое поведение вы подключили? Если вы подключили WebScriptEnhibitedBehavior (поскольку WebScriptServiceHostFactory подключает его автоматически), ваша проблема может заключаться в том, что WSEB подключает собственный обработчик ошибок, который выполняет МНОГО тех же действий, что вы пытаетесь сделать.

Я бы также использовал Reflector и посмотрел на обработчик ошибок, встроенный в WebScriptEnhibitedBehavior, и посмотрел, что вы делаете по-другому, и можете ли вы делать что-то еще, чего вы еще не делаете. Это очень, очень сложная и проблемная область, изобилующая множеством тонкостей, поэтому вы, вероятно, не правильно поняли обработчик ошибок с первого раза.

Возможно, вам также придется вообще прекратить использование WebScriptEnhibitedBehavior (если вы его используете), поэтому просто убедитесь, что вы этого не делаете. Возможно, вам придется заново реализовать WebScriptEnhibitedBehavior самостоятельно, с нуля и подключить его с нуля в фабрике хоста службы, вместо того, чтобы подключать только обработчик ошибок JSON.

Надеюсь это поможет!

person krisragh MSFT    schedule 26.04.2012