/* * \brief Test for XML parser * \author Norman Feske * \date 2007-08-21 */ #include #include using namespace Genode; /**************** ** Test cases ** ****************/ /* valid example of XML structure */ static const char *xml_test_valid = "" " " " init" " 16M" " " " " " " " timer" " 64K" " " " " " " " framebuffer" " 8M" " " ""; /* the first 'program' tag is broken */ static const char *xml_test_broken_tag = "" " " " init" " 16M" " " " " " " " timer" " 64K" " " " " " framebuffer" " 8M" " " ""; /* end tag is missing */ static const char *xml_test_truncated = "" " " " init" " 16M" " " " " " " " timer" " 64K" " " " " " framebuffer" " 8M" " "; /* comment end tag is missing */ static const char *xml_test_truncated_comment = "" " " " init" " 16M" " " " " " " " timer" " 64K" " " " " " \"unfinished string" " 64K" " " " " " framebuffer" " 8M" " " ""; /* valid XML structure attributes */ static const char *xml_test_attributes = "" " " " init" " 16M" " " " " " " ""; /* valid example of XML structure with text between nodes */ static const char *xml_test_text_between_nodes = "" " sometext1" " " " sometext2" " inProgram" " sometext3" ""; /* strange but valid XML comments */ static const char *xml_test_comments = "" "" "" "" "" "" "" ""; /****************** ** Test program ** ******************/ /** * Print attributes of XML token */ template static const char *token_type_string(typename Token::Type token_type) { switch (token_type) { case Token::SINGLECHAR: return "SINGLECHAR"; case Token::NUMBER : return "NUMBER"; case Token::IDENT : return "IDENT"; case Token::STRING : return "STRING"; case Token::WHITESPACE: return "WHITESPACE"; case Token::END : return "END"; } return ""; } /** * Print attributes of XML token */ template static void print_xml_token_info(Token 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(xml_token.type()), xml_token.len(), content_buf); } template static void print_xml_tokens(const char *xml_string) { Token token(xml_string); while (token.type() != Token::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(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; }