xml2js
Version:
Simple XML to JavaScript object converter.
234 lines (206 loc) • 10.8 kB
text/coffeescript
# use zap to run tests, it also detects CoffeeScript files
xml2js = require '../lib/xml2js'
fs = require 'fs'
util = require 'util'
assert = require 'assert'
path = require 'path'
fileName = path.join __dirname, '/fixtures/sample.xml'
skeleton = (options, checks) ->
(test) ->
xmlString = options?.__xmlString
delete options?.__xmlString
x2js = new xml2js.Parser options
x2js.addListener 'end', (r) ->
checks r
test.finish()
if not xmlString
fs.readFile fileName, (err, data) ->
x2js.parseString data
else
x2js.parseString xmlString
###
The `validator` function validates the value at the XPath. It also transforms the value
if necessary to conform to the schema or other validation information being used. If there
is an existing value at this path it is supplied in `currentValue` (e.g. this is the second or
later item in an array).
If the validation fails it should throw a `ValidationError`.
###
validator = (xpath, currentValue, newValue) ->
if xpath == '/sample/validatortest/numbertest'
return Number(newValue)
else if xpath in ['/sample/arraytest', '/sample/validatortest/emptyarray', '/sample/validatortest/oneitemarray']
if not ('item' of newValue)
return {'item': []}
else if xpath in ['/sample/arraytest/item', '/sample/validatortest/emptyarray/item', '/sample/validatortest/oneitemarray/item']
if not currentValue
return newValue
else if xpath == '/validationerror'
throw new xml2js.ValidationError("Validation error!")
return newValue
# shortcut, because it is quite verbose
equ = assert.equal
module.exports =
'test parse with defaults': skeleton(undefined, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.chartest[0].$.desc, 'Test for CHARs'
equ r.sample.chartest[0]._, 'Character data here!'
equ r.sample.cdatatest[0].$.desc, 'Test for CDATA'
equ r.sample.cdatatest[0].$.misc, 'true'
equ r.sample.cdatatest[0]._, 'CDATA here!'
equ r.sample.nochartest[0].$.desc, 'No data'
equ r.sample.nochartest[0].$.misc, 'false'
equ r.sample.listtest[0].item[0]._, '\n This is\n \n character\n \n data!\n \n '
equ r.sample.listtest[0].item[0].subitem[0], 'Foo(1)'
equ r.sample.listtest[0].item[0].subitem[1], 'Foo(2)'
equ r.sample.listtest[0].item[0].subitem[2], 'Foo(3)'
equ r.sample.listtest[0].item[0].subitem[3], 'Foo(4)'
equ r.sample.listtest[0].item[1], 'Qux.'
equ r.sample.listtest[0].item[2], 'Quux.'
# determine number of items in object
equ Object.keys(r.sample.tagcasetest[0]).length, 3)
'test parse with explicitCharkey': skeleton(explicitCharkey: true, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.chartest[0].$.desc, 'Test for CHARs'
equ r.sample.chartest[0]._, 'Character data here!'
equ r.sample.cdatatest[0].$.desc, 'Test for CDATA'
equ r.sample.cdatatest[0].$.misc, 'true'
equ r.sample.cdatatest[0]._, 'CDATA here!'
equ r.sample.nochartest[0].$.desc, 'No data'
equ r.sample.nochartest[0].$.misc, 'false'
equ r.sample.listtest[0].item[0]._, '\n This is\n \n character\n \n data!\n \n '
equ r.sample.listtest[0].item[0].subitem[0]._, 'Foo(1)'
equ r.sample.listtest[0].item[0].subitem[1]._, 'Foo(2)'
equ r.sample.listtest[0].item[0].subitem[2]._, 'Foo(3)'
equ r.sample.listtest[0].item[0].subitem[3]._, 'Foo(4)'
equ r.sample.listtest[0].item[1]._, 'Qux.'
equ r.sample.listtest[0].item[2]._, 'Quux.')
'test parse with mergeAttrs': skeleton(mergeAttrs: true, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.chartest[0].desc, 'Test for CHARs'
equ r.sample.chartest[0]._, 'Character data here!'
equ r.sample.cdatatest[0].desc, 'Test for CDATA'
equ r.sample.cdatatest[0].misc, 'true'
equ r.sample.cdatatest[0]._, 'CDATA here!'
equ r.sample.nochartest[0].desc, 'No data'
equ r.sample.nochartest[0].misc, 'false'
equ r.sample.listtest[0].item[0].subitem[0], 'Foo(1)'
equ r.sample.listtest[0].item[0].subitem[1], 'Foo(2)'
equ r.sample.listtest[0].item[0].subitem[2], 'Foo(3)'
equ r.sample.listtest[0].item[0].subitem[3], 'Foo(4)'
equ r.sample.listtest[0].item[1], 'Qux.'
equ r.sample.listtest[0].item[2], 'Quux.')
'test text trimming, normalize': skeleton(trim: true, normalize: true, (r) ->
equ r.sample.whitespacetest[0]._, 'Line One Line Two')
'test text trimming, no normalizing': skeleton(trim: true, normalize: false, (r) ->
equ r.sample.whitespacetest[0]._, 'Line One\n Line Two')
'test text no trimming, normalize': skeleton(trim: false, normalize: true, (r) ->
equ r.sample.whitespacetest[0]._, 'Line One Line Two')
'test text no trimming, no normalize': skeleton(trim: false, normalize: false, (r) ->
equ r.sample.whitespacetest[0]._, '\n Line One\n Line Two\n ')
'test enabled root node elimination': skeleton(__xmlString: '<root></root>', explicitRoot: false, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
assert.deepEqual r, {})
'test disabled root node elimination': skeleton(__xmlString: '<root></root>', explicitRoot: true, (r) ->
assert.deepEqual r, {root: {}})
'test default empty tag result': skeleton(undefined, (r) ->
assert.deepEqual r.sample.emptytest, [{}])
'test empty tag result specified null': skeleton(emptyTag: null, (r) ->
equ r.sample.emptytest[0], null)
'test invalid empty XML file': skeleton(__xmlString: ' ', (r) ->
equ r, null)
'test enabled normalizeTags': skeleton(normalizeTags: true, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ Object.keys(r.sample.tagcasetest).length, 1)
'test parse with custom char and attribute object keys': skeleton(attrkey: 'attrobj', charkey: 'charobj', (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.chartest[0].attrobj.desc, 'Test for CHARs'
equ r.sample.chartest[0].charobj, 'Character data here!'
equ r.sample.cdatatest[0].attrobj.desc, 'Test for CDATA'
equ r.sample.cdatatest[0].attrobj.misc, 'true'
equ r.sample.cdatatest[0].charobj, 'CDATA here!'
equ r.sample.nochartest[0].attrobj.desc, 'No data'
equ r.sample.nochartest[0].attrobj.misc, 'false')
'test child node without explicitArray': skeleton(explicitArray: false, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.arraytest.item[0].subitem, 'Baz.'
equ r.sample.arraytest.item[1].subitem[0], 'Foo.'
equ r.sample.arraytest.item[1].subitem[1], 'Bar.')
'test child node with explicitArray': skeleton(explicitArray: true, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.arraytest[0].item[0].subitem[0], 'Baz.'
equ r.sample.arraytest[0].item[1].subitem[0], 'Foo.'
equ r.sample.arraytest[0].item[1].subitem[1], 'Bar.')
'test ignore attributes': skeleton(ignoreAttrs: true, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample.chartest, 'Character data here!'
equ r.sample.cdatatest, 'CDATA here!'
assert.deepEqual r.sample.nochartest[0], {}
equ r.sample.listtest[0].item[0]._, '\n This is\n \n character\n \n data!\n \n '
equ r.sample.listtest[0].item[0].subitem[0], 'Foo(1)'
equ r.sample.listtest[0].item[0].subitem[1], 'Foo(2)'
equ r.sample.listtest[0].item[0].subitem[2], 'Foo(3)'
equ r.sample.listtest[0].item[0].subitem[3], 'Foo(4)'
equ r.sample.listtest[0].item[1], 'Qux.'
equ r.sample.listtest[0].item[2], 'Quux.')
'test simple callback mode': (test) ->
x2js = new xml2js.Parser()
fs.readFile fileName, (err, data) ->
equ err, null
x2js.parseString data, (err, r) ->
equ err, null
# just a single test to check whether we parsed anything
equ r.sample.chartest[0]._, 'Character data here!'
test.finish()
'test double parse': (test) ->
x2js = new xml2js.Parser()
fs.readFile fileName, (err, data) ->
equ err, null
x2js.parseString data, (err, r) ->
equ err, null
# make sure we parsed anything
equ r.sample.chartest[0]._, 'Character data here!'
x2js.parseString data, (err, r) ->
equ err, null
equ r.sample.chartest[0]._, 'Character data here!'
test.finish()
'test simple function without options': (test) ->
fs.readFile fileName, (err, data) ->
xml2js.parseString data, (err, r) ->
equ err, null
equ r.sample.chartest[0]._, 'Character data here!'
test.finish()
'test simple function with options': (test) ->
fs.readFile fileName, (err, data) ->
# well, {} still counts as option, right?
xml2js.parseString data, {}, (err, r) ->
equ err, null
equ r.sample.chartest[0]._, 'Character data here!'
test.finish()
'test validator': skeleton(validator: validator, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ typeof r.sample.validatortest[0].stringtest[0], 'string'
equ typeof r.sample.validatortest[0].numbertest[0], 'number'
assert.ok r.sample.validatortest[0].emptyarray[0].item instanceof Array
equ r.sample.validatortest[0].emptyarray[0].item.length, 0
assert.ok r.sample.validatortest[0].oneitemarray[0].item instanceof Array
equ r.sample.validatortest[0].oneitemarray[0].item.length, 1
equ r.sample.validatortest[0].oneitemarray[0].item[0], 'Bar.'
assert.ok r.sample.arraytest[0].item instanceof Array
equ r.sample.arraytest[0].item.length, 2
equ r.sample.arraytest[0].item[0].subitem[0], 'Baz.'
equ r.sample.arraytest[0].item[1].subitem[0], 'Foo.'
equ r.sample.arraytest[0].item[1].subitem[1], 'Bar.')
'test validation error': (test) ->
x2js = new xml2js.Parser({validator: validator})
x2js.parseString '<validationerror/>', (err, r) ->
equ err, 'Validation error!'
test.finish()
'test xmlns': skeleton(xmlns: true, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
equ r.sample["pfx:top"][0].$ns.local, 'top'
equ r.sample["pfx:top"][0].$ns.uri, 'http://foo.com'
equ r.sample["pfx:top"][0].$["pfx:attr"].value, 'baz'
equ r.sample["pfx:top"][0].$["pfx:attr"].local, 'attr'
equ r.sample["pfx:top"][0].$["pfx:attr"].uri, 'http://foo.com'
equ r.sample["pfx:top"][0].middle[0].$ns.local, 'middle'
equ r.sample["pfx:top"][0].middle[0].$ns.uri, 'http://bar.com')