File

util/xml.lua @ 6990:f476e2497568

MUC: Fire per-field events for the config form, including those with value == nil (no <value/> element in form)
author Matthew Wild <mwild1@gmail.com>
date Fri, 11 Dec 2015 15:27:01 +0000
parent 6978:30c96a5db360
child 7239:c9af793b2d8f
line wrap: on
line source


local st = require "util.stanza";
local lxp = require "lxp";

local _ENV = nil;

local parse_xml = (function()
	local ns_prefixes = {
		["http://www.w3.org/XML/1998/namespace"] = "xml";
	};
	local ns_separator = "\1";
	local ns_pattern = "^([^"..ns_separator.."]*)"..ns_separator.."?(.*)$";
	return function(xml)
		--luacheck: ignore 212/self
		local handler = {};
		local stanza = st.stanza("root");
		local namespaces = {}
		function handler:StartNamespaceDecl(prefix, url)
			if prefix ~= nil then
				namespaces[prefix] = url
			end
		end
		function handler:EndNamespaceDecl(prefix)
			if prefix ~= nil then
				namespaces[prefix] = nil
			end
		end
		function handler:StartElement(tagname, attr)
			local curr_ns,name = tagname:match(ns_pattern);
			if name == "" then
				curr_ns, name = "", curr_ns;
			end
			if curr_ns ~= "" then
				attr.xmlns = curr_ns;
			end
			for i=1,#attr do
				local k = attr[i];
				attr[i] = nil;
				local ns, nm = k:match(ns_pattern);
				if nm ~= "" then
					ns = ns_prefixes[ns];
					if ns then
						attr[ns..":"..nm] = attr[k];
						attr[k] = nil;
					end
				end
			end
			local n = {}
			for prefix, url in pairs(namespaces) do
				n[prefix] = url
			end
			stanza:tag(name, attr, n);
		end
		function handler:CharacterData(data)
			stanza:text(data);
		end
		function handler:EndElement()
			stanza:up();
		end
		local parser = lxp.new(handler, "\1");
		local ok, err, line, col = parser:parse(xml);
		if ok then ok, err, line, col = parser:parse(); end
		--parser:close();
		if ok then
			return stanza.tags[1];
		else
			return ok, err.." (line "..line..", col "..col..")";
		end
	end;
end)();

return {
	parse = parse_xml;
};