mirror of
https://github.com/bjc/prosody.git
synced 2025-04-03 21:27:38 +03:00
util.stanza: Fix :top_tag() handling of namespaced attributes
This commit is contained in:
parent
096ebc3bcf
commit
e1b559853f
2 changed files with 61 additions and 32 deletions
|
@ -381,4 +381,35 @@ describe("util.stanza", function()
|
||||||
end);
|
end);
|
||||||
end);
|
end);
|
||||||
end);
|
end);
|
||||||
|
|
||||||
|
describe("top_tag", function ()
|
||||||
|
local xml_parse = require "util.xml".parse;
|
||||||
|
it("works", function ()
|
||||||
|
local s = st.message({type="chat"}, "Hello");
|
||||||
|
local top_tag = s:top_tag();
|
||||||
|
assert.is_string(top_tag);
|
||||||
|
assert.not_equal("/>", top_tag:sub(-2, -1));
|
||||||
|
assert.equal(">", top_tag:sub(-1, -1));
|
||||||
|
local s2 = xml_parse(top_tag.."</message>");
|
||||||
|
assert(st.is_stanza(s2));
|
||||||
|
assert.equal("message", s2.name);
|
||||||
|
assert.equal(0, #s2);
|
||||||
|
assert.equal(0, #s2.tags);
|
||||||
|
assert.equal("chat", s2.attr.type);
|
||||||
|
end);
|
||||||
|
|
||||||
|
it("works with namespaced attributes", function ()
|
||||||
|
local s = xml_parse[[<message foo:bar='true' xmlns:foo='my-awesome-ns'/>]];
|
||||||
|
local top_tag = s:top_tag();
|
||||||
|
assert.is_string(top_tag);
|
||||||
|
assert.not_equal("/>", top_tag:sub(-2, -1));
|
||||||
|
assert.equal(">", top_tag:sub(-1, -1));
|
||||||
|
local s2 = xml_parse(top_tag.."</message>");
|
||||||
|
assert(st.is_stanza(s2));
|
||||||
|
assert.equal("message", s2.name);
|
||||||
|
assert.equal(0, #s2);
|
||||||
|
assert.equal(0, #s2.tags);
|
||||||
|
assert.equal("true", s2.attr["my-awesome-ns\1bar"]);
|
||||||
|
end);
|
||||||
|
end);
|
||||||
end);
|
end);
|
||||||
|
|
|
@ -270,6 +270,34 @@ function stanza_mt:find(path)
|
||||||
until not self
|
until not self
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local function _clone(stanza, only_top)
|
||||||
|
local attr, tags = {}, {};
|
||||||
|
for k,v in pairs(stanza.attr) do attr[k] = v; end
|
||||||
|
local old_namespaces, namespaces = stanza.namespaces;
|
||||||
|
if old_namespaces then
|
||||||
|
namespaces = {};
|
||||||
|
for k,v in pairs(old_namespaces) do namespaces[k] = v; end
|
||||||
|
end
|
||||||
|
local new = { name = stanza.name, attr = attr, namespaces = namespaces, tags = tags };
|
||||||
|
if not only_top then
|
||||||
|
for i=1,#stanza do
|
||||||
|
local child = stanza[i];
|
||||||
|
if child.name then
|
||||||
|
child = _clone(child);
|
||||||
|
t_insert(tags, child);
|
||||||
|
end
|
||||||
|
t_insert(new, child);
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return setmetatable(new, stanza_mt);
|
||||||
|
end
|
||||||
|
|
||||||
|
local function clone(stanza, only_top)
|
||||||
|
if not is_stanza(stanza) then
|
||||||
|
error("bad argument to clone: expected stanza, got "..type(stanza));
|
||||||
|
end
|
||||||
|
return _clone(stanza, only_top);
|
||||||
|
end
|
||||||
|
|
||||||
local escape_table = { ["'"] = "'", ["\""] = """, ["<"] = "<", [">"] = ">", ["&"] = "&" };
|
local escape_table = { ["'"] = "'", ["\""] = """, ["<"] = "<", [">"] = ">", ["&"] = "&" };
|
||||||
local function xml_escape(str) return (s_gsub(str, "['&<>\"]", escape_table)); end
|
local function xml_escape(str) return (s_gsub(str, "['&<>\"]", escape_table)); end
|
||||||
|
@ -310,11 +338,8 @@ function stanza_mt.__tostring(t)
|
||||||
end
|
end
|
||||||
|
|
||||||
function stanza_mt.top_tag(t)
|
function stanza_mt.top_tag(t)
|
||||||
local attr_string = "";
|
local top_tag_clone = clone(t, true);
|
||||||
if t.attr then
|
return tostring(top_tag_clone):sub(1,-3)..">";
|
||||||
for k, v in pairs(t.attr) do if type(k) == "string" then attr_string = attr_string .. s_format(" %s='%s'", k, xml_escape(tostring(v))); end end
|
|
||||||
end
|
|
||||||
return s_format("<%s%s>", t.name, attr_string);
|
|
||||||
end
|
end
|
||||||
|
|
||||||
function stanza_mt.get_text(t)
|
function stanza_mt.get_text(t)
|
||||||
|
@ -388,33 +413,6 @@ local function deserialize(serialized)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
local function _clone(stanza)
|
|
||||||
local attr, tags = {}, {};
|
|
||||||
for k,v in pairs(stanza.attr) do attr[k] = v; end
|
|
||||||
local old_namespaces, namespaces = stanza.namespaces;
|
|
||||||
if old_namespaces then
|
|
||||||
namespaces = {};
|
|
||||||
for k,v in pairs(old_namespaces) do namespaces[k] = v; end
|
|
||||||
end
|
|
||||||
local new = { name = stanza.name, attr = attr, namespaces = namespaces, tags = tags };
|
|
||||||
for i=1,#stanza do
|
|
||||||
local child = stanza[i];
|
|
||||||
if child.name then
|
|
||||||
child = _clone(child);
|
|
||||||
t_insert(tags, child);
|
|
||||||
end
|
|
||||||
t_insert(new, child);
|
|
||||||
end
|
|
||||||
return setmetatable(new, stanza_mt);
|
|
||||||
end
|
|
||||||
|
|
||||||
local function clone(stanza)
|
|
||||||
if not is_stanza(stanza) then
|
|
||||||
error("bad argument to clone: expected stanza, got "..type(stanza));
|
|
||||||
end
|
|
||||||
return _clone(stanza);
|
|
||||||
end
|
|
||||||
|
|
||||||
local function message(attr, body)
|
local function message(attr, body)
|
||||||
if not body then
|
if not body then
|
||||||
return new_stanza("message", attr);
|
return new_stanza("message", attr);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue