UPDATE 4/12/2013: Tool included
Download WSDL Flattener Tool
I’m not including source because it’s messy (I prefer to cleanup my code before sharing…) but feel free to decompile with ILSpy if you don’t trust me.
UPDATE 8/11/2015: Command line tool included
WSDL Flattener Command Line Utility
Somehow I forgot to save my update – thanks Luke for pointing it out!
WCF is great – its flexible, standards compliant, super easy to work with, and just altogether nice. That is, until you try to use it with, say, Oracle products. WCF generates so-called modular wsdl files, with separate xsd files for types, and even sometimes multiple wsdl files. Not technically in violation of any standards, but kind-of pushing the limits.
There are plenty of articles out there about modifying a WCF service to generate a flat WSDL. Also if you upgrade to .NET 4.5, you can use the URL parameter ?singleWsdl to get a flattened WSDL.
But what if you don’t have control of the web service? What if you cannot upgrade or modify the code?
For some odd reason, Microsoft added the internal ability to generate a single WSDL but only exposed it through the WCF Service implementation. They didn’t add it to svcutil.exe, and they didn’t expose it in the Framework… publicly that is *evil grin*
Turns out, in the System.ServiceModel.Description namespace, there is an internal static class called “WsdlHelper”, and this class has a public static Method called “GetSingleWsdl” that takes a MetadataSet for a parameter, and returns a System.Web.Services.Description.ServiceDescription object representing the Wsdl as a single entity. All you have to do is call .Write(filename) and you have your file.
“But I don’t see any WsdlHelper class!!!!!!”
It’s internal – you have to use reflection to invoke it.
First, you have to load the metadata from the existing WCF service into a MetadataSet using a MetadataExchangeClient. (Hint: you will probably need to use a custom binding). Then you run that set into something like this:
            var asy = System.Reflection.Assembly.GetAssembly(typeof(WsdlExporter));
            Type t = asy.GetType("System.ServiceModel.Description.WsdlHelper", true);
            var method = t.GetMethod("GetSingleWsdl", System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Static);
            object retVal = method.Invoke(null, new object[] { metadataSet });
            //cast retVal into System.Web.Services.Description.ServiceDescription, and call the .Write() method. All done
Using this, I was able to create a tool that takes a WCF url, and allows you to save a single .wsdl file. You want the tool? Later perhaps. For now put your thinking cap and Google gloves on, and write your own.
 
								
You cruel man, making us try & work it out for ourselves! 😛
I’ve tried a number of ways to flatten the WSDL from a WCF service I’m hosting under SharePoint.
But, the custom factory solutions don’t seem to work (on the host side), and none of the WSDL file tools can handle multiple WSDLs (on the client side).
Hosting under SharePoint seems to complicate matters, but as it’s processing SharePoint data, it seems the easiest way.
So… how much begging will it take for you to provide the tool? 😉
hey dave, post a comment using your real email (unless the one you used is real) and I’ll send it your way
Thanks for making this tool available.
Tried a few approaches that didn’t work that well but this does, thanks.
Can this run from command line?
I would like to run this as a post deploy task and generate the wsdl file. Is there a way to run this as a command line tool?
In the current version no, but it could be easily turned into a command line tool. Let me see if I can find my source code and I’ll hook you up
Turns out I had already written a command line version! Check the post – I’ve updated it to include the command line version
There is still no link there for the command-line tool, but fortunately you left browsing on in the content folders, so I found it here…
http://www.shulerent.com/wp-content/uploads/2013/03/flatwcf.zip
Many thanks for this, Jason. Very useful.
your tool is life saver dude. Thank you very much.
#Very useful tool .Thanks for your code.
Thanks Luke Webber for providing an updated version of the link!
Jason – looks like the links you’d posted aren’t working anymore. Perhaps you could put the code up on github or something – hopefully you’ve had time to clean it up a bit 🙂