]> mj.ucw.cz Git - libucw.git/blob - sherlock/xml/xml.h
8e416dbc8a775966ac2fe83c66a0227c7408aeb7
[libucw.git] / sherlock / xml / xml.h
1 /*
2  *      Sherlock Library -- A simple XML parser
3  *
4  *      (c) 2007--2008 Pavel Charvat <pchar@ucw.cz>
5  *
6  *      This software may be freely distributed and used according to the terms
7  *      of the GNU Lesser General Public License.
8  */
9
10 #ifndef _SHERLOCK_XML_XML_H
11 #define _SHERLOCK_XML_XML_H
12
13 #include "lib/clists.h"
14 #include "lib/slists.h"
15 #include "lib/mempool.h"
16 #include "lib/fastbuf.h"
17
18 struct xml_context;
19 struct xml_source;
20 struct xml_dtd_entity;
21
22 enum xml_error {
23   XML_ERR_OK = 0,
24   XML_ERR_WARN = 1000,                                  /* Warning */
25   XML_ERR_ERROR = 2000,                                 /* Recoverable error */
26   XML_ERR_FATAL = 3000,                                 /* Unrecoverable error */
27   XML_ERR_EOF,
28 };
29
30 enum xml_state {
31   XML_STATE_EOF,                                        /* EOF or a fatal error */
32   XML_STATE_START,                                      /* Initial state */
33   XML_STATE_XML_DECL,                                   /* XML_PULL_XML_DECL */
34   XML_STATE_DOCTYPE_DECL,                               /* XML_PULL_DOCTYPE_DECL */
35   XML_STATE_CHARS,                                      /* XML_PULL_CHARS */
36   XML_STATE_STAG,                                       /* XML_PULL_STAG */
37   XML_STATE_ETAG,                                       /* XML_PULL_ETAG */
38   XML_STATE_COMMENT,                                    /* XML_PULL_COMMENT */
39   XML_STATE_PI,                                         /* XML_PULL_PI */
40
41   /* Internal states */
42   XML_STATE_CHARS_BEFORE_STAG,
43   XML_STATE_CHARS_BEFORE_ETAG,
44   XML_STATE_CHARS_BEFORE_CDATA,
45   XML_STATE_CHARS_BEFORE_COMMENT,
46   XML_STATE_CHARS_BEFORE_PI,
47   XML_STATE_PROLOG_COMMENT,
48   XML_STATE_PROLOG_PI,
49   XML_STATE_EPILOG_COMMENT,
50   XML_STATE_EPILOG_PI,
51 };
52
53 enum xml_pull {
54   XML_PULL_XML_DECL =                   0x00000001,     /* Stop after the XML declaration */
55   XML_PULL_DOCTYPE_DECL =               0x00000002,     /* Stop in the doctype declaration (before optional internal subset) */
56   XML_PULL_CHARS =                      0x00000004,
57   XML_PULL_STAG =                       0x00000008,
58   XML_PULL_ETAG =                       0x00000010,
59   XML_PULL_COMMENT =                    0x00000020,
60   XML_PULL_PI =                         0x00000040,
61   XML_PULL_ALL =                        0xffffffff,
62 };
63
64 enum xml_flags {
65   /* Enable reporting of various events via SAX and/or PUSH interface */
66   XML_REPORT_COMMENTS =                 0x00000001,     /* Report comments */
67   XML_REPORT_PIS =                      0x00000002,     /* Report processing instructions */
68   XML_REPORT_CHARS =                    0x00000004,     /* Report characters */
69   XML_REPORT_TAGS =                     0x00000008,     /* Report element starts/ends */
70   XML_REPORT_MISC = XML_REPORT_COMMENTS | XML_REPORT_PIS,
71   XML_REPORT_ALL = XML_REPORT_MISC | XML_REPORT_CHARS | XML_REPORT_TAGS,
72
73   /* Enable construction of DOM for these types */
74   XML_ALLOC_COMMENTS =                  0x00000010,     /* Create comment nodes */
75   XML_ALLOC_PIS =                       0x00000020,     /* Create processing instruction nodes */
76   XML_ALLOC_CHARS =                     0x00000040,     /* Create character nodes */
77   XML_ALLOC_TAGS =                      0x00000080,     /* Create element nodes */
78   XML_ALLOC_MISC = XML_ALLOC_COMMENTS | XML_ALLOC_PIS,
79   XML_ALLOC_ALL = XML_ALLOC_MISC | XML_ALLOC_CHARS | XML_ALLOC_TAGS,
80
81   /* Other parameters */
82   XML_VALIDATING =                      0x00000100,     /* Validate everything (not fully implemented!) */
83   XML_PARSE_DTD =                       0x00000200,     /* Enable parsing of DTD */
84
85   /* Internals, do not change! */
86   XML_EMPTY_ELEM_TAG =                  0x00010000,     /* The current element match EmptyElemTag */
87   XML_VERSION_1_1 =                     0x00020000,     /* XML version is 1.1, otherwise 1.0 */
88   XML_HAS_EXTERNAL_SUBSET =             0x00040000,     /* The document contains a reference to external DTD subset */
89   XML_HAS_INTERNAL_SUBSET =             0x00080000,     /* The document contains an internal subset */
90   XML_SRC_EOF =                         0x00100000,     /* EOF reached */
91   XML_SRC_EXPECTED_DECL =               0x00200000,     /* Just before optional or required XMLDecl/TextDecl */
92   XML_SRC_NEW_LINE =                    0x00400000,     /* The last read character is 0xD */
93   XML_SRC_SURROUND =                    0x00800000,     /* Surround the text with 0x20 (references to parameter entities) */
94   XML_SRC_DOCUMENT =                    0x01000000,     /* The document entity */
95   XML_SRC_EXTERNAL =                    0x02000000,     /* An external entity */
96 };
97
98 enum xml_node_type {
99   XML_NODE_ELEM,
100   XML_NODE_COMMENT,
101   XML_NODE_CHARS,
102   XML_NODE_PI,
103 };
104
105 #define XML_NODE_FOR_EACH(var, node) CLIST_FOR_EACH(struct xml_node *, var, (node)->sons)
106 #define XML_ATTR_FOR_EACH(var, node) SLIST_FOR_EACH(struct xml_attr *, var, (node)->attrs)
107
108 struct xml_node {
109   cnode n;                                              /* Node for list of parent's sons */
110   uns type;                                             /* XML_NODE_x */
111   struct xml_node *parent;                              /* Parent node */
112   char *name;                                           /* Element name / PI target */
113   clist sons;                                           /* Children nodes */
114   union {
115     struct {
116       char *text;                                       /* PI text / Comment / CDATA */
117       uns len;                                          /* Text length in bytes */
118     };
119     struct {
120       struct xml_dtd_elem *dtd;                         /* Element DTD */
121       slist attrs;                                      /* Link list of element attributes */
122     };
123   };
124   void *user;                                           /* User-defined (initialized to NULL) */
125 };
126
127 struct xml_attr {
128   snode n;                                              /* Node for elem->attrs */
129   struct xml_node *elem;                                /* Parent element */
130   struct xml_dtd_attr *dtd;                             /* Attribute DTD */
131   char *name;                                           /* Attribute name */
132   char *val;                                            /* Attribute value */
133   void *user;                                           /* User-defined (initialized to NULL) */
134 };
135
136 struct xml_context {
137   /* Error handling */
138   char *err_msg;                                        /* Last error message */
139   enum xml_error err_code;                              /* Last error code */
140   void *throw_buf;                                      /* Where to jump on error */
141   void (*h_warn)(struct xml_context *ctx);              /* Warning callback */
142   void (*h_error)(struct xml_context *ctx);             /* Recoverable error callback */
143   void (*h_fatal)(struct xml_context *ctx);             /* Unrecoverable error callback */
144
145   /* Memory management */
146   struct mempool *pool;                                 /* DOM pool */
147   struct mempool *stack;                                /* Stack pool (freed as soon as possible) */
148   struct xml_stack *stack_list;                         /* See xml_push(), xml_pop() */
149   uns flags;                                            /* XML_FLAG_x (restored on xml_pop()) */
150   uns depth;                                            /* Nesting level (for checking of valid source nesting -> valid pushes/pops on memory pools) */
151   struct fastbuf chars;                                 /* Character data / attribute value */
152   struct mempool_state chars_state;                     /* Mempool state before the current character block has started */
153   char *chars_trivial;                                  /* If not empty, it will be appended to chars */
154   void *tab_attrs;                                      /* Hash table of element attributes */
155
156   /* Input */
157   struct xml_source *src;                               /* Current source */
158   u32 *bptr, *bstop;                                    /* Buffer with preprocessed characters (validated UCS-4 + category flags) */
159   uns cat_chars;                                        /* Unicode range of supported characters (cdata, attribute values, ...) */
160   uns cat_unrestricted;                                 /* Unrestricted characters (may appear in document/external entities) */
161   uns cat_new_line;                                     /* New line characters */
162   uns cat_name;                                         /* Characters that may appear in names */
163   uns cat_sname;                                        /* Characters that may begin a name */
164
165   /* SAX-like interface */
166   void (*h_document_start)(struct xml_context *ctx);    /* Called before entering prolog */
167   void (*h_document_end)(struct xml_context *ctx);      /* Called after leaving epilog */
168   void (*h_xml_decl)(struct xml_context *ctx);          /* Called after the XML declaration */
169   void (*h_doctype_decl)(struct xml_context *ctx);      /* Called in the doctype declaration (before optional internal subset) */
170   void (*h_comment)(struct xml_context *ctx);           /* Called after a comment (only with XML_REPORT_COMMENTS) */
171   void (*h_pi)(struct xml_context *ctx);                /* Called after a processing instruction (only with XML_REPORT_PIS) */
172   void (*h_stag)(struct xml_context *ctx);              /* Called after STag or EmptyElemTag (only with XML_REPORT_TAGS) */
173   void (*h_etag)(struct xml_context *ctx);              /* Called before ETag or after EmptyElemTag (only with XML_REPORT_TAGS) */
174   void (*h_chars)(struct xml_context *ctx);             /* Called after some characters (only with XML_REPORT_CHARS) */
175   void (*h_block)(struct xml_context *ctx, char *text, uns len);        /* Called for each continuous block of characters not reported by h_cdata() (only with XML_REPORT_CHARS) */
176   void (*h_cdata)(struct xml_context *ctx, char *text, uns len);        /* Called for each CDATA section (only with XML_REPORT_CHARS) */
177   void (*h_dtd_start)(struct xml_context *ctx);         /* Called just after the DTD structure is initialized */
178   void (*h_dtd_end)(struct xml_context *ctx);           /* Called after DTD subsets subsets */
179   struct xml_dtd_entity *(*h_find_entity)(struct xml_context *ctx, char *name);         /* Called when needed to resolve a general entity */
180   void (*h_resolve_entity)(struct xml_context *ctx, struct xml_dtd_entity *ent);        /* User should push source fastbuf for a parsed external entity (either general or parameter) */
181
182   /* DOM */
183   struct xml_node *dom;                                 /* DOM root */
184   struct xml_node *node;                                /* Current DOM node */
185
186   char *version_str;
187   uns standalone;
188   char *doctype;                                        /* The document type (or NULL if unknown) */
189   char *system_id;                                      /* DTD external id */
190   char *public_id;                                      /* DTD public id */
191   struct xml_dtd *dtd;                                  /* The DTD structure (or NULL) */
192   uns state;                                            /* Current state for the PULL interface (XML_STATE_x) */
193   uns pull;                                             /* Parameters for the PULL interface (XML_PULL_x) */
194
195   void (*start_entity)(struct xml_context *ctx);
196   void (*end_entity)(struct xml_context *ctx);
197   void (*notation_decl)(struct xml_context *ctx);
198   void (*unparsed_entity_decl)(struct xml_context *ctx);
199 };
200
201 /* Initialize XML context */
202 void xml_init(struct xml_context *ctx);
203
204 /* Clean up all internal structures */
205 void xml_cleanup(struct xml_context *ctx);
206
207 /* Reuse XML context */
208 void xml_reset(struct xml_context *ctx);
209
210 /* Add XML source (fastbuf will be automatically closed) */
211 struct xml_source *xml_push_fastbuf(struct xml_context *ctx, struct fastbuf *fb);
212
213 /* Parse without the PUSH interface, return XML_ERR_x code (zero on success) */
214 uns xml_parse(struct xml_context *ctx);
215
216 /* Parse with the PUSH interface, return XML_STATE_x (zero on EOF or fatal error) */
217 uns xml_next(struct xml_context *ctx);
218
219 /* Returns the current row number in the document entity */
220 uns xml_row(struct xml_context *ctx);
221
222 /* Finds a given attribute value in a XML_NODE_ELEM node */
223 struct xml_attr *xml_attr_find(struct xml_context *ctx, struct xml_node *node, char *name);
224
225 /* The default value of h_find_entity(), knows &lt;, &gt;, &amp;, &apos; and &quot; */
226 struct xml_dtd_entity *xml_def_find_entity(struct xml_context *ctx, char *name);
227
228 /* The default value of h_resolve_entity(), throws an error */
229 void xml_def_resolve_entity(struct xml_context *ctx, struct xml_dtd_entity *ent);
230
231 /* Remove leading/trailing spaces and replaces sequences of spaces to a single space character (non-CDATA attribute normalization) */
232 uns xml_normalize_white(struct xml_context *ctx, char *value);
233
234 #endif