JSR 181: a Java Simplification Request

JSR 181: A Java Simplification Request

By Michael Easter, OCI Senior Software Engineer

June 2007


Introduction

Historically, web services in Java have vexed developers with myriad specifications and complex solutions. Surrounded by legions of W3C specs, JSRs, and XML descriptors, these developers endured complexity while other Java communities (such as EJB3, Seam and Guice) celebrate the elegance of Java 5 and beyond. Moreover, with the success of the sleek REST school of web architecture, there is a fitting rallying cry: "hear our Java Simplification Request for web services!"

If only there were a world in which Java developers express web services in a compact manner, where tools assist with the mundane, "plumbing" details. This world is a far cry from the status quo, and certainly appeals.  For example, what if web services were as easy as:

  1. @WebService
  2. public class Concatenator {
  3.  
  4. @WebMethod
  5. public String concatenate(String a, String b) {
  6. return a + " " + b;
  7. }
  8. }

Welcome to the world of JSR 181: the above is a bona fide web serviceJSR 181, "Web Services Metadata for Java", rescues downtrodden web services developers with a simple, annotation-based development model for common web service tasks. Combined with a lightweight HTTP server in Java 6, JSR 181 enables rapid prototyping of web services (WS) and brings WS development into the modern era of Java.

This article has two goals:

It is assumed that the reader understands web services as an approach to distributed computing, and has a basic knowledge of web service constructs (e.g. WSDLSOAP). All examples require JDK 1.6.x.

Setting the Stage

There are two main eras of web services in Java: JAX-RPC and JAX-WS. Articles on both topics refer to several JSRs and their implicit relationships. This section provides a bird's eye view of the landscape.

JAX-RPC 1.x

JAX-RPC represents the first version of web services in Java. See the graphical overview below. Some key points:

JAX-RPC Version 1.1

JAX-WS 2.x

Though initially met with considerable excitement, there are problems with JAX-RPC 1.x: the implementation is complex and competing technologies/philosophies (e.g. Apache Axis, REST) offer simplicity that is seductive, despite being non-standard. Also, new specs and standards render some of the machinery of JAX-RPC 1.x (e.g. data binding) as obsolete. Most of all, since the release of JAX-RPC, the Java landscape itself underwent seismic shifts, particularly with respect to EJB.

Seismic Shifts

Initially, tools (e.g. XDoclet) assisted with the generation of deployment descriptors. Later, Java 5 brought the promise of annotations. In recent years, prominent authors railed against the complexity of J2EE and EJB. Alternative frameworks (e.g. Ruby on Rails) introduced simple, Zen-like philosophies, such as "prefer convention over configuration". All of these forces culimated in a shift against the explicit use of complex deployment time descriptors, and even XML configuration. EJB3, Seam and Guice clearly reflect this shift.

JSR 224

For version 2 of WS in Java, the technology was renamed as JAX-WS 2.0. Just as JSR 109 is the primary spec for JAX-RPC, JSR 224 "Java API for XML-Based Web Services", is the central document for JAX-WS. Some of the spec's stated goals pertain to evolving standards:

However, the following goals are more interesting:

See the graphical overview below. There are several themes here:

JAX-WS 2.0

JSR 181 within JAX-WS

The full JAX-WS spec (JSR 224) exceeds the scope of this article but, as noted, it defines the architecture for "WS Version 2." Clearly, it relates to JSR 181. What is the relationship? Some answers come from the introduction of the JSR 181 spec:

In short, JSR 181 is "Web Services Lite": all of the common WS functionality, without that complex aftertaste. In the parlance of OO design, JSR 181 is a facade design pattern: a simpler interface to the full-blown JSR 224 (see diagram below). It truly represents a "front-end" to JAX-WS as it requires the presence of JSRs 109, 224 and 175.

Mercifully, the JAX-WS 2.0 reference implementation contains all of the WS-oriented JSR solutions and is bundled in Java 6, along with a lightweight HTTP server. This allows us to experiment with JSR 181 out-of-the-box. 

