123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946 |
- #
- #
- # Nim's Runtime Library
- # (c) Copyright 2012 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## A simple XML tree generator.
- ##
- runnableExamples:
- var g = newElement("myTag")
- g.add newText("some text")
- g.add newComment("this is comment")
- var h = newElement("secondTag")
- h.add newEntity("some entity")
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- let k = newXmlTree("treeTag", [g, h], att)
- doAssert $k == """<treeTag key1="first value" key2="second value">
- <myTag>some text<!-- this is comment --></myTag>
- <secondTag>&some entity;</secondTag>
- </treeTag>"""
- ## **See also:**
- ## * `xmlparser module <xmlparser.html>`_ for high-level XML parsing
- ## * `parsexml module <parsexml.html>`_ for low-level XML parsing
- ## * `htmlgen module <htmlgen.html>`_ for html code generator
- import std/private/since
- import macros, strtabs, strutils, sequtils
- when defined(nimPreviewSlimSystem):
- import std/assertions
- type
- XmlNode* = ref XmlNodeObj ## An XML tree consisting of XML nodes.
- ##
- ## Use `newXmlTree proc <#newXmlTree,string,openArray[XmlNode],XmlAttributes>`_
- ## for creating a new tree.
- XmlNodeKind* = enum ## Different kinds of XML nodes.
- xnText, ## a text element
- xnVerbatimText, ##
- xnElement, ## an element with 0 or more children
- xnCData, ## a CDATA node
- xnEntity, ## an entity (like ``&thing;``)
- xnComment ## an XML comment
- XmlAttributes* = StringTableRef ## An alias for a string to string mapping.
- ##
- ## Use `toXmlAttributes proc <#toXmlAttributes,varargs[tuple[string,string]]>`_
- ## to create `XmlAttributes`.
- XmlNodeObj {.acyclic.} = object
- case k: XmlNodeKind # private, use the kind() proc to read this field.
- of xnText, xnVerbatimText, xnComment, xnCData, xnEntity:
- fText: string
- of xnElement:
- fTag: string
- s: seq[XmlNode]
- fAttr: XmlAttributes
- fClientData: int ## for other clients
- const
- xmlHeader* = "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n"
- ## Header to use for complete XML output.
- template expect(node: XmlNode, kind: set[XmlNodeKind]) =
- ## Check the node's kind is within a set of values
- assert node.k in kind, "Got " & $node.k
- template expect(node: XmlNode, kind: XmlNodeKind) =
- ## Check the node's kind equals a value
- assert node.k == kind, "Got " & $node.k
- proc newXmlNode(kind: XmlNodeKind): XmlNode =
- ## Creates a new ``XmlNode``.
- result = XmlNode(k: kind)
- proc newElement*(tag: sink string): XmlNode =
- ## Creates a new ``XmlNode`` of kind ``xnElement`` with the given `tag`.
- ##
- ## See also:
- ## * `newXmlTree proc <#newXmlTree,string,openArray[XmlNode],XmlAttributes>`_
- ## * [<> macro](#<>.m,untyped)
- runnableExamples:
- var a = newElement("firstTag")
- a.add newElement("childTag")
- assert a.kind == xnElement
- assert $a == """<firstTag>
- <childTag />
- </firstTag>"""
- result = newXmlNode(xnElement)
- result.fTag = tag
- result.s = @[]
- # init attributes lazily to save memory
- proc newText*(text: sink string): XmlNode =
- ## Creates a new ``XmlNode`` of kind ``xnText`` with the text `text`.
- runnableExamples:
- var b = newText("my text")
- assert b.kind == xnText
- assert $b == "my text"
- result = newXmlNode(xnText)
- result.fText = text
- proc newVerbatimText*(text: sink string): XmlNode {.since: (1, 3).} =
- ## Creates a new ``XmlNode`` of kind ``xnVerbatimText`` with the text `text`.
- ## **Since**: Version 1.3.
- result = newXmlNode(xnVerbatimText)
- result.fText = text
- proc newComment*(comment: sink string): XmlNode =
- ## Creates a new ``XmlNode`` of kind ``xnComment`` with the text `comment`.
- runnableExamples:
- var c = newComment("my comment")
- assert c.kind == xnComment
- assert $c == "<!-- my comment -->"
- result = newXmlNode(xnComment)
- result.fText = comment
- proc newCData*(cdata: sink string): XmlNode =
- ## Creates a new ``XmlNode`` of kind ``xnCData`` with the text `cdata`.
- runnableExamples:
- var d = newCData("my cdata")
- assert d.kind == xnCData
- assert $d == "<![CDATA[my cdata]]>"
- result = newXmlNode(xnCData)
- result.fText = cdata
- proc newEntity*(entity: string): XmlNode =
- ## Creates a new ``XmlNode`` of kind ``xnEntity`` with the text `entity`.
- runnableExamples:
- var e = newEntity("my entity")
- assert e.kind == xnEntity
- assert $e == "&my entity;"
- result = newXmlNode(xnEntity)
- result.fText = entity
- proc newXmlTree*(tag: sink string, children: openArray[XmlNode],
- attributes: XmlAttributes = nil): XmlNode =
- ## Creates a new XML tree with `tag`, `children` and `attributes`.
- ##
- ## See also:
- ## * `newElement proc <#newElement,string>`_
- ## * [<> macro](#<>.m,untyped)
- runnableExamples:
- var g = newElement("myTag")
- g.add newText("some text")
- g.add newComment("this is comment")
- var h = newElement("secondTag")
- h.add newEntity("some entity")
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- let k = newXmlTree("treeTag", [g, h], att)
- doAssert $k == """<treeTag key1="first value" key2="second value">
- <myTag>some text<!-- this is comment --></myTag>
- <secondTag>&some entity;</secondTag>
- </treeTag>"""
- result = newXmlNode(xnElement)
- result.fTag = tag
- newSeq(result.s, children.len)
- for i in 0..children.len-1: result.s[i] = children[i]
- result.fAttr = attributes
- proc text*(n: XmlNode): lent string {.inline.} =
- ## Gets the associated text with the node `n`.
- ##
- ## `n` can be a CDATA, Text, comment, or entity node.
- ##
- ## See also:
- ## * `text= proc <#text=,XmlNode,string>`_ for text setter
- ## * `tag proc <#tag,XmlNode>`_ for tag getter
- ## * `tag= proc <#tag=,XmlNode,string>`_ for tag setter
- ## * `innerText proc <#innerText,XmlNode>`_
- runnableExamples:
- var c = newComment("my comment")
- assert $c == "<!-- my comment -->"
- assert c.text == "my comment"
- n.expect {xnText, xnVerbatimText, xnComment, xnCData, xnEntity}
- result = n.fText
- proc `text=`*(n: XmlNode, text: sink string) {.inline.} =
- ## Sets the associated text with the node `n`.
- ##
- ## `n` can be a CDATA, Text, comment, or entity node.
- ##
- ## See also:
- ## * `text proc <#text,XmlNode>`_ for text getter
- ## * `tag proc <#tag,XmlNode>`_ for tag getter
- ## * `tag= proc <#tag=,XmlNode,string>`_ for tag setter
- runnableExamples:
- var e = newEntity("my entity")
- assert $e == "&my entity;"
- e.text = "a new entity text"
- assert $e == "&a new entity text;"
- n.expect {xnText, xnVerbatimText, xnComment, xnCData, xnEntity}
- n.fText = text
- proc tag*(n: XmlNode): lent string {.inline.} =
- ## Gets the tag name of `n`.
- ##
- ## `n` has to be an ``xnElement`` node.
- ##
- ## See also:
- ## * `text proc <#text,XmlNode>`_ for text getter
- ## * `text= proc <#text=,XmlNode,string>`_ for text setter
- ## * `tag= proc <#tag=,XmlNode,string>`_ for tag setter
- ## * `innerText proc <#innerText,XmlNode>`_
- runnableExamples:
- var a = newElement("firstTag")
- a.add newElement("childTag")
- assert $a == """<firstTag>
- <childTag />
- </firstTag>"""
- assert a.tag == "firstTag"
- n.expect xnElement
- result = n.fTag
- proc `tag=`*(n: XmlNode, tag: sink string) {.inline.} =
- ## Sets the tag name of `n`.
- ##
- ## `n` has to be an ``xnElement`` node.
- ##
- ## See also:
- ## * `text proc <#text,XmlNode>`_ for text getter
- ## * `text= proc <#text=,XmlNode,string>`_ for text setter
- ## * `tag proc <#tag,XmlNode>`_ for tag getter
- runnableExamples:
- var a = newElement("firstTag")
- a.add newElement("childTag")
- assert $a == """<firstTag>
- <childTag />
- </firstTag>"""
- a.tag = "newTag"
- assert $a == """<newTag>
- <childTag />
- </newTag>"""
- n.expect xnElement
- n.fTag = tag
- proc rawText*(n: XmlNode): string {.inline.} =
- ## Returns the underlying 'text' string by reference.
- ##
- ## This is only used for speed hacks.
- when defined(gcDestructors):
- result = move(n.fText)
- else:
- shallowCopy(result, n.fText)
- proc rawTag*(n: XmlNode): string {.inline.} =
- ## Returns the underlying 'tag' string by reference.
- ##
- ## This is only used for speed hacks.
- when defined(gcDestructors):
- result = move(n.fTag)
- else:
- shallowCopy(result, n.fTag)
- proc innerText*(n: XmlNode): string =
- ## Gets the inner text of `n`:
- ##
- ## - If `n` is `xnText` or `xnEntity`, returns its content.
- ## - If `n` is `xnElement`, runs recursively on each child node and
- ## concatenates the results.
- ## - Otherwise returns an empty string.
- ##
- ## See also:
- ## * `text proc <#text,XmlNode>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newText("my text")
- f.add newComment("my comment")
- f.add newEntity("my entity")
- assert $f == "<myTag>my text<!-- my comment -->&my entity;</myTag>"
- assert innerText(f) == "my textmy entity"
- proc worker(res: var string, n: XmlNode) =
- case n.k
- of xnText, xnEntity:
- res.add(n.fText)
- of xnElement:
- for sub in n.s:
- worker(res, sub)
- else:
- discard
- result = ""
- worker(result, n)
- proc add*(father, son: XmlNode) {.inline.} =
- ## Adds the child `son` to `father`.
- ## `father` must be of `xnElement` type
- ##
- ## See also:
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- ## * `delete proc <#delete,XmlNode,Natural>`_
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newText("my text")
- f.add newElement("sonTag")
- f.add newEntity("my entity")
- assert $f == "<myTag>my text<sonTag />&my entity;</myTag>"
- father.expect xnElement
- add(father.s, son)
- proc add*(father: XmlNode, sons: openArray[XmlNode]) {.inline.} =
- ## Adds the children `sons` to `father`.
- ## `father` must be of `xnElement` type
- ##
- ## See also:
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- ## * `delete proc <#delete,XmlNode,Natural>`_
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add(@[newText("my text"), newElement("sonTag"), newEntity("my entity")])
- assert $f == "<myTag>my text<sonTag />&my entity;</myTag>"
- father.expect xnElement
- add(father.s, sons)
- proc insert*(father, son: XmlNode, index: int) {.inline.} =
- ## Inserts the child `son` to a given position in `father`.
- ##
- ## `father` must be of `xnElement` kind.
- ##
- ## See also:
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `delete proc <#delete,XmlNode,Natural>`_
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert(newElement("second"), 0)
- assert $f == """<myTag>
- <second />
- <first />
- </myTag>"""
- father.expect xnElement
- if len(father.s) > index:
- insert(father.s, son, index)
- else:
- insert(father.s, son, len(father.s))
- proc insert*(father: XmlNode, sons: openArray[XmlNode], index: int) {.inline.} =
- ## Inserts the children openArray[`sons`] to a given position in `father`.
- ##
- ## `father` must be of `xnElement` kind.
- ##
- ## See also:
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `delete proc <#delete,XmlNode,Natural>`_
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert([newElement("second"), newElement("third")], 0)
- assert $f == """<myTag>
- <second />
- <third />
- <first />
- </myTag>"""
- father.expect xnElement
- if len(father.s) > index:
- insert(father.s, sons, index)
- else:
- insert(father.s, sons, len(father.s))
- proc delete*(n: XmlNode, i: Natural) =
- ## Deletes the `i`'th child of `n`.
- ##
- ## See also:
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert(newElement("second"), 0)
- f.delete(0)
- assert $f == """<myTag>
- <first />
- </myTag>"""
- n.expect xnElement
- n.s.delete(i)
- proc delete*(n: XmlNode, slice: Slice[int]) =
- ## Deletes the items `n[slice]` of `n`.
- ##
- ## See also:
- ## * `delete proc <#delete.XmlNode,int>`_
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert([newElement("second"), newElement("third")], 0)
- f.delete(0..1)
- assert $f == """<myTag>
- <first />
- </myTag>"""
- n.expect xnElement
- n.s.delete(slice)
- proc replace*(n: XmlNode, i: Natural, replacement: openArray[XmlNode]) =
- ## Replaces the `i`'th child of `n` with `replacement` openArray.
- ##
- ## `n` must be of `xnElement` kind.
- ##
- ## See also:
- ## * `replace proc <#replace.XmlNode,Slice[int],openArray[XmlNode]>`_
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `delete proc <#delete,XmlNode,Natural>`_
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert(newElement("second"), 0)
- f.replace(0, @[newElement("third"), newElement("fourth")])
- assert $f == """<myTag>
- <third />
- <fourth />
- <first />
- </myTag>"""
- n.expect xnElement
- n.s.delete(i)
- n.s.insert(replacement, i)
- proc replace*(n: XmlNode, slice: Slice[int], replacement: openArray[XmlNode]) =
- ## Deletes the items `n[slice]` of `n`.
- ##
- ## `n` must be of `xnElement` kind.
- ##
- ## See also:
- ## * `replace proc <#replace.XmlNode,int,openArray[XmlNode]>`_
- ## * `add proc <#add,XmlNode,XmlNode>`_
- ## * `add proc <#add,XmlNode,openArray[XmlNode]>`_
- ## * `delete proc <#delete,XmlNode,Natural>`_
- ## * `delete proc <#delete.XmlNode,Slice[int]>`_
- ## * `insert proc <#insert,XmlNode,XmlNode,int>`_
- ## * `insert proc <#insert,XmlNode,openArray[XmlNode],int>`_
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert([newElement("second"), newElement("fifth")], 0)
- f.replace(0..1, @[newElement("third"), newElement("fourth")])
- assert $f == """<myTag>
- <third />
- <fourth />
- <first />
- </myTag>"""
- n.expect xnElement
- n.s.delete(slice)
- n.s.insert(replacement, slice.a)
- proc len*(n: XmlNode): int {.inline.} =
- ## Returns the number of `n`'s children.
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert(newElement("second"), 0)
- assert len(f) == 2
- if n.k == xnElement: result = len(n.s)
- proc kind*(n: XmlNode): XmlNodeKind {.inline.} =
- ## Returns `n`'s kind.
- runnableExamples:
- var a = newElement("firstTag")
- assert a.kind == xnElement
- var b = newText("my text")
- assert b.kind == xnText
- result = n.k
- proc `[]`*(n: XmlNode, i: int): XmlNode {.inline.} =
- ## Returns the `i`'th child of `n`.
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("first")
- f.insert(newElement("second"), 0)
- assert $f[1] == "<first />"
- assert $f[0] == "<second />"
- n.expect xnElement
- result = n.s[i]
- proc `[]`*(n: var XmlNode, i: int): var XmlNode {.inline.} =
- ## Returns the `i`'th child of `n` so that it can be modified.
- n.expect xnElement
- result = n.s[i]
- proc clear*(n: var XmlNode) =
- ## Recursively clears all children of an XmlNode.
- ##
- runnableExamples:
- var g = newElement("myTag")
- g.add newText("some text")
- g.add newComment("this is comment")
- var h = newElement("secondTag")
- h.add newEntity("some entity")
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- var k = newXmlTree("treeTag", [g, h], att)
- doAssert $k == """<treeTag key1="first value" key2="second value">
- <myTag>some text<!-- this is comment --></myTag>
- <secondTag>&some entity;</secondTag>
- </treeTag>"""
- clear(k)
- doAssert $k == """<treeTag key1="first value" key2="second value" />"""
- for i in 0 ..< n.len:
- clear(n[i])
- if n.k == xnElement:
- n.s.setLen(0)
- iterator items*(n: XmlNode): XmlNode {.inline.} =
- ## Iterates over all direct children of `n`.
- runnableExamples:
- var g = newElement("myTag")
- g.add newText("some text")
- g.add newComment("this is comment")
- var h = newElement("secondTag")
- h.add newEntity("some entity")
- g.add h
- assert $g == "<myTag>some text<!-- this is comment --><secondTag>&some entity;</secondTag></myTag>"
- # for x in g: # the same as `for x in items(g):`
- # echo x
- # some text
- # <!-- this is comment -->
- # <secondTag>&some entity;<![CDATA[some cdata]]></secondTag>
- n.expect xnElement
- for i in 0 .. n.len-1: yield n[i]
- iterator mitems*(n: var XmlNode): var XmlNode {.inline.} =
- ## Iterates over all direct children of `n` so that they can be modified.
- n.expect xnElement
- for i in 0 .. n.len-1: yield n[i]
- proc toXmlAttributes*(keyValuePairs: varargs[tuple[key,
- val: string]]): XmlAttributes =
- ## Converts `{key: value}` pairs into `XmlAttributes`.
- ##
- runnableExamples:
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- var j = newElement("myTag")
- j.attrs = att
- doAssert $j == """<myTag key1="first value" key2="second value" />"""
- newStringTable(keyValuePairs)
- proc attrs*(n: XmlNode): XmlAttributes {.inline.} =
- ## Gets the attributes belonging to `n`.
- ##
- ## Returns `nil` if attributes have not been initialised for this node.
- ##
- ## See also:
- ## * `attrs= proc <#attrs=,XmlNode,XmlAttributes>`_ for XmlAttributes setter
- ## * `attrsLen proc <#attrsLen,XmlNode>`_ for number of attributes
- ## * `attr proc <#attr,XmlNode,string>`_ for finding an attribute
- runnableExamples:
- var j = newElement("myTag")
- assert j.attrs == nil
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- j.attrs = att
- assert j.attrs == att
- n.expect xnElement
- result = n.fAttr
- proc `attrs=`*(n: XmlNode, attr: XmlAttributes) {.inline.} =
- ## Sets the attributes belonging to `n`.
- ##
- ## See also:
- ## * `attrs proc <#attrs,XmlNode>`_ for XmlAttributes getter
- ## * `attrsLen proc <#attrsLen,XmlNode>`_ for number of attributes
- ## * `attr proc <#attr,XmlNode,string>`_ for finding an attribute
- runnableExamples:
- var j = newElement("myTag")
- assert j.attrs == nil
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- j.attrs = att
- assert j.attrs == att
- n.expect xnElement
- n.fAttr = attr
- proc attrsLen*(n: XmlNode): int {.inline.} =
- ## Returns the number of `n`'s attributes.
- ##
- ## See also:
- ## * `attrs proc <#attrs,XmlNode>`_ for XmlAttributes getter
- ## * `attrs= proc <#attrs=,XmlNode,XmlAttributes>`_ for XmlAttributes setter
- ## * `attr proc <#attr,XmlNode,string>`_ for finding an attribute
- runnableExamples:
- var j = newElement("myTag")
- assert j.attrsLen == 0
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- j.attrs = att
- assert j.attrsLen == 2
- n.expect xnElement
- if not isNil(n.fAttr): result = len(n.fAttr)
- proc attr*(n: XmlNode, name: string): string =
- ## Finds the first attribute of `n` with a name of `name`.
- ## Returns "" on failure.
- ##
- ## See also:
- ## * `attrs proc <#attrs,XmlNode>`_ for XmlAttributes getter
- ## * `attrs= proc <#attrs=,XmlNode,XmlAttributes>`_ for XmlAttributes setter
- ## * `attrsLen proc <#attrsLen,XmlNode>`_ for number of attributes
- runnableExamples:
- var j = newElement("myTag")
- let att = {"key1": "first value", "key2": "second value"}.toXmlAttributes
- j.attrs = att
- assert j.attr("key1") == "first value"
- assert j.attr("key2") == "second value"
- n.expect xnElement
- if n.attrs == nil: return ""
- return n.attrs.getOrDefault(name)
- proc clientData*(n: XmlNode): int {.inline.} =
- ## Gets the client data of `n`.
- ##
- ## The client data field is used by the HTML parser and generator.
- result = n.fClientData
- proc `clientData=`*(n: XmlNode, data: int) {.inline.} =
- ## Sets the client data of `n`.
- ##
- ## The client data field is used by the HTML parser and generator.
- n.fClientData = data
- proc addEscaped*(result: var string, s: string) =
- ## The same as `result.add(escape(s)) <#escape,string>`_, but more efficient.
- for c in items(s):
- case c
- of '<': result.add("<")
- of '>': result.add(">")
- of '&': result.add("&")
- of '"': result.add(""")
- of '\'': result.add("'")
- else: result.add(c)
- proc escape*(s: string): string =
- ## Escapes `s` for inclusion into an XML document.
- ##
- ## Escapes these characters:
- ##
- ## ============ ===================
- ## char is converted to
- ## ============ ===================
- ## ``<`` ``<``
- ## ``>`` ``>``
- ## ``&`` ``&``
- ## ``"`` ``"``
- ## ``'`` ``'``
- ## ============ ===================
- ##
- ## You can also use `addEscaped proc <#addEscaped,string,string>`_.
- result = newStringOfCap(s.len)
- addEscaped(result, s)
- proc addIndent(result: var string, indent: int, addNewLines: bool) =
- if addNewLines:
- result.add("\n")
- for i in 1 .. indent:
- result.add(' ')
- proc addImpl(result: var string, n: XmlNode, indent = 0, indWidth = 2,
- addNewLines = true, lastNodeIsText = false) =
- proc noWhitespace(n: XmlNode): bool =
- for i in 0 ..< n.len:
- if n[i].kind in {xnText, xnVerbatimText, xnEntity}: return true
- proc addEscapedAttr(result: var string, s: string) =
- # `addEscaped` alternative with less escaped characters.
- # Only to be used for escaping attribute values enclosed in double quotes!
- for c in items(s):
- case c
- of '<': result.add("<")
- of '>': result.add(">")
- of '&': result.add("&")
- of '"': result.add(""")
- else: result.add(c)
- if n == nil: return
- case n.k
- of xnElement:
- if indent > 0 and not lastNodeIsText:
- result.addIndent(indent, addNewLines)
- let
- addNewLines = if n.noWhitespace():
- false
- else:
- addNewLines
- result.add('<')
- result.add(n.fTag)
- if not isNil(n.fAttr):
- for key, val in pairs(n.fAttr):
- result.add(' ')
- result.add(key)
- result.add("=\"")
- result.addEscapedAttr(val)
- result.add('"')
- if n.len == 0:
- result.add(" />")
- return
- let
- indentNext = if n.noWhitespace():
- indent
- else:
- indent+indWidth
- result.add('>')
- var lastNodeIsText = false
- for i in 0 ..< n.len:
- result.addImpl(n[i], indentNext, indWidth, addNewLines, lastNodeIsText)
- lastNodeIsText = (n[i].kind == xnText) or (n[i].kind == xnVerbatimText)
- if not n.noWhitespace():
- result.addIndent(indent, addNewLines)
- result.add("</")
- result.add(n.fTag)
- result.add(">")
- of xnText:
- result.addEscaped(n.fText)
- of xnVerbatimText:
- result.add(n.fText)
- of xnComment:
- result.add("<!-- ")
- result.addEscaped(n.fText)
- result.add(" -->")
- of xnCData:
- result.add("<![CDATA[")
- result.add(n.fText)
- result.add("]]>")
- of xnEntity:
- result.add('&')
- result.add(n.fText)
- result.add(';')
- proc add*(result: var string, n: XmlNode, indent = 0, indWidth = 2,
- addNewLines = true) {.inline.} =
- ## Adds the textual representation of `n` to string `result`.
- runnableExamples:
- var
- a = newElement("firstTag")
- b = newText("my text")
- c = newComment("my comment")
- s = ""
- s.add(c)
- s.add(a)
- s.add(b)
- assert s == "<!-- my comment --><firstTag />my text"
- result.addImpl(n, indent, indWidth, addNewLines)
- proc `$`*(n: XmlNode): string =
- ## Converts `n` into its string representation.
- ##
- ## No ``<$xml ...$>`` declaration is produced, so that the produced
- ## XML fragments are composable.
- result = ""
- result.add(n)
- proc child*(n: XmlNode, name: string): XmlNode =
- ## Finds the first child element of `n` with a name of `name`.
- ## Returns `nil` on failure.
- runnableExamples:
- var f = newElement("myTag")
- f.add newElement("firstSon")
- f.add newElement("secondSon")
- f.add newElement("thirdSon")
- assert $(f.child("secondSon")) == "<secondSon />"
- n.expect xnElement
- for i in items(n):
- if i.kind == xnElement:
- if i.tag == name:
- return i
- proc findAll*(n: XmlNode, tag: string, result: var seq[XmlNode],
- caseInsensitive = false) =
- ## Iterates over all the children of `n` returning those matching `tag`.
- ##
- ## Found nodes satisfying the condition will be appended to the `result`
- ## sequence.
- runnableExamples:
- var
- b = newElement("good")
- c = newElement("bad")
- d = newElement("BAD")
- e = newElement("GOOD")
- b.add newText("b text")
- c.add newText("c text")
- d.add newText("d text")
- e.add newText("e text")
- let a = newXmlTree("father", [b, c, d, e])
- var s = newSeq[XmlNode]()
- a.findAll("good", s)
- assert $s == "@[<good>b text</good>]"
- s.setLen(0)
- a.findAll("good", s, caseInsensitive = true)
- assert $s == "@[<good>b text</good>, <GOOD>e text</GOOD>]"
- s.setLen(0)
- a.findAll("BAD", s)
- assert $s == "@[<BAD>d text</BAD>]"
- s.setLen(0)
- a.findAll("BAD", s, caseInsensitive = true)
- assert $s == "@[<bad>c text</bad>, <BAD>d text</BAD>]"
- n.expect xnElement
- for child in n.items():
- if child.k != xnElement:
- continue
- if child.tag == tag or
- (caseInsensitive and cmpIgnoreCase(child.tag, tag) == 0):
- result.add(child)
- child.findAll(tag, result)
- proc findAll*(n: XmlNode, tag: string, caseInsensitive = false): seq[XmlNode] =
- ## A shortcut version to assign in let blocks.
- runnableExamples:
- var
- b = newElement("good")
- c = newElement("bad")
- d = newElement("BAD")
- e = newElement("GOOD")
- b.add newText("b text")
- c.add newText("c text")
- d.add newText("d text")
- e.add newText("e text")
- let a = newXmlTree("father", [b, c, d, e])
- assert $(a.findAll("good")) == "@[<good>b text</good>]"
- assert $(a.findAll("BAD")) == "@[<BAD>d text</BAD>]"
- assert $(a.findAll("good", caseInsensitive = true)) == "@[<good>b text</good>, <GOOD>e text</GOOD>]"
- assert $(a.findAll("BAD", caseInsensitive = true)) == "@[<bad>c text</bad>, <BAD>d text</BAD>]"
- newSeq(result, 0)
- findAll(n, tag, result, caseInsensitive)
- proc xmlConstructor(a: NimNode): NimNode =
- if a.kind == nnkCall:
- result = newCall("newXmlTree", toStrLit(a[0]))
- var attrs = newNimNode(nnkBracket, a)
- var newStringTabCall = newCall(bindSym"newStringTable", attrs,
- bindSym"modeCaseSensitive")
- var elements = newNimNode(nnkBracket, a)
- for i in 1..a.len-1:
- if a[i].kind == nnkExprEqExpr:
- # In order to support attributes like `data-lang` we have to
- # replace whitespace because `toStrLit` gives `data - lang`.
- let attrName = toStrLit(a[i][0]).strVal.replace(" ", "")
- attrs.add(newStrLitNode(attrName))
- attrs.add(a[i][1])
- #echo repr(attrs)
- else:
- elements.add(a[i])
- result.add(elements)
- if attrs.len > 1:
- #echo repr(newStringTabCall)
- result.add(newStringTabCall)
- else:
- result = newCall("newXmlTree", toStrLit(a))
- macro `<>`*(x: untyped): untyped =
- ## Constructor macro for XML. Example usage:
- ##
- ## .. code-block:: nim
- ## <>a(href="http://nim-lang.org", newText("Nim rules."))
- ##
- ## Produces an XML tree for:
- ##
- ## <a href="http://nim-lang.org">Nim rules.</a>
- ##
- result = xmlConstructor(x)
|