从支持AJAX的WCF服务返回错误详细信息

Awe*_*own 11 asp.net ajax wcf json

简短版本:在启用AJAX的WCF服务中抛出异常时,是否有建议的方法将错误详细信息返回给客户端(除了打开大门并发回所有异常详细信息)?

长版:

我有一个相对简单的启用AJAX的WCF服务,我使用默认服务代理从客户端调用.我在下面提供了代码片段,但我不相信代码本身有任何问题.

我的问题是,如果我在服务中抛出异常,返回给客户端的错误对象总是通用的:

{
    "ExceptionDetail":null,
    "ExceptionType":null,
    "Message":"The server was unable to process the request..."
    "StackTrace":null
}
Run Code Online (Sandbox Code Playgroud)

理想情况下,我想在客户端上显示不同的错误消息,具体取决于出错的地方.

一种选择是允许WCF故障中的异常,这将为我提供完整的堆栈跟踪和所有内容,但我很欣赏这方面的安全问题,这实际上是比我需要的更多信息.我能做的就是能够发回一个描述问题的字符串,但是我没有办法做到这一点.

我的服务代码:

[ServiceContract(Namespace = "MyNamespace")]
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
public class MyService
{
    [OperationContract]
    public void DoStuff(string param1, string etc)
    {
        //Do some stuff that maybe causes an exception
    }
}
Run Code Online (Sandbox Code Playgroud)

在客户端:

MyNamespace.MyService.DoStuff(
    param1,
    etc,
    function() { alert("success"); },
    HandleError);
Run Code Online (Sandbox Code Playgroud)

其中"HandleError"只是一个通用的错误处理方法,可以显示有关错误的详细信息.

Fed*_*rne 18

编辑:使用适当的自定义json错误处理程序更新了帖子

快速但非优先的方式.

<serviceDebug includeExceptionDetailInFaults="true"/>
Run Code Online (Sandbox Code Playgroud)

在您的服务行为中,将为您提供所需的所有详细信息.

好的方式

应用程序中的所有异常都将转换为a JsonError并使用a进行序列化DataContractJsonSerializer.的Exception.Message是直接使用.FaultExceptions提供FaultCode,其他异常被故障代码-1威胁为未知.

FaultException以HTTP状态代码400发送,其他异常是HTTP代码500 - 内部服务器错误.这不是必要的,因为故障代码可以用来判断它是否是未知错误.这在我的应用程序中很方便.

错误处理程序

internal class CustomErrorHandler : IErrorHandler
{
    public bool HandleError(Exception error)
    {
        //Tell the system that we handle all errors here.
        return true;
    }

    public void ProvideFault(Exception error, System.ServiceModel.Channels.MessageVersion version, ref System.ServiceModel.Channels.Message fault)
    {
        if (error is FaultException<int>)
        {
            FaultException<int> fe = (FaultException<int>)error;

            //Detail for the returned value
            int faultCode = fe.Detail;
            string cause = fe.Message;

            //The json serializable object
            JsonError msErrObject = new JsonError { Message = cause, FaultCode = faultCode };

            //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();

            // return custom error code, 400.
            rmp.StatusCode = System.Net.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);
        }
        else
        {
            //Arbitraty error
            JsonError msErrObject = new JsonError { Message = error.Message, FaultCode = -1};

            // create a fault message containing our FaultContract object
            fault = Message.CreateMessage(version, "", msErrObject, new DataContractJsonSerializer(msErrObject.GetType()));

            // tell WCF to use JSON encoding rather than default XML
            var wbf = new WebBodyFormatMessageProperty(WebContentFormat.Json);
            fault.Properties.Add(WebBodyFormatMessageProperty.Name, wbf);

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

            rmp.Headers[HttpResponseHeader.ContentType] = "application/json";
            rmp.Headers["jsonerror"] = "true";

            //Internal server error with exception mesasage as status.
            rmp.StatusCode = System.Net.HttpStatusCode.InternalServerError;
            rmp.StatusDescription = error.Message;

            fault.Properties.Add(HttpResponseMessageProperty.Name, rmp);
        }
    }

    #endregion
}
Run Code Online (Sandbox Code Playgroud)

Webbehaviour用于安装上面的错误处理程序

internal class AddErrorHandlerBehavior : WebHttpBehavior
{
    protected override void AddServerErrorHandlers(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
    {
        base.AddServerErrorHandlers(endpoint, endpointDispatcher);

        //Remove all other error handlers
        endpointDispatcher.ChannelDispatcher.ErrorHandlers.Clear();
        //Add our own
        endpointDispatcher.ChannelDispatcher.ErrorHandlers.Add(new CustomErrorHandler());
    }
}
Run Code Online (Sandbox Code Playgroud)

json错误数据合同

指定json错误格式.在此处添加属性以更改错误格式.

[DataContractFormat]
public class JsonError
{
    [DataMember]
    public string Message { get; set; }

    [DataMember]
    public int FaultCode { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

使用错误处理程序

自托管

ServiceHost wsHost = new ServiceHost(new Webservice1(), new Uri("http://localhost/json")); 

ServiceEndpoint wsEndpoint = wsHost.AddServiceEndpoint(typeof(IWebservice1), new WebHttpBinding(), string.Empty);

wsEndpoint.Behaviors.Add(new AddErrorHandlerBehavior());
Run Code Online (Sandbox Code Playgroud)

App.config中

<extensions>  
  <behaviorExtensions>  
    <add name="errorHandler"  
        type="WcfServiceLibrary1.ErrorHandlerElement, WcfServiceLibrary1" />  
  </behaviorExtensions>  
</extensions> 
Run Code Online (Sandbox Code Playgroud)