JSR 181

A Closer Look

The stage is set for JSR 181: let's dig in.

The Development Model in JSR 181

In the JAX-WS world, a web service requires these artifacts:

JSR 181 requires only the SIB (i.e. the business logic). Annotations are used to generate the other artifacts. No doubt, annotation of business POJOs should be music to the ears of the long-suffering WS developers.

There are several programming models defined in JSR 181 (only the first is required by implementations):

The examples in this article use the "Start with Java" model.

Example 1: Concatenator

Before listing all of the available annotations in JSR 181, we'll start small, to get a view of the landscape of the JAX-WS tools. The first example (available here) is a trivial "string concatenation" service.

Server Side Concatenation

Let's begin with the service implementation bean in Java:

  1. package com.ociweb.demo;
  2.  
  3. import javax.jws.WebService;
  4. import javax.jws.WebMethod;
  5. import javax.xml.ws.Endpoint;
  6.  
  7. @WebService
  8. public class Concatenator {
  9.  
  10. @WebMethod
  11. public String concatenate(String a, String b) {
  12. return a + " " + b;
  13. }
  14. }

Note the annotations on this simple class, which clearly denote a web service and a "web method." Amazingly, we are done with development and can start testing the web service! The JSR 181 processor will take care of the rest. To prove it, we write a main method to publish the service to the HTTP server contained in Java 6:

  1. public static void main(String[] args) {
  2. // e.g.
  3. // publish.url = http://localhost:9998/concatservice
  4.  
  5. String publishUrl = System.getProperty("publish.url");
  6. if( publishUrl != null & publishUrl.length() > 0 ) {
  7. System.out.println("publishing service at: " + publishUrl);
  8. Concatenator concatenator = new Concatenator();
  9. Endpoint endpoint = Endpoint.publish(publishUrl,concatenator);
  10. } else {
  11. System.err.println("usage: Concatenator publishUrl");
  12. }
  13. }

The main method creates an instance of the SIB, and calls Endpoint.publish(). The first parameter, a URI, specifies the address and transport/protocol. The current javadoc states that for a 'http:' URI, it is assumed that the SOAP 1.1/HTTP binding is used. The publish method uses a default configuration to create the necessary server infrastructure for publication.

Ant takes care of the next steps (see build_server.xml for details):

Once the server is running, we can point a browser to http://localhost:9998/concatservice?wsdl:

Web Service

Presto! With stunning ease, the web service is available, and follows these JSR 224 conventions:

If we point the browser to the URI of the schema types:

URI of the Scheme Type

We see that the types follow a similar, logical convention. However, there's more: the JSR 181 processor generates these classes for us. In the example directory, you can find the generated Java classes Concatenate and ConcatenateResponse. Note how they are annotated with JAXB meta-data. These classes are entirely inline with the sophisticated JSR 224 conventions.

Client Side Concatenation

Here is the simple client class:

  1. package com.ociweb.demo;
  2.  
  3. class Client {
  4. public static void main(String args[]) {
  5. // create service
  6. ConcatenatorService service = new ConcatenatorService();
  7.  
  8. // get the port
  9. Concatenator concatProxy = service.getConcatenatorPort();
  10.  
  11. // Invoke the remote method
  12. String a = "Hello";
  13. String b = "Web Services!";
  14. String resultStr = concatProxy.concatenate(a,b);
  15. System.out.println("result = " + resultStr);
  16. }
  17. }

This class simply uses a ConcatenatorService object to acquire a proxy for the web service operation, and then uses it with concatProxy.concatenate. Note there are no annotations on this class: where do the other classes originate? The answer is another JAX-WS tool, wsimport. The Ant build on the client side works like this:

The output from the client is:

$ ant -f build_client.xml
Buildfile: build_client.xml
 
clean:
 [delete] Deleting directory C:\measter\src\jsr181\ex1\client\classes
 [delete] Deleting directory C:\measter\src\jsr181\ex1\client\generated
 
