Sometimes I hear the question “Why not decouple the service from the caller by using generic interfaces?”
To better understand this question let’s have a look at the following pseudo code example:
BookingService
{
Flight[] getAvailableFlights(date flightDate, string destination);
bool bookFlight(int flightId);
}
The Web Service BookingService has two methods, which can be called by the clients and is described in an BookingService.wsdl file.
If we would like to add a new method for instance cancelFlight(int flightId) we have to change the caller of the service. In most cases this means rebuilding the client side proxy.
Why not using a generic interface like one of the following:
Generic parameter approach:
BookingService
{
Map getAvailableFlights(Map params);
Map bookFlight(Map params);
}
This approach uses only generic parameters but no generic service operations.
Generic service approach:
GenericService
{
Map execute(int commandId,Map params);
}
This approcach uses the common command pattern. It means the client sends a command id toghether with the required parameters to the service.
Dependent of the command id the service dispatches the call to the service implementation.
In order to add the service method cancelFlight we simply have to add a new command. The contract remains unchanged and the proxy hasn’t to be rebuild.
And even better the parameters can be processed in a generic way. That’s cool, isn’t it?
At first glance this approach looks promising. But at what price?
A common misunderstanding is that the Web Service Signature described in the WSDL file forms the service contract. This is only one option.
More precisely said the name of the service operation (or command id) in conjunction with the required in and out parameters form the contract.
It doesn’t matter where this information is stored.
What really happens when using generic interfaces is that the contract is shifted from a standarized to a custom layer.
That means the contract is not described in the BookingService.wsdl file anymore. Now the service and the caller need a proprietary way to describe and exchange the contract.
Moreover the type safety is lost because only generic types are used.
The command dispatching must be implemented. This is usually handled by the Web Service infrastructure for instance by using the SoapAction in the HTTP-Header.
Generic parameter processing is not a reason to use generic interfaces. If necessary reflection can be used to process typed parameters in a generic way, but without loosing type safety.
I think the drawbacks of generic interfaces outweight the benefits.
What do you think?
I agree with you.
I was on a project that followed that approach and it was a terrible mess.
Now I prefer fixed "versioned" endpoints, not generic ones…