WSDL.exe - generate interface as well as concrete class for easy fake/mocks later

為{幸葍}努か 提交于 2019-12-05 04:37:26

Right, prompted by Philip's answer I set off on one, and may have come up with a working solution. Using WSDL.exe I generated the interfaces (using the /si switch) and normal proxy classes, and added both to my business project.

I then created a new class which inherits from the concrete class AND implements the interface. This little class contained basically no code, as the inherited concrete members provided an implicit implementation of the interface members. The code compiled first time and I've been able to substitute this little "shim" (?adaptor?) class into my integration tests, and execute calls against the live 3rd party server.

I can now create other classes (mocks or fakes) which implement the interface, and substitue them instead of the "shim" class.

Edit: OK, I've worked a little further on this, and barring a few complications it's working.

The first significant issue is that the proxy classes are still marked "internal", so the derived (adaptor/shim) class has to be internal too. This isn't a problem if you put a Factory class into your business project/assembly which new's up the proxy classes, and returns them as the interface.

The second issue I found was that we were setting the URL and timeout properties of the webserice explicitly, but these are not included in the interface instead they are inherited from System.Web.Services.Protocols.WebClientProtocol via SoapHttpClientProtocol. Again, I dealt with this in the factory, as it's an implementation detail that I'm happy isn't in the interface.

Edit: This is still working out prety well for me while testing and developing our Facade. Since getting the proxy behind an interface I've also created a logging decorator class, which is capturing loads of example calls for use debugging, and when the 3rd party server is offline.

I've written up what I did in a little more detail here: http://www.flowerchild.org.uk/archive/2010/09/21/mocking-or-faking-or-maybe-stubbing-a-wsdl-exe-soap.html

You can run wsdl with the /serverinterface or /si switch to get interfaces for each binding in the wsdl. This is meant to give you the server-side skeleton from a wsdl document, but the interfaces should get you on your way - if I understand the question correctly.

EDIT -- After reading the comment, I believe that I misunderstood the question - you want client interface/concrete so that you can code to contract not implementation. The /si switch will probably not give you what you want at all.

I'm not aware of any switch that can give you this behavior, as wsdl basically creates one of three things: 1) client proxy classes: 2) server abstract classes (for creating a server implementation) 3) server interfaces (again, for creating a server implementation - this is just iface instead of abstract class) I don't see any way to force the client proxy classes to implement interfaces (other than INotifyPropertyChanged on data types)

I've always found that these generated classes are too large to mock easily (too many methods / property).

Rather, create a Facade or Repository that implements just the calls you need, and passes back DTO style objects with just the properties you care about.

Write some simple integration tests against the real Webservice, and then mock the simpler Facade in the rest of the tests.

An added benefit of this approach is that you effectively get an anticorruption layer, so changes to the 3rd party will only impact a small area of your code.

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!