init:
 [mkdir] Created dir: C:\measter\src\jsr181\ex1\client\classes
 [mkdir] Created dir: C:\measter\src\jsr181\ex1\client\generated
 
build_client:
 [javac] Compiling 1 source file to C:\measter\src\jsr181\ex1\client\classes
 
run_client:
 [exec] result = Hello Web Services!
 
BUILD SUCCESSFUL
Total time: 11 seconds

Recap

Example 1 represents a complete round-trip from client to server for a trivial web service. Note the compact amount of code and how it concentrates on the business logic for the task at hand. With JSR 181, Java WS development joins EJB3 in the modern era of annotations.

The Annotations

The last example demonstrates the JAX-WS tools. However, the annotations are the heart of JSR 181. The table below gives a broad look at the available annotations:

AnnotationDescription
@WebService   Marks a class/interface as a web service
    Allows overrides for outer WSDL attributes: targetNamespace, wsdlLocation, portName
@WebMethod   Customizes an exposed WS operation (or excludes a method from WS publication)
    Allows overrides for the name of the wsdl:operation
@Oneway   Denotes a web method as having one input message and no output
@SOAPBinding   Allows a WS or web method to customize mapping from WS to SOAP protocol
    Defaults are DOCUMENT/LITERAL/WRAPPED but each dimension can be altered, e.g. setting style=RPC  
@WebParam   Customizes the mapping from a Java param to a WS message part
    The legal usage and behavior depends on RPC vs DOCUMENT style, etc
@WebResult   Analogous to WebParam, but for the return value
@HandlerChain   Advanced functionality to associate WS with external handler chain


The most effective way to learn a given annotation is to modify the server Java class in Example 1, run the HTTP server from Ant and view the resulting WSDL in the browser. Also, one should examine the generated Java classes. Here is a starting example to try in this interactive manner:

  1. package com.ociweb.demo;
  2.  
  3. import javax.jws.WebService;
  4. import javax.jws.WebMethod;
  5. import javax.jws.soap.SOAPBinding;
  6. import javax.xml.ws.Endpoint;
  7.  
  8. @WebService(
  9. name="myConcatenatorType",
  10. serviceName="myConcatService",
  11. targetNamespace="http://reversed.when.normally.reads"
  12. )
  13.  
  14. @SOAPBinding(
  15. style=SOAPBinding.Style.RPC
  16. )
  17.  
  18. public class Concatenator {
  19. // as before
  20. }

Example 2: Acme Web Service

This example, though unabashedly contrived, illustrates more of the JSR 181 annotations.

The scenario: Superheroes, like many enterprises, have concentrated on their core competencies and have farmed out the manufacturing of their super-materials to specialty factories. Acme is one such manufacturer; it has a division to support the "arachnid class" of superhero, which has strong demand for custom-made super silk. In the past, the superheroes have interfaced with Acme's Arachnid Division through legacy technologies. Now, Acme wants to join the modern era with a new, well, "web service."

Acme has selected Java and JSR 181 as their technology. The specific goals include:

Do Try This At Home (Not in Production)

Before examining the code, note:

A Rapid Prototype

This example (available here) uses the same structure and Ant tasks as Example 1. The server side exists entirely in Acme.java, which we analyze here. First, the legacy record:

  1. class OrderRecord {
  2. // very poor design! assume "legacy" record
  3. // estimate fields
  4. public int numUnits;
  5. public String customerId;
  6. public float cost;
  7. // order fields
  8. public String orderCode;
  9. public Date expectedArrival;
  10. }

Here is the legacy interface, decorated with JSR 181 meta-data:

  1. @WebService (
  2. name="AcmeWebOrderInterface"
  3. )
  4. @SOAPBinding(style=SOAPBinding.Style.RPC)
  5. interface LegacyInterface {
  6.  
  7. @WebMethod
  8. @WebResult(name="Estimate")
  9. OrderRecord getEstimate( String customerId, int numUnits );
  10.  
  11. @WebMethod
  12. boolean placeOrder( @WebParam(mode=WebParam.Mode.INOUT)
  13. Holder<OrderRecord> orderHolder );
  14.  
  15. @WebMethod
  16. @Oneway
  17. void confirmReceipt(String orderCode);
  18. }

Recall in JAX-WS terminology, this interface is a Service Endpoint Interface (SEI). Note the following points:

A stated goal of JSR 181 is to separate public contracts from private implementations; this dovetails with an Acme requirement. The following Service Implementation Bean (SIB) demonstrates this:

  1. // Note that this SIB defers meta-data information to the SEI
  2. @WebService(
  3. endpointInterface="com.ociweb.demo.LegacyInterface",
  4. serviceName="AcmeWebService",
  5. portName="AcmeWebOrderPort"
  6. )
  7. public class Acme implements LegacyInterface {
  8.  
  9. public OrderRecord getEstimate( String customerId, int numUnits ) {
  10. OrderRecord record = new OrderRecord();
  11. System.out.println("received estimate request. customerId = " + customerId);
  12. record.numUnits = numUnits;
  13. record.customerId = customerId;
  14. record.cost = numUnits * 100.0f;
  15. return record;
  16. }
  17.  
  18. public boolean placeOrder( Holder<OrderRecord> orderHolder ) {
  19. OrderRecord order = orderHolder.value;
  20. final int arbitraryLimit = 10000;
  21. int randomInt = (new Random()).nextInt(arbitraryLimit);
  22. order.orderCode = Integer.toString(randomInt);
  23. order.expectedArrival = new Date();
  24. System.out.println("placing order with orderCode = " + order.orderCode);
  25. return true;
  26. }
  27.  
  28. public void confirmReceipt(String orderCode) {
  29. // oneway methods should generally spawn a thread to do
  30. // the business logic
  31. System.out.println("writing confirmation to DB for: " + orderCode);
  32. }
  33.  
  34. // arbitrary unit-test method -- not exposed to the web service
  35. boolean testingMethod() { return true; }
  36. }
  37.  

Note the following points:

That's it for the server! The client steps through the use case of getting an estimate, making an order and confirming receipt. As with Example 1, JSR 181 and JAX-WS perform an astounding amount of heavy lifting.

In Production

As noted, the HTTP server in JDK 6 is not appropriate for production. In order to move Example 2 to a production enviroment, it is necessary to use a web/app server that is enabled for JAX-WS.Here is a list of suitable application servers which are Java EE 5 compatible (and thus JAX-WS enabled). Note that Tomcat does not yet work out-of-the-box: here are instructions to enable Tomcat for JAX-WS.

Even with the appropriate server, another aspect to consider is the generation of the JSR-109 deployment descriptors, especially as this is the major boon of JAX-WS. The wsgen tool (here) performs this task: it accepts a SEI and generates the required artifacts for deployment and invocation.

JAX-WS RI

The reference implemenation (RI) for JAX-WS is located here. Here is a quick road-map regarding the RI:

Summary

Java 5's original annotation facility was not a solution, but a promise. Eventually, the solutions poured in for EJB3, JAXB and other communities within Java.

Now, JAX-WS brings the same power of development time metadata to the WS space, leveraging annotations and JAXB. Moreover, JSR 181 acts as a facade pattern for JAX-WS and brings remarkable simplicity for common WS tasks. Combined with the innate HTTP server in Java 6, WS developers finally enjoy simplicity and agility. The woeful cry of a "Java Simplification Request" may well soar into a joyful (if grammatically incorrect) cheer: "we never met a data we didn't like!"

Example Notes

The examples in this article use Java 1.6.0, and Ant 1.6.x. It is assumed that Java and Ant are installed on the machine. The first example is inspired by a blog post by Vivek Pandey.

References and Resources

Michael Easter thanks Eric Burke, Tom Wheeler, Dean Wette and Jeremy Ford for reviewing this article and providing useful suggestions.