lukas-krecan / json2xml

Java JSON to XML converter
Apache License 2.0
64 stars 26 forks source link

Java Json to XML conversion Apache License 2 CI Maven Central

Json2Xml project is a simple implementation of JSON to XML conversion. Under the hood it uses Jackson pull parser and generates XML SAX events. This way the conversion has low memory consumption and is pretty fast.

There is already Jettison project that has similar objective, unfortunately it can not handle JSON arrays properly.

Json2Xml converts the following JSON

       {
       "document":{
          "a":1,
          "b":2,
          "c":{
             "d":"text"
          },
          "e":[1, 2, 3],
          "f":[[1, 2, 3], [4, 5,6]],
          "g":null,
          "h":[
             {
                "i":true,
                "j":false
             }
          ],
          "k":[
             [
                {"l":1, "m":2}
             ],
             [
                {"n":3, "o":4},
                {"p":5, "q":6}
             ]
          ]
       }
    }
     
    <document>
       <a>1</a>
       <b>2</b>
       <c>
          <d>text</d>
       </c>
       <e>
          <e>1</e>
          <e>2</e>
          <e>3</e>
       </e>
       <f>
          <f>
             <f>1</f>
             <f>2</f>
             <f>3</f>
          </f>
          <f>
             <f>4</f>
             <f>5</f>
             <f>6</f>
          </f>
       </f>
       <g />
       <h>
          <h>
             <i>true</i>
             <j>false</j>
          </h>
       </h>
       <k>
          <k>
             <k>
                <l>1</l>
                <m>2</m>
             </k>
          </k>
          <k>
             <k>
                <n>3</n>
                <o>4</o>
             </k>
             <k>
                <p>5</p>
                <q>6</q>
             </k>
          </k>
       </k>
    </document>
     

Usage

If you have SAX content handler, you can use net.javacrumbs.json2xml.JsonSaxAdapter class directly.

ContentHandler ch = ...;
JsonSaxAdapter adapter = new JsonSaxAdapter(json, ch);
adapter.parse();

Otherwise it's possible to use net.javacrumbs.json2xml.JsonXmlReader together with standard Java transformation.

Transformer transformer = TransformerFactory.newInstance().newTransformer();
InputSource source = new InputSource(...);
Result result = ...;
transformer.transform(new SAXSource(new JsonXmlReader(),source), result);

For example

Transformer transformer = TransformerFactory.newInstance().newTransformer();
InputSource source = new InputSource(new StringReader(json));
DOMResult result = new DOMResult();
transformer.transform(new SAXSource(new JsonXmlReader(namespace, addTypeAttributes, artificialRootName), source), result);
result.getNode();

Type attributes

Since XML does not have any mechanism to reflect JSON type information, there is a new feature since json2xml version 1.2. You can switch on the addTypeAttributes flag using a constructor argument. Then you will get the type information in XML attributes like this:

<?xml version="1.0" encoding="UTF-8"?>
<document xmlns="http://javacrumbs.net/test">
   <a type="int">1</a>
   <b type="int">2</b>
   <c>
      <d type="string">text</d>
   </c>
   <e type="array">
      <e type="int">1</e>
      <e type="int">2</e>
      <e type="int">3</e>
   </e>
   <f type="array">
      <f type="array">
         <f type="int">1</f>
         <f type="int">2</f>
         <f type="int">3</f>
      </f>
      <f type="array">
         <f type="int">4</f>
         <f type="int">5</f>
         <f type="int">6</f>
      </f>
   </f>
   <g type="null" />
   <h type="array">
      <h>
         <i type="boolean">true</i>
         <j type="boolean">false</j>
      </h>
   </h>
   <k type="array">
      <k type="array">
         <k>
            <l type="int">1</l>
            <m type="int">2</m>
         </k>
      </k>
      <k type="array">
         <k>
            <n type="int">3</n>
            <o type="int">4</o>
         </k>
         <k>
            <p type="int">5</p>
            <q type="int">6</q>
         </k>
      </k>
   </k>
</document>

Artificial root

XML support only one root element but JSON documents may have multiple roots. To overcome this mismatch, you can specify artificialRootName which will generate artificial XML root with given name. new JsonXmlReader(null, false, "artificialRoot") will transform

{"a":1, "b":2}

to

<artificialRoot>
  <a>1</a>
  <b>2</b>
</artificialRoot>

Converting XML DOM back to JSON

If you have created an XML DOM node from JSON content with addTypeAttributes then you can convert it back to JSON using net.javacrumbs.json2xml.JsonXmlHelper.

// convert JSON to DOM Node
Node node = JsonXmlHelper.convertToDom(...);
// do whatever on the DOM Node (eventually modify using XPATH it while respecting the type attributes)
String json = JsonXmlHelper.convertToJson(node);

This method has proven very useful to work on huge JSON document using XPATH and converting it back to JSON afterward.

Name transformation

Other difference between JSON and XML are allowed names. In cases, when your JSON contains names not allowed as XML element names, an ElementNameConverter can be used. For example to convert '@' to '_' create the following reader

new JsonXmlReader("", false, null, new ElementNameConverter() {
    public String convertName(String name) {
        return name.replaceAll("@","_");
    }
})

Compatibility notes:

Version 4.1 handles arrays differently than the previous version. The change is in handling of arrays of JSON objects. In older version it behaved erratically. Version 4 fixes the behavior, but is backwards-incompatible.

Also note that version 2.0 and higher require Jackson 2.x If you need new features in Jackson 1.x, just file a ticket and I will backport the changes.

Please do not use version 4.0. It has broken handling of arrays. Since it has been in the wild for two days only, I do not assume anyone is using it.

Maven

To use with Maven, add this dependency

<dependency>
    <groupId>net.javacrumbs</groupId>
    <artifactId>json-xml</artifactId>
    <version>4.2</version><!-- for Jackson >= 2.0 -->
    <!--<version>1.3</version>--><!-- older version for Jackson < 2.0 -->
</dependency>