Skip to content

Working with JSON Message Payloads

WSO2 Micro Integrator provides support for JavaScript Object Notation (JSON) payloads in messages. The following sections describe how to work with JSON via the Micro Integrator.

Handling JSON to XML conversion

When building the XML tree, JSON builders attach the converted XML infoset to a special XML element that acts as the root element of the final XML tree. If the original JSON payload is of type object , the special element is <jsonObject/>. If it is an array, the special element is <jsonArray/>. Following are examples of JSON and XML representations of various objects and arrays.

Empty objects

{"object":{}}
<jsonObject>
  <object></object>
</jsonObject>

Empty strings

{"object":""}
<jsonObject>
  <object></object>
</jsonObject>

Empty array

[]
<jsonArray></jsonArray>
<jsonArray>
  <?xml-multiple jsonElement?>
</jsonArray>

Named arrays

{"array":[1,2]}
<jsonObject>
  <array>1</array>
  <array>2</array>
</jsonObject>
<jsonObject>
  <?xml-multiple array?>
  <array>1</array>
  <array>2</array>
</jsonObject>
{"array":[]}
<jsonObject></jsonObject>
<jsonObject>
  <?xml-multiple array?>
</jsonObject>

Anonymous arrays

[1,2]
<jsonArray>
   <jsonElement>1</jsonElement>
   <jsonElement>2</jsonElement>
</jsonArray>
<jsonArray>
  <?xml-multiple jsonElement?>
  <jsonElement>1</jsonElement>
  <jsonElement>2</jsonElement>
</jsonArray>
[1, []]
<jsonArray>
  <jsonElement>1</jsonElement>
  <jsonElement>
     <jsonArray></jsonArray>
  </jsonElement>
</jsonArray>
<jsonArray>
  <?xml-multiple jsonElement?>
  <jsonElement>1</jsonElement>
  <jsonElement>
     <jsonArray>
       <?xml-multiple jsonElement?>
     </jsonArray>
  </jsonElement>
</jsonArray>

XML processing instructions (PIs)

Note the addition of xml-multiple processing instructions to the XML payloads whose JSON representations contain arrays. JsonBuilder (via StAXON) adds these instructions to the XML payload that it builds during the JSON to XML conversion so that during the XML to JSON conversion, JsonFormatter can reconstruct the arrays that are present in the original JSON payload. JsonFormatter interprets the elements immediately following a processing instruction to construct an array.

Special characters

When building XML elements, the EI handles the $ character and digits in a special manner when they appear as the first character of a JSON key. Following are examples of two such occurrences. Note the addition of the _JsonReader_PS_ and _JsonReader_PD_ prefixes in place of the $ and digit characters, respectively.

{"$key":1234}
<jsonObject>
  <_JsonReader_PS_key>1234</_JsonReader_PS_key>
</jsonObject>
{"32X32":"image_32x32.png"}
<jsonObject>
  <_JsonReader_PD_32X32>image_32x32.png</_JsonReader_PD_32X32>
</jsonObject>

Converting spaces

Although you can have spaces in JSON elements, you cannot have them when converted to XML. Therefore, you can handle spaces when converting JSON message payloads to XML, by adding the following property to the MI_HOME/conf/deployment.toml file in the [mediation] section: synapse.build_valid_nc_name

For example, consider the following JSON message:

{
  "abc def" : "this is a sample value"
}

The output converted to XML is as follows:

<abc_jsonreader_32_def>this is a sample value</abc_jsonreader_32_def> 

Tip

The value 32 represents the standard char value of the space. This works other way around as well. When you need to convert XML to JSON, with a JSON element that needs to have a space within it. Then, you use " _JsonReader_32_ " in the XML element, to get the space in the JSON output. For example, if you consider the following XML payload;

<abc_jsonreader_32_def>this is a sample value</abc_jsonreader_32_def>

The JSON output will be as follows:

{ "abc def" : "this is a sample value"}

Handling XML to JSON conversion

When an XML element is converted to JSON, the following rules apply:

Empty XML elements

Consider the following empty XML elements:

<jsonObject>
  <object></object>
</jsonObject>
<jsonObject>
  <object/>
</jsonObject>

By default, empty XML elements convert to JSON as null objects as shown below.

{"object":null}

JSON representation of empty XML element will change as below by adding 'synapse.commons.enableXmlNullForEmptyElement' = false under [synapse_properties] section in MI_HOME/conf/deployment.toml file.

{"object":""}

Info

'synapse.commons.enableXmlNullForEmptyElement property surrounded with single quotation to identify it as whole string rather dot separated TOML object.

Empty XML elements with the 'nil' attribute

Consider the following XML element that has the 'nil' attribute set to true.

<jsonObject>
  <object nil=true></object>
</jsonObject>

By default, the above XML element converts to JSON as shown below.

{"object":{"@nil":"true"}}

If you set the synapse.enable_xml_nil=true property in the deployment.toml file [mediation] section (stored in the MI_HOME/conf/ directory), XML elements where the 'nil' attribute is set to true will be represented in JSON as null objects as shown below.

{"object":null}

Converting a payload between XML and JSON

To convert an XML payload to JSON, set the messageType property to application/json in the axis2 scope before sending message to an endpoint. Similarly, to convert a JSON payload to XML, set the messageType property to application/xml or text/xml. For example:

<proxy xmlns="http://ws.apache.org/ns/synapse"
          name="tojson"
          transports="https,http"
          statistics="disable"
          trace="disable"
          startOnLoad="true">
  <target>
    <inSequence>
       <property name="messageType" value="application/json" scope="axis2"/>
       <respond/>
    </inSequence>
  </target>
  <description/>
</proxy>
If the request payload is as follows:

<coordinates>
       <location>
           <name>Bermuda Triangle</name>
           <n>25.0000</n>
           <w>71.0000</w>
       </location>
       <location>
           <name>Eiffel Tower</name>
           <n>48.8582</n>
           <e>2.2945</e>
       </location>
</coordinates>

Save the payload in request.xml file and use the following command to invoke this proxy service:

curl -v -X POST -H "Content-Type:application/xml" [email protected] "http://localhost:8290/services/tojson"

The response payload will look like this:

{
      "coordinates":{
         "location":[
            {
               "name":"Bermuda Triangle",
               "n":25.0000,
               "w":71.0000
            },
            {
               "name":"Eiffel Tower",
               "n":48.8582,
               "e":2.2945
            }
         ]
      }
}

Note that we have used the Property mediator to mark the outgoing payload to be formatted as JSON:

<property name="messageType" value="application/json" scope="axis2"/>

Note

JSON requests cannot be converted to XML if it contains invalid XML characters.

Info

If you need to convert complex XML responses (e.g., XML with with xsi:type values), you will need to set the message type using the Property mediator as follows: <property name="messageType" value="application/json/badgerfish" scope="axis2" type="STRING"/> You will also need to ensure you register the following message builder and formatter as specified in Message Builders and Formatters.

<messageBuilder contentType="text/javascript" class="org.apache.axis2.json.JSONBadgerfishOMBuilder"/>
<messageFormatter contentType="text/javascript" class="org.apache.axis2.json.JSONBadgerfishMessageFormatter"/> 

Accessing content from JSON payloads

There are two ways to access the content of a JSON payload within the MI.

  • JSONPath expressions (with json-eval() method)
  • XPath expressions

JSONPath allows you to access fields of JSON payloads with faster results and less processing overhead. Although it is possible to evaluate XPath expressions on JSON payloads by assuming the XML representation of the JSON payload, we recommend that you use JSONPath to query JSON payloads. It is also possible to evaluate both JSONPath and XPath expressions on a payload (XML/JSON) at the same time.

You can use JSON path expressions with following mediators:

Mediator Usage
Log

As a log property:

<log>
    <property name="location" 
              expression="json-eval($.coordinates.location[0].name)"/>
</log>
Property

As a standalone property:

<property name="location" 
              expression="json-eval($.coordinates.location[0].name)"/>
PayloadFactory

As the payload arguments:

<payloadFactory media-type="json">
    <format>{"RESPONSE":"$1"}</format>
    <args>
        <arg evaluator="json" expression="$.coordinates.location[0].name"/>
    </args>
</payloadFactory>

IMPORTANT : You MUST omit the json-eval() method within the payload arguments to evaluate JSON paths within the PayloadFactory mediator. Instead, you MUST select the correct expression evaluator ( xml or json ) for a given argument.

