2015-04-08 09:40:57 +00:00
|
|
|
/*
|
|
|
|
* \brief Test for XML parser
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2007-08-21
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <util/xml_node.h>
|
|
|
|
#include <base/printf.h>
|
|
|
|
|
|
|
|
using namespace Genode;
|
|
|
|
|
|
|
|
|
|
|
|
/****************
|
|
|
|
** Test cases **
|
|
|
|
****************/
|
|
|
|
|
|
|
|
/* valid example of XML structure */
|
|
|
|
static const char *xml_test_valid =
|
|
|
|
"<config>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>init</filename>"
|
|
|
|
" <quota>16M</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <!-- comment -->"
|
|
|
|
" <program>"
|
|
|
|
" <filename>timer</filename>"
|
|
|
|
" <quota>64K</quota>"
|
|
|
|
" <!-- <quota>32K</quota> -->"
|
|
|
|
" </program>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>framebuffer</filename>"
|
|
|
|
" <quota>8M</quota>"
|
|
|
|
" </program>"
|
|
|
|
"</config>";
|
|
|
|
|
|
|
|
/* the first 'program' tag is broken */
|
|
|
|
static const char *xml_test_broken_tag =
|
|
|
|
"<config>"
|
|
|
|
" <program >"
|
|
|
|
" <filename>init</filename>"
|
|
|
|
" <quota>16M</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <!-- comment -->"
|
|
|
|
" <program>"
|
|
|
|
" <filename>timer</filename>"
|
|
|
|
" <quota>64K</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>framebuffer</filename>"
|
|
|
|
" <quota>8M</quota>"
|
|
|
|
" </program>"
|
|
|
|
"</config>";
|
|
|
|
|
|
|
|
/* end tag is missing */
|
|
|
|
static const char *xml_test_truncated =
|
|
|
|
"<config>"
|
|
|
|
" <program >"
|
|
|
|
" <filename>init</filename>"
|
|
|
|
" <quota>16M</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <!-- comment -->"
|
|
|
|
" <program>"
|
|
|
|
" <filename>timer</filename>"
|
|
|
|
" <quota>64K</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>framebuffer</filename>"
|
|
|
|
" <quota>8M</quota>"
|
|
|
|
" </program>";
|
|
|
|
|
|
|
|
/* comment end tag is missing */
|
|
|
|
static const char *xml_test_truncated_comment =
|
|
|
|
"<config>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>init</filename>"
|
|
|
|
" <quota>16M</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <!-- comment -->"
|
|
|
|
" <program>"
|
|
|
|
" <filename>timer</filename>"
|
|
|
|
" <quota>64K</quota>"
|
|
|
|
" <!-- truncated comment"
|
|
|
|
" </program>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>framebuffer</filename>"
|
|
|
|
" <quota>8M</quota>"
|
|
|
|
" </program>"
|
|
|
|
"</config>";
|
|
|
|
|
|
|
|
/* contains unfinished string */
|
|
|
|
static const char *xml_test_unfinished_string =
|
|
|
|
"<config>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>init</filename>"
|
|
|
|
" <quota>16M</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <!-- comment -->"
|
|
|
|
" <program>"
|
|
|
|
" <filename>\"unfinished string</filename>"
|
|
|
|
" <quota>64K</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <program>"
|
|
|
|
" <filename>framebuffer</filename>"
|
|
|
|
" <quota>8M</quota>"
|
|
|
|
" </program>"
|
|
|
|
"</config>";
|
|
|
|
|
|
|
|
/* valid XML structure attributes */
|
|
|
|
static const char *xml_test_attributes =
|
|
|
|
"<config priolevels=\"4\">"
|
|
|
|
" <program>"
|
|
|
|
" <filename>init</filename>"
|
|
|
|
" <quota>16M</quota>"
|
|
|
|
" </program>"
|
|
|
|
" <single-tag/>"
|
|
|
|
" <single-tag-with-attr name=\"ein_name\" quantum=\"2K\" />"
|
|
|
|
"</config>";
|
|
|
|
|
2015-04-19 21:08:31 +00:00
|
|
|
/* valid example of XML structure with text between nodes */
|
|
|
|
static const char *xml_test_text_between_nodes =
|
|
|
|
"<config>"
|
|
|
|
" sometext1"
|
|
|
|
" <program attr=\"abcd\"/>"
|
|
|
|
" sometext2"
|
|
|
|
" <program>inProgram</program>"
|
|
|
|
" sometext3"
|
|
|
|
"</config>";
|
|
|
|
|
2015-04-08 09:40:57 +00:00
|
|
|
|
|
|
|
/******************
|
|
|
|
** Test program **
|
|
|
|
******************/
|
|
|
|
|
2015-04-19 21:08:31 +00:00
|
|
|
/**
|
|
|
|
* Print attributes of XML token
|
|
|
|
*/
|
|
|
|
template <typename SCANNER_POLICY>
|
|
|
|
static const char *token_type_string(typename Token<SCANNER_POLICY>::Type token_type)
|
|
|
|
{
|
|
|
|
switch (token_type) {
|
|
|
|
case Token<SCANNER_POLICY>::SINGLECHAR: return "SINGLECHAR";
|
|
|
|
case Token<SCANNER_POLICY>::NUMBER : return "NUMBER";
|
|
|
|
case Token<SCANNER_POLICY>::IDENT : return "IDENT";
|
|
|
|
case Token<SCANNER_POLICY>::STRING : return "STRING";
|
|
|
|
case Token<SCANNER_POLICY>::WHITESPACE: return "WHITESPACE";
|
|
|
|
case Token<SCANNER_POLICY>::END : return "END";
|
|
|
|
}
|
|
|
|
return "<invalid>";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Print attributes of XML token
|
|
|
|
*/
|
|
|
|
template <typename SCANNER_POLICY>
|
|
|
|
static void print_xml_token_info(Token<SCANNER_POLICY> xml_token)
|
|
|
|
{
|
|
|
|
static char content_buf[128];
|
|
|
|
xml_token.string(content_buf, sizeof(content_buf));
|
|
|
|
printf("token type=\"%s\", len=%zd, content=\"%s\"\n",
|
|
|
|
token_type_string<SCANNER_POLICY>(xml_token.type()),
|
|
|
|
xml_token.len(), content_buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <typename SCANNER_POLICY>
|
|
|
|
static void print_xml_tokens(const char *xml_string)
|
|
|
|
{
|
|
|
|
Token<SCANNER_POLICY> token(xml_string);
|
|
|
|
while (token.type() != Token<SCANNER_POLICY>::END) {
|
|
|
|
print_xml_token_info(token);
|
|
|
|
token = token.next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-08 09:40:57 +00:00
|
|
|
/**
|
|
|
|
* Print attributes of XML node
|
|
|
|
*/
|
|
|
|
static void print_xml_attr_info(Xml_node xml_node, int indent = 0)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
for (Xml_node::Attribute a = xml_node.attribute(0U); ; a = a.next()) {
|
|
|
|
|
|
|
|
/* indentation */
|
|
|
|
for (int i = 0; i < indent; i++)
|
|
|
|
printf(" ");
|
|
|
|
|
|
|
|
/* read attribute name and value */
|
|
|
|
char name[32]; name[0] = 0;
|
2015-04-19 21:08:31 +00:00
|
|
|
a.type(name, sizeof(name));
|
2015-04-08 09:40:57 +00:00
|
|
|
char value[32]; value[0] = 0;
|
|
|
|
a.value(value, sizeof(value));
|
|
|
|
|
|
|
|
printf("attribute name=\"%s\", value=\"%s\"\n", name, value);
|
|
|
|
}
|
|
|
|
} catch (Xml_node::Nonexistent_attribute) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Print information about XML node and its sub nodes
|
|
|
|
*
|
|
|
|
* \param xml_node root fo XML sub tree to print
|
|
|
|
* \param indent current indentation level
|
|
|
|
*/
|
|
|
|
static void print_xml_node_info(Xml_node xml_node, int indent = 0)
|
|
|
|
{
|
|
|
|
char buf[128];
|
|
|
|
xml_node.type_name(buf, sizeof(buf));
|
|
|
|
|
|
|
|
/* indentation */
|
|
|
|
for (int i = 0; i < indent; i++)
|
|
|
|
printf(" ");
|
|
|
|
|
|
|
|
/* print node information */
|
|
|
|
printf("XML node: name = \"%s\", ", buf);
|
|
|
|
if (xml_node.num_sub_nodes() == 0) {
|
|
|
|
xml_node.value(buf, sizeof(buf));
|
|
|
|
printf("leaf content = \"%s\"\n", buf);
|
|
|
|
} else
|
2015-04-19 21:08:31 +00:00
|
|
|
printf("number of subnodes = %zd\n",
|
2015-04-08 09:40:57 +00:00
|
|
|
xml_node.num_sub_nodes());
|
|
|
|
|
|
|
|
print_xml_attr_info(xml_node, indent + 2);
|
|
|
|
|
|
|
|
/* print information of sub nodes */
|
|
|
|
for (unsigned i = 0; i < xml_node.num_sub_nodes(); i++) {
|
|
|
|
try {
|
|
|
|
Xml_node sub_node = xml_node.sub_node(i);
|
|
|
|
print_xml_node_info(sub_node, indent + 2);
|
|
|
|
} catch (Xml_node::Invalid_syntax) {
|
|
|
|
printf("invalid syntax of sub node %d\n", i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Print content of sub node with specified type
|
|
|
|
*/
|
|
|
|
static void print_key(Xml_node node, const char *key)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
Xml_node sub_node = node.sub_node(key);
|
|
|
|
char buf[32];
|
|
|
|
sub_node.value(buf, sizeof(buf));
|
|
|
|
printf("content of sub node \"%s\" = \"%s\"\n", key, buf);
|
|
|
|
} catch (Xml_node::Nonexistent_sub_node) {
|
|
|
|
printf("sub node \"%s\" is not defined\n", key);
|
|
|
|
} catch (Xml_node::Invalid_syntax) {
|
|
|
|
printf("invalid syntax of node \"%s\"\n", key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void print_xml_info(const char *xml_string)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
print_xml_node_info(Xml_node(xml_string));
|
|
|
|
} catch (Xml_node::Invalid_syntax) {
|
|
|
|
printf("string has invalid XML syntax\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
2015-04-19 21:08:31 +00:00
|
|
|
printf("--- XML-token test ---\n");
|
|
|
|
print_xml_tokens<Scanner_policy_identifier_with_underline>(xml_test_text_between_nodes);
|
|
|
|
|
2015-04-08 09:40:57 +00:00
|
|
|
printf("--- XML-parser test ---\n");
|
|
|
|
|
|
|
|
printf("-- Test valid XML structure --\n");
|
|
|
|
print_xml_info(xml_test_valid);
|
|
|
|
|
|
|
|
printf("-- Test invalid XML structure (broken tag) --\n");
|
|
|
|
print_xml_info(xml_test_broken_tag);
|
|
|
|
|
|
|
|
printf("-- Test invalid XML structure (truncated) --\n");
|
|
|
|
print_xml_info(xml_test_truncated);
|
|
|
|
|
|
|
|
printf("-- Test invalid XML structure (truncated comment) --\n");
|
|
|
|
print_xml_info(xml_test_truncated_comment);
|
|
|
|
|
|
|
|
printf("-- Test invalid XML structure (unfinished string) --\n");
|
|
|
|
print_xml_info(xml_test_unfinished_string);
|
|
|
|
|
|
|
|
printf("-- Test node access by key --\n");
|
|
|
|
Xml_node prg(Xml_node(xml_test_valid).sub_node(0U));
|
|
|
|
print_key(prg, "filename");
|
|
|
|
print_key(prg, "quota");
|
|
|
|
print_key(prg, "info");
|
|
|
|
|
|
|
|
printf("-- Test access to XML attributes --\n");
|
|
|
|
print_xml_info(xml_test_attributes);
|
|
|
|
|
2015-04-19 21:08:31 +00:00
|
|
|
printf("-- Test parsing XML with nodes mixed with text --\n");
|
|
|
|
print_xml_info(xml_test_text_between_nodes);
|
|
|
|
|
2015-04-08 09:40:57 +00:00
|
|
|
printf("--- End of XML-parser test ---\n");
|
|
|
|
return 0;
|
|
|
|
}
|