XML Builder can be used to parse a JS object into XML. It supports following options;
const {XMLBuilder} = require('fast-xml-parser');
const options = {
ignoreAttributes : false
};
const builder = new XMLBuilder(options);
let xmlDataStr = builder.build(jObj);
When you build XML from an array, it's better to set arrayNodeName
option to some name.
const cars = [
{
"color": "purple",
"type": "minivan",
"registration": "2020-02-03",
"capacity": 7
},
{
"color": "orange",
"type": "SUV",
"registration": "2021-05-17",
"capacity": 4
},
];
const builder = new XMLBuilder({
arrayNodeName: "car"
});
const output = builder.build(cars);
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>
<car>
<color>orange</color>
<type>SUV</type>
<registration>2021-05-17</registration>
<capacity>4</capacity>
</car>
To recognize attribute properties in the JS object so that they can be trimmed.
To recognize attribute properties group in the JS object so that they can be trimmed and can form attribute expression for a tag.
This property is not supported when preserveOrder: true
because attributes are already grouped.
To customize the behavior of parsing an attribute value to the string value. It accepts attribute name and value.
To recognize CDATA properties in a JS object so that they can be transformed correctly.
Eg Input
{
"any_name": {
"person": {
"phone": [
122233344550,
122233344551,
""
],
"name": [
"<some>Jack</some>Jack",
"<some>Mohan</some>"
],
"blank": "",
"regx": "^[ ].*$"
}
}
};
code
const options = {
processEntities:false,
format: true,
ignoreAttributes: false,
cdataPropName: "phone"
};
const builder = new XMLBuilder(options);
const xmlOutput = builder.build(input);
Output
<any_name>
<person>
<![CDATA[122233344550]]>
<![CDATA[122233344551]]>
<![CDATA[]]>
<name><some>Jack</some>Jack</name>
<name><some>Mohan</some></name>
<blank></blank>
<regx>^[ ].*$</regx>
</person>
</any_name>`;
It is recommended to use preserveOrder: true
when you're parsing XML to js object and building the XML back. So that the order of CDATA is maintained.
To recognize comments in a JS object so that they can be transformed correctly.
Eg Input
{
"any_name": {
"person": {
"phone": [
122233344550,
122233344551,
""
],
"name": [
"<some>Jack</some>Jack",
"<some>Mohan</some>"
],
"blank": "",
"regx": "^[ ].*$"
}
}
};
code
const options = {
processEntities:false,
format: true,
ignoreAttributes: false,
commentPropName: "phone"
};
const builder = new XMLBuilder(options);
const xmlOutput = builder.build(input);
Output
<any_name>
<person>
<!--122233344550-->
<!--122233344551-->
<!---->
<name><some>Jack</some>Jack</name>
<name><some>Mohan</some></name>
<blank></blank>
<regx>^[ ].*$</regx>
</person>
</any_name>`;
It is recommended to use preserveOrder: true
when you're parsing XML to js object and building the XML back. So that the order of comment is maintained.
By default, parsed XML is single line XML string. By format: true
, you can format it for better view.
By default, the ignoreAttributes
option skips attributes while building XML. However, you can specify an array of strings, regular expressions, or a callback function to selectively ignore specific attributes during the building process.
The ignoreAttributes
option supports:
- Array of Strings: Ignore specific attributes by name while building XML.
- Array of Regular Expressions: Ignore attributes that match a pattern while building XML.
- Callback Function: Ignore attributes based on custom logic during the building process.
{
"tag": {
"$ns:attr1": "a1-value",
"$ns:attr2": "a2-value",
"$ns2:attr3": "a3-value",
"$ns2:attr4": "a4-value",
"tag2": {
"$ns:attr1": "a1-value",
"$ns:attr2": "a2-value",
"$ns2:attr3": "a3-value",
"$ns2:attr4": "a4-value"
}
}
}
const options = {
attributeNamePrefix: "$",
ignoreAttributes: ['ns:attr1', 'ns:attr2']
};
const builder = new XMLBuilder(options);
const xmlOutput = builder.build(jsonData);
Result:
<tag ns2:attr3="a3-value" ns2:attr4="a4-value">
<tag2 ns2:attr3="a3-value" ns2:attr4="a4-value"></tag2>
</tag>
const options = {
attributeNamePrefix: "$",
ignoreAttributes: [/^ns2:/]
};
const builder = new XMLBuilder(options);
const xmlOutput = builder.build(jsonData);
Result:
<tag ns:attr1="a1-value" ns:attr2="a2-value">
<tag2 ns:attr1="a1-value" ns:attr2="a2-value"></tag2>
</tag>
const options = {
attributeNamePrefix: "$",
ignoreAttributes: (aName, jPath) => aName.startsWith('ns:') || jPath === 'tag.tag2'
};
const builder = new XMLBuilder(options);
const xmlOutput = builder.build(jsonData);
Result:
<tag ns2:attr3="a3-value" ns2:attr4="a4-value">
<tag2></tag2>
</tag>
Applicable only if format:true
is set.
When you parse a XML using XMLParser with preserveOrder: true
, the result JS object has different structure. So parse that structure in original XML, you should set the same option while building the XML from that js object.
Set it to true
(default) to process XML entities. Check Entities section for more detail. If you don't have entities in your XML document then it is recommended to disable it processEntities: false
for better performance.
This property group all the children tags in single parent tag.
{
"Attributes": [
{
"Attribute": "1"
},
{
"Attribute": "2"
}
]
}
const builder = new XMLBuilder({oneListGroup:"true"})
const output = builder.build(json)
Output
<Attributes>
<Attribute>1</Attribute>
<Attribute>2</Attribute>
</Attributes>
As you set stopNodes
to the XML parser configuration to avoid parsing and processing of any tag, you can set it builder to avoid parsing and entity processing. Check HTML Document Parsing for more detail.
This property is currently supported with preserveOrder: true
option only.
You can parse attributes with value true
without their value.
Input
const jsOrderedObj = [
{
"?textinfo": [
{
"#text": ""
}
],
":@": {
"@_whitespace": true,
"@_is": true,
"@_allowed": true
}
}
];
const options = {
ignoreAttributes: false,
preserveOrder: true,
allowBooleanAttributes: true,
suppressBooleanAttributes: true
};
const builder = new XMLBuilder(options);
const output = builder.build(result);
Output
<?textinfo whitespace is allowed?>
Tags with no text value would be parsed as empty tags. Input
const builder = new XMLBuilder({
arrayNodeName: "any", //not effective
suppressEmptyNode: true
});
const output = builder.build({
a: 32,
b: ""
});
Outout
<a>32</a>
<b/>
To suppress an unpaired tag from <br/>
to <br>
.
To customize the behavior of parsing the text value of a tag to the string value. It accepts tag name and value.
To recognize text value for a tag in the JS object so that they can be properly assigned to the tag.
Unpaired Tags are the tags which don't have matching closing tag. Eg <br>
in HTML. You can parse unpaired tags by providing their list to the parser, validator and builder.
Eg
const xmlData = `
<rootNode>
<tag>value</tag>
<empty />
<unpaired>
<unpaired />
<unpaired></unpaired>
</rootNode>`;
const options = {
// suppressUnpairedNode: true,
unpairedTags: ["unpaired"]
};
const parser = new XMLParser(options);
const result = parser.parse(xmlData);
const builder = new XMLBuilder(options);
const output = builder.build(result);
Output
<rootNode>
<tag>value</tag>
<empty></empty>
<unpaired>
<unpaired>
<unpaired>
</rootNode>
Now if you set `suppressUnpairedNode: false`. You'll get following output
```xml
<rootNode>
<tag>value</tag>
<empty></empty>
<unpaired/>
<unpaired/>
<unpaired/>
</rootNode>
Example 1
When you use XML Parser with alwaysCreateTextNode: true
, it doesn't impact XMLBuilder result
const XMLdata = `
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>`;
const parser = new XMLParser({
alwaysCreateTextNode: true,
});
let result = parser.parse(XMLdata);
// console.log(JSON.stringify(result, null,4));
const builder = new XMLBuilder({ format: true });
const output = builder.build(result);
Output
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>
Example 2
When you use XML Parser with isArray
, it doesn't impact XMLBuilder result under some extent.
const XMLdata = `
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>`;
const parser = new XMLParser({
isArray: (tagName, jPath, isLeafNode, isAttribute) => {
if(isLeafNode) return true;
}
});
let result = parser.parse(XMLdata);
// console.log(JSON.stringify(result, null,4));
const builder = new XMLBuilder();
const output = builder.build(result);
Output
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>
Example 3
When you use XML Parser with preserveOrder
, you should use the same option with XMLBuilder.
const XMLdata = `
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>`;
const parser = new XMLParser({
preserveOrder: true
});
let result = parser.parse(XMLdata);
// console.log(JSON.stringify(result, null,4));
const builder = new XMLBuilder({ preserveOrder: true });
const output = builder.build(result);
Output
<car>
<color>purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>
Example 4
You should set attributeNamePrefix
and other properties to the same value for XML Parser and XMLBuilder.
const XMLdata = `
<car>
<color alpha="7">purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>`;
const options = {
ignoreAttributes: false,
attributeNamePrefix: "@@",
format: true
};
const parser = new XMLParser(options);
let result = parser.parse(XMLdata);
// console.log(JSON.stringify(result, null,4));
const builder = new XMLBuilder(options);
const output = builder.build(result);
Output
<car>
<color alpha="7">purple</color>
<type>minivan</type>
<registration>2020-02-03</registration>
<capacity>7</capacity>
</car>