# 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: '', explicitRoot: false, (r) ->
console.log 'Result object: ' + util.inspect r, false, 10
assert.deepEqual r, {})
'test disabled root node elimination': skeleton(__xmlString: '', 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) ->
assert.equal err, null
x2js.parseString data, (err, r) ->
assert.equal err, null
# just a single test to check whether we parsed anything
assert.equal r.sample.chartest[0]._, 'Character data here!'
test.finish()
'test double parse': (test) ->
x2js = new xml2js.Parser()
fs.readFile fileName, (err, data) ->
assert.equal err, null
x2js.parseString data, (err, r) ->
assert.equal err, null
# make sure we parsed anything
assert.equal r.sample.chartest[0]._, 'Character data here!'
x2js.parseString data, (err, r) ->
assert.equal err, null
assert.equal 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 '', (err, r) ->
assert.equal 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')