genode/repos/os/src/test/xml_node/test.cc
2016-10-21 12:39:29 +02:00

318 lines
7.6 KiB
C++

/*
* \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>";
/* 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>";
/* strange but valid XML comments */
static const char *xml_test_comments =
"<config>"
"<visible-tag/>"
"<!---->"
"<!-- <invisible-tag/> -->"
"<!--<invisible-tag/>-->"
"<!--invisible-tag></invisible-tag-->"
"<visible-tag/>"
"</config>";
/******************
** Test program **
******************/
/**
* 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=%ld, 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();
}
}
/**
* 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;
a.type(name, sizeof(name));
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
printf("number of subnodes = %ld\n",
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()
{
printf("--- XML-token test ---\n");
print_xml_tokens<Scanner_policy_identifier_with_underline>(xml_test_text_between_nodes);
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);
printf("-- Test parsing XML with nodes mixed with text --\n");
print_xml_info(xml_test_text_between_nodes);
printf("-- Test parsing XML with comments --\n");
print_xml_info(xml_test_comments);
printf("--- End of XML-parser test ---\n");
return 0;
}