#LuaRock "htmlparser"
Parse HTML text into a tree of elements with selectors
##Install
Htmlparser is a listed LuaRock. Install using LuaRocks: luarocks install htmlparser
###Dependencies Htmlparser depends on Lua 5.2, and on the "set" LuaRock, which is installed along automatically. To be able to run the tests, lunitx also comes along as a LuaRock
##Usage Start off with
require("luarocks.loader")
local htmlparser = require("htmlparser")
Then, parse some html:
local root = htmlparser.parse(htmlstring)
The input to parse may be the contents of a complete html document, or any valid html snippet, as long as all tags are correctly opened and closed. Now, find sepcific contained elements by selecting:
local elements = root:select(selectorstring)
Or in shorthand:
local elements = root(selectorstring)
This wil return a Set of elements, all of which are of the same type as the root element, and thus support selecting as well, if ever needed:
for e in pairs(elements) do
	print(e.name)
	local subs = e(subselectorstring)
	for sub in pairs(subs) do
		print("", sub.name)
	end
end
The root element is a container for the top level elements in the parsed text, i.e. the <html> element in a parsed html document would be a child of the returned root element.
##Selectors Supported selectors are a subset of jQuery's selectors:
- "*"all contained elements
- "element"elements with the given tagname
- "#id"elements with the given id attribute value
- ".class"elements with the given classname in the class attribute
- "[attribute]"elements with an attribute of the given name
- "[attribute='value']"equals: elements with the given value for the attribute with the given name
- "[attribute!='value']"not equals: elements without an attribute of the given name, or with that attribute, but with a value that is different from the given value
- "[attribute|='value']"prefix: attribute's value is given value, or starts with given value, followed by a hyphen (- -)
- "[attribute*='value']"contains: attribute's value contains given value
- "[attribute~='value']"word: attribute's value is a space-separated token, where one of the tokens is the given value
- "[attribute^='value']"starts with: attribute's value starts with given value
- "[attribute$='value']"ends with: attribute's value ends with given value
- ":not(selectorstring)"elements not selected by given selector string
- "ancestor descendant"elements selected by the- descendantselector string, that are a descendant of any element selected by the- ancestorselector string
- "parent > child"elements selected by the- childselector string, that are a child element of any element selected by the- parentselector string
Selectors can be combined; e.g. ".class:not([attribute]) element.class"
##Element type
All tree elements provide, apart from :select and (), the following accessors:
###Basic
- .namethe element's tagname
- .attributesa table with keys and values for the element's attributes;- {}if none
- .idthe value of the element's id attribute;- nilif not present
- .classesan array with the classes listed in element's class attribute;- {}if none
- :getcontent()the raw text between the opening and closing tags of the element;- ""if none
- .nodesan array with the element's child elements,- {}if none
- .parentthe elements that contains this element;- root.parentis- nil
###Other
- :gettext()the raw text of the complete element, starting with- "<tagname"and ending with- "/>"or- "</tagname>"
- .levelhow deep the element is in the tree; root level is- 0
- .rootthe root element of the tree;- root.rootis- root
- .deepernodesa Set containing all elements in the tree beneath this element, including this element's- .nodes;- {}if none
- .deeperelementsa table with a key for each distinct tagname in- .deepernodes, containing a Set of all deeper element nodes with that name;- {}in none
- .deeperattributesas- .deeperelements, but keyed on attribute name
- .deeperidsas- .deeperelements, but keyed on id value
- .deeperclassesas- .deeperelements, but keyed on class name
##Limitations
- Attribute values in selectors currently cannot contain any spaces, since space is interpreted as a delimiter between the ancestoranddescendant,parentand>, or>andchildparts of the selector
- Consequently, for the parent > childrelation, the spaces before and after the>are mandatory
- Attribute values in selectors currently also cannot contain any of #,.,[,],:,(, or)
- <!elements are not parsed, including doctype, comments, and CDATA
- Textnodes are not seperate entries in the tree, so the content of <p>line1<br />line2</p>is plainly"line1<br />line2"
- All start and end tags should be explicitly specified in the text to be parsed; omitted tags (as permitted by the the HTML spec) are NOT implied. Only the void elements naturally don't need (and mustn't have) an end tag
- The HTML text is not validated in any way; tag and attribute names and the nesting of different tags is completely arbitrary. The only HTML-specific part of the parser is that it knows which tags are void elements
##Examples
See ./doc/sample.lua
##Tests
See ./tst/init.lua
##License
MIT; see ./doc/LICENSE