Switch

As the switch source:

<switch source="json-eval($.coordinates.location[0].name)">
Filter

As the filter source:

<filter source="json-eval($.coordinates.location[0].name)" 
        regex="Eiffel.*">

JSONpath syntax

JSONPath provides a way to navigate, filter, and extract data from JSON structures. To learn more about how to use JSONPath as an expression, see the JSONPath expressions documentation.

Logging JSON payloads

To log JSON payloads as JSON, use the Log mediator as shown below. The json-eval() method returns the java.lang.String representation of the existing JSON payload.

<log>
  <property name="JSON-Payload" expression="json-eval($.)"/>
</log>

To log JSON payloads as XML, use the Log mediator as shown below:

<log level="full"/>

For more information on logging, see Troubleshooting, debugging, and logging below.

Constructing and transforming JSON payloads

To construct and transform JSON payloads, you can use the PayloadFactory mediator or Script mediator as described in the rest of this section.

PayloadFactory mediator

The PayloadFactory mediator provides the simplest way to work with JSON payloads. Suppose we have a service that returns the following response for a search query:

{
       "geometry":{
          "location":{
             "lat":-33.867260,
             "lng":151.1958130
          }
       },
       "icon":"bar-71.png",
       "id":"7eaf7",
       "name":"Biaggio Cafe",
       "opening_hours":{
          "open_now":true
       },
       "photos":[
          {
             "height":600,
             "html_attributions":[
             ],
             "photo_reference":"CoQBegAAAI",
             "width":900
          }
       ],
       "price_level":1,
       "reference":"CnRqAAAAtz",
       "types":[
          "bar",
          "restaurant",
          "food",
          "establishment"
       ],
       "vicinity":"48 Pirrama Road, Pyrmont"
}

We can create a proxy service that consumes the above response and creates a new response containing the location name and tags associated with the location based on several fields from the above response.

<proxy xmlns="http://ws.apache.org/ns/synapse"
         name="singleresponse"
         transports="https,http"
         statistics="disable"
         trace="disable"
         startOnLoad="true">
         <target>
             <inSequence>
                 <payloadFactory media-type="json">
                     <format>{
                                 "location_response" : {
                                     "name" : "$1",
                                     "tags" : "$2"
                             }}
                     </format>
                     <args>
                         <arg evaluator="json" expression="$.name"/>
                         <arg evaluator="json" expression="$.types"/>
                     </args>
                 </payloadFactory>
                 <respond/>
             </inSequence>
         </target>
     <description/>
</proxy>

Save the above payload in request.json file and use the following command to invoke this service:

curl -v POST -H "Content-Type:application/json" [email protected] "http://localhost:8290/services/singleresponse"

The response payload would look like this:

{
    "location_response":{
      "name":"Biaggio Cafe",
      "tags":["bar", "restaurant", "food", "establishment"]
    }
}

Note the following aspects of the proxy service configuration:

  • We use the payloadFactory mediator to construct the new JSON payload.
  • The media-type attribute is set to json .
  • Because JSONPath expressions are used in arguments, the json evaluators are specified.
Configuring the payload format

The <format> section of the proxy service configuration defines the format of the response. Notice that in the example above, the name and tags field values are enclosed by double quotes ("), which creates a string value in the final response. If you do not use quotes, the value that gets assigned uses the real type evaluated by the expression (boolean, number, object, array, or null).

It is also possible to instruct the PayloadFactory mediator to load a payload format definition from the registry. This approach is particularly useful when using large/complex payload formats in the definitions. To load a format from the registry, click Pick From Registry instead of Define inline when defining the PayloadFactory mediator.

For example, suppose we have saved the following text content in the following registry location: conf:/repository/MI/transform.txt .

{
    "location_response" : {
        "name" : "$1",
        "tags" : "$2"
    }
}

We can now modify the definition of the PayloadFactory mediator to use this format text saved as a registry resource as the payload format. The new configuration would look as follows (note that the <format> element now uses the key attribute to point to the registry resource key):

<payloadFactory media-type="json">
  <format key="conf:/repository/MI/transform.txt"/>
  ... 
</payloadFactory>

Note

When saving format text for the PayloadFactory mediator as a registry resource, be sure to save it as text content with the “text/plain” media type.

Script mediator

The Script mediator in JavaScript is useful when you need to create payloads that have recurring structures such as arrays of objects. The Script mediator defines the following important methods that can be used to manipulate payloads in many different ways:

  • getPayloadJSON
  • setPayloadJSON
  • getPayloadXML
  • setPayloadXML

By combining any of the setters with a getter, we can handle almost any type of content transformation within the MI. For example, by combining getPayloadXML and setPayloadJSON , we can easily implement an XML to JSON transformation scenario. In addition, we can perform various operations (such as deleting individual keys, modifying selected values, and inserting new objects) on JSON payloads to transform from one JSON format to another JSON format by using the getPayloadJSON and setPayloadJSON methods.

Note

  • If you are using nashornJS as the JavaScript language, and also if you have JSON operations defined in the Script mediator, you need to have JDK version 8u112 or a later version in your environment. If your environment has an older JDK version, the Script mediator (that uses nashornJS and JSON operations) will not function properly because of this JDK bug. That is, you will encounter server exceptions in the Micro Integrator.

  • If you are using JDK 15 or above, you need to manually copy the nashorn-core and asm-util jars to the <MI_HOME>/lib directory since Nashorn was removed from the JDK in Java 15.

Example

Following is an example of a JSON to JSON transformation performed by the Script mediator. Suppose a second service returns the following response:

{
    "results" : [
          {
             "geometry" : {
                "location" : {
                   "lat" : -33.867260,
                   "lng" : 151.1958130
                }
             },
             "icon" : "bar-71.png",
             "id" : "7eaf7",
             "name" : "Biaggio Cafe",
             "opening_hours" : {
                "open_now" : true
             },
             "photos" : [
                {
                   "height" : 600,
                   "html_attributions" : [],
                   "photo_reference" : "CoQBegAAAI",
                   "width" : 900
                }
             ],
             "price_level" : 1,
             "reference" : "CnRqAAAAtz",
             "types" : [ "bar", "restaurant", "food", "establishment" ],
             "vicinity" : "48 Pirrama Road, Pyrmont"
          },
          {
             "geometry" : {
                "location" : {
                   "lat" : -33.8668040,
                   "lng" : 151.1955790
                }
             },
             "icon" : "generic_business-71.png",
             "id" : "3ef98",
             "name" : "Doltone House",
             "photos" : [
                {
                   "height" : 600,
                   "html_attributions" : [],
                   "photo_reference" : "CqQBmgAAAL",
                   "width" : 900
                }
             ],
             "reference" : "CnRrAAAAV",
             "types" : [ "food", "establishment" ],
             "vicinity" : "48 Pirrama Road, Pyrmont"
          }
       ],
       "status" : "OK"
}

The following proxy service shows how we can transform the above response using JavaScript with the Script mediator.

<proxy xmlns="http://ws.apache.org/ns/synapse"
           name="locations"
           transports="https,http"
           statistics="disable"
           trace="disable"
           startOnLoad="true">
       <target>
          <inSequence>
             <script language="js"
                     key="conf:/repository/MI/transform.js"
                     function="transform"/>
             <respond/>
          </inSequence>
       </target>
       <description/>
</proxy>

The registry resource transform.js contains the JavaScript function that performs the transformation:

function transform(mc) {
        payload = mc.getPayloadJSON();
        results = payload.results;
        var response = new Array();
        for (i = 0; i < results.length; ++i) {
            location_object = results[i];
            l = new Object();
            l.name = location_object.name;
            l.tags = location_object.types;
            l.id = "ID:" + (location_object.id);
            response[i] = l;
        }
        mc.setPayloadJSON(response);
}

mc.getPayloadJSON() returns the current JSON payload as a JavaScript object. This object can be manipulated as a normal JavaScript variable within a script as shown in the above JavaScript code. The mc.setPayloadJSON() method can be used to replace the existing payload with a new payload. In the above script, we build a new array object by using the fields of the incoming JSON payload and set that array object as the new payload (see the response payload returned by the final proxy service below.)

Save the above payload in request.json file and use the following command to invoke the proxy service:

curl -v POST -H "Content-Type:application/json" [email protected] "http://localhost:8290/services/locations"

The response payload would look like this:

[
        {
            "id":"ID:7eaf7", 
            "tags":["bar", "restaurant", "food", "establishment"], 
            "name":"Biaggio Cafe"
        }, 
        {
           "id":"ID:3ef98", 
           "tags":["food", "establishment"], 
           "name":"Doltone House"
        }
]

If you want to get the response in XML instead of JSON, you would modify the service by adding the Property mediator as follows:

<inSequence>
    <script language="js" 
                key="conf:/repository/MI/transform.js" 
                function="transform"/>
    <property name="messageType" value="application/xml" scope="axis2"/>
    <respond/>
</inSequence>

The response will then look like this:

<jsonArray>
       <jsonElement>
          <id>ID:7eaf7</id>
          <tags>bar</tags>
          <tags>restaurant</tags>
          <tags>food</tags>
          <tags>establishment</tags>
          <name>Biaggio Cafe</name>
       </jsonElement>
       <jsonElement>
          <id>ID:3ef98</id>
          <tags>food</tags>
          <tags>establishment</tags>
          <name>Doltone House</name>
       </jsonElement>
</jsonArray>

If you are not getting the results you want when the Script mediator converts the JSON payload directly into XML, you can build the XML payload iteratively with the Script mediator as shown in the following script.

function transformXML(mc) {
        payload = mc.getPayloadJSON();
        results = payload.results;
        var response = <locations/>;
        for (i = 0; i < results.length; ++i) {
            var elem = results[i];
            response.locations += <location>
                <id>{elem.id}</id>
                <name>{elem.name}</name>
                <tags>{elem.types}</tags>
            </location>
        }
        mc.setPayloadXML(response);
}

The response would now look like this:

<locations>
       <location>
          <id>7eaf7</id>
          <name>Biaggio Cafe</name>
          <tags>bar,restaurant,food,establishment</tags>
       </location>
       <location>
          <id>3ef98</id>
          <name>Doltone House</name>
          <tags>food,establishment</tags>
       </location>
</locations>

Finally, let's look at how you can perform delete, modify, and add field operations on JSON payloads with the Script mediator in JavaScript. Let's send the JSON message returned by the locations proxy service as the request for the following proxy service, transformjson :

<proxy xmlns="http://ws.apache.org/ns/synapse"
           name="transformjson"
           transports="https,http"
           statistics="disable"
           trace="disable"
           startOnLoad="true">
       <target>
          <inSequence>
             <script language="js">
               payload = mc.getPayloadJSON();
               for (i = 0; i &lt; payload.length; ++i) {
                   payload[i].id_str = payload[i].id;
                   delete payload[i].id;
                   payload[i].tags[payload[i].tags.length] = "pub";
               }
               mc.setPayloadJSON(payload);
             </script>
             <log>
                <property name="JSON-Payload" expression="json-eval($.)"/>
             </log>
             <respond/>
          </inSequence>
       </target>
       <description/>
</proxy>

The proxy service will convert the request into the following format:

[
       {
          "name":"Biaggio Cafe",
          "tags":["bar", "restaurant", "food", "establishment", "pub"],
          "id_str":"ID:7eaf7"
       },
       {
          "name":"Doltone House",
          "tags":["food", "establishment", "pub"],
          "id_str":"ID:3ef98"
       }
]

Note that the transformation (line 9 through 17) has added a new field id_str and removed the old field id from the request, and it has added a new tag pub to the existing tags list of the payload.

XML to JSON transformation parameters

You can use XML to JSON transformation parameters when you need to transform XML formatted data into the JSON format.

Following are the XML to JSON transformation parameters and their descriptions:

Parameter

Description

Default Value

synapse.commons.json.preserve.namespace

Preserves the namespace declarations in the JSON output in XML to JSON transformations.

false

synapse.commons.json.buildValidNCNames

Builds valid XML NCNames when building XML element names in XML to JSON transformations.

false

synapse.commons.json.output.autoPrimitive

Allows primitive types in the JSON output in XML to JSON transformations.

true

synapse.commons.json.output.namespaceSepChar

The namespace prefix separation character for the JSON output in XML to JSON transformations.

The default separation character is -

synapse.commons.json.output.enableNSDeclarations

Adds XML namespace declarations in the JSON output in XML to JSON transformations.

false

synapse.commons.json.output.disableAutoPrimitive.regex

Disables auto primitive conversion in XML to JSON transformations.

null

synapse.commons.json.output.jsonoutAutoArray

Sets the JSON output to an array element in XML to JSON transformations.

true

synapse.commons.json.output.jsonoutMultiplePI

Sets the JSON output to an xml multiple processing instruction in XML to JSON transformations.

true

synapse.commons.json.output.xmloutAutoArray

Sets the XML output to an array element in XML to JSON transformations.

true

synapse.commons.json.output.xmloutMultiplePI

Sets the XML output to an xml multiple processing instruction in XML to JSON transformations.

false
synapse.commons.enableXmlNilReadWrite Handles how empty XML elements with the 'nil' attribute are converted to JSON. false
synapse.commons.enableXmlNullForEmptyElement
Handles how empty XML elements are converted to JSON. true

Validating JSON messages

You can use the Validate mediator to validate JSON messages against a specified JSON schema as described in the rest of this section.

Validate mediator

The parameters available in this section are as follows.

Parameter Name Description
Schema keys defined for Validate Mediator This section is used to specify the key to access the main schema based on which validation is carried out, as well as to specify the JSON, which needs to be validated.
Source The JSONPath expression to extract the JSON that needs to be validated. E.g: json-eval($.msg)"

Following example use the below sample schema StockQuoteSchema.json file. Add this sample schema file (i.e. StockQuoteSchema.json ) to the following Registry path: conf:/schema/StockQuoteSchema . json. For instructions on adding the schema file to the Registry path, see Adding a Resource.

Tip

When adding this sample schema file to the Registry, specify the Media Type as application/json.

{
  "$schema": "http://json-schema.org/draft-04/schema#",
      "type": "object",
      "properties": {
        "getQuote": {
          "type": "object",
          "properties": {
            "request": {
              "type": "object",
              "properties": {
                "symbol": {
                  "type": "string"
                }
              },
              "required": [
                "symbol"
              ]
            }
          },
          "required": [
            "request"
          ]
        }
      },
      "required": [
        "getQuote"
      ]
}   

In this example, the required schema for validating messages going through the Validate mediator is given as a registry key (i.e. schema\StockQuoteSchema.json ). You do not have any source attributes specified. Therefore, the schema will be used to validate the complete JSON body. The mediation logic to follow if the validation fails is defined within the on-fail element. In this example, the PayloadFactory mediator creates a fault to be sent back to the party, which sends the message.

<validate>
  <schema key="conf:/schema/StockQuoteSchema.json"/>
    <on-fail>
            <payloadFactory media-type="json">
                <format>{"Error":"$1"}</format>
                <args>
                    <arg evaluator="xml" expression="$ctx:ERROR_MESSAGE"/>
                </args>
            </payloadFactory>
            <property name="HTTP_SC" value="500" scope="axis2"/>
            <respond/>
    </on-fail>
</validate>

An example for a valid JSON payload request is given below.

{
  "getQuote": {
   "request": {
      "symbol": "WSO2"
    }
  }
}

Troubleshooting, debugging, and logging

To assist with troubleshooting, you can enable debug logging at several stages of the mediation of a JSON payload by adding one or more of the following loggers to the MI_HOME/conf/log4j2.properties file and restarting the MI.

Info

Be sure to turn off these loggers when running the MI in a production environment, as logging every message will significantly reduce performance.

Following are the available logger components:

Message builders and formatters

  • org.apache.synapse.commons.json.JsonStreamBuilder
  • org.apache.synapse.commons.json.JsonStreamFormatter
  • org.apache.synapse.commons.json.JsonBuilder
  • org.apache.synapse.commons.json.JsonFormatter

JSON utility class

org.apache.synapse.commons.json.JsonUtil

PayloadFactory mediator

org.apache.synapse.mediators.transform.PayloadFactoryMediator

JSONPath evaluator

org.apache.synapse.util.xpath.SynapseJsonPath

Debug logging for the mediation of a JSON payload can be enabled by adding these loggers in log4j2.properties file.

For example: logger.JsonStreamBuilder.name = org.apache.synapse.commons.json.JsonStreamBuilder logger.JsonStreamBuilder.level = DEBUG For more instructions on adding loggers, see Configuring Log4j Properties.