| /** |
| * rngparser.c: parser for the Relax-NG compact syntax. |
| * |
| * Based on: |
| * RELAX NG Compact Syntax |
| * Committee Specification 21 November 2002 |
| * http://www.oasis-open.org/committees/relax-ng/compact-20021121.html |
| * |
| * See Copyright for the status of this software. |
| * |
| * Daniel Veillard <veillard@redhat.com> |
| */ |
| |
| #include <string.h> |
| |
| #include <libxml/parser.h> |
| #include <libxml/parserInternals.h> |
| #include <libxml/relaxng.h> |
| #include <libxml/dict.h> |
| |
| #define TODO \ |
| xmlGenericError(xmlGenericErrorContext, \ |
| "Unimplemented block at %s:%d\n", \ |
| __FILE__, __LINE__); |
| |
| #define MAX_TOKEN 10 |
| |
| typedef enum { |
| CRNG_NONE = 0, |
| CRNG_OP = 1, |
| CRNG_KEYWORD, |
| CRNG_IDENTIFIER, |
| CRNG_LITERAL_SEGMENT, |
| CRNG_CNAME, |
| CRNG_QNAME, |
| CRNG_NSNAME, |
| CRNG_DOCUMENTATION |
| } xmlCRNGTokType; |
| |
| typedef enum { |
| CRNG_OKAY = 0, |
| CRNG_MEMORY_ERROR, |
| CRNG_INVALID_CHAR_ERROR, |
| CRNG_END_ERROR, |
| CRNG_ENCODING_ERROR |
| } xmlCRNGError; |
| |
| typedef enum { |
| XML_CRNG_ERROR = -1, |
| XML_CRNG_OK = 0, |
| XML_CRNG_EOF = 1 |
| } xmlCRelaxNGParserState; |
| |
| typedef struct _token _token; |
| typedef _token *tokenPtr; |
| struct _token { |
| xmlCRNGTokType toktype; |
| int toklen; |
| const xmlChar *token; |
| const xmlChar *prefix; |
| }; |
| |
| typedef struct _xmlCRelaxNGParserCtxt xmlCRelaxNGParserCtxt; |
| typedef xmlCRelaxNGParserCtxt *xmlCRelaxNGParserCtxtPtr; |
| struct _xmlCRelaxNGParserCtxt { |
| void *userData; /* user specific data block */ |
| xmlRelaxNGValidityErrorFunc error; /* the callback in case of errors */ |
| xmlRelaxNGValidityWarningFunc warning;/* the callback in case of warning */ |
| xmlRelaxNGValidErr err; |
| |
| const xmlChar *compact; |
| const xmlChar *end; |
| const xmlChar *cur; |
| int isElem; |
| int lineno; |
| const xmlChar *linestart; |
| const char *filename; |
| |
| int nbTokens; |
| int firstToken; |
| _token tokens[MAX_TOKEN]; |
| int totalToken; |
| |
| xmlCRelaxNGParserState state; |
| |
| int nbErrors; |
| |
| xmlDocPtr res; /* the result */ |
| xmlNodePtr ins; /* the current insertion node */ |
| |
| xmlNsPtr nsDef; |
| tokenPtr token; |
| |
| xmlHashTablePtr namespaces; |
| xmlHashTablePtr datatypes; |
| |
| /* |
| * dictionnary and keywords |
| */ |
| xmlDictPtr dict; |
| const xmlChar *key_attribute; |
| const xmlChar *key_default; |
| const xmlChar *key_datatypes; |
| const xmlChar *key_div; |
| const xmlChar *key_element; |
| const xmlChar *key_empty; |
| const xmlChar *key_external; |
| const xmlChar *key_grammar; |
| const xmlChar *key_include; |
| const xmlChar *key_inherit; |
| const xmlChar *key_list; |
| const xmlChar *key_mixed; |
| const xmlChar *key_namespace; |
| const xmlChar *key_notAllowed; |
| const xmlChar *key_parent; |
| const xmlChar *key_start; |
| const xmlChar *key_string; |
| const xmlChar *key_text; |
| const xmlChar *key_token; |
| const xmlChar *key_equal; |
| const xmlChar *key_orequal; |
| const xmlChar *key_andequal; |
| const xmlChar *key_combine; |
| const xmlChar *key_or; |
| const xmlChar *key_comma; |
| const xmlChar *key_and; |
| const xmlChar *key_choice; |
| const xmlChar *key_group; |
| const xmlChar *key_interleave; |
| const xmlChar *key_ref; |
| const xmlChar *key_define; |
| |
| /* results */ |
| xmlDocPtr doc; /* the resulting doc */ |
| xmlNodePtr insert; /* the insertion point */ |
| xmlAttrPtr attrs; /* pending attributes */ |
| }; |
| |
| static const xmlChar *xmlCRelaxNGInherit = BAD_CAST "Inherit string"; |
| static const xmlChar *xmlCRelaxNGDefault = BAD_CAST "Default string"; |
| |
| #define CUR_CHAR(l) xmlXPathCurrentChar(ctxt, &l) |
| /** |
| * IS_BLANK: |
| * @c: an UNICODE value (int) |
| * |
| * Macro to check the following production in the XML spec: |
| * |
| * [3] S ::= (#x20 | #x9 | #xD | #xA)+ |
| */ |
| #ifndef IS_BLANK |
| #define IS_BLANK(c) (((c) == 0x20) || ((c) == 0x09) || ((c) == 0xA) || \ |
| ((c) == 0x0D)) |
| #endif |
| #define IS_SEPARATOR(c) (((c) == 0x20) || ((c) == 0x09) || ((c) == 0xA) || \ |
| ((c) == 0x0D) || (c == '#')) |
| |
| #define CRNG_ERROR0(X) \ |
| { xmlCRNGErr(ctxt, X, NULL); return(0); } |
| #define CRNG_ERROR(X) \ |
| { xmlCRNGErr(ctxt, X, NULL); } |
| |
| #define CRNG_MEM_ERROR0() \ |
| { xmlCRNGErr(ctxt, CRNG_MEMORY_ERROR, NULL); return(0); } |
| #define CRNG_MEM_ERROR() \ |
| { xmlCRNGErr(ctxt, CRNG_MEMORY_ERROR, NULL); } |
| |
| #define ERROR(str) xmlCRNGErr(ctxt, 0, str); |
| |
| static void |
| xmlCRNGErr(xmlCRelaxNGParserCtxtPtr ctxt, int err_no, const char *err_msg) { |
| const xmlChar *cur; |
| xmlChar buffer[150]; |
| int i, l; |
| |
| if (ctxt != NULL) { |
| if (ctxt->filename != NULL) |
| fprintf(stderr, "%s:%d ", ctxt->filename, ctxt->lineno); |
| } |
| if (err_msg != NULL) { |
| fprintf(stderr, "error: %s\n", err_msg); |
| } else if (err_no != 0) |
| fprintf(stderr, "error %d\n", err_no); |
| cur = ctxt->cur; |
| while ((*cur != '\n') && (*cur != '\r') && (ctxt->cur - cur < 80)) cur--; |
| l = ctxt->cur - cur; |
| cur++; |
| for (i = 0; i < 100;i++) { |
| if ((*cur == '\n') || (*cur == '\r')) break; |
| buffer[i] = *cur++; |
| } |
| buffer[i] = 0; |
| fprintf(stderr, "%s\n", buffer); |
| for (i = 0; i < l;i++) buffer[i] = ' '; |
| buffer[i++] = '^'; |
| buffer[i++] = 0; |
| fprintf(stderr, "%s\n", buffer); |
| } |
| |
| /** |
| * IS_OP |
| * @c: an UNICODE value (int) |
| * |
| * Macro to check for operator value |
| */ |
| #ifndef IS_OP |
| #define IS_OP(c) (((c) == ',') || ((c) == '&') || ((c) == '|') || \ |
| ((c) == '?') || ((c) == '-') || ((c) == '*') || \ |
| ((c) == '{') || ((c) == '}') || ((c) == '(') || \ |
| ((c) == ')') || ((c) == '+') || ((c) == '=') || \ |
| ((c) == ':')) |
| #endif |
| |
| static int |
| xmlCRNGIsKeyword(xmlCRelaxNGParserCtxtPtr ctxt, const xmlChar *str) { |
| if ((str == ctxt->key_attribute) || |
| (str == ctxt->key_default) || |
| (str == ctxt->key_datatypes) || |
| (str == ctxt->key_div) || |
| (str == ctxt->key_element) || |
| (str == ctxt->key_empty) || |
| (str == ctxt->key_external) || |
| (str == ctxt->key_grammar) || |
| (str == ctxt->key_include) || |
| (str == ctxt->key_inherit) || |
| (str == ctxt->key_list) || |
| (str == ctxt->key_mixed) || |
| (str == ctxt->key_namespace) || |
| (str == ctxt->key_notAllowed) || |
| (str == ctxt->key_parent) || |
| (str == ctxt->key_start) || |
| (str == ctxt->key_string) || |
| (str == ctxt->key_text) || |
| (str == ctxt->key_token)) |
| return(1); |
| return(0); |
| |
| } |
| |
| /* |
| * xmlCRNGNextToken: |
| * ctxt: a compact RNG parser context |
| * |
| * Scan the schema to get the next token |
| * |
| * Return 0 if success and -1 in case of error |
| */ |
| |
| static int |
| xmlCRNGNextToken(xmlCRelaxNGParserCtxtPtr ctxt) { |
| const xmlChar *cur; |
| tokenPtr token; |
| |
| if (ctxt == NULL) return(-1); |
| if (ctxt->nbTokens >= MAX_TOKEN) return(-1); |
| token = &(ctxt->tokens[(ctxt->firstToken + ctxt->nbTokens) % MAX_TOKEN]); |
| token->toktype = CRNG_NONE; |
| |
| if (ctxt->cur == NULL) { |
| ctxt->cur = ctxt->compact; |
| } |
| retry: |
| if (ctxt->cur >= ctxt->end) { |
| ctxt->state = XML_CRNG_EOF; |
| return(-1); |
| } |
| while ((ctxt->cur < ctxt->end) && |
| (IS_BLANK(*ctxt->cur))) ctxt->cur++; |
| if (ctxt->cur >= ctxt->end) { |
| ctxt->state = XML_CRNG_EOF; |
| return(-1); |
| } |
| if (*ctxt->cur == '#') { |
| cur = ctxt->cur; |
| cur++; |
| while ((cur < ctxt->end) && (*cur != '\n') && (*cur != '\r')) |
| cur++; |
| ctxt->cur = cur; |
| goto retry; |
| } else if (*ctxt->cur == '"') { |
| /* string, check for '"""' */ |
| ctxt->cur++; |
| if (ctxt->cur >= ctxt->end) goto eof; |
| cur = ctxt->cur; |
| if ((ctxt->end - ctxt->end > 2) && |
| (*cur == '"') && (cur[1] == '"')) { |
| TODO |
| } else { |
| while ((cur < ctxt->end) && (*cur != '"')) cur++; |
| if (cur >= ctxt->end) goto eof; |
| token->toklen = cur - ctxt->cur; |
| token->token = xmlDictLookup(ctxt->dict, ctxt->cur, token->toklen); |
| token->toktype = CRNG_LITERAL_SEGMENT; |
| token->prefix = NULL; |
| cur++; |
| ctxt->cur = cur; |
| } |
| } else if (*ctxt->cur == '\'') { |
| /* string, check for "'''" */ |
| TODO |
| } else if ((IS_OP(*ctxt->cur)) || (*ctxt->cur == ':')) { |
| cur = ctxt->cur; |
| cur++; |
| if ((cur < ctxt->end) && |
| (((*cur == '=') && |
| ((*ctxt->cur == '|') || (*ctxt->cur == '&'))) || |
| ((*cur == '*') && (*ctxt->cur == ':')))) { |
| token->toklen = 2; |
| } else { |
| token->toklen = 1; |
| } |
| token->token = xmlDictLookup(ctxt->dict, ctxt->cur, token->toklen); |
| token->toktype = CRNG_OP; |
| token->prefix = NULL; |
| ctxt->cur += token->toklen; |
| } else { |
| int escape = 0; |
| |
| cur = ctxt->cur; |
| if (*cur == '\\') { |
| escape = 1; |
| cur++; |
| ctxt->cur++; |
| } |
| while ((cur < ctxt->end) && |
| (!(IS_SEPARATOR(*cur))) && (!(IS_OP(*cur)))) cur++; |
| |
| token->toklen = cur - ctxt->cur; |
| token->token = xmlDictLookup(ctxt->dict, ctxt->cur, token->toklen); |
| token->prefix = NULL; |
| ctxt->cur = cur; |
| if ((escape == 0) && (xmlCRNGIsKeyword(ctxt, token->token))) |
| token->toktype = CRNG_KEYWORD; |
| else { |
| token->toktype = CRNG_IDENTIFIER; |
| } |
| if (*ctxt->cur == ':') { |
| ctxt->cur++; |
| if (*ctxt->cur == '*') { |
| ctxt->cur++; |
| token->toktype = CRNG_NSNAME; |
| } else { |
| cur = ctxt->cur; |
| while ((cur < ctxt->end) && |
| (!(IS_SEPARATOR(*cur))) && (!(IS_OP(*cur)))) cur++; |
| token->prefix = token->token; |
| token->toklen = cur - ctxt->cur; |
| token->token = xmlDictLookup(ctxt->dict, ctxt->cur, |
| token->toklen); |
| ctxt->cur = cur; |
| if (xmlValidateNCName(token->token, 0) == 0) |
| token->toktype = CRNG_QNAME; |
| else { |
| TODO /* sounds like an error ! */ |
| token->toktype = CRNG_IDENTIFIER; |
| } |
| } |
| } |
| } |
| ctxt->nbTokens++; |
| return(0); |
| eof: |
| ctxt->state = XML_CRNG_EOF; |
| CRNG_ERROR(CRNG_END_ERROR); |
| return(-1); |
| } |
| |
| /** |
| * xmlParseCRNGGetToken: |
| * @ctxt: a compact RNG parser context |
| * @no: the number of the token from 1 for the first one |
| * and 2, 3 ... for read-ahead |
| * |
| * Token reading interface |
| * |
| * returns a pointer to the new token, or NULL in case of error or EOF |
| */ |
| static tokenPtr |
| xmlParseCRNGGetToken(xmlCRelaxNGParserCtxtPtr ctxt, int no) { |
| tokenPtr ret; |
| int res; |
| |
| if ((no <= 0) || (no >= MAX_TOKEN)) return(NULL); |
| no--; |
| while (ctxt->nbTokens <= no) { |
| res = xmlCRNGNextToken(ctxt); |
| if (res < 0) |
| return(NULL); |
| } |
| ret = &(ctxt->tokens[(ctxt->firstToken + no) % MAX_TOKEN]); |
| return(ret); |
| } |
| |
| /** |
| * xmlParseCRNGDropTokens: |
| * @ctxt: a compact RNG parser context |
| * @nr: the number of token marked as read |
| * |
| * mark a number of token as read and consumed. |
| * |
| * Returns -1 in case of error and 0 otherwise |
| */ |
| static int |
| xmlParseCRNGDropTokens(xmlCRelaxNGParserCtxtPtr ctxt, int nr) { |
| if ((nr <= 0) || (nr >= MAX_TOKEN)) return(-1); |
| while ((ctxt->nbTokens >0) && (nr > 0)) { |
| ctxt->firstToken++; |
| nr--; |
| ctxt->nbTokens--; |
| ctxt->totalToken++; |
| if (ctxt->totalToken == 384) |
| fprintf(stderr, "found\n"); |
| } |
| ctxt->firstToken = ctxt->firstToken % MAX_TOKEN; |
| return(0); |
| } |
| |
| static void |
| xmlParseCRNGTokenize(xmlCRelaxNGParserCtxtPtr ctxt) { |
| tokenPtr token; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| while (token != NULL) { |
| switch (token->toktype) { |
| case CRNG_NONE: printf("none"); break; |
| case CRNG_OP: printf("op"); break; |
| case CRNG_KEYWORD: printf("keyword"); break; |
| case CRNG_IDENTIFIER: printf("identifier"); break; |
| case CRNG_LITERAL_SEGMENT: printf("literal"); break; |
| case CRNG_CNAME: printf("cname"); break; |
| case CRNG_QNAME: printf("qname"); break; |
| case CRNG_NSNAME: printf("nsname"); break; |
| case CRNG_DOCUMENTATION: printf("doc"); break; |
| } |
| printf(":%s\n", token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| } |
| } |
| |
| /** |
| * xmlParseCRNG_attribute: |
| * @ctxt: a compact RNG parser context |
| * @name: the attribute name |
| * @ns: the attribute namespace |
| * @value: the attribute value |
| * |
| * implements attribute of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_attribute(xmlCRelaxNGParserCtxtPtr ctxt, |
| const xmlChar *name, |
| xmlNsPtr ns, |
| const xmlChar *value) |
| { |
| xmlAttrPtr attr; |
| |
| attr = xmlNewNsPropEatName(NULL, ns, (xmlChar *) name, value); |
| if (attr == NULL) CRNG_MEM_ERROR0(); |
| attr->next = ctxt->attrs; |
| if (ctxt->attrs != NULL) |
| ctxt->attrs->prev = attr; |
| ctxt->attrs = attr; |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_bindPrefix: |
| * @ctxt: a compact RNG parser context |
| * @prefix: the namespace prefix or NULL |
| * @namespace: the namespace name |
| * |
| * implements bindPrefix of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_bindPrefix(xmlCRelaxNGParserCtxtPtr ctxt, |
| const xmlChar *prefix, |
| const xmlChar *namespace) |
| { |
| int ret; |
| |
| if ((prefix != NULL) && (xmlStrEqual(prefix, BAD_CAST "xml")) && |
| (!xmlStrEqual(namespace, XML_XML_NAMESPACE))) { |
| ERROR("The \"xml\" prefix must be bound to \"http://www.w3.org/XML/1998/namespace\""); |
| return(-1); |
| } else if ((xmlStrEqual(namespace, XML_XML_NAMESPACE)) && |
| (!xmlStrEqual(prefix, BAD_CAST "xml"))) { |
| ERROR("The \"http://www.w3.org/XML/1998/namespace\" name must be bound to \"xml\" prefix"); |
| return(-1); |
| } |
| if (ctxt->namespaces == NULL) |
| ctxt->namespaces = xmlHashCreate(10); |
| if (ctxt->namespaces == NULL) { |
| ERROR("Failed to create namespace hash table"); |
| return(-1); |
| } |
| if (prefix == NULL) |
| ret = xmlHashAddEntry(ctxt->namespaces, xmlCRelaxNGDefault, |
| (void *) namespace); |
| else |
| ret = xmlHashAddEntry(ctxt->namespaces, prefix, |
| (void *) namespace); |
| if (ret < 0) { |
| if (prefix == NULL) { |
| ERROR("Redefinition of default namespace"); |
| } else { |
| ERROR("Redefinition of namespace"); |
| } |
| return(-1); |
| } |
| |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_bindDatatypePrefix: |
| * @ctxt: a compact RNG parser context |
| * @prefix: the datatype prefix |
| * @namespace: the datatype identifier |
| * |
| * implements bindDatatypePrefix of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_bindDatatypePrefix(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED, |
| const xmlChar *prefix, |
| const xmlChar *namespace) |
| { |
| int ret; |
| |
| if ((prefix != NULL) && (xmlStrEqual(prefix, BAD_CAST "xsd")) && |
| (!xmlStrEqual(namespace, |
| BAD_CAST "http://www.w3.org/2001/XMLSchema-datatypes"))) { |
| ERROR("The \"xsd\" prefix must be bound to \"http://www.w3.org/2001/XMLSchema-datatypes\""); |
| return(-1); |
| } |
| if (ctxt->datatypes == NULL) |
| ctxt->datatypes = xmlHashCreate(10); |
| if (ctxt->datatypes == NULL) { |
| ERROR("Failed to create namespace hash table"); |
| return(-1); |
| } |
| ret = xmlHashAddEntry(ctxt->datatypes, prefix, |
| (void *) namespace); |
| if (ret < 0) { |
| ERROR("Redefinition of datatype"); |
| return(-1); |
| } |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_lookupPrefix: |
| * @ctxt: a compact RNG parser context |
| * @prefix: the namespace prefix or NULL |
| * |
| * implements lookupPrefix of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns the prefix in case of success or NULL in case of error |
| */ |
| static const xmlChar * |
| xmlParseCRNG_lookupPrefix(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED, |
| const xmlChar *prefix) |
| { |
| const xmlChar *ret; |
| |
| if (prefix == NULL) |
| ret = xmlHashLookup(ctxt->namespaces, xmlCRelaxNGDefault); |
| else |
| ret = xmlHashLookup(ctxt->namespaces, prefix); |
| return(ret); |
| } |
| |
| /** |
| * xmlParseCRNG_lookupDatatypePrefix: |
| * @ctxt: a compact RNG parser context |
| * @prefix: the namespace prefix or NULL |
| * |
| * implements lookupDatatypePrefix of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns the prefix in case of success or NULL in case of error |
| */ |
| static const xmlChar * |
| xmlParseCRNG_lookupDatatypePrefix(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED, |
| const xmlChar *prefix) |
| { |
| const xmlChar *ret; |
| ret = xmlHashLookup(ctxt->datatypes, prefix); |
| return(ret); |
| } |
| |
| /** |
| * xmlParseCRNG_datatypeAttributes: |
| * @ctxt: a compact RNG parser context |
| * @prefix: the namespace prefix or NULL |
| * |
| * implements lookupPrefix of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns the prefix in case of success or NULL in case of error |
| */ |
| static xmlAttrPtr |
| xmlParseCRNG_datatypeAttributes(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED, |
| const xmlChar *library, const xmlChar *type) |
| { |
| xmlAttrPtr lib, typ; |
| |
| lib = xmlNewNsProp(NULL, NULL, BAD_CAST "datatypeLibrary", library); |
| if (lib == NULL) { |
| CRNG_MEM_ERROR(); |
| return(NULL); |
| } |
| typ = xmlNewNsProp(NULL, NULL, BAD_CAST "type", type); |
| if (typ == NULL) { |
| CRNG_MEM_ERROR(); |
| return(lib); |
| } |
| lib->next = typ; |
| |
| return(lib); |
| } |
| |
| /** |
| * xmlParseCRNG_XXX: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse XXX of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_XXX(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED) |
| { |
| return(0); |
| } |
| |
| static int xmlParseCRNG_pattern(xmlCRelaxNGParserCtxtPtr ctxt); |
| static int xmlParseCRNG_nameClass(xmlCRelaxNGParserCtxtPtr ctxt); |
| |
| /** |
| * xmlParseCRNG_params: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse params of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_params(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED) |
| { |
| TODO |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_exceptNameClass: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse exceptNameClass of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_exceptNameClass(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED) |
| { |
| tokenPtr token; |
| xmlNodePtr insert = ctxt->insert, cur; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype == CRNG_OP) && |
| (token->token[0] == '-') && (token->token[1] == 0)) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| cur = xmlNewNode(NULL, BAD_CAST "except"); |
| if (cur == NULL) CRNG_MEM_ERROR0(); |
| if (ctxt->insert != NULL) |
| xmlAddChild(ctxt->insert, cur); |
| ctxt->insert = cur; |
| xmlParseCRNG_nameClass(ctxt); |
| } |
| ctxt->insert = insert; |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_innerNameClass: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse innerNameClass of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_innerNameClass(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| xmlNodePtr cur; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype == CRNG_OP) { |
| if ((token->token[0] == '(') && (token->token[1] == 0)) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| xmlParseCRNG_nameClass(ctxt); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != ')') || (token->token[1] != 0)) { |
| ERROR("Expecting \")\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } else if ((token->token[0] == '*') && (token->token[1] == 0)) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| cur = xmlNewNode(NULL, BAD_CAST "anyName"); |
| if (cur == NULL) CRNG_MEM_ERROR0(); |
| if (ctxt->insert != NULL) |
| xmlAddChild(ctxt->insert, cur); |
| ctxt->insert = cur; |
| xmlParseCRNG_exceptNameClass(ctxt); |
| } else { |
| TODO |
| } |
| } else if ((token->toktype == CRNG_IDENTIFIER) || |
| (token->toktype == CRNG_KEYWORD)) { |
| cur = xmlNewNode(NULL, BAD_CAST "name"); |
| if (cur == NULL) CRNG_MEM_ERROR0(); |
| if (ctxt->isElem) { |
| xmlSetProp(cur, BAD_CAST "ns", |
| xmlParseCRNG_lookupPrefix(ctxt, NULL)); |
| } else { |
| xmlSetProp(cur, BAD_CAST "ns", BAD_CAST ""); |
| } |
| xmlNodeAddContent(cur, token->token); |
| if (ctxt->insert != NULL) |
| xmlAddChild(ctxt->insert, cur); |
| ctxt->insert = cur; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } else if (token->toktype == CRNG_CNAME) { |
| TODO |
| } else if (token->toktype == CRNG_NSNAME) { |
| cur = xmlNewNode(NULL, BAD_CAST "nsName"); |
| if (cur == NULL) CRNG_MEM_ERROR0(); |
| xmlSetProp(cur, BAD_CAST "ns", |
| xmlParseCRNG_lookupPrefix(ctxt, token->token)); |
| if (ctxt->insert != NULL) |
| xmlAddChild(ctxt->insert, cur); |
| ctxt->insert = cur; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| xmlParseCRNG_exceptNameClass(ctxt); |
| } else { |
| TODO /* probably an error */ |
| } |
| |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_nameClass: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse nameClass of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_nameClass(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| xmlNodePtr insert = ctxt->insert, last, choice; |
| |
| ctxt->insert = NULL; |
| xmlParseCRNG_innerNameClass(ctxt); |
| last = ctxt->insert; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| while ((token->toktype == CRNG_OP) && |
| (token->token[0] == '|') && (token->token[1] == 0)) { |
| choice = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_choice); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (choice == NULL) CRNG_MEM_ERROR0(); |
| ctxt->insert = NULL; |
| xmlParseCRNG_innerNameClass(ctxt); |
| xmlAddChild(choice, last); |
| xmlAddChild(choice, ctxt->insert); |
| last = choice; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| } |
| xmlAddChild(insert, last); |
| |
| ctxt->insert = insert; |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_patternBlock: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse a pattern block of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_patternBlock(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != '{') || (token->token[1] != 0)) { |
| ERROR("Expecting \"{\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| xmlParseCRNG_pattern(ctxt); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != '}') || (token->token[1] != 0)) { |
| ERROR("Expecting \"}\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_datatype: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse datatype of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_datatype(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED) |
| { |
| tokenPtr token; |
| xmlAttrPtr attrs = NULL; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype == CRNG_KEYWORD) { |
| if (token->token == ctxt->key_string) { |
| attrs = xmlParseCRNG_datatypeAttributes(ctxt, BAD_CAST "", |
| token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } else if (token->token == ctxt->key_token) { |
| attrs = xmlParseCRNG_datatypeAttributes(ctxt, BAD_CAST "", |
| token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } else { |
| TODO /* probably an error */ |
| } |
| } else if (token->toktype == CRNG_LITERAL_SEGMENT) { |
| ctxt->insert = xmlNewNode(NULL, BAD_CAST "value"); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| xmlNodeAddContent(ctxt->insert, token->token); |
| } else if (token->toktype == CRNG_QNAME) { |
| attrs = xmlParseCRNG_datatypeAttributes(ctxt, |
| xmlParseCRNG_lookupDatatypePrefix(ctxt, token->prefix), |
| token->token); |
| } else { |
| TODO |
| } |
| if (attrs != NULL) { |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype == CRNG_LITERAL_SEGMENT) { |
| ctxt->insert = xmlNewNode(NULL, BAD_CAST "value"); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) { |
| xmlFreePropList(attrs); |
| CRNG_MEM_ERROR0(); |
| } |
| ctxt->insert->properties = attrs; |
| xmlNodeAddContent(ctxt->insert, token->token); |
| } else if ((token->toktype == CRNG_OP) && |
| (token->token[0] == '{') && (token->token[0] == 0)) { |
| ctxt->insert = xmlNewNode(NULL, BAD_CAST "data"); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) { |
| xmlFreePropList(attrs); |
| CRNG_MEM_ERROR0(); |
| } |
| ctxt->insert->properties = attrs; |
| xmlParseCRNG_params(ctxt); |
| } else { |
| ctxt->insert = xmlNewNode(NULL, BAD_CAST "data"); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) { |
| xmlFreePropList(attrs); |
| CRNG_MEM_ERROR0(); |
| } |
| ctxt->insert->properties = attrs; |
| xmlNodeAddContent(ctxt->insert, token->token); |
| } |
| } |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_primary: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse primary of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_primary(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED) |
| { |
| tokenPtr token; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token == NULL) |
| return(0); |
| if (token->toktype == CRNG_KEYWORD) { |
| if (token->token == ctxt->key_element) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| ctxt->isElem = 1; |
| xmlParseCRNG_nameClass(ctxt); |
| xmlParseCRNG_patternBlock(ctxt); |
| } else if (token->token == ctxt->key_attribute) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| ctxt->isElem = 0; |
| xmlParseCRNG_nameClass(ctxt); |
| xmlParseCRNG_patternBlock(ctxt); |
| } else if (token->token == ctxt->key_mixed) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| xmlParseCRNG_patternBlock(ctxt); |
| } else if (token->token == ctxt->key_list) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| xmlParseCRNG_patternBlock(ctxt); |
| } else if (token->token == ctxt->key_empty) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| } else if (token->token == ctxt->key_notAllowed) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| } else if (token->token == ctxt->key_text) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| } else if (token->token == ctxt->key_parent) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| TODO |
| } else if (token->token == ctxt->key_grammar) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| TODO |
| } else if (token->token == ctxt->key_external) { |
| ctxt->insert = xmlNewNode(NULL, BAD_CAST "externalRef"); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| TODO |
| } else { |
| TODO |
| } |
| } else if (token->toktype == CRNG_IDENTIFIER) { |
| ctxt->insert = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_ref); |
| if (ctxt->insert == NULL) CRNG_MEM_ERROR0(); |
| xmlSetProp(ctxt->insert, BAD_CAST "name", token->token); |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } else if (token->toktype == CRNG_QNAME) { |
| xmlParseCRNG_datatype(ctxt); |
| } else if (token->toktype == CRNG_LITERAL_SEGMENT) { |
| xmlParseCRNG_datatype(ctxt); |
| } else if ((token->toktype == CRNG_OP) && |
| (token->token[0] == '(') && (token->token[1] == 0)) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| xmlParseCRNG_pattern(ctxt); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != ')') || (token->token[1] != 0)) { |
| ERROR("Expecting \")\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_particle: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse particle of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_particle(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| xmlNodePtr insert = ctxt->insert, res, tmp = NULL; |
| |
| ctxt->insert = NULL; |
| xmlParseCRNG_primary(ctxt); |
| res = ctxt->insert; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token != NULL) && (token->toktype == CRNG_OP)) { |
| if ((token->token[0] == '*') && (token->token[1] == 0)) { |
| tmp = xmlNewNode(NULL, BAD_CAST "zeroOrMore"); |
| if (tmp == NULL) CRNG_MEM_ERROR0(); |
| } else if ((token->token[0] == '+') && (token->token[1] == 0)) { |
| tmp = xmlNewNode(NULL, BAD_CAST "oneOrMore"); |
| if (tmp == NULL) CRNG_MEM_ERROR0(); |
| } else if ((token->token[0] == '?') && (token->token[1] == 0)) { |
| tmp = xmlNewNode(NULL, BAD_CAST "optional"); |
| if (tmp == NULL) CRNG_MEM_ERROR0(); |
| } |
| if (tmp != NULL) { |
| xmlAddChild(tmp, res); |
| res = tmp; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| } |
| } |
| if (insert != NULL) { |
| xmlAddChild(insert, res); |
| ctxt->insert = insert; |
| } else |
| ctxt->insert = res; |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_pattern: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse pattern of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_pattern(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| xmlNodePtr insert = ctxt->insert, prev, grp; |
| |
| ctxt->insert = NULL; |
| xmlParseCRNG_particle(ctxt); |
| prev = ctxt->insert; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| while ((prev != NULL) && (token != NULL) && (token->toktype == CRNG_OP)) { |
| if (token->token == ctxt->key_or) { |
| grp = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_choice); |
| if (grp == NULL) CRNG_MEM_ERROR0(); |
| } else if (token->token == ctxt->key_and) { |
| grp = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_interleave); |
| if (grp == NULL) CRNG_MEM_ERROR0(); |
| } else if (token->token == ctxt->key_comma) { |
| grp = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_group); |
| if (grp == NULL) CRNG_MEM_ERROR0(); |
| } else |
| break; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| ctxt->insert = NULL; |
| xmlParseCRNG_particle(ctxt); |
| xmlAddChild(grp, prev); |
| xmlAddChild(grp, ctxt->insert); |
| prev = grp; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| } |
| if (insert != NULL) { |
| xmlAddChild(insert, prev); |
| ctxt->insert = insert; |
| } else { |
| ctxt->insert = prev; |
| } |
| |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_component: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse component of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_component(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token, tok2; |
| xmlNodePtr insert = ctxt->insert; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token == NULL) |
| return(0); |
| if (token->toktype == CRNG_KEYWORD) { |
| if (token->token == ctxt->key_start) { |
| xmlNodePtr start; |
| |
| start = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_start); |
| if (start == NULL) CRNG_MEM_ERROR0(); |
| if (ctxt->insert != NULL) |
| xmlAddChild(ctxt->insert, start); |
| ctxt->insert = start; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| |
| if ((token->toktype == CRNG_OP) && |
| (token->token == ctxt->key_equal)) { |
| } else if ((token->toktype == CRNG_OP) && |
| (token->token == ctxt->key_orequal)) { |
| xmlParseCRNG_attribute(ctxt, ctxt->key_combine, NULL, |
| BAD_CAST "choice"); |
| } else if ((token->toktype == CRNG_OP) && |
| (token->token == ctxt->key_andequal)) { |
| xmlParseCRNG_attribute(ctxt, ctxt->key_combine, NULL, |
| BAD_CAST "interleave"); |
| } else { |
| ERROR("expecting \"=\" or \"&=\" or \"|=\" here") |
| return(-1); |
| } |
| start->properties = ctxt->attrs; |
| ctxt->attrs = NULL; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| xmlParseCRNG_pattern(ctxt); |
| |
| } else if (token->token == ctxt->key_include) { |
| TODO |
| } else if (token->token == ctxt->key_div) { |
| TODO |
| } else { |
| return(-1); |
| } |
| } else if (token->toktype == CRNG_IDENTIFIER) { |
| xmlNodePtr define; |
| const xmlChar *identifier; |
| |
| identifier = token->token; |
| tok2 = xmlParseCRNGGetToken(ctxt, 2); |
| if ((tok2->toktype == CRNG_OP) && |
| (tok2->token == ctxt->key_equal)) { |
| } else if ((tok2->toktype == CRNG_OP) && |
| (tok2->token == ctxt->key_orequal)) { |
| xmlParseCRNG_attribute(ctxt, ctxt->key_combine, NULL, |
| BAD_CAST "choice"); |
| } else if ((tok2->toktype == CRNG_OP) && |
| (tok2->token == ctxt->key_andequal)) { |
| xmlParseCRNG_attribute(ctxt, ctxt->key_combine, NULL, |
| BAD_CAST "interleave"); |
| } else { |
| ERROR("expecting \"=\" or \"&=\" or \"|=\" here") |
| return(-1); |
| } |
| xmlParseCRNGDropTokens(ctxt, 2); |
| |
| define = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_define); |
| if (define == NULL) CRNG_MEM_ERROR0(); |
| define->properties = ctxt->attrs; |
| ctxt->attrs = NULL; |
| xmlSetProp(define, BAD_CAST "name", identifier); |
| if (ctxt->insert != NULL) |
| xmlAddChild(ctxt->insert, define); |
| ctxt->insert = define; |
| xmlParseCRNG_pattern(ctxt); |
| } else { |
| return(-1); |
| } |
| ctxt->insert = insert; |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_grammar: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse grammar of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_grammar(xmlCRelaxNGParserCtxtPtr ctxt ATTRIBUTE_UNUSED) |
| { |
| tokenPtr token; |
| int ret; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| while (token != NULL) { |
| ret = xmlParseCRNG_component(ctxt); |
| if (ret != 0) |
| break; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| } |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_topLevelBody: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse topLevelBody of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_topLevelBody(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token, tok2; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype == CRNG_KEYWORD) { |
| if ((token->token == ctxt->key_start) || |
| (token->token == ctxt->key_include) || |
| (token->token == ctxt->key_div)) { |
| xmlNodePtr grammar; |
| |
| grammar = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_grammar); |
| if (grammar == NULL) CRNG_MEM_ERROR0(); |
| xmlDocSetRootElement(ctxt->doc, grammar); |
| ctxt->insert = grammar; |
| |
| xmlParseCRNG_grammar(ctxt); |
| } else { |
| xmlParseCRNG_pattern(ctxt); |
| } |
| } else { |
| tok2 = xmlParseCRNGGetToken(ctxt, 2); |
| if ((tok2->toktype == CRNG_OP) && |
| ((tok2->token == ctxt->key_equal) || |
| (tok2->token == ctxt->key_orequal) || |
| (tok2->token == ctxt->key_andequal))) { |
| xmlNodePtr grammar; |
| |
| grammar = xmlNewNodeEatName(NULL, (xmlChar *) ctxt->key_grammar); |
| if (grammar == NULL) CRNG_MEM_ERROR0(); |
| xmlDocSetRootElement(ctxt->doc, grammar); |
| ctxt->insert = grammar; |
| |
| xmlParseCRNG_grammar(ctxt); |
| } else { |
| xmlParseCRNG_pattern(ctxt); |
| } |
| } |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_namespacePrefix: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse namespacePrefix of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns the prefix or NULL in case of error |
| */ |
| static const xmlChar * |
| xmlParseCRNG_namespacePrefix(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| const xmlChar *prefix = NULL; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype == CRNG_IDENTIFIER) { |
| prefix = token->token; |
| } else if (token->toktype == CRNG_OP) { |
| if ((token->token[0] == '=') && (token->token[1] == 0)) |
| return(NULL); |
| prefix = token->token; |
| } else { |
| ERROR("Expecting a namespace prefix"); |
| return(NULL); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| |
| if (xmlStrEqual(prefix, BAD_CAST "xmlns")) { |
| ERROR("Namespace prefix \"xmlns\" is forbidden"); |
| } |
| return(prefix); |
| } |
| |
| /** |
| * xmlParseCRNG_decl: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse decl of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_decl(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| const xmlChar *prefix = NULL; |
| const xmlChar *namespace = NULL; |
| tokenPtr token; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype != CRNG_KEYWORD) return(-1); |
| if (token->token == ctxt->key_default) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_KEYWORD) || |
| (token->token != ctxt->key_namespace)) { |
| ERROR("Expecting keyword \"namespace\" after \"default\""); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| prefix = xmlParseCRNG_namespacePrefix(ctxt); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != '=') || (token->token[1] != 0)) { |
| ERROR("Expecting keyword \"=\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype == CRNG_KEYWORD) && |
| (token->token == ctxt->key_inherit)) { |
| namespace = xmlCRelaxNGInherit; |
| } else if (token->toktype == CRNG_LITERAL_SEGMENT) { |
| namespace = token->token; |
| } else { |
| ERROR("Expecting an URI or \"inherit\" value"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (namespace != NULL) { |
| if (prefix != NULL) |
| xmlParseCRNG_bindPrefix(ctxt, prefix, namespace); |
| xmlParseCRNG_bindPrefix(ctxt, NULL, namespace); |
| } |
| } else if (token->token == ctxt->key_namespace) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| prefix = xmlParseCRNG_namespacePrefix(ctxt); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != '=') || (token->token[1] != 0)) { |
| ERROR("Expecting keyword \"=\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype == CRNG_KEYWORD) && |
| (token->token == ctxt->key_inherit)) { |
| namespace = xmlCRelaxNGInherit; |
| } else if (token->toktype == CRNG_LITERAL_SEGMENT) { |
| namespace = token->token; |
| } else { |
| ERROR("Expecting an URI or \"inherit\" value"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if (namespace != NULL) |
| xmlParseCRNG_bindPrefix(ctxt, prefix, namespace); |
| } else if (token->token == ctxt->key_datatypes) { |
| xmlParseCRNGDropTokens(ctxt, 1); |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_KEYWORD) && |
| (token->toktype != CRNG_IDENTIFIER)) { |
| ERROR("Expecting a datatype prefix identifier here"); |
| } else |
| prefix = token->token; |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if ((token->toktype != CRNG_OP) || |
| (token->token[0] != '=') || (token->token[1] != 0)) { |
| ERROR("Expecting keyword \"=\" here"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| if (token->toktype == CRNG_LITERAL_SEGMENT) { |
| namespace = token->token; |
| } else { |
| ERROR("Expecting a literal value for the datatype identifier"); |
| } |
| xmlParseCRNGDropTokens(ctxt, 1); |
| if ((namespace != NULL) && (prefix != NULL)) |
| xmlParseCRNG_bindDatatypePrefix(ctxt, prefix, namespace); |
| } |
| |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_preamble: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse preamble of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_preamble(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| tokenPtr token; |
| |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| while (token != NULL) { |
| if (token == NULL) return(-1); |
| if ((token->toktype == CRNG_KEYWORD) && |
| ((token->token == ctxt->key_default) || |
| (token->token == ctxt->key_namespace) || |
| (token->token == ctxt->key_datatypes))) { |
| xmlParseCRNG_decl(ctxt); |
| } else |
| break; |
| token = xmlParseCRNGGetToken(ctxt, 1); |
| } |
| return(0); |
| } |
| |
| /** |
| * xmlParseCRNG_topLevel: |
| * @ctxt: a compact RNG parser context |
| * |
| * Parse topLevel of the RELAX NG Compact Syntax Appendix A |
| * |
| * Returns 0 in case of success and -1 in case of error |
| */ |
| static int |
| xmlParseCRNG_topLevel(xmlCRelaxNGParserCtxtPtr ctxt) |
| { |
| xmlParseCRNG_preamble(ctxt); |
| xmlParseCRNG_topLevelBody(ctxt); |
| return(0); |
| } |
| |
| /** |
| * xmlConvertCRNG: |
| * @schemas: pointer to the text of the compact schemas |
| * @len: length of the schemas in bytes (or 0) |
| * @encoding: encoding indicated by the context or NULL |
| * |
| * Compiles the schemas into the equivalent Relax-NG XML structure |
| * |
| * Returns the xmlDocPtr resulting from the compilation or |
| * NULL in case of error |
| */ |
| xmlDocPtr |
| xmlConvertCRNG(const char *schemas, int len, const char *encoding) { |
| struct _xmlCRelaxNGParserCtxt ctxt; |
| xmlDocPtr ret = NULL; |
| |
| if (schemas == NULL) return(NULL); |
| if (len <= 5) len = xmlStrlen((const unsigned char *) schemas); |
| if (len <= 0) return(NULL); |
| |
| memset(&ctxt, 0, sizeof(ctxt)); |
| ctxt.compact = (const unsigned char *) schemas; |
| ctxt.cur = (const unsigned char *) schemas; |
| ctxt.end = (const unsigned char *) &schemas[len]; |
| ctxt.dict = xmlDictCreate(); |
| if (ctxt.dict == NULL) |
| return(NULL); |
| ctxt.doc = xmlNewDoc(NULL); |
| if (ctxt.doc == NULL) { |
| xmlDictFree(ctxt.dict); |
| return(NULL); |
| } |
| ctxt.doc->dict = ctxt.dict; |
| xmlDictReference(ctxt.dict); |
| |
| ctxt.nbTokens = 0; |
| ctxt.firstToken = 0; |
| ctxt.key_attribute = xmlDictLookup(ctxt.dict, BAD_CAST "attribute", -1); |
| ctxt.key_default = xmlDictLookup(ctxt.dict, BAD_CAST "default", -1); |
| ctxt.key_datatypes = xmlDictLookup(ctxt.dict, BAD_CAST "datatypes", -1); |
| ctxt.key_div = xmlDictLookup(ctxt.dict, BAD_CAST "div", -1); |
| ctxt.key_element = xmlDictLookup(ctxt.dict, BAD_CAST "element", -1); |
| ctxt.key_empty = xmlDictLookup(ctxt.dict, BAD_CAST "empty", -1); |
| ctxt.key_external = xmlDictLookup(ctxt.dict, BAD_CAST "external", -1); |
| ctxt.key_grammar = xmlDictLookup(ctxt.dict, BAD_CAST "grammar", -1); |
| ctxt.key_include = xmlDictLookup(ctxt.dict, BAD_CAST "include", -1); |
| ctxt.key_inherit = xmlDictLookup(ctxt.dict, BAD_CAST "inherit", -1); |
| ctxt.key_list = xmlDictLookup(ctxt.dict, BAD_CAST "list", -1); |
| ctxt.key_mixed = xmlDictLookup(ctxt.dict, BAD_CAST "mixed", -1); |
| ctxt.key_namespace = xmlDictLookup(ctxt.dict, BAD_CAST "namespace", -1); |
| ctxt.key_notAllowed = xmlDictLookup(ctxt.dict, BAD_CAST "notAllowed", -1); |
| ctxt.key_parent = xmlDictLookup(ctxt.dict, BAD_CAST "parent", -1); |
| ctxt.key_start = xmlDictLookup(ctxt.dict, BAD_CAST "start", -1); |
| ctxt.key_string = xmlDictLookup(ctxt.dict, BAD_CAST "string", -1); |
| ctxt.key_text = xmlDictLookup(ctxt.dict, BAD_CAST "text", -1); |
| ctxt.key_token = xmlDictLookup(ctxt.dict, BAD_CAST "token", -1); |
| ctxt.key_equal = xmlDictLookup(ctxt.dict, BAD_CAST "=", 1); |
| ctxt.key_orequal = xmlDictLookup(ctxt.dict, BAD_CAST "|=", 2); |
| ctxt.key_andequal = xmlDictLookup(ctxt.dict, BAD_CAST "&=", 2); |
| ctxt.key_combine = xmlDictLookup(ctxt.dict, BAD_CAST "&=", 2); |
| ctxt.key_or = xmlDictLookup(ctxt.dict, BAD_CAST "|", 1); |
| ctxt.key_comma = xmlDictLookup(ctxt.dict, BAD_CAST ",", 1); |
| ctxt.key_and = xmlDictLookup(ctxt.dict, BAD_CAST "&", 1); |
| ctxt.key_choice = xmlDictLookup(ctxt.dict, BAD_CAST "choice", -1); |
| ctxt.key_group = xmlDictLookup(ctxt.dict, BAD_CAST "group", -1); |
| ctxt.key_interleave = xmlDictLookup(ctxt.dict, BAD_CAST "interleave", -1); |
| ctxt.key_ref = xmlDictLookup(ctxt.dict, BAD_CAST "ref", 3); |
| ctxt.key_define = xmlDictLookup(ctxt.dict, BAD_CAST "define", 6); |
| |
| /* xmlConvertCRNGTokenize(&ctxt); */ |
| xmlConvertCRNG_topLevel(&ctxt); |
| |
| xmlDictFree(ctxt.dict); |
| |
| ret = ctxt.doc; |
| return(ret); |
| } |
| |
| /** |
| * xmlConvertCRNGFile: |
| * @URL: URL or filename for the resource |
| * @encoding: encoding indicated by the context or NULL |
| * |
| * Compiles the schemas into the equivalent Relax-NG XML structure |
| * |
| * Returns the xmlDocPtr resulting from the compilation or |
| * NULL in case of error |
| */ |
| xmlDocPtr |
| xmlConvertCRNGFile(const char *URL, const char *encoding) { |
| } |
| |
| #ifdef STANDALONE |
| const xmlChar *schemas = |
| "# RELAX NG XML syntax specified in compact syntax.\n\ |
| \n\ |
| default namespace rng = \"http://relaxng.org/ns/structure/1.0\"\n\ |
| namespace local = \"\"\n\ |
| datatypes xsd = \"http://www.w3.org/2001/XMLSchema-datatypes\"\n\ |
| \n\ |
| start = pattern\n\ |
| \n\ |
| pattern =\n\ |
| element element { (nameQName | nameClass), (common & pattern+) }\n\ |
| | element attribute { (nameQName | nameClass), (common & pattern?) }\n\ |
| | element group|interleave|choice|optional\n\ |
| |zeroOrMore|oneOrMore|list|mixed { common & pattern+ }\n\ |
| | element ref|parentRef { nameNCName, common }\n\ |
| | element empty|notAllowed|text { common }\n\ |
| | element data { type, param*, (common & exceptPattern?) }\n\ |
| | element value { commonAttributes, type?, xsd:string }\n\ |
| | element externalRef { href, common }\n\ |
| | element grammar { common & grammarContent* }\n\ |
| \n\ |
| param = element param { commonAttributes, nameNCName, xsd:string }\n\ |
| \n\ |
| exceptPattern = element except { common & pattern+ }\n\ |
| \n\ |
| grammarContent =\n\ |
| definition\n\ |
| | element div { common & grammarContent* }\n\ |
| | element include { href, (common & includeContent*) }\n\ |
| \n\ |
| includeContent =\n\ |
| definition\n\ |
| | element div { common & includeContent* }\n\ |
| \n\ |
| definition =\n\ |
| element start { combine?, (common & pattern+) }\n\ |
| | element define { nameNCName, combine?, (common & pattern+) }\n\ |
| \n\ |
| combine = attribute combine { \"choice\" | \"interleave\" }\n\ |
| \n\ |
| nameClass =\n\ |
| element name { commonAttributes, xsd:QName }\n\ |
| | element anyName { common & exceptNameClass? }\n\ |
| | element nsName { common & exceptNameClass? }\n\ |
| | element choice { common & nameClass+ }\n\ |
| \n\ |
| exceptNameClass = element except { common & nameClass+ }\n\ |
| \n\ |
| nameQName = attribute name { xsd:QName }\n\ |
| nameNCName = attribute name { xsd:NCName }\n\ |
| href = attribute href { xsd:anyURI }\n\ |
| type = attribute type { xsd:NCName }\n\ |
| \n\ |
| common = commonAttributes, foreignElement*\n\ |
| \n\ |
| commonAttributes =\n\ |
| attribute ns { xsd:string }?,\n\ |
| attribute datatypeLibrary { xsd:anyURI }?,\n\ |
| foreignAttribute*\n\ |
| \n\ |
| foreignElement = element * - rng:* { (anyAttribute | text | anyElement)* }\n\ |
| foreignAttribute = attribute * - (rng:*|local:*) { text }\n\ |
| anyElement = element * { (anyAttribute | text | anyElement)* }\n\ |
| anyAttribute = attribute * { text }\n\ |
| "; |
| |
| int main(int argc ATTRIBUTE_UNUSED, char **argv ATTRIBUTE_UNUSED) { |
| xmlDocPtr res; |
| |
| res = xmlConvertCRNG(schemas, -1); |
| if (res != NULL) { |
| xmlDocFormatDump(stdout, res, 1); |
| xmlFreeDoc(res); |
| } |
| return(0); |
| } |
| #endif |