mirror of
https://github.com/bjc/prosody.git
synced 2025-04-04 13:47:41 +03:00
util.datamapper: Finally implement support for parsing arrays
This commit is contained in:
parent
87474145e5
commit
f5962d7193
3 changed files with 85 additions and 30 deletions
|
@ -9,6 +9,7 @@ end);
|
||||||
describe("util.datampper", function()
|
describe("util.datampper", function()
|
||||||
|
|
||||||
local s, x, d
|
local s, x, d
|
||||||
|
local disco, disco_info, disco_schema
|
||||||
setup(function()
|
setup(function()
|
||||||
|
|
||||||
local function attr() return {type = "string"; xml = {attribute = true}} end
|
local function attr() return {type = "string"; xml = {attribute = true}} end
|
||||||
|
@ -91,12 +92,74 @@ describe("util.datampper", function()
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
disco_schema = {
|
||||||
|
type = "object";
|
||||||
|
xml = {
|
||||||
|
name = "iq";
|
||||||
|
namespace = "jabber:client"
|
||||||
|
};
|
||||||
|
properties = {
|
||||||
|
to = attr();
|
||||||
|
from = attr();
|
||||||
|
type = attr();
|
||||||
|
id = attr();
|
||||||
|
disco = {
|
||||||
|
type = "object";
|
||||||
|
xml = {
|
||||||
|
name = "query";
|
||||||
|
namespace = "http://jabber.org/protocol/disco#info"
|
||||||
|
};
|
||||||
|
properties = {
|
||||||
|
features = {
|
||||||
|
type = "array";
|
||||||
|
items = {
|
||||||
|
type = "string";
|
||||||
|
xml = {
|
||||||
|
name = "feature";
|
||||||
|
x_single_attribute = "var";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
disco_info = xml.parse[[
|
||||||
|
<iq type="result" id="disco1" from="example.com">
|
||||||
|
<query xmlns="http://jabber.org/protocol/disco#info">
|
||||||
|
<feature var="urn:example:feature:1">wrong</feature>
|
||||||
|
<feature var="urn:example:feature:2"/>
|
||||||
|
<feature var="urn:example:feature:3"/>
|
||||||
|
<unrelated var="urn:example:feature:not"/>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
]];
|
||||||
|
|
||||||
|
disco = {
|
||||||
|
type="result";
|
||||||
|
id="disco1";
|
||||||
|
from="example.com";
|
||||||
|
disco = {
|
||||||
|
features = {
|
||||||
|
"urn:example:feature:1";
|
||||||
|
"urn:example:feature:2";
|
||||||
|
"urn:example:feature:3";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
end);
|
end);
|
||||||
|
|
||||||
describe("parse", function()
|
describe("parse", function()
|
||||||
it("works", function()
|
it("works", function()
|
||||||
assert.same(d, map.parse(s, x));
|
assert.same(d, map.parse(s, x));
|
||||||
end);
|
end);
|
||||||
|
|
||||||
|
it("handles arrays", function ()
|
||||||
|
assert.same(disco, map.parse(disco_schema, disco_info));
|
||||||
|
end);
|
||||||
|
|
||||||
end);
|
end);
|
||||||
|
|
||||||
describe("unparse", function()
|
describe("unparse", function()
|
||||||
|
|
|
@ -60,7 +60,7 @@ end
|
||||||
local function unpack_propschema( propschema : schema_t, propname : string, current_ns : string )
|
local function unpack_propschema( propschema : schema_t, propname : string, current_ns : string )
|
||||||
: json_type_name, value_goes, string, string, string, string, { any }
|
: json_type_name, value_goes, string, string, string, string, { any }
|
||||||
local proptype : json_type_name = "string"
|
local proptype : json_type_name = "string"
|
||||||
local value_where : value_goes = "in_text_tag"
|
local value_where : value_goes = propname and "in_text_tag" or "in_text"
|
||||||
local name = propname
|
local name = propname
|
||||||
local namespace = current_ns
|
local namespace = current_ns
|
||||||
local prefix : string
|
local prefix : string
|
||||||
|
@ -191,21 +191,17 @@ function parse_object (schema : schema_t, s : st.stanza_t) : { string : any }
|
||||||
end
|
end
|
||||||
|
|
||||||
function parse_array (schema : json_schema_object, s : st.stanza_t) : { any }
|
function parse_array (schema : json_schema_object, s : st.stanza_t) : { any }
|
||||||
local proptype, value_where, child_name, namespace = unpack_propschema(schema.items, nil, s.attr.xmlns)
|
local proptype, value_where, child_name, namespace, prefix, single_attribute, enums = unpack_propschema(schema.items, nil, s.attr.xmlns)
|
||||||
|
local attr_name : string
|
||||||
|
if value_where == "in_single_attribute" then -- FIXME this shouldn't be needed
|
||||||
|
value_where = "in_attribute";
|
||||||
|
attr_name = single_attribute;
|
||||||
|
end
|
||||||
local out : { any } = {}
|
local out : { any } = {}
|
||||||
for c in s:childtags(child_name, namespace) do
|
for c in s:childtags(child_name, namespace) do
|
||||||
local value : string;
|
local value : string = extract_value (c, value_where, proptype, attr_name or child_name, namespace, prefix, single_attribute, enums)
|
||||||
if value_where == "in_text_tag" then
|
|
||||||
value = c:get_text();
|
|
||||||
else
|
|
||||||
error "NYI"
|
|
||||||
end
|
|
||||||
|
|
||||||
value = totype(proptype, value)
|
table.insert(out, totype(proptype, value));
|
||||||
|
|
||||||
if value ~= nil then
|
|
||||||
table.insert(out, value);
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
return out;
|
return out;
|
||||||
end
|
end
|
||||||
|
@ -329,10 +325,10 @@ local function unparse ( schema : json_schema_object, t : table, current_name :
|
||||||
if proptype == "string" then
|
if proptype == "string" then
|
||||||
for _, item in ipairs(t as { string }) do
|
for _, item in ipairs(t as { string }) do
|
||||||
if value_where == "in_text_tag" then
|
if value_where == "in_text_tag" then
|
||||||
out:text_tag(name, item, { xmlns = namespace });
|
out:text_tag(name, item, { xmlns = namespace });
|
||||||
else
|
else
|
||||||
error "NYI"
|
error "NYI"
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
error "NYI"
|
error "NYI"
|
||||||
|
|
|
@ -27,7 +27,7 @@ local value_goes = {}
|
||||||
local function unpack_propschema(propschema, propname, current_ns)
|
local function unpack_propschema(propschema, propname, current_ns)
|
||||||
|
|
||||||
local proptype = "string"
|
local proptype = "string"
|
||||||
local value_where = "in_text_tag"
|
local value_where = propname and "in_text_tag" or "in_text"
|
||||||
local name = propname
|
local name = propname
|
||||||
local namespace = current_ns
|
local namespace = current_ns
|
||||||
local prefix
|
local prefix
|
||||||
|
@ -158,21 +158,17 @@ function parse_object(schema, s)
|
||||||
end
|
end
|
||||||
|
|
||||||
function parse_array(schema, s)
|
function parse_array(schema, s)
|
||||||
local proptype, value_where, child_name, namespace = unpack_propschema(schema.items, nil, s.attr.xmlns)
|
local proptype, value_where, child_name, namespace, prefix, single_attribute, enums = unpack_propschema(schema.items, nil, s.attr.xmlns)
|
||||||
|
local attr_name
|
||||||
|
if value_where == "in_single_attribute" then
|
||||||
|
value_where = "in_attribute";
|
||||||
|
attr_name = single_attribute;
|
||||||
|
end
|
||||||
local out = {}
|
local out = {}
|
||||||
for c in s:childtags(child_name, namespace) do
|
for c in s:childtags(child_name, namespace) do
|
||||||
local value;
|
local value = extract_value(c, value_where, proptype, attr_name or child_name, namespace, prefix, single_attribute, enums)
|
||||||
if value_where == "in_text_tag" then
|
|
||||||
value = c:get_text();
|
|
||||||
else
|
|
||||||
error("NYI")
|
|
||||||
end
|
|
||||||
|
|
||||||
value = totype(proptype, value)
|
table.insert(out, totype(proptype, value));
|
||||||
|
|
||||||
if value ~= nil then
|
|
||||||
table.insert(out, value);
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
return out
|
return out
|
||||||
end
|
end
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue