A short story of something new we've learned about how exactly WCF serializes the data that is sent over the wire.
Introduction
Before WCF, the default way to serialize objects to XML was to use the XmlSerializer. It works and of course has its shortcomings when it comes to serialization of complex types and collections.
When WCF was introduced, a couple of new serializers were brought into the Base Class Library, including the DataContractSerializer and NetDataContractSerializer. New serializers mean new features, comparision charts are available (e.g. this one by Sebasian Krysmanski).
If you, like we did, live in a simple world where WCF just uses the new set of serializers, read on.
Usually, where both the service and the client are .NET apps, web services can be designed by writing down C# interfaces and data models first. I'd call this common approach the code first approach - you share a code between the service and the client:
// common, shared between the service and the client [DataContract] public class DataModel { [DataMember] public string Whatever { get; set; } } public interface IServiceContract { void DoWork( DataModel model ); }Then, the server just implements the interface and exposes the service using a service host (IIS/self-host):
[ServiceBehavior(...)] public class ServiceImpl : IServiceContract { ... }and the client uses the ChannelFactory or the ClientBase to easily have the proxy based on the same interface.
A case of a unit test
Working on a complex integration project involving interoperable calls between a .NET client and a Java WebService, we were faced with an approach we haven't followed often before. Instead of the usual code first approach, we were given a couple of *.WSDL/*.XSD files, which makes a valid model first approach. Given these, you use an automated tool like the xsd.exe or the newer svcutil.exe to automatically create code from models:
svcutil.exe /syncOnly /n:*,Test *.wsdl *.xsd
This approach was used, the code has been generated and someone tried to write a unit test to make sure the request body is correctly serialized so that it meets the XML structure expected at the Java's side. The unit test code first used the DataContractSerializer as we believed this is what WCF uses under the hood. The test code was basically something like:
DoWorkRequest request = new DoWorkRequest(); request.model = var serializer = new DataContractSerializer(); var ms = new MemoryStream(); serializer.WriteObject( ms, request ); var requestXML = Encoding.UTF8.GetString( ms.ToArray() ); Assert.....
As it turned out, the serializer's output was something like
<DoWorkRequest ....while the server's expectation was
<DoWork ....(note the Request suffix missing from the root's name)
The test was obviously failing. We started an investigation.
What is really going on under the hood
After a couple of different trials and errors involving other serializers and their settings, we've found something that we never manually put into the code in the code-first approach. It was the MessageContractAttribute put over the request class by the generator:
[MessageContractAttribute(WrapperName="DoWork"....] public class DoWorkRequest {Things started to get interesting, it looks like there's yet another serializer, not mentioned that much, that obviously respects this attribute. Googling around reveals that there is indeed yet another layer used by WCF on top of different serializers to have even more control on how your data is serialized when a web service is called. This directly leads to the TypedMessageConverter class and code snippets people already posted (e.g this one by Stanislav Dvoychenko).
A solution, finally
The solution was to rewrite the unit test to actually use the TypedMessageConverter:
var request = new DoWorkRequest(...); var converter = TypedMessageConverter.Create( request.GetType(), "*", string.Empty, new XmlSerializerFormatAttribute()); var message = converter.ToMessage(request, MessageVersion.Soap11WSAddressing10); var writerSettings = new XmlWriterSettings(); writerSettings.OmitXmlDeclaration = true; var stream = new MemoryStream(); var writer = XmlWriter.Create(stream, writerSettings); message.WriteMessage(writer); writer.Flush(); var requsetXML = Encoding.UTF8.GetString(stream.ToArray());which gives the exact SOAP message that can be peeked using an HTTP debugger (you can possibly unpack the soap envelope it's wrapped into in your unit test code).
No comments:
Post a Comment