Java API override
Mechanism with java Endorsed Standards
This Topic is about a very rarely used feature of java to
override some of the java API with newer version of those APIs. I java world it is called ‘Java Endorsed Standards Override Mechanism’.
I got to know about this when I got stuck in one of the
scenario where I had some java version problem.
I was using Apache CXF 3.x runtime for Webservice. By
default, it generates Jaxb 2.2 api implementation classes, which is according
to java 7 standards. But, I was using java 6.x for generating client stub from
my ‘wsdl’. While doing so I was getting some exceptions:
For Example:
Caused by: java.lang.NoSuchMethodException:
javax.xml.bind.annotation.XmlElementRef.required()
at
java.lang.Class.getDeclaredMethod(Class.java:1937)
at
com.sun.codemodel.TypedAnnotationWriter.invoke(TypedAnnotationWriter.java:112)….
As you can see above exception shows that it is unable to
find required method in javax.xml.bind.annotation.XmlElementRef
class, because there is a change in java 6 to java 7 version. Java 7 has this
required method in javax.xml.bind.annotation.XmlElementRef, but not in Java 6,
which I was using.
Now,
What to do in this scenario. After doing little research I got to know about ‘Java Endorsed Standards Override Mechanism’. This provides a means whereby later versions
of classes and interfaces that implement Endorsed Standards or Standalone
Technologies may be incorporated into the Java Platform.
All you need to do is deploy updated packages, by placing
the packages in jar and pointing those jar using system property java.endorsed.dirs.
Eg:
-Djava.endorsed.dirs=/some/path/directory
containing packaged jars.
Or,
Put
those packaged jars in jdk’s jre/lib/endorsed
folder
One you are done with this, your updated API will come into
effect.
This overriding mechanism can be done only with Endorsed Standards
APIs and Standalone Technologies till now. More details of these can be found
at Java
7 endorsed-standard-apis.
Let us understand
this feature by an example
You can access code of this example at my git location: Test Java
Overriding Mechanism.
In this example I just wanted to show how I have written
some code which is supported by java 7 which has JAXB 2.2 API but I am trying to compile and run using
Java 6, which has JAXB 2.1 .
For this I create a JAXB Application , and for this created 3
model classes.
ContactInfo.java
public abstract class ContactInfo {
}
Address.java
import
javax.xml.bind.annotation.XmlElement;
import
javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement(name="address")
public class Address extends ContactInfo {
@XmlElement(required=false)
private String street;
public void setStreet(String street) {
this.street = street;
}
}
PhoneNumber.java
@XmlRootElement(name="phoneNumber")
public class PhoneNumber extends ContactInfo {
}
Customer.java
@XmlRootElement
public class Customer {
@XmlElementRefs({
@XmlElementRef(name="address",type=Address.class,required=true),
@XmlElementRef(name="phoneNumber",type=PhoneNumber.class,required=true)
})
private ContactInfo contactInfo;
public void setContactInfo(ContactInfo
contactInfo) {
this.contactInfo = contactInfo;
}
}
Here,
You can see I have used ‘required’ attribute in @XmlElementRef(name="address",type=Address.class,required=true), which is there since Java7 with JAXB 2.1. So when you try to
compile it using Java 6 it would give compilation error, because it comes with
JAXB 2.1.
cannot find symbol
symbol : method required()
location: @interface
javax.xml.bind.annotation.XmlElementRef
@XmlElementRef(name="phoneNumber",type=PhoneNumber.class,required=true)
^
Approach using ‘Java Endorsed Standards Override Mechanism’,
we can provide JAXB 2.2 jars in jdk’s
jre/lib/endorsed folder or -Djava.endorsed.dirs=/some/path/directory
containing JAXB 2.1 jars. It should compile and run successfully.
-----------------------------------------------------------------------------------------------------------------------
import java.io.File;
import java.io.IOException;
import javax.xml.XMLConstants;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import
javax.xml.bind.SchemaOutputResolver;
import javax.xml.transform.Result;
import
javax.xml.transform.stream.StreamResult;
import javax.xml.validation.Schema;
import
javax.xml.validation.SchemaFactory;
import com.test.model.Address;
import com.test.model.Customer;
import com.test.model.PhoneNumber;
public class TestEndorsed {
public static void main(String[] args) throws Exception {
Customer
customer = new Customer();
Address
address = new Address();
address.setStreet("1 A Street");
customer.setContactInfo(address);
JAXBContext
jc = JAXBContext.newInstance(Customer.class, Address.class, PhoneNumber.class);
jc.generateSchema(new SchemaOutputResolver() {
@Override
public Result createOutput(String
namespaceUri, String suggestedFileName)
throws IOException {
suggestedFileName
= "schema.xsd";
File
file = new File(suggestedFileName);
StreamResult result = new StreamResult(file);
result.setSystemId(file.toURI().toURL().toString());
return result;
}
});
Marshaller
marshaller = jc.createMarshaller();
SchemaFactory sf = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
System.out.println(marshaller.getSchema());
Schema schema = sf.newSchema(new File("schema.xsd"));
marshaller.setSchema(schema);
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(customer, System.out);
}
}
This
is the Main program which you can test for marshalling and generating schema
out of customer object.
You
can also use it in build scripts like ANT and Maven. For Example,
<target name="compile">
<mkdir dir="build/classes"/>
<javac srcdir="src" destdir="build/classes">
<compilerarg value="
-Djava.endorsed.dirs=C:\software\jdk1.6.0_21\jdk1.6.0_21\jre\lib\endorsed1"/>
</javac>
</target>
I Hope this
helps many who are struggling with API compatibility and its usage.
Suggestions and
Comments are most welcome.
CHEERS!!!