I want to convert a RELAX-NG schema to a schemaInfo object so that it can be used in codemirror for xml-completion.
https://codemirror.net/demo/xmlcomplete.html
xmllint usage
libxml2 already has support for a multi-document relax-NG schema when used to validate a document like this:
xmllint --schema myschema.rng mydoc.xml
Question
Can libxml2 also be used to parse a multi-document schema file?
Here is an example for a multi-document schema:
here is some libxml2 functionality i don't understand but which could be helpful:
Assumption
I think I have to convert the multi-document schema into a single document schema using tools like: https://github.com/h4l/rnginline/tree/master/rnginline
Using libxml2 directly would be great since I could then support schemas without pre-processing.
update 3.5.2016
as you can see parsing the relax-NG schema shows only the top level file and it will not contain any files which are included using the include
directive from the relax-NG main file (note: relax-NG schemas can be spilit into several files).
<!-- XHTML Basic -->
<grammar ns="http://www.w3.org/1999/xhtml"
xmlns="http://relaxng.org/ns/structure/1.0">
<include href="modules/datatypes.rng"/>
<include href="modules/attribs.rng"/>
<include href="modules/struct.rng"/>
<include href="modules/text.rng"/>
<include href="modules/hypertext.rng"/>
<include href="modules/list.rng"/>
<include href="modules/basic-form.rng"/>
<include href="modules/basic-table.rng"/>
<include href="modules/image.rng"/>
<include href="modules/param.rng"/>
<include href="modules/object.rng"/>
<include href="modules/meta.rng"/>
<include href="modules/link.rng"/>
<include href="modules/base.rng"/>
</grammar>
source code
/**
* section: Tree
* synopsis: Navigates a tree to print element names
* purpose: Parse a file to a tree, use xmlDocGetRootElement() to
* get the root element, then walk the document and print
* all the element name in document order.
* usage: tree1 filename_or_URL
* test: tree1 test2.xml > tree1.tmp && diff tree1.tmp $(srcdir)/tree1.res
* author: Dodji Seketeli
* copy: see Copyright for the status of this software.
*/
#include <stdio.h>
#include <libxml/parser.h>
#include <libxml/tree.h>
#ifdef LIBXML_TREE_ENABLED
#define ANSI_COLOR_RED "\x1b[31m"
#define ANSI_COLOR_GREEN "\x1b[32m"
#define ANSI_COLOR_YELLOW "\x1b[33m"
#define ANSI_COLOR_BLUE "\x1b[34m"
#define ANSI_COLOR_MAGENTA "\x1b[35m"
#define ANSI_COLOR_CYAN "\x1b[36m"
#define ANSI_COLOR_RESET "\x1b[0m"
/*
*To compile this file using gcc you can type
*gcc `xml2-config --cflags --libs` -o xmlexample libxml2-example.c
*/
/**
* print_element_names:
* @a_node: the initial xml node to consider.
*
* Prints the names of the all the xml elements
* that are siblings or children of a given xml node.
*/
char* pad(int depth) {
// if (depth <= 0)
// return "";
char str[2000];
// sprintf(str, "%*s", " ", depth);
for (int i=0; i <= depth; i++) {
str[i] = ' ';
}
str[depth+1] = 0;
return &str;
}
static void
print_element_names(xmlNode * a_node, int depth)
{
xmlNode *cur_node = NULL;
for (cur_node = a_node; cur_node; cur_node = cur_node->next) {
if (cur_node->type == XML_ELEMENT_NODE) {
// if (strcmp(cur_node->name, "element") == 0) {
// printf("node type: Element, name: %s\n", cur_node->name);
printf("%s %s\n", pad(depth), cur_node->name);
for(xmlAttrPtr attr = cur_node->properties; NULL != attr; attr = attr->next)
{
printf("%s", ANSI_COLOR_MAGENTA);
printf("%s %s: ", pad(depth), attr->name);
xmlChar* value = xmlNodeListGetString(cur_node->doc, attr->children, 1);
printf("%s \n", value);
printf("%s", ANSI_COLOR_RESET);
}
// }
}
print_element_names(cur_node->children, depth+1);
}
}
/**
* Simple example to parse a file called "file.xml",
* walk down the DOM, and print the name of the
* xml elements nodes.
*/
int
main(int argc, char **argv)
{
xmlDoc *doc = NULL;
xmlNode *root_element = NULL;
if (argc != 2)
return(1);
/*
* this initialize the library and check potential ABI mismatches
* between the version it was compiled for and the actual shared
* library used.
*/
LIBXML_TEST_VERSION
/*parse the file and get the DOM */
doc = xmlReadFile(argv[1], NULL, 0);
if (doc == NULL) {
printf("error: could not parse file %s\n", argv[1]);
}
/*Get the root element node */
root_element = xmlDocGetRootElement(doc);
print_element_names(root_element, 0);
/*free the document */
xmlFreeDoc(doc);
/*
*Free the global variables that may
*have been allocated by the parser.
*/
xmlCleanupParser();
return 0;
}
#else
int main(void) {
fprintf(stderr, "Tree support not compiled in\n");
exit(1);
}
#endif
example usage
[nix-shell:~/Desktop/projects/nlnet/nlnet]$ ./tree1 html5-rng/xhtml-basic.rng
grammar
ns: http://www.w3.org/1999/xhtml
include
href: modules/datatypes.rng
include
href: modules/attribs.rng
include
href: modules/struct.rng
include
href: modules/text.rng
include
href: modules/hypertext.rng
include
href: modules/list.rng
include
href: modules/basic-form.rng
include
href: modules/basic-table.rng
include
href: modules/image.rng
include
href: modules/param.rng
include
href: modules/object.rng
include
href: modules/meta.rng
include
href: modules/link.rng
include
href: modules/base.rng