So I'm trying to use Enterprise Library in my WCF service to do some of the exception-related work for me.
My idea is to set up a "Custom Exception Handler" for say "NullReferenceException" and in the "Custom Exception Handler" create FaultException exception.
My understanding is that this "new" exception will then make it across the wire and I will catch it at the client.
Some code for better understanding:
WCF service:
[ServiceContract(Name="MyService", ConfigurationName="MyNamespace.MyService")]
[ExceptionShielding("PolicyName")]
public interface IMyService
{
[OperationContract]
[FaultContract(typeof(MyFaultContract))]
string Method(String Param);
}
public class MyService : IMyService
{
public string Method(String Param)
{
throw new NullReferenceException("code-created null message here");
}
}
Custom Exception Handler: (Enterprise library)
public class MyExceptionHandler : IExceptionHandler
{
public MyExceptionHandler(NameValueCollection collection) { }
public MyExceptionHandler() { }
public System.Exception HandleException(System.Exception exception,
Guid handlingInstanceId)
{
MyFaultContract details = new MyFaultContract();
if (exception is NullReferenceException)
{
details.ErrorCode = MyFaultCode.NullReferenceException;
details.OriginalMessage = exception.Message;
details.MyMessage = "Null Reference exception here!";
}
return new FaultException<MyFaultContract>(details);
}
}
App Config File that maps NullReferenceException to "Custom Exception Handler":
<exceptionPolicies>
<add name="PolicyName">
<exceptionTypes>
<add name="NullReferenceException" type="System.NullReferenceException, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"
postHandlingAction="ThrowNewException">
<exceptionHandlers>
<add type="MyExceptionHandler, MyApplication, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null"
name="MyExceptionHandler" />
</exceptionHandlers>
</add>
</exceptionTypes>
</add>
</exceptionPolicies>
And finally Client code that expects to catch this FaultException:
MyService.MyServiceClient client = new MyService.MyServiceClient();
client.Open();
try
{
string result = client.Method(string parameter);
}
catch (System.ServiceModel.FaultException<MyService.MyFaultContract> ex)
{
// where I think exception should end up
}
catch (System.ServiceModel.FaultException ex)
{
// general FaultException
}
But instead I get a ProtocolException, stating that reply action is required:
A reply message was received for operation 'Method' with action ''. However, your client code requires action 'http://tempuri.org/MyService/MethodResponse'.
What am I doing wrong? Is it possible to return (not explicitly throw) an FaultException with custom FaultContract in the "Custom Exception Handler"?
Any advice is appreciated.
Update: As you can see, post-handling action in the "Custom Exception Handler" is set to "ThrowNewException". If I change it to "NotifyRethrow" I don't het "ProtocolException" any more! Instead, client catches a regular "FaultException" (not custom-typed).
Now the question is why original custom-typed FaultException doesn't make it across the wire.
Update 2 One thing I forgot to mention is the fact that my WCF service is running within the ServiceHost, not under IIS. So basically I have a Windows service that creates a ServiceHost and exposes Service1 interface through this ServiceHost.
The reason why I think this may have something to do with this problem is that Enterprise Library claims it deals with exceptions throughout entire application, not only within service boundaries. Maybe this cause the exception to be thrown too late? Or not at the correct level?
Update 3
Thanks for the post!
You're right, the only reason why I'm messing with custom handler is because I want to set MyFaultCode value.
I tried your advice - having 2 handlers configured.
First is a custom handler and catches NullReference exception. It than throws a new exception - MyApplicationException, with MyFaultContract field.
Then I configured second handler - built-in "Fault Contract Exception Handler" that catches MyApplicationException, creates new FaultException and automatically maps MyFaultcontract from MyApplicationException to the newly created FaultException.
WCF client still catches generic FaultException, not the custom-contract one.