1 /*
2 * runtest.c: C program to run libxml2 regression tests without
3 * requiring make or Python, and reducing platform dependancies
4 * to a strict minimum.
5 *
6 * To compile on Unixes:
7 * cc -o runtest `xml2-config --cflags` runtest.c `xml2-config --libs` -lpthread
8 *
9 * See Copyright for the status of this software.
10 *
11 * daniel@veillard.com
12 */
13
14 #include "libxml.h"
15 #include <stdio.h>
16
17 #if !defined(_WIN32) || defined(__CYGWIN__)
18 #include <unistd.h>
19 #endif
20 #include <string.h>
21 #include <sys/types.h>
22 #include <sys/stat.h>
23 #include <fcntl.h>
24
25 #include <libxml/parser.h>
26 #include <libxml/parserInternals.h>
27 #include <libxml/tree.h>
28 #include <libxml/uri.h>
29 #include <libxml/encoding.h>
30
31 #ifdef LIBXML_OUTPUT_ENABLED
32 #ifdef LIBXML_READER_ENABLED
33 #include <libxml/xmlreader.h>
34 #endif
35
36 #ifdef LIBXML_XINCLUDE_ENABLED
37 #include <libxml/xinclude.h>
38 #endif
39
40 #ifdef LIBXML_XPATH_ENABLED
41 #include <libxml/xpath.h>
42 #include <libxml/xpathInternals.h>
43 #ifdef LIBXML_XPTR_ENABLED
44 #include <libxml/xpointer.h>
45 #endif
46 #endif
47
48 #ifdef LIBXML_SCHEMAS_ENABLED
49 #include <libxml/relaxng.h>
50 #include <libxml/xmlschemas.h>
51 #include <libxml/xmlschemastypes.h>
52 #endif
53
54 #ifdef LIBXML_PATTERN_ENABLED
55 #include <libxml/pattern.h>
56 #endif
57
58 #ifdef LIBXML_C14N_ENABLED
59 #include <libxml/c14n.h>
60 #endif
61
62 #ifdef LIBXML_HTML_ENABLED
63 #include <libxml/HTMLparser.h>
64 #include <libxml/HTMLtree.h>
65
66 /*
67 * pseudo flag for the unification of HTML and XML tests
68 */
69 #define XML_PARSE_HTML 1 << 24
70 #endif
71
72 #if defined(LIBXML_THREAD_ENABLED) && defined(LIBXML_CATALOG_ENABLED)
73 #include <libxml/globals.h>
74 #include <libxml/threads.h>
75 #include <libxml/parser.h>
76 #include <libxml/catalog.h>
77 #include <string.h>
78 #endif
79
80 /*
81 * O_BINARY is just for Windows compatibility - if it isn't defined
82 * on this system, avoid any compilation error
83 */
84 #ifdef O_BINARY
85 #define RD_FLAGS O_RDONLY | O_BINARY
86 #define WR_FLAGS O_WRONLY | O_CREAT | O_TRUNC | O_BINARY
87 #else
88 #define RD_FLAGS O_RDONLY
89 #define WR_FLAGS O_WRONLY | O_CREAT | O_TRUNC
90 #endif
91
92 typedef int (*functest) (const char *filename, const char *result,
93 const char *error, int options);
94
95 typedef struct testDesc testDesc;
96 typedef testDesc *testDescPtr;
97 struct testDesc {
98 const char *desc; /* descripton of the test */
99 functest func; /* function implementing the test */
100 const char *in; /* glob to path for input files */
101 const char *out; /* output directory */
102 const char *suffix;/* suffix for output files */
103 const char *err; /* suffix for error output files */
104 int options; /* parser options for the test */
105 };
106
107 static int update_results = 0;
108 static int checkTestFile(const char *filename);
109
110 #if defined(_WIN32) && !defined(__CYGWIN__)
111
112 #include <windows.h>
113 #include <io.h>
114
115 typedef struct
116 {
117 size_t gl_pathc; /* Count of paths matched so far */
118 char **gl_pathv; /* List of matched pathnames. */
119 size_t gl_offs; /* Slots to reserve in 'gl_pathv'. */
120 } glob_t;
121
122 #define GLOB_DOOFFS 0
glob(const char * pattern,ATTRIBUTE_UNUSED int flags,ATTRIBUTE_UNUSED int errfunc (const char * epath,int eerrno),glob_t * pglob)123 static int glob(const char *pattern, ATTRIBUTE_UNUSED int flags,
124 ATTRIBUTE_UNUSED int errfunc(const char *epath, int eerrno),
125 glob_t *pglob) {
126 glob_t *ret;
127 WIN32_FIND_DATA FindFileData;
128 HANDLE hFind;
129 unsigned int nb_paths = 0;
130 char directory[500];
131 int len;
132
133 if ((pattern == NULL) || (pglob == NULL)) return(-1);
134
135 strncpy(directory, pattern, 499);
136 for (len = strlen(directory);len >= 0;len--) {
137 if (directory[len] == '/') {
138 len++;
139 directory[len] = 0;
140 break;
141 }
142 }
143 if (len <= 0)
144 len = 0;
145
146
147 ret = pglob;
148 memset(ret, 0, sizeof(glob_t));
149
150 hFind = FindFirstFileA(pattern, &FindFileData);
151 if (hFind == INVALID_HANDLE_VALUE)
152 return(0);
153 nb_paths = 20;
154 ret->gl_pathv = (char **) malloc(nb_paths * sizeof(char *));
155 if (ret->gl_pathv == NULL) {
156 FindClose(hFind);
157 return(-1);
158 }
159 strncpy(directory + len, FindFileData.cFileName, 499 - len);
160 ret->gl_pathv[ret->gl_pathc] = strdup(directory);
161 if (ret->gl_pathv[ret->gl_pathc] == NULL)
162 goto done;
163 ret->gl_pathc++;
164 while(FindNextFileA(hFind, &FindFileData)) {
165 if (FindFileData.cFileName[0] == '.')
166 continue;
167 if (ret->gl_pathc + 2 > nb_paths) {
168 char **tmp = realloc(ret->gl_pathv, nb_paths * 2 * sizeof(char *));
169 if (tmp == NULL)
170 break;
171 ret->gl_pathv = tmp;
172 nb_paths *= 2;
173 }
174 strncpy(directory + len, FindFileData.cFileName, 499 - len);
175 ret->gl_pathv[ret->gl_pathc] = strdup(directory);
176 if (ret->gl_pathv[ret->gl_pathc] == NULL)
177 break;
178 ret->gl_pathc++;
179 }
180 ret->gl_pathv[ret->gl_pathc] = NULL;
181
182 done:
183 FindClose(hFind);
184 return(0);
185 }
186
187
188
globfree(glob_t * pglob)189 static void globfree(glob_t *pglob) {
190 unsigned int i;
191 if (pglob == NULL)
192 return;
193
194 for (i = 0;i < pglob->gl_pathc;i++) {
195 if (pglob->gl_pathv[i] != NULL)
196 free(pglob->gl_pathv[i]);
197 }
198 }
199
200 #else
201 #include <glob.h>
202 #endif
203
204 /************************************************************************
205 * *
206 * Libxml2 specific routines *
207 * *
208 ************************************************************************/
209
210 static int nb_tests = 0;
211 static int nb_errors = 0;
212 static int nb_leaks = 0;
213 static int extraMemoryFromResolver = 0;
214
215 static int
fatalError(void)216 fatalError(void) {
217 fprintf(stderr, "Exitting tests on fatal error\n");
218 exit(1);
219 }
220
221 /*
222 * We need to trap calls to the resolver to not account memory for the catalog
223 * which is shared to the current running test. We also don't want to have
224 * network downloads modifying tests.
225 */
226 static xmlParserInputPtr
testExternalEntityLoader(const char * URL,const char * ID,xmlParserCtxtPtr ctxt)227 testExternalEntityLoader(const char *URL, const char *ID,
228 xmlParserCtxtPtr ctxt) {
229 xmlParserInputPtr ret;
230
231 if (checkTestFile(URL)) {
232 ret = xmlNoNetExternalEntityLoader(URL, ID, ctxt);
233 } else {
234 int memused = xmlMemUsed();
235 ret = xmlNoNetExternalEntityLoader(URL, ID, ctxt);
236 extraMemoryFromResolver += xmlMemUsed() - memused;
237 }
238
239 return(ret);
240 }
241
242 /*
243 * Trapping the error messages at the generic level to grab the equivalent of
244 * stderr messages on CLI tools.
245 */
246 static char testErrors[32769];
247 static int testErrorsSize = 0;
248
249 static void XMLCDECL
testErrorHandler(void * ctx ATTRIBUTE_UNUSED,const char * msg,...)250 testErrorHandler(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...) {
251 va_list args;
252 int res;
253
254 if (testErrorsSize >= 32768)
255 return;
256 va_start(args, msg);
257 res = vsnprintf(&testErrors[testErrorsSize],
258 32768 - testErrorsSize,
259 msg, args);
260 va_end(args);
261 if (testErrorsSize + res >= 32768) {
262 /* buffer is full */
263 testErrorsSize = 32768;
264 testErrors[testErrorsSize] = 0;
265 } else {
266 testErrorsSize += res;
267 }
268 testErrors[testErrorsSize] = 0;
269 }
270
271 static void XMLCDECL
channel(void * ctx ATTRIBUTE_UNUSED,const char * msg,...)272 channel(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...) {
273 va_list args;
274 int res;
275
276 if (testErrorsSize >= 32768)
277 return;
278 va_start(args, msg);
279 res = vsnprintf(&testErrors[testErrorsSize],
280 32768 - testErrorsSize,
281 msg, args);
282 va_end(args);
283 if (testErrorsSize + res >= 32768) {
284 /* buffer is full */
285 testErrorsSize = 32768;
286 testErrors[testErrorsSize] = 0;
287 } else {
288 testErrorsSize += res;
289 }
290 testErrors[testErrorsSize] = 0;
291 }
292
293 /**
294 * xmlParserPrintFileContext:
295 * @input: an xmlParserInputPtr input
296 *
297 * Displays current context within the input content for error tracking
298 */
299
300 static void
xmlParserPrintFileContextInternal(xmlParserInputPtr input,xmlGenericErrorFunc chanl,void * data)301 xmlParserPrintFileContextInternal(xmlParserInputPtr input ,
302 xmlGenericErrorFunc chanl, void *data ) {
303 const xmlChar *cur, *base;
304 unsigned int n, col; /* GCC warns if signed, because compared with sizeof() */
305 xmlChar content[81]; /* space for 80 chars + line terminator */
306 xmlChar *ctnt;
307
308 if (input == NULL) return;
309 cur = input->cur;
310 base = input->base;
311 /* skip backwards over any end-of-lines */
312 while ((cur > base) && ((*(cur) == '\n') || (*(cur) == '\r'))) {
313 cur--;
314 }
315 n = 0;
316 /* search backwards for beginning-of-line (to max buff size) */
317 while ((n++ < (sizeof(content)-1)) && (cur > base) &&
318 (*(cur) != '\n') && (*(cur) != '\r'))
319 cur--;
320 if ((*(cur) == '\n') || (*(cur) == '\r')) cur++;
321 /* calculate the error position in terms of the current position */
322 col = input->cur - cur;
323 /* search forward for end-of-line (to max buff size) */
324 n = 0;
325 ctnt = content;
326 /* copy selected text to our buffer */
327 while ((*cur != 0) && (*(cur) != '\n') &&
328 (*(cur) != '\r') && (n < sizeof(content)-1)) {
329 *ctnt++ = *cur++;
330 n++;
331 }
332 *ctnt = 0;
333 /* print out the selected text */
334 chanl(data ,"%s\n", content);
335 /* create blank line with problem pointer */
336 n = 0;
337 ctnt = content;
338 /* (leave buffer space for pointer + line terminator) */
339 while ((n<col) && (n++ < sizeof(content)-2) && (*ctnt != 0)) {
340 if (*(ctnt) != '\t')
341 *(ctnt) = ' ';
342 ctnt++;
343 }
344 *ctnt++ = '^';
345 *ctnt = 0;
346 chanl(data ,"%s\n", content);
347 }
348
349 static void
testStructuredErrorHandler(void * ctx ATTRIBUTE_UNUSED,xmlErrorPtr err)350 testStructuredErrorHandler(void *ctx ATTRIBUTE_UNUSED, xmlErrorPtr err) {
351 char *file = NULL;
352 int line = 0;
353 int code = -1;
354 int domain;
355 void *data = NULL;
356 const char *str;
357 const xmlChar *name = NULL;
358 xmlNodePtr node;
359 xmlErrorLevel level;
360 xmlParserInputPtr input = NULL;
361 xmlParserInputPtr cur = NULL;
362 xmlParserCtxtPtr ctxt = NULL;
363
364 if (err == NULL)
365 return;
366
367 file = err->file;
368 line = err->line;
369 code = err->code;
370 domain = err->domain;
371 level = err->level;
372 node = err->node;
373 if ((domain == XML_FROM_PARSER) || (domain == XML_FROM_HTML) ||
374 (domain == XML_FROM_DTD) || (domain == XML_FROM_NAMESPACE) ||
375 (domain == XML_FROM_IO) || (domain == XML_FROM_VALID)) {
376 ctxt = err->ctxt;
377 }
378 str = err->message;
379
380 if (code == XML_ERR_OK)
381 return;
382
383 if ((node != NULL) && (node->type == XML_ELEMENT_NODE))
384 name = node->name;
385
386 /*
387 * Maintain the compatibility with the legacy error handling
388 */
389 if (ctxt != NULL) {
390 input = ctxt->input;
391 if ((input != NULL) && (input->filename == NULL) &&
392 (ctxt->inputNr > 1)) {
393 cur = input;
394 input = ctxt->inputTab[ctxt->inputNr - 2];
395 }
396 if (input != NULL) {
397 if (input->filename)
398 channel(data, "%s:%d: ", input->filename, input->line);
399 else if ((line != 0) && (domain == XML_FROM_PARSER))
400 channel(data, "Entity: line %d: ", input->line);
401 }
402 } else {
403 if (file != NULL)
404 channel(data, "%s:%d: ", file, line);
405 else if ((line != 0) && (domain == XML_FROM_PARSER))
406 channel(data, "Entity: line %d: ", line);
407 }
408 if (name != NULL) {
409 channel(data, "element %s: ", name);
410 }
411 if (code == XML_ERR_OK)
412 return;
413 switch (domain) {
414 case XML_FROM_PARSER:
415 channel(data, "parser ");
416 break;
417 case XML_FROM_NAMESPACE:
418 channel(data, "namespace ");
419 break;
420 case XML_FROM_DTD:
421 case XML_FROM_VALID:
422 channel(data, "validity ");
423 break;
424 case XML_FROM_HTML:
425 channel(data, "HTML parser ");
426 break;
427 case XML_FROM_MEMORY:
428 channel(data, "memory ");
429 break;
430 case XML_FROM_OUTPUT:
431 channel(data, "output ");
432 break;
433 case XML_FROM_IO:
434 channel(data, "I/O ");
435 break;
436 case XML_FROM_XINCLUDE:
437 channel(data, "XInclude ");
438 break;
439 case XML_FROM_XPATH:
440 channel(data, "XPath ");
441 break;
442 case XML_FROM_XPOINTER:
443 channel(data, "parser ");
444 break;
445 case XML_FROM_REGEXP:
446 channel(data, "regexp ");
447 break;
448 case XML_FROM_MODULE:
449 channel(data, "module ");
450 break;
451 case XML_FROM_SCHEMASV:
452 channel(data, "Schemas validity ");
453 break;
454 case XML_FROM_SCHEMASP:
455 channel(data, "Schemas parser ");
456 break;
457 case XML_FROM_RELAXNGP:
458 channel(data, "Relax-NG parser ");
459 break;
460 case XML_FROM_RELAXNGV:
461 channel(data, "Relax-NG validity ");
462 break;
463 case XML_FROM_CATALOG:
464 channel(data, "Catalog ");
465 break;
466 case XML_FROM_C14N:
467 channel(data, "C14N ");
468 break;
469 case XML_FROM_XSLT:
470 channel(data, "XSLT ");
471 break;
472 default:
473 break;
474 }
475 if (code == XML_ERR_OK)
476 return;
477 switch (level) {
478 case XML_ERR_NONE:
479 channel(data, ": ");
480 break;
481 case XML_ERR_WARNING:
482 channel(data, "warning : ");
483 break;
484 case XML_ERR_ERROR:
485 channel(data, "error : ");
486 break;
487 case XML_ERR_FATAL:
488 channel(data, "error : ");
489 break;
490 }
491 if (code == XML_ERR_OK)
492 return;
493 if (str != NULL) {
494 int len;
495 len = xmlStrlen((const xmlChar *)str);
496 if ((len > 0) && (str[len - 1] != '\n'))
497 channel(data, "%s\n", str);
498 else
499 channel(data, "%s", str);
500 } else {
501 channel(data, "%s\n", "out of memory error");
502 }
503 if (code == XML_ERR_OK)
504 return;
505
506 if (ctxt != NULL) {
507 xmlParserPrintFileContextInternal(input, channel, data);
508 if (cur != NULL) {
509 if (cur->filename)
510 channel(data, "%s:%d: \n", cur->filename, cur->line);
511 else if ((line != 0) && (domain == XML_FROM_PARSER))
512 channel(data, "Entity: line %d: \n", cur->line);
513 xmlParserPrintFileContextInternal(cur, channel, data);
514 }
515 }
516 if ((domain == XML_FROM_XPATH) && (err->str1 != NULL) &&
517 (err->int1 < 100) &&
518 (err->int1 < xmlStrlen((const xmlChar *)err->str1))) {
519 xmlChar buf[150];
520 int i;
521
522 channel(data, "%s\n", err->str1);
523 for (i=0;i < err->int1;i++)
524 buf[i] = ' ';
525 buf[i++] = '^';
526 buf[i] = 0;
527 channel(data, "%s\n", buf);
528 }
529 }
530
531 static void
initializeLibxml2(void)532 initializeLibxml2(void) {
533 xmlGetWarningsDefaultValue = 0;
534 xmlPedanticParserDefault(0);
535
536 xmlMemSetup(xmlMemFree, xmlMemMalloc, xmlMemRealloc, xmlMemoryStrdup);
537 xmlInitParser();
538 xmlSetExternalEntityLoader(testExternalEntityLoader);
539 xmlSetStructuredErrorFunc(NULL, testStructuredErrorHandler);
540 #ifdef LIBXML_SCHEMAS_ENABLED
541 xmlSchemaInitTypes();
542 xmlRelaxNGInitTypes();
543 #endif
544 }
545
546
547 /************************************************************************
548 * *
549 * File name and path utilities *
550 * *
551 ************************************************************************/
552
baseFilename(const char * filename)553 static const char *baseFilename(const char *filename) {
554 const char *cur;
555 if (filename == NULL)
556 return(NULL);
557 cur = &filename[strlen(filename)];
558 while ((cur > filename) && (*cur != '/'))
559 cur--;
560 if (*cur == '/')
561 return(cur + 1);
562 return(cur);
563 }
564
resultFilename(const char * filename,const char * out,const char * suffix)565 static char *resultFilename(const char *filename, const char *out,
566 const char *suffix) {
567 const char *base;
568 char res[500];
569 char suffixbuff[500];
570
571 /*************
572 if ((filename[0] == 't') && (filename[1] == 'e') &&
573 (filename[2] == 's') && (filename[3] == 't') &&
574 (filename[4] == '/'))
575 filename = &filename[5];
576 *************/
577
578 base = baseFilename(filename);
579 if (suffix == NULL)
580 suffix = ".tmp";
581 if (out == NULL)
582 out = "";
583
584 strncpy(suffixbuff,suffix,499);
585 #ifdef VMS
586 if(strstr(base,".") && suffixbuff[0]=='.')
587 suffixbuff[0]='_';
588 #endif
589
590 snprintf(res, 499, "%s%s%s", out, base, suffixbuff);
591 res[499] = 0;
592 return(strdup(res));
593 }
594
checkTestFile(const char * filename)595 static int checkTestFile(const char *filename) {
596 struct stat buf;
597
598 if (stat(filename, &buf) == -1)
599 return(0);
600
601 #if defined(_WIN32) && !defined(__CYGWIN__)
602 if (!(buf.st_mode & _S_IFREG))
603 return(0);
604 #else
605 if (!S_ISREG(buf.st_mode))
606 return(0);
607 #endif
608
609 return(1);
610 }
611
compareFiles(const char * r1,const char * r2)612 static int compareFiles(const char *r1 /* temp */, const char *r2 /* result */) {
613 int res1, res2;
614 int fd1, fd2;
615 char bytes1[4096];
616 char bytes2[4096];
617
618 if (update_results) {
619 fd1 = open(r1, RD_FLAGS);
620 if (fd1 < 0)
621 return(-1);
622 fd2 = open(r2, WR_FLAGS, 0644);
623 if (fd2 < 0) {
624 close(fd1);
625 return(-1);
626 }
627 do {
628 res1 = read(fd1, bytes1, 4096);
629 if (res1 <= 0)
630 break;
631 res2 = write(fd2, bytes1, res1);
632 if (res2 <= 0 || res2 != res1)
633 break;
634 } while (1);
635 close(fd2);
636 close(fd1);
637 return(res1 != 0);
638 }
639
640 fd1 = open(r1, RD_FLAGS);
641 if (fd1 < 0)
642 return(-1);
643 fd2 = open(r2, RD_FLAGS);
644 if (fd2 < 0) {
645 close(fd1);
646 return(-1);
647 }
648 while (1) {
649 res1 = read(fd1, bytes1, 4096);
650 res2 = read(fd2, bytes2, 4096);
651 if ((res1 != res2) || (res1 < 0)) {
652 close(fd1);
653 close(fd2);
654 return(1);
655 }
656 if (res1 == 0)
657 break;
658 if (memcmp(bytes1, bytes2, res1) != 0) {
659 close(fd1);
660 close(fd2);
661 return(1);
662 }
663 }
664 close(fd1);
665 close(fd2);
666 return(0);
667 }
668
compareFileMem(const char * filename,const char * mem,int size)669 static int compareFileMem(const char *filename, const char *mem, int size) {
670 int res;
671 int fd;
672 char bytes[4096];
673 int idx = 0;
674 struct stat info;
675
676 if (update_results) {
677 fd = open(filename, WR_FLAGS, 0644);
678 if (fd < 0) {
679 fprintf(stderr, "failed to open %s for writing", filename);
680 return(-1);
681 }
682 res = write(fd, mem, size);
683 close(fd);
684 return(res != size);
685 }
686
687 if (stat(filename, &info) < 0) {
688 fprintf(stderr, "failed to stat %s\n", filename);
689 return(-1);
690 }
691 if (info.st_size != size) {
692 fprintf(stderr, "file %s is %ld bytes, result is %d bytes\n",
693 filename, (long) info.st_size, size);
694 return(-1);
695 }
696 fd = open(filename, RD_FLAGS);
697 if (fd < 0) {
698 fprintf(stderr, "failed to open %s for reading", filename);
699 return(-1);
700 }
701 while (idx < size) {
702 res = read(fd, bytes, 4096);
703 if (res <= 0)
704 break;
705 if (res + idx > size)
706 break;
707 if (memcmp(bytes, &mem[idx], res) != 0) {
708 int ix;
709 for (ix=0; ix<res; ix++)
710 if (bytes[ix] != mem[idx+ix])
711 break;
712 fprintf(stderr,"Compare error at position %d\n", idx+ix);
713 close(fd);
714 return(1);
715 }
716 idx += res;
717 }
718 close(fd);
719 if (idx != size) {
720 fprintf(stderr,"Compare error index %d, size %d\n", idx, size);
721 }
722 return(idx != size);
723 }
724
loadMem(const char * filename,const char ** mem,int * size)725 static int loadMem(const char *filename, const char **mem, int *size) {
726 int fd, res;
727 struct stat info;
728 char *base;
729 int siz = 0;
730 if (stat(filename, &info) < 0)
731 return(-1);
732 base = malloc(info.st_size + 1);
733 if (base == NULL)
734 return(-1);
735 if ((fd = open(filename, RD_FLAGS)) < 0) {
736 free(base);
737 return(-1);
738 }
739 while ((res = read(fd, &base[siz], info.st_size - siz)) > 0) {
740 siz += res;
741 }
742 close(fd);
743 #if !defined(_WIN32)
744 if (siz != info.st_size) {
745 free(base);
746 return(-1);
747 }
748 #endif
749 base[siz] = 0;
750 *mem = base;
751 *size = siz;
752 return(0);
753 }
754
unloadMem(const char * mem)755 static int unloadMem(const char *mem) {
756 free((char *)mem);
757 return(0);
758 }
759
760 /************************************************************************
761 * *
762 * Tests implementations *
763 * *
764 ************************************************************************/
765
766 /************************************************************************
767 * *
768 * Parse to SAX based tests *
769 * *
770 ************************************************************************/
771
772 static FILE *SAXdebug = NULL;
773
774 /*
775 * empty SAX block
776 */
777 static xmlSAXHandler emptySAXHandlerStruct = {
778 NULL, /* internalSubset */
779 NULL, /* isStandalone */
780 NULL, /* hasInternalSubset */
781 NULL, /* hasExternalSubset */
782 NULL, /* resolveEntity */
783 NULL, /* getEntity */
784 NULL, /* entityDecl */
785 NULL, /* notationDecl */
786 NULL, /* attributeDecl */
787 NULL, /* elementDecl */
788 NULL, /* unparsedEntityDecl */
789 NULL, /* setDocumentLocator */
790 NULL, /* startDocument */
791 NULL, /* endDocument */
792 NULL, /* startElement */
793 NULL, /* endElement */
794 NULL, /* reference */
795 NULL, /* characters */
796 NULL, /* ignorableWhitespace */
797 NULL, /* processingInstruction */
798 NULL, /* comment */
799 NULL, /* xmlParserWarning */
800 NULL, /* xmlParserError */
801 NULL, /* xmlParserError */
802 NULL, /* getParameterEntity */
803 NULL, /* cdataBlock; */
804 NULL, /* externalSubset; */
805 1,
806 NULL,
807 NULL, /* startElementNs */
808 NULL, /* endElementNs */
809 NULL /* xmlStructuredErrorFunc */
810 };
811
812 static xmlSAXHandlerPtr emptySAXHandler = &emptySAXHandlerStruct;
813 static int callbacks = 0;
814 static int quiet = 0;
815
816 /**
817 * isStandaloneDebug:
818 * @ctxt: An XML parser context
819 *
820 * Is this document tagged standalone ?
821 *
822 * Returns 1 if true
823 */
824 static int
isStandaloneDebug(void * ctx ATTRIBUTE_UNUSED)825 isStandaloneDebug(void *ctx ATTRIBUTE_UNUSED)
826 {
827 callbacks++;
828 if (quiet)
829 return(0);
830 fprintf(SAXdebug, "SAX.isStandalone()\n");
831 return(0);
832 }
833
834 /**
835 * hasInternalSubsetDebug:
836 * @ctxt: An XML parser context
837 *
838 * Does this document has an internal subset
839 *
840 * Returns 1 if true
841 */
842 static int
hasInternalSubsetDebug(void * ctx ATTRIBUTE_UNUSED)843 hasInternalSubsetDebug(void *ctx ATTRIBUTE_UNUSED)
844 {
845 callbacks++;
846 if (quiet)
847 return(0);
848 fprintf(SAXdebug, "SAX.hasInternalSubset()\n");
849 return(0);
850 }
851
852 /**
853 * hasExternalSubsetDebug:
854 * @ctxt: An XML parser context
855 *
856 * Does this document has an external subset
857 *
858 * Returns 1 if true
859 */
860 static int
hasExternalSubsetDebug(void * ctx ATTRIBUTE_UNUSED)861 hasExternalSubsetDebug(void *ctx ATTRIBUTE_UNUSED)
862 {
863 callbacks++;
864 if (quiet)
865 return(0);
866 fprintf(SAXdebug, "SAX.hasExternalSubset()\n");
867 return(0);
868 }
869
870 /**
871 * internalSubsetDebug:
872 * @ctxt: An XML parser context
873 *
874 * Does this document has an internal subset
875 */
876 static void
internalSubsetDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,const xmlChar * ExternalID,const xmlChar * SystemID)877 internalSubsetDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name,
878 const xmlChar *ExternalID, const xmlChar *SystemID)
879 {
880 callbacks++;
881 if (quiet)
882 return;
883 fprintf(SAXdebug, "SAX.internalSubset(%s,", name);
884 if (ExternalID == NULL)
885 fprintf(SAXdebug, " ,");
886 else
887 fprintf(SAXdebug, " %s,", ExternalID);
888 if (SystemID == NULL)
889 fprintf(SAXdebug, " )\n");
890 else
891 fprintf(SAXdebug, " %s)\n", SystemID);
892 }
893
894 /**
895 * externalSubsetDebug:
896 * @ctxt: An XML parser context
897 *
898 * Does this document has an external subset
899 */
900 static void
externalSubsetDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,const xmlChar * ExternalID,const xmlChar * SystemID)901 externalSubsetDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name,
902 const xmlChar *ExternalID, const xmlChar *SystemID)
903 {
904 callbacks++;
905 if (quiet)
906 return;
907 fprintf(SAXdebug, "SAX.externalSubset(%s,", name);
908 if (ExternalID == NULL)
909 fprintf(SAXdebug, " ,");
910 else
911 fprintf(SAXdebug, " %s,", ExternalID);
912 if (SystemID == NULL)
913 fprintf(SAXdebug, " )\n");
914 else
915 fprintf(SAXdebug, " %s)\n", SystemID);
916 }
917
918 /**
919 * resolveEntityDebug:
920 * @ctxt: An XML parser context
921 * @publicId: The public ID of the entity
922 * @systemId: The system ID of the entity
923 *
924 * Special entity resolver, better left to the parser, it has
925 * more context than the application layer.
926 * The default behaviour is to NOT resolve the entities, in that case
927 * the ENTITY_REF nodes are built in the structure (and the parameter
928 * values).
929 *
930 * Returns the xmlParserInputPtr if inlined or NULL for DOM behaviour.
931 */
932 static xmlParserInputPtr
resolveEntityDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * publicId,const xmlChar * systemId)933 resolveEntityDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *publicId, const xmlChar *systemId)
934 {
935 callbacks++;
936 if (quiet)
937 return(NULL);
938 /* xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx; */
939
940
941 fprintf(SAXdebug, "SAX.resolveEntity(");
942 if (publicId != NULL)
943 fprintf(SAXdebug, "%s", (char *)publicId);
944 else
945 fprintf(SAXdebug, " ");
946 if (systemId != NULL)
947 fprintf(SAXdebug, ", %s)\n", (char *)systemId);
948 else
949 fprintf(SAXdebug, ", )\n");
950 /*********
951 if (systemId != NULL) {
952 return(xmlNewInputFromFile(ctxt, (char *) systemId));
953 }
954 *********/
955 return(NULL);
956 }
957
958 /**
959 * getEntityDebug:
960 * @ctxt: An XML parser context
961 * @name: The entity name
962 *
963 * Get an entity by name
964 *
965 * Returns the xmlParserInputPtr if inlined or NULL for DOM behaviour.
966 */
967 static xmlEntityPtr
getEntityDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name)968 getEntityDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name)
969 {
970 callbacks++;
971 if (quiet)
972 return(NULL);
973 fprintf(SAXdebug, "SAX.getEntity(%s)\n", name);
974 return(NULL);
975 }
976
977 /**
978 * getParameterEntityDebug:
979 * @ctxt: An XML parser context
980 * @name: The entity name
981 *
982 * Get a parameter entity by name
983 *
984 * Returns the xmlParserInputPtr
985 */
986 static xmlEntityPtr
getParameterEntityDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name)987 getParameterEntityDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name)
988 {
989 callbacks++;
990 if (quiet)
991 return(NULL);
992 fprintf(SAXdebug, "SAX.getParameterEntity(%s)\n", name);
993 return(NULL);
994 }
995
996
997 /**
998 * entityDeclDebug:
999 * @ctxt: An XML parser context
1000 * @name: the entity name
1001 * @type: the entity type
1002 * @publicId: The public ID of the entity
1003 * @systemId: The system ID of the entity
1004 * @content: the entity value (without processing).
1005 *
1006 * An entity definition has been parsed
1007 */
1008 static void
entityDeclDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,int type,const xmlChar * publicId,const xmlChar * systemId,xmlChar * content)1009 entityDeclDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name, int type,
1010 const xmlChar *publicId, const xmlChar *systemId, xmlChar *content)
1011 {
1012 const xmlChar *nullstr = BAD_CAST "(null)";
1013 /* not all libraries handle printing null pointers nicely */
1014 if (publicId == NULL)
1015 publicId = nullstr;
1016 if (systemId == NULL)
1017 systemId = nullstr;
1018 if (content == NULL)
1019 content = (xmlChar *)nullstr;
1020 callbacks++;
1021 if (quiet)
1022 return;
1023 fprintf(SAXdebug, "SAX.entityDecl(%s, %d, %s, %s, %s)\n",
1024 name, type, publicId, systemId, content);
1025 }
1026
1027 /**
1028 * attributeDeclDebug:
1029 * @ctxt: An XML parser context
1030 * @name: the attribute name
1031 * @type: the attribute type
1032 *
1033 * An attribute definition has been parsed
1034 */
1035 static void
attributeDeclDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * elem,const xmlChar * name,int type,int def,const xmlChar * defaultValue,xmlEnumerationPtr tree)1036 attributeDeclDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar * elem,
1037 const xmlChar * name, int type, int def,
1038 const xmlChar * defaultValue, xmlEnumerationPtr tree)
1039 {
1040 callbacks++;
1041 if (quiet)
1042 return;
1043 if (defaultValue == NULL)
1044 fprintf(SAXdebug, "SAX.attributeDecl(%s, %s, %d, %d, NULL, ...)\n",
1045 elem, name, type, def);
1046 else
1047 fprintf(SAXdebug, "SAX.attributeDecl(%s, %s, %d, %d, %s, ...)\n",
1048 elem, name, type, def, defaultValue);
1049 xmlFreeEnumeration(tree);
1050 }
1051
1052 /**
1053 * elementDeclDebug:
1054 * @ctxt: An XML parser context
1055 * @name: the element name
1056 * @type: the element type
1057 * @content: the element value (without processing).
1058 *
1059 * An element definition has been parsed
1060 */
1061 static void
elementDeclDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,int type,xmlElementContentPtr content ATTRIBUTE_UNUSED)1062 elementDeclDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name, int type,
1063 xmlElementContentPtr content ATTRIBUTE_UNUSED)
1064 {
1065 callbacks++;
1066 if (quiet)
1067 return;
1068 fprintf(SAXdebug, "SAX.elementDecl(%s, %d, ...)\n",
1069 name, type);
1070 }
1071
1072 /**
1073 * notationDeclDebug:
1074 * @ctxt: An XML parser context
1075 * @name: The name of the notation
1076 * @publicId: The public ID of the entity
1077 * @systemId: The system ID of the entity
1078 *
1079 * What to do when a notation declaration has been parsed.
1080 */
1081 static void
notationDeclDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,const xmlChar * publicId,const xmlChar * systemId)1082 notationDeclDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name,
1083 const xmlChar *publicId, const xmlChar *systemId)
1084 {
1085 callbacks++;
1086 if (quiet)
1087 return;
1088 fprintf(SAXdebug, "SAX.notationDecl(%s, %s, %s)\n",
1089 (char *) name, (char *) publicId, (char *) systemId);
1090 }
1091
1092 /**
1093 * unparsedEntityDeclDebug:
1094 * @ctxt: An XML parser context
1095 * @name: The name of the entity
1096 * @publicId: The public ID of the entity
1097 * @systemId: The system ID of the entity
1098 * @notationName: the name of the notation
1099 *
1100 * What to do when an unparsed entity declaration is parsed
1101 */
1102 static void
unparsedEntityDeclDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,const xmlChar * publicId,const xmlChar * systemId,const xmlChar * notationName)1103 unparsedEntityDeclDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name,
1104 const xmlChar *publicId, const xmlChar *systemId,
1105 const xmlChar *notationName)
1106 {
1107 const xmlChar *nullstr = BAD_CAST "(null)";
1108
1109 if (publicId == NULL)
1110 publicId = nullstr;
1111 if (systemId == NULL)
1112 systemId = nullstr;
1113 if (notationName == NULL)
1114 notationName = nullstr;
1115 callbacks++;
1116 if (quiet)
1117 return;
1118 fprintf(SAXdebug, "SAX.unparsedEntityDecl(%s, %s, %s, %s)\n",
1119 (char *) name, (char *) publicId, (char *) systemId,
1120 (char *) notationName);
1121 }
1122
1123 /**
1124 * setDocumentLocatorDebug:
1125 * @ctxt: An XML parser context
1126 * @loc: A SAX Locator
1127 *
1128 * Receive the document locator at startup, actually xmlDefaultSAXLocator
1129 * Everything is available on the context, so this is useless in our case.
1130 */
1131 static void
setDocumentLocatorDebug(void * ctx ATTRIBUTE_UNUSED,xmlSAXLocatorPtr loc ATTRIBUTE_UNUSED)1132 setDocumentLocatorDebug(void *ctx ATTRIBUTE_UNUSED, xmlSAXLocatorPtr loc ATTRIBUTE_UNUSED)
1133 {
1134 callbacks++;
1135 if (quiet)
1136 return;
1137 fprintf(SAXdebug, "SAX.setDocumentLocator()\n");
1138 }
1139
1140 /**
1141 * startDocumentDebug:
1142 * @ctxt: An XML parser context
1143 *
1144 * called when the document start being processed.
1145 */
1146 static void
startDocumentDebug(void * ctx ATTRIBUTE_UNUSED)1147 startDocumentDebug(void *ctx ATTRIBUTE_UNUSED)
1148 {
1149 callbacks++;
1150 if (quiet)
1151 return;
1152 fprintf(SAXdebug, "SAX.startDocument()\n");
1153 }
1154
1155 /**
1156 * endDocumentDebug:
1157 * @ctxt: An XML parser context
1158 *
1159 * called when the document end has been detected.
1160 */
1161 static void
endDocumentDebug(void * ctx ATTRIBUTE_UNUSED)1162 endDocumentDebug(void *ctx ATTRIBUTE_UNUSED)
1163 {
1164 callbacks++;
1165 if (quiet)
1166 return;
1167 fprintf(SAXdebug, "SAX.endDocument()\n");
1168 }
1169
1170 /**
1171 * startElementDebug:
1172 * @ctxt: An XML parser context
1173 * @name: The element name
1174 *
1175 * called when an opening tag has been processed.
1176 */
1177 static void
startElementDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,const xmlChar ** atts)1178 startElementDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name, const xmlChar **atts)
1179 {
1180 int i;
1181
1182 callbacks++;
1183 if (quiet)
1184 return;
1185 fprintf(SAXdebug, "SAX.startElement(%s", (char *) name);
1186 if (atts != NULL) {
1187 for (i = 0;(atts[i] != NULL);i++) {
1188 fprintf(SAXdebug, ", %s='", atts[i++]);
1189 if (atts[i] != NULL)
1190 fprintf(SAXdebug, "%s'", atts[i]);
1191 }
1192 }
1193 fprintf(SAXdebug, ")\n");
1194 }
1195
1196 /**
1197 * endElementDebug:
1198 * @ctxt: An XML parser context
1199 * @name: The element name
1200 *
1201 * called when the end of an element has been detected.
1202 */
1203 static void
endElementDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name)1204 endElementDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name)
1205 {
1206 callbacks++;
1207 if (quiet)
1208 return;
1209 fprintf(SAXdebug, "SAX.endElement(%s)\n", (char *) name);
1210 }
1211
1212 /**
1213 * charactersDebug:
1214 * @ctxt: An XML parser context
1215 * @ch: a xmlChar string
1216 * @len: the number of xmlChar
1217 *
1218 * receiving some chars from the parser.
1219 * Question: how much at a time ???
1220 */
1221 static void
charactersDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * ch,int len)1222 charactersDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *ch, int len)
1223 {
1224 char output[40];
1225 int i;
1226
1227 callbacks++;
1228 if (quiet)
1229 return;
1230 for (i = 0;(i<len) && (i < 30);i++)
1231 output[i] = ch[i];
1232 output[i] = 0;
1233
1234 fprintf(SAXdebug, "SAX.characters(%s, %d)\n", output, len);
1235 }
1236
1237 /**
1238 * referenceDebug:
1239 * @ctxt: An XML parser context
1240 * @name: The entity name
1241 *
1242 * called when an entity reference is detected.
1243 */
1244 static void
referenceDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name)1245 referenceDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name)
1246 {
1247 callbacks++;
1248 if (quiet)
1249 return;
1250 fprintf(SAXdebug, "SAX.reference(%s)\n", name);
1251 }
1252
1253 /**
1254 * ignorableWhitespaceDebug:
1255 * @ctxt: An XML parser context
1256 * @ch: a xmlChar string
1257 * @start: the first char in the string
1258 * @len: the number of xmlChar
1259 *
1260 * receiving some ignorable whitespaces from the parser.
1261 * Question: how much at a time ???
1262 */
1263 static void
ignorableWhitespaceDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * ch,int len)1264 ignorableWhitespaceDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *ch, int len)
1265 {
1266 char output[40];
1267 int i;
1268
1269 callbacks++;
1270 if (quiet)
1271 return;
1272 for (i = 0;(i<len) && (i < 30);i++)
1273 output[i] = ch[i];
1274 output[i] = 0;
1275 fprintf(SAXdebug, "SAX.ignorableWhitespace(%s, %d)\n", output, len);
1276 }
1277
1278 /**
1279 * processingInstructionDebug:
1280 * @ctxt: An XML parser context
1281 * @target: the target name
1282 * @data: the PI data's
1283 * @len: the number of xmlChar
1284 *
1285 * A processing instruction has been parsed.
1286 */
1287 static void
processingInstructionDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * target,const xmlChar * data)1288 processingInstructionDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *target,
1289 const xmlChar *data)
1290 {
1291 callbacks++;
1292 if (quiet)
1293 return;
1294 if (data != NULL)
1295 fprintf(SAXdebug, "SAX.processingInstruction(%s, %s)\n",
1296 (char *) target, (char *) data);
1297 else
1298 fprintf(SAXdebug, "SAX.processingInstruction(%s, NULL)\n",
1299 (char *) target);
1300 }
1301
1302 /**
1303 * cdataBlockDebug:
1304 * @ctx: the user data (XML parser context)
1305 * @value: The pcdata content
1306 * @len: the block length
1307 *
1308 * called when a pcdata block has been parsed
1309 */
1310 static void
cdataBlockDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * value,int len)1311 cdataBlockDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *value, int len)
1312 {
1313 callbacks++;
1314 if (quiet)
1315 return;
1316 fprintf(SAXdebug, "SAX.pcdata(%.20s, %d)\n",
1317 (char *) value, len);
1318 }
1319
1320 /**
1321 * commentDebug:
1322 * @ctxt: An XML parser context
1323 * @value: the comment content
1324 *
1325 * A comment has been parsed.
1326 */
1327 static void
commentDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * value)1328 commentDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *value)
1329 {
1330 callbacks++;
1331 if (quiet)
1332 return;
1333 fprintf(SAXdebug, "SAX.comment(%s)\n", value);
1334 }
1335
1336 /**
1337 * warningDebug:
1338 * @ctxt: An XML parser context
1339 * @msg: the message to display/transmit
1340 * @...: extra parameters for the message display
1341 *
1342 * Display and format a warning messages, gives file, line, position and
1343 * extra parameters.
1344 */
1345 static void XMLCDECL
warningDebug(void * ctx ATTRIBUTE_UNUSED,const char * msg,...)1346 warningDebug(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...)
1347 {
1348 va_list args;
1349
1350 callbacks++;
1351 if (quiet)
1352 return;
1353 va_start(args, msg);
1354 fprintf(SAXdebug, "SAX.warning: ");
1355 vfprintf(SAXdebug, msg, args);
1356 va_end(args);
1357 }
1358
1359 /**
1360 * errorDebug:
1361 * @ctxt: An XML parser context
1362 * @msg: the message to display/transmit
1363 * @...: extra parameters for the message display
1364 *
1365 * Display and format a error messages, gives file, line, position and
1366 * extra parameters.
1367 */
1368 static void XMLCDECL
errorDebug(void * ctx ATTRIBUTE_UNUSED,const char * msg,...)1369 errorDebug(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...)
1370 {
1371 va_list args;
1372
1373 callbacks++;
1374 if (quiet)
1375 return;
1376 va_start(args, msg);
1377 fprintf(SAXdebug, "SAX.error: ");
1378 vfprintf(SAXdebug, msg, args);
1379 va_end(args);
1380 }
1381
1382 /**
1383 * fatalErrorDebug:
1384 * @ctxt: An XML parser context
1385 * @msg: the message to display/transmit
1386 * @...: extra parameters for the message display
1387 *
1388 * Display and format a fatalError messages, gives file, line, position and
1389 * extra parameters.
1390 */
1391 static void XMLCDECL
fatalErrorDebug(void * ctx ATTRIBUTE_UNUSED,const char * msg,...)1392 fatalErrorDebug(void *ctx ATTRIBUTE_UNUSED, const char *msg, ...)
1393 {
1394 va_list args;
1395
1396 callbacks++;
1397 if (quiet)
1398 return;
1399 va_start(args, msg);
1400 fprintf(SAXdebug, "SAX.fatalError: ");
1401 vfprintf(SAXdebug, msg, args);
1402 va_end(args);
1403 }
1404
1405 static xmlSAXHandler debugSAXHandlerStruct = {
1406 internalSubsetDebug,
1407 isStandaloneDebug,
1408 hasInternalSubsetDebug,
1409 hasExternalSubsetDebug,
1410 resolveEntityDebug,
1411 getEntityDebug,
1412 entityDeclDebug,
1413 notationDeclDebug,
1414 attributeDeclDebug,
1415 elementDeclDebug,
1416 unparsedEntityDeclDebug,
1417 setDocumentLocatorDebug,
1418 startDocumentDebug,
1419 endDocumentDebug,
1420 startElementDebug,
1421 endElementDebug,
1422 referenceDebug,
1423 charactersDebug,
1424 ignorableWhitespaceDebug,
1425 processingInstructionDebug,
1426 commentDebug,
1427 warningDebug,
1428 errorDebug,
1429 fatalErrorDebug,
1430 getParameterEntityDebug,
1431 cdataBlockDebug,
1432 externalSubsetDebug,
1433 1,
1434 NULL,
1435 NULL,
1436 NULL,
1437 NULL
1438 };
1439
1440 static xmlSAXHandlerPtr debugSAXHandler = &debugSAXHandlerStruct;
1441
1442 /*
1443 * SAX2 specific callbacks
1444 */
1445 /**
1446 * startElementNsDebug:
1447 * @ctxt: An XML parser context
1448 * @name: The element name
1449 *
1450 * called when an opening tag has been processed.
1451 */
1452 static void
startElementNsDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * localname,const xmlChar * prefix,const xmlChar * URI,int nb_namespaces,const xmlChar ** namespaces,int nb_attributes,int nb_defaulted,const xmlChar ** attributes)1453 startElementNsDebug(void *ctx ATTRIBUTE_UNUSED,
1454 const xmlChar *localname,
1455 const xmlChar *prefix,
1456 const xmlChar *URI,
1457 int nb_namespaces,
1458 const xmlChar **namespaces,
1459 int nb_attributes,
1460 int nb_defaulted,
1461 const xmlChar **attributes)
1462 {
1463 int i;
1464
1465 callbacks++;
1466 if (quiet)
1467 return;
1468 fprintf(SAXdebug, "SAX.startElementNs(%s", (char *) localname);
1469 if (prefix == NULL)
1470 fprintf(SAXdebug, ", NULL");
1471 else
1472 fprintf(SAXdebug, ", %s", (char *) prefix);
1473 if (URI == NULL)
1474 fprintf(SAXdebug, ", NULL");
1475 else
1476 fprintf(SAXdebug, ", '%s'", (char *) URI);
1477 fprintf(SAXdebug, ", %d", nb_namespaces);
1478
1479 if (namespaces != NULL) {
1480 for (i = 0;i < nb_namespaces * 2;i++) {
1481 fprintf(SAXdebug, ", xmlns");
1482 if (namespaces[i] != NULL)
1483 fprintf(SAXdebug, ":%s", namespaces[i]);
1484 i++;
1485 fprintf(SAXdebug, "='%s'", namespaces[i]);
1486 }
1487 }
1488 fprintf(SAXdebug, ", %d, %d", nb_attributes, nb_defaulted);
1489 if (attributes != NULL) {
1490 for (i = 0;i < nb_attributes * 5;i += 5) {
1491 if (attributes[i + 1] != NULL)
1492 fprintf(SAXdebug, ", %s:%s='", attributes[i + 1], attributes[i]);
1493 else
1494 fprintf(SAXdebug, ", %s='", attributes[i]);
1495 fprintf(SAXdebug, "%.4s...', %d", attributes[i + 3],
1496 (int)(attributes[i + 4] - attributes[i + 3]));
1497 }
1498 }
1499 fprintf(SAXdebug, ")\n");
1500 }
1501
1502 /**
1503 * endElementDebug:
1504 * @ctxt: An XML parser context
1505 * @name: The element name
1506 *
1507 * called when the end of an element has been detected.
1508 */
1509 static void
endElementNsDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * localname,const xmlChar * prefix,const xmlChar * URI)1510 endElementNsDebug(void *ctx ATTRIBUTE_UNUSED,
1511 const xmlChar *localname,
1512 const xmlChar *prefix,
1513 const xmlChar *URI)
1514 {
1515 callbacks++;
1516 if (quiet)
1517 return;
1518 fprintf(SAXdebug, "SAX.endElementNs(%s", (char *) localname);
1519 if (prefix == NULL)
1520 fprintf(SAXdebug, ", NULL");
1521 else
1522 fprintf(SAXdebug, ", %s", (char *) prefix);
1523 if (URI == NULL)
1524 fprintf(SAXdebug, ", NULL)\n");
1525 else
1526 fprintf(SAXdebug, ", '%s')\n", (char *) URI);
1527 }
1528
1529 static xmlSAXHandler debugSAX2HandlerStruct = {
1530 internalSubsetDebug,
1531 isStandaloneDebug,
1532 hasInternalSubsetDebug,
1533 hasExternalSubsetDebug,
1534 resolveEntityDebug,
1535 getEntityDebug,
1536 entityDeclDebug,
1537 notationDeclDebug,
1538 attributeDeclDebug,
1539 elementDeclDebug,
1540 unparsedEntityDeclDebug,
1541 setDocumentLocatorDebug,
1542 startDocumentDebug,
1543 endDocumentDebug,
1544 NULL,
1545 NULL,
1546 referenceDebug,
1547 charactersDebug,
1548 ignorableWhitespaceDebug,
1549 processingInstructionDebug,
1550 commentDebug,
1551 warningDebug,
1552 errorDebug,
1553 fatalErrorDebug,
1554 getParameterEntityDebug,
1555 cdataBlockDebug,
1556 externalSubsetDebug,
1557 XML_SAX2_MAGIC,
1558 NULL,
1559 startElementNsDebug,
1560 endElementNsDebug,
1561 NULL
1562 };
1563
1564 static xmlSAXHandlerPtr debugSAX2Handler = &debugSAX2HandlerStruct;
1565
1566 #ifdef LIBXML_HTML_ENABLED
1567 /**
1568 * htmlstartElementDebug:
1569 * @ctxt: An XML parser context
1570 * @name: The element name
1571 *
1572 * called when an opening tag has been processed.
1573 */
1574 static void
htmlstartElementDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * name,const xmlChar ** atts)1575 htmlstartElementDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *name, const xmlChar **atts)
1576 {
1577 int i;
1578
1579 fprintf(SAXdebug, "SAX.startElement(%s", (char *) name);
1580 if (atts != NULL) {
1581 for (i = 0;(atts[i] != NULL);i++) {
1582 fprintf(SAXdebug, ", %s", atts[i++]);
1583 if (atts[i] != NULL) {
1584 unsigned char output[40];
1585 const unsigned char *att = atts[i];
1586 int outlen, attlen;
1587 fprintf(SAXdebug, "='");
1588 while ((attlen = strlen((char*)att)) > 0) {
1589 outlen = sizeof output - 1;
1590 htmlEncodeEntities(output, &outlen, att, &attlen, '\'');
1591 output[outlen] = 0;
1592 fprintf(SAXdebug, "%s", (char *) output);
1593 att += attlen;
1594 }
1595 fprintf(SAXdebug, "'");
1596 }
1597 }
1598 }
1599 fprintf(SAXdebug, ")\n");
1600 }
1601
1602 /**
1603 * htmlcharactersDebug:
1604 * @ctxt: An XML parser context
1605 * @ch: a xmlChar string
1606 * @len: the number of xmlChar
1607 *
1608 * receiving some chars from the parser.
1609 * Question: how much at a time ???
1610 */
1611 static void
htmlcharactersDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * ch,int len)1612 htmlcharactersDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *ch, int len)
1613 {
1614 unsigned char output[40];
1615 int inlen = len, outlen = 30;
1616
1617 htmlEncodeEntities(output, &outlen, ch, &inlen, 0);
1618 output[outlen] = 0;
1619
1620 fprintf(SAXdebug, "SAX.characters(%s, %d)\n", output, len);
1621 }
1622
1623 /**
1624 * htmlcdataDebug:
1625 * @ctxt: An XML parser context
1626 * @ch: a xmlChar string
1627 * @len: the number of xmlChar
1628 *
1629 * receiving some cdata chars from the parser.
1630 * Question: how much at a time ???
1631 */
1632 static void
htmlcdataDebug(void * ctx ATTRIBUTE_UNUSED,const xmlChar * ch,int len)1633 htmlcdataDebug(void *ctx ATTRIBUTE_UNUSED, const xmlChar *ch, int len)
1634 {
1635 unsigned char output[40];
1636 int inlen = len, outlen = 30;
1637
1638 htmlEncodeEntities(output, &outlen, ch, &inlen, 0);
1639 output[outlen] = 0;
1640
1641 fprintf(SAXdebug, "SAX.cdata(%s, %d)\n", output, len);
1642 }
1643
1644 static xmlSAXHandler debugHTMLSAXHandlerStruct = {
1645 internalSubsetDebug,
1646 isStandaloneDebug,
1647 hasInternalSubsetDebug,
1648 hasExternalSubsetDebug,
1649 resolveEntityDebug,
1650 getEntityDebug,
1651 entityDeclDebug,
1652 notationDeclDebug,
1653 attributeDeclDebug,
1654 elementDeclDebug,
1655 unparsedEntityDeclDebug,
1656 setDocumentLocatorDebug,
1657 startDocumentDebug,
1658 endDocumentDebug,
1659 htmlstartElementDebug,
1660 endElementDebug,
1661 referenceDebug,
1662 htmlcharactersDebug,
1663 ignorableWhitespaceDebug,
1664 processingInstructionDebug,
1665 commentDebug,
1666 warningDebug,
1667 errorDebug,
1668 fatalErrorDebug,
1669 getParameterEntityDebug,
1670 htmlcdataDebug,
1671 externalSubsetDebug,
1672 1,
1673 NULL,
1674 NULL,
1675 NULL,
1676 NULL
1677 };
1678
1679 static xmlSAXHandlerPtr debugHTMLSAXHandler = &debugHTMLSAXHandlerStruct;
1680 #endif /* LIBXML_HTML_ENABLED */
1681
1682 /**
1683 * saxParseTest:
1684 * @filename: the file to parse
1685 * @result: the file with expected result
1686 * @err: the file with error messages
1687 *
1688 * Parse a file using the SAX API and check for errors.
1689 *
1690 * Returns 0 in case of success, an error code otherwise
1691 */
1692 static int
saxParseTest(const char * filename,const char * result,const char * err ATTRIBUTE_UNUSED,int options)1693 saxParseTest(const char *filename, const char *result,
1694 const char *err ATTRIBUTE_UNUSED,
1695 int options) {
1696 int ret;
1697 char *temp;
1698
1699 nb_tests++;
1700 temp = resultFilename(filename, "", ".res");
1701 if (temp == NULL) {
1702 fprintf(stderr, "out of memory\n");
1703 fatalError();
1704 }
1705 SAXdebug = fopen(temp, "wb");
1706 if (SAXdebug == NULL) {
1707 fprintf(stderr, "Failed to write to %s\n", temp);
1708 free(temp);
1709 return(-1);
1710 }
1711
1712 /* for SAX we really want the callbacks though the context handlers */
1713 xmlSetStructuredErrorFunc(NULL, NULL);
1714 xmlSetGenericErrorFunc(NULL, testErrorHandler);
1715
1716 #ifdef LIBXML_HTML_ENABLED
1717 if (options & XML_PARSE_HTML) {
1718 htmlSAXParseFile(filename, NULL, emptySAXHandler, NULL);
1719 ret = 0;
1720 } else
1721 #endif
1722 {
1723 xmlParserCtxtPtr ctxt = xmlCreateFileParserCtxt(filename);
1724 memcpy(ctxt->sax, emptySAXHandler, sizeof(xmlSAXHandler));
1725 xmlCtxtUseOptions(ctxt, options);
1726 xmlParseDocument(ctxt);
1727 ret = ctxt->wellFormed ? 0 : ctxt->errNo;
1728 xmlFreeDoc(ctxt->myDoc);
1729 xmlFreeParserCtxt(ctxt);
1730 }
1731 if (ret == XML_WAR_UNDECLARED_ENTITY) {
1732 fprintf(SAXdebug, "xmlSAXUserParseFile returned error %d\n", ret);
1733 ret = 0;
1734 }
1735 if (ret != 0) {
1736 fprintf(stderr, "Failed to parse %s\n", filename);
1737 ret = 1;
1738 goto done;
1739 }
1740 #ifdef LIBXML_HTML_ENABLED
1741 if (options & XML_PARSE_HTML) {
1742 htmlSAXParseFile(filename, NULL, debugHTMLSAXHandler, NULL);
1743 ret = 0;
1744 } else
1745 #endif
1746 {
1747 xmlParserCtxtPtr ctxt = xmlCreateFileParserCtxt(filename);
1748 if (options & XML_PARSE_SAX1) {
1749 memcpy(ctxt->sax, debugSAXHandler, sizeof(xmlSAXHandler));
1750 options -= XML_PARSE_SAX1;
1751 } else {
1752 memcpy(ctxt->sax, debugSAX2Handler, sizeof(xmlSAXHandler));
1753 }
1754 xmlCtxtUseOptions(ctxt, options);
1755 xmlParseDocument(ctxt);
1756 ret = ctxt->wellFormed ? 0 : ctxt->errNo;
1757 xmlFreeDoc(ctxt->myDoc);
1758 xmlFreeParserCtxt(ctxt);
1759 }
1760 if (ret == XML_WAR_UNDECLARED_ENTITY) {
1761 fprintf(SAXdebug, "xmlSAXUserParseFile returned error %d\n", ret);
1762 ret = 0;
1763 }
1764 fclose(SAXdebug);
1765 if (compareFiles(temp, result)) {
1766 fprintf(stderr, "Got a difference for %s\n", filename);
1767 ret = 1;
1768 }
1769
1770 done:
1771 if (temp != NULL) {
1772 unlink(temp);
1773 free(temp);
1774 }
1775
1776 /* switch back to structured error handling */
1777 xmlSetGenericErrorFunc(NULL, NULL);
1778 xmlSetStructuredErrorFunc(NULL, testStructuredErrorHandler);
1779
1780 return(ret);
1781 }
1782
1783 /************************************************************************
1784 * *
1785 * Parse to tree based tests *
1786 * *
1787 ************************************************************************/
1788 /**
1789 * oldParseTest:
1790 * @filename: the file to parse
1791 * @result: the file with expected result
1792 * @err: the file with error messages: unused
1793 *
1794 * Parse a file using the old xmlParseFile API, then serialize back
1795 * reparse the result and serialize again, then check for deviation
1796 * in serialization.
1797 *
1798 * Returns 0 in case of success, an error code otherwise
1799 */
1800 static int
oldParseTest(const char * filename,const char * result,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)1801 oldParseTest(const char *filename, const char *result,
1802 const char *err ATTRIBUTE_UNUSED,
1803 int options ATTRIBUTE_UNUSED) {
1804 xmlDocPtr doc;
1805 char *temp;
1806 int res = 0;
1807
1808 nb_tests++;
1809 /*
1810 * base of the test, parse with the old API
1811 */
1812 #ifdef LIBXML_SAX1_ENABLED
1813 doc = xmlParseFile(filename);
1814 #else
1815 doc = xmlReadFile(filename, NULL, 0);
1816 #endif
1817 if (doc == NULL)
1818 return(1);
1819 temp = resultFilename(filename, "", ".res");
1820 if (temp == NULL) {
1821 fprintf(stderr, "out of memory\n");
1822 fatalError();
1823 }
1824 xmlSaveFile(temp, doc);
1825 if (compareFiles(temp, result)) {
1826 res = 1;
1827 }
1828 xmlFreeDoc(doc);
1829
1830 /*
1831 * Parse the saved result to make sure the round trip is okay
1832 */
1833 #ifdef LIBXML_SAX1_ENABLED
1834 doc = xmlParseFile(temp);
1835 #else
1836 doc = xmlReadFile(temp, NULL, 0);
1837 #endif
1838 if (doc == NULL)
1839 return(1);
1840 xmlSaveFile(temp, doc);
1841 if (compareFiles(temp, result)) {
1842 res = 1;
1843 }
1844 xmlFreeDoc(doc);
1845
1846 if (temp != NULL) {
1847 unlink(temp);
1848 free(temp);
1849 }
1850 return(res);
1851 }
1852
1853 #ifdef LIBXML_PUSH_ENABLED
1854 /**
1855 * pushParseTest:
1856 * @filename: the file to parse
1857 * @result: the file with expected result
1858 * @err: the file with error messages: unused
1859 *
1860 * Parse a file using the Push API, then serialize back
1861 * to check for content.
1862 *
1863 * Returns 0 in case of success, an error code otherwise
1864 */
1865 static int
pushParseTest(const char * filename,const char * result,const char * err ATTRIBUTE_UNUSED,int options)1866 pushParseTest(const char *filename, const char *result,
1867 const char *err ATTRIBUTE_UNUSED,
1868 int options) {
1869 xmlParserCtxtPtr ctxt;
1870 xmlDocPtr doc;
1871 const char *base;
1872 int size, res;
1873 int cur = 0;
1874 int chunkSize = 4;
1875
1876 nb_tests++;
1877 /*
1878 * load the document in memory and work from there.
1879 */
1880 if (loadMem(filename, &base, &size) != 0) {
1881 fprintf(stderr, "Failed to load %s\n", filename);
1882 return(-1);
1883 }
1884
1885 if (chunkSize > size)
1886 chunkSize = size;
1887
1888 #ifdef LIBXML_HTML_ENABLED
1889 if (options & XML_PARSE_HTML)
1890 ctxt = htmlCreatePushParserCtxt(NULL, NULL, base + cur, chunkSize, filename,
1891 XML_CHAR_ENCODING_NONE);
1892 else
1893 #endif
1894 ctxt = xmlCreatePushParserCtxt(NULL, NULL, base + cur, chunkSize, filename);
1895 xmlCtxtUseOptions(ctxt, options);
1896 cur += chunkSize;
1897 chunkSize = 1024;
1898 do {
1899 if (cur + chunkSize >= size) {
1900 #ifdef LIBXML_HTML_ENABLED
1901 if (options & XML_PARSE_HTML)
1902 htmlParseChunk(ctxt, base + cur, size - cur, 1);
1903 else
1904 #endif
1905 xmlParseChunk(ctxt, base + cur, size - cur, 1);
1906 break;
1907 } else {
1908 #ifdef LIBXML_HTML_ENABLED
1909 if (options & XML_PARSE_HTML)
1910 htmlParseChunk(ctxt, base + cur, chunkSize, 0);
1911 else
1912 #endif
1913 xmlParseChunk(ctxt, base + cur, chunkSize, 0);
1914 cur += chunkSize;
1915 }
1916 } while (cur < size);
1917 doc = ctxt->myDoc;
1918 #ifdef LIBXML_HTML_ENABLED
1919 if (options & XML_PARSE_HTML)
1920 res = 1;
1921 else
1922 #endif
1923 res = ctxt->wellFormed;
1924 xmlFreeParserCtxt(ctxt);
1925 free((char *)base);
1926 if (!res) {
1927 xmlFreeDoc(doc);
1928 fprintf(stderr, "Failed to parse %s\n", filename);
1929 return(-1);
1930 }
1931 #ifdef LIBXML_HTML_ENABLED
1932 if (options & XML_PARSE_HTML)
1933 htmlDocDumpMemory(doc, (xmlChar **) &base, &size);
1934 else
1935 #endif
1936 xmlDocDumpMemory(doc, (xmlChar **) &base, &size);
1937 xmlFreeDoc(doc);
1938 res = compareFileMem(result, base, size);
1939 if ((base == NULL) || (res != 0)) {
1940 if (base != NULL)
1941 xmlFree((char *)base);
1942 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
1943 return(-1);
1944 }
1945 xmlFree((char *)base);
1946 if (err != NULL) {
1947 res = compareFileMem(err, testErrors, testErrorsSize);
1948 if (res != 0) {
1949 fprintf(stderr, "Error for %s failed\n", filename);
1950 return(-1);
1951 }
1952 }
1953 return(0);
1954 }
1955 #endif
1956
1957 /**
1958 * memParseTest:
1959 * @filename: the file to parse
1960 * @result: the file with expected result
1961 * @err: the file with error messages: unused
1962 *
1963 * Parse a file using the old xmlReadMemory API, then serialize back
1964 * reparse the result and serialize again, then check for deviation
1965 * in serialization.
1966 *
1967 * Returns 0 in case of success, an error code otherwise
1968 */
1969 static int
memParseTest(const char * filename,const char * result,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)1970 memParseTest(const char *filename, const char *result,
1971 const char *err ATTRIBUTE_UNUSED,
1972 int options ATTRIBUTE_UNUSED) {
1973 xmlDocPtr doc;
1974 const char *base;
1975 int size, res;
1976
1977 nb_tests++;
1978 /*
1979 * load and parse the memory
1980 */
1981 if (loadMem(filename, &base, &size) != 0) {
1982 fprintf(stderr, "Failed to load %s\n", filename);
1983 return(-1);
1984 }
1985
1986 doc = xmlReadMemory(base, size, filename, NULL, 0);
1987 unloadMem(base);
1988 if (doc == NULL) {
1989 return(1);
1990 }
1991 xmlDocDumpMemory(doc, (xmlChar **) &base, &size);
1992 xmlFreeDoc(doc);
1993 res = compareFileMem(result, base, size);
1994 if ((base == NULL) || (res != 0)) {
1995 if (base != NULL)
1996 xmlFree((char *)base);
1997 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
1998 return(-1);
1999 }
2000 xmlFree((char *)base);
2001 return(0);
2002 }
2003
2004 /**
2005 * noentParseTest:
2006 * @filename: the file to parse
2007 * @result: the file with expected result
2008 * @err: the file with error messages: unused
2009 *
2010 * Parse a file with entity resolution, then serialize back
2011 * reparse the result and serialize again, then check for deviation
2012 * in serialization.
2013 *
2014 * Returns 0 in case of success, an error code otherwise
2015 */
2016 static int
noentParseTest(const char * filename,const char * result,const char * err ATTRIBUTE_UNUSED,int options)2017 noentParseTest(const char *filename, const char *result,
2018 const char *err ATTRIBUTE_UNUSED,
2019 int options) {
2020 xmlDocPtr doc;
2021 char *temp;
2022 int res = 0;
2023
2024 nb_tests++;
2025 /*
2026 * base of the test, parse with the old API
2027 */
2028 doc = xmlReadFile(filename, NULL, options);
2029 if (doc == NULL)
2030 return(1);
2031 temp = resultFilename(filename, "", ".res");
2032 if (temp == NULL) {
2033 fprintf(stderr, "Out of memory\n");
2034 fatalError();
2035 }
2036 xmlSaveFile(temp, doc);
2037 if (compareFiles(temp, result)) {
2038 res = 1;
2039 }
2040 xmlFreeDoc(doc);
2041
2042 /*
2043 * Parse the saved result to make sure the round trip is okay
2044 */
2045 doc = xmlReadFile(filename, NULL, options);
2046 if (doc == NULL)
2047 return(1);
2048 xmlSaveFile(temp, doc);
2049 if (compareFiles(temp, result)) {
2050 res = 1;
2051 }
2052 xmlFreeDoc(doc);
2053
2054 if (temp != NULL) {
2055 unlink(temp);
2056 free(temp);
2057 }
2058 return(res);
2059 }
2060
2061 /**
2062 * errParseTest:
2063 * @filename: the file to parse
2064 * @result: the file with expected result
2065 * @err: the file with error messages
2066 *
2067 * Parse a file using the xmlReadFile API and check for errors.
2068 *
2069 * Returns 0 in case of success, an error code otherwise
2070 */
2071 static int
errParseTest(const char * filename,const char * result,const char * err,int options)2072 errParseTest(const char *filename, const char *result, const char *err,
2073 int options) {
2074 xmlDocPtr doc;
2075 const char *base = NULL;
2076 int size, res = 0;
2077
2078 nb_tests++;
2079 #ifdef LIBXML_HTML_ENABLED
2080 if (options & XML_PARSE_HTML) {
2081 doc = htmlReadFile(filename, NULL, options);
2082 } else
2083 #endif
2084 #ifdef LIBXML_XINCLUDE_ENABLED
2085 if (options & XML_PARSE_XINCLUDE) {
2086 doc = xmlReadFile(filename, NULL, options);
2087 xmlXIncludeProcessFlags(doc, options);
2088 } else
2089 #endif
2090 {
2091 xmlGetWarningsDefaultValue = 1;
2092 doc = xmlReadFile(filename, NULL, options);
2093 }
2094 xmlGetWarningsDefaultValue = 0;
2095 if (result) {
2096 if (doc == NULL) {
2097 base = "";
2098 size = 0;
2099 } else {
2100 #ifdef LIBXML_HTML_ENABLED
2101 if (options & XML_PARSE_HTML) {
2102 htmlDocDumpMemory(doc, (xmlChar **) &base, &size);
2103 } else
2104 #endif
2105 xmlDocDumpMemory(doc, (xmlChar **) &base, &size);
2106 }
2107 res = compareFileMem(result, base, size);
2108 if (res != 0) {
2109 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
2110 return(-1);
2111 }
2112 }
2113 if (doc != NULL) {
2114 if (base != NULL)
2115 xmlFree((char *)base);
2116 xmlFreeDoc(doc);
2117 }
2118 if (err != NULL) {
2119 res = compareFileMem(err, testErrors, testErrorsSize);
2120 if (res != 0) {
2121 fprintf(stderr, "Error for %s failed\n", filename);
2122 return(-1);
2123 }
2124 } else if (options & XML_PARSE_DTDVALID) {
2125 if (testErrorsSize != 0)
2126 fprintf(stderr, "Validation for %s failed\n", filename);
2127 }
2128
2129 return(0);
2130 }
2131
2132 #ifdef LIBXML_READER_ENABLED
2133 /************************************************************************
2134 * *
2135 * Reader based tests *
2136 * *
2137 ************************************************************************/
2138
processNode(FILE * out,xmlTextReaderPtr reader)2139 static void processNode(FILE *out, xmlTextReaderPtr reader) {
2140 const xmlChar *name, *value;
2141 int type, empty;
2142
2143 type = xmlTextReaderNodeType(reader);
2144 empty = xmlTextReaderIsEmptyElement(reader);
2145
2146 name = xmlTextReaderConstName(reader);
2147 if (name == NULL)
2148 name = BAD_CAST "--";
2149
2150 value = xmlTextReaderConstValue(reader);
2151
2152
2153 fprintf(out, "%d %d %s %d %d",
2154 xmlTextReaderDepth(reader),
2155 type,
2156 name,
2157 empty,
2158 xmlTextReaderHasValue(reader));
2159 if (value == NULL)
2160 fprintf(out, "\n");
2161 else {
2162 fprintf(out, " %s\n", value);
2163 }
2164 }
2165 static int
streamProcessTest(const char * filename,const char * result,const char * err,xmlTextReaderPtr reader,const char * rng,int options ATTRIBUTE_UNUSED)2166 streamProcessTest(const char *filename, const char *result, const char *err,
2167 xmlTextReaderPtr reader, const char *rng,
2168 int options ATTRIBUTE_UNUSED) {
2169 int ret;
2170 char *temp = NULL;
2171 FILE *t = NULL;
2172
2173 if (reader == NULL)
2174 return(-1);
2175
2176 nb_tests++;
2177 if (result != NULL) {
2178 temp = resultFilename(filename, "", ".res");
2179 if (temp == NULL) {
2180 fprintf(stderr, "Out of memory\n");
2181 fatalError();
2182 }
2183 t = fopen(temp, "wb");
2184 if (t == NULL) {
2185 fprintf(stderr, "Can't open temp file %s\n", temp);
2186 free(temp);
2187 return(-1);
2188 }
2189 }
2190 #ifdef LIBXML_SCHEMAS_ENABLED
2191 if (rng != NULL) {
2192 ret = xmlTextReaderRelaxNGValidate(reader, rng);
2193 if (ret < 0) {
2194 testErrorHandler(NULL, "Relax-NG schema %s failed to compile\n",
2195 rng);
2196 fclose(t);
2197 if (temp != NULL) {
2198 unlink(temp);
2199 free(temp);
2200 }
2201 return(0);
2202 }
2203 }
2204 #endif
2205 xmlGetWarningsDefaultValue = 1;
2206 ret = xmlTextReaderRead(reader);
2207 while (ret == 1) {
2208 if ((t != NULL) && (rng == NULL))
2209 processNode(t, reader);
2210 ret = xmlTextReaderRead(reader);
2211 }
2212 if (ret != 0) {
2213 testErrorHandler(NULL, "%s : failed to parse\n", filename);
2214 }
2215 if (rng != NULL) {
2216 if (xmlTextReaderIsValid(reader) != 1) {
2217 testErrorHandler(NULL, "%s fails to validate\n", filename);
2218 } else {
2219 testErrorHandler(NULL, "%s validates\n", filename);
2220 }
2221 }
2222 xmlGetWarningsDefaultValue = 0;
2223 if (t != NULL) {
2224 fclose(t);
2225 ret = compareFiles(temp, result);
2226 if (temp != NULL) {
2227 unlink(temp);
2228 free(temp);
2229 }
2230 if (ret) {
2231 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
2232 return(-1);
2233 }
2234 }
2235 if (err != NULL) {
2236 ret = compareFileMem(err, testErrors, testErrorsSize);
2237 if (ret != 0) {
2238 fprintf(stderr, "Error for %s failed\n", filename);
2239 printf("%s", testErrors);
2240 return(-1);
2241 }
2242 }
2243
2244 return(0);
2245 }
2246
2247 /**
2248 * streamParseTest:
2249 * @filename: the file to parse
2250 * @result: the file with expected result
2251 * @err: the file with error messages
2252 *
2253 * Parse a file using the reader API and check for errors.
2254 *
2255 * Returns 0 in case of success, an error code otherwise
2256 */
2257 static int
streamParseTest(const char * filename,const char * result,const char * err,int options)2258 streamParseTest(const char *filename, const char *result, const char *err,
2259 int options) {
2260 xmlTextReaderPtr reader;
2261 int ret;
2262
2263 reader = xmlReaderForFile(filename, NULL, options);
2264 ret = streamProcessTest(filename, result, err, reader, NULL, options);
2265 xmlFreeTextReader(reader);
2266 return(ret);
2267 }
2268
2269 /**
2270 * walkerParseTest:
2271 * @filename: the file to parse
2272 * @result: the file with expected result
2273 * @err: the file with error messages
2274 *
2275 * Parse a file using the walker, i.e. a reader built from a atree.
2276 *
2277 * Returns 0 in case of success, an error code otherwise
2278 */
2279 static int
walkerParseTest(const char * filename,const char * result,const char * err,int options)2280 walkerParseTest(const char *filename, const char *result, const char *err,
2281 int options) {
2282 xmlDocPtr doc;
2283 xmlTextReaderPtr reader;
2284 int ret;
2285
2286 doc = xmlReadFile(filename, NULL, options);
2287 if (doc == NULL) {
2288 fprintf(stderr, "Failed to parse %s\n", filename);
2289 return(-1);
2290 }
2291 reader = xmlReaderWalker(doc);
2292 ret = streamProcessTest(filename, result, err, reader, NULL, options);
2293 xmlFreeTextReader(reader);
2294 xmlFreeDoc(doc);
2295 return(ret);
2296 }
2297
2298 /**
2299 * streamMemParseTest:
2300 * @filename: the file to parse
2301 * @result: the file with expected result
2302 * @err: the file with error messages
2303 *
2304 * Parse a file using the reader API from memory and check for errors.
2305 *
2306 * Returns 0 in case of success, an error code otherwise
2307 */
2308 static int
streamMemParseTest(const char * filename,const char * result,const char * err,int options)2309 streamMemParseTest(const char *filename, const char *result, const char *err,
2310 int options) {
2311 xmlTextReaderPtr reader;
2312 int ret;
2313 const char *base;
2314 int size;
2315
2316 /*
2317 * load and parse the memory
2318 */
2319 if (loadMem(filename, &base, &size) != 0) {
2320 fprintf(stderr, "Failed to load %s\n", filename);
2321 return(-1);
2322 }
2323 reader = xmlReaderForMemory(base, size, filename, NULL, options);
2324 ret = streamProcessTest(filename, result, err, reader, NULL, options);
2325 free((char *)base);
2326 xmlFreeTextReader(reader);
2327 return(ret);
2328 }
2329 #endif
2330
2331 #ifdef LIBXML_XPATH_ENABLED
2332 #ifdef LIBXML_DEBUG_ENABLED
2333 /************************************************************************
2334 * *
2335 * XPath and XPointer based tests *
2336 * *
2337 ************************************************************************/
2338
2339 static FILE *xpathOutput;
2340 static xmlDocPtr xpathDocument;
2341
2342 static void
ignoreGenericError(void * ctx ATTRIBUTE_UNUSED,const char * msg ATTRIBUTE_UNUSED,...)2343 ignoreGenericError(void *ctx ATTRIBUTE_UNUSED,
2344 const char *msg ATTRIBUTE_UNUSED, ...) {
2345 }
2346
2347 static void
testXPath(const char * str,int xptr,int expr)2348 testXPath(const char *str, int xptr, int expr) {
2349 xmlGenericErrorFunc handler = ignoreGenericError;
2350 xmlXPathObjectPtr res;
2351 xmlXPathContextPtr ctxt;
2352
2353 /* Don't print generic errors to stderr. */
2354 initGenericErrorDefaultFunc(&handler);
2355
2356 nb_tests++;
2357 #if defined(LIBXML_XPTR_ENABLED)
2358 if (xptr) {
2359 ctxt = xmlXPtrNewContext(xpathDocument, NULL, NULL);
2360 res = xmlXPtrEval(BAD_CAST str, ctxt);
2361 } else {
2362 #endif
2363 ctxt = xmlXPathNewContext(xpathDocument);
2364 ctxt->node = xmlDocGetRootElement(xpathDocument);
2365 if (expr)
2366 res = xmlXPathEvalExpression(BAD_CAST str, ctxt);
2367 else {
2368 /* res = xmlXPathEval(BAD_CAST str, ctxt); */
2369 xmlXPathCompExprPtr comp;
2370
2371 comp = xmlXPathCompile(BAD_CAST str);
2372 if (comp != NULL) {
2373 res = xmlXPathCompiledEval(comp, ctxt);
2374 xmlXPathFreeCompExpr(comp);
2375 } else
2376 res = NULL;
2377 }
2378 #if defined(LIBXML_XPTR_ENABLED)
2379 }
2380 #endif
2381 xmlXPathDebugDumpObject(xpathOutput, res, 0);
2382 xmlXPathFreeObject(res);
2383 xmlXPathFreeContext(ctxt);
2384
2385 /* Reset generic error handler. */
2386 initGenericErrorDefaultFunc(NULL);
2387 }
2388
2389 /**
2390 * xpathExprTest:
2391 * @filename: the file to parse
2392 * @result: the file with expected result
2393 * @err: the file with error messages
2394 *
2395 * Parse a file containing XPath standalone expressions and evaluate them
2396 *
2397 * Returns 0 in case of success, an error code otherwise
2398 */
2399 static int
xpathCommonTest(const char * filename,const char * result,int xptr,int expr)2400 xpathCommonTest(const char *filename, const char *result,
2401 int xptr, int expr) {
2402 FILE *input;
2403 char expression[5000];
2404 int len, ret = 0;
2405 char *temp;
2406
2407 temp = resultFilename(filename, "", ".res");
2408 if (temp == NULL) {
2409 fprintf(stderr, "Out of memory\n");
2410 fatalError();
2411 }
2412 xpathOutput = fopen(temp, "wb");
2413 if (xpathOutput == NULL) {
2414 fprintf(stderr, "failed to open output file %s\n", temp);
2415 free(temp);
2416 return(-1);
2417 }
2418
2419 input = fopen(filename, "rb");
2420 if (input == NULL) {
2421 xmlGenericError(xmlGenericErrorContext,
2422 "Cannot open %s for reading\n", filename);
2423 free(temp);
2424 return(-1);
2425 }
2426 while (fgets(expression, 4500, input) != NULL) {
2427 len = strlen(expression);
2428 len--;
2429 while ((len >= 0) &&
2430 ((expression[len] == '\n') || (expression[len] == '\t') ||
2431 (expression[len] == '\r') || (expression[len] == ' '))) len--;
2432 expression[len + 1] = 0;
2433 if (len >= 0) {
2434 fprintf(xpathOutput,
2435 "\n========================\nExpression: %s\n",
2436 expression) ;
2437 testXPath(expression, xptr, expr);
2438 }
2439 }
2440
2441 fclose(input);
2442 fclose(xpathOutput);
2443 if (result != NULL) {
2444 ret = compareFiles(temp, result);
2445 if (ret) {
2446 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
2447 }
2448 }
2449
2450 if (temp != NULL) {
2451 unlink(temp);
2452 free(temp);
2453 }
2454 return(ret);
2455 }
2456
2457 /**
2458 * xpathExprTest:
2459 * @filename: the file to parse
2460 * @result: the file with expected result
2461 * @err: the file with error messages
2462 *
2463 * Parse a file containing XPath standalone expressions and evaluate them
2464 *
2465 * Returns 0 in case of success, an error code otherwise
2466 */
2467 static int
xpathExprTest(const char * filename,const char * result,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)2468 xpathExprTest(const char *filename, const char *result,
2469 const char *err ATTRIBUTE_UNUSED,
2470 int options ATTRIBUTE_UNUSED) {
2471 return(xpathCommonTest(filename, result, 0, 1));
2472 }
2473
2474 /**
2475 * xpathDocTest:
2476 * @filename: the file to parse
2477 * @result: the file with expected result
2478 * @err: the file with error messages
2479 *
2480 * Parse a file containing XPath expressions and evaluate them against
2481 * a set of corresponding documents.
2482 *
2483 * Returns 0 in case of success, an error code otherwise
2484 */
2485 static int
xpathDocTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options)2486 xpathDocTest(const char *filename,
2487 const char *resul ATTRIBUTE_UNUSED,
2488 const char *err ATTRIBUTE_UNUSED,
2489 int options) {
2490
2491 char pattern[500];
2492 char result[500];
2493 glob_t globbuf;
2494 size_t i;
2495 int ret = 0, res;
2496
2497 xpathDocument = xmlReadFile(filename, NULL,
2498 options | XML_PARSE_DTDATTR | XML_PARSE_NOENT);
2499 if (xpathDocument == NULL) {
2500 fprintf(stderr, "Failed to load %s\n", filename);
2501 return(-1);
2502 }
2503
2504 snprintf(pattern, 499, "./test/XPath/tests/%s*", baseFilename(filename));
2505 pattern[499] = 0;
2506 globbuf.gl_offs = 0;
2507 glob(pattern, GLOB_DOOFFS, NULL, &globbuf);
2508 for (i = 0;i < globbuf.gl_pathc;i++) {
2509 snprintf(result, 499, "result/XPath/tests/%s",
2510 baseFilename(globbuf.gl_pathv[i]));
2511 res = xpathCommonTest(globbuf.gl_pathv[i], &result[0], 0, 0);
2512 if (res != 0)
2513 ret = res;
2514 }
2515 globfree(&globbuf);
2516
2517 xmlFreeDoc(xpathDocument);
2518 return(ret);
2519 }
2520
2521 #ifdef LIBXML_XPTR_ENABLED
2522 /**
2523 * xptrDocTest:
2524 * @filename: the file to parse
2525 * @result: the file with expected result
2526 * @err: the file with error messages
2527 *
2528 * Parse a file containing XPath expressions and evaluate them against
2529 * a set of corresponding documents.
2530 *
2531 * Returns 0 in case of success, an error code otherwise
2532 */
2533 static int
xptrDocTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options)2534 xptrDocTest(const char *filename,
2535 const char *resul ATTRIBUTE_UNUSED,
2536 const char *err ATTRIBUTE_UNUSED,
2537 int options) {
2538
2539 char pattern[500];
2540 char result[500];
2541 glob_t globbuf;
2542 size_t i;
2543 int ret = 0, res;
2544
2545 xpathDocument = xmlReadFile(filename, NULL,
2546 options | XML_PARSE_DTDATTR | XML_PARSE_NOENT);
2547 if (xpathDocument == NULL) {
2548 fprintf(stderr, "Failed to load %s\n", filename);
2549 return(-1);
2550 }
2551
2552 snprintf(pattern, 499, "./test/XPath/xptr/%s*", baseFilename(filename));
2553 pattern[499] = 0;
2554 globbuf.gl_offs = 0;
2555 glob(pattern, GLOB_DOOFFS, NULL, &globbuf);
2556 for (i = 0;i < globbuf.gl_pathc;i++) {
2557 snprintf(result, 499, "result/XPath/xptr/%s",
2558 baseFilename(globbuf.gl_pathv[i]));
2559 res = xpathCommonTest(globbuf.gl_pathv[i], &result[0], 1, 0);
2560 if (res != 0)
2561 ret = res;
2562 }
2563 globfree(&globbuf);
2564
2565 xmlFreeDoc(xpathDocument);
2566 return(ret);
2567 }
2568 #endif /* LIBXML_XPTR_ENABLED */
2569
2570 /**
2571 * xmlidDocTest:
2572 * @filename: the file to parse
2573 * @result: the file with expected result
2574 * @err: the file with error messages
2575 *
2576 * Parse a file containing xml:id and check for errors and verify
2577 * that XPath queries will work on them as expected.
2578 *
2579 * Returns 0 in case of success, an error code otherwise
2580 */
2581 static int
xmlidDocTest(const char * filename,const char * result,const char * err,int options)2582 xmlidDocTest(const char *filename,
2583 const char *result,
2584 const char *err,
2585 int options) {
2586
2587 int res = 0;
2588 int ret = 0;
2589 char *temp;
2590
2591 xpathDocument = xmlReadFile(filename, NULL,
2592 options | XML_PARSE_DTDATTR | XML_PARSE_NOENT);
2593 if (xpathDocument == NULL) {
2594 fprintf(stderr, "Failed to load %s\n", filename);
2595 return(-1);
2596 }
2597
2598 temp = resultFilename(filename, "", ".res");
2599 if (temp == NULL) {
2600 fprintf(stderr, "Out of memory\n");
2601 fatalError();
2602 }
2603 xpathOutput = fopen(temp, "wb");
2604 if (xpathOutput == NULL) {
2605 fprintf(stderr, "failed to open output file %s\n", temp);
2606 xmlFreeDoc(xpathDocument);
2607 free(temp);
2608 return(-1);
2609 }
2610
2611 testXPath("id('bar')", 0, 0);
2612
2613 fclose(xpathOutput);
2614 if (result != NULL) {
2615 ret = compareFiles(temp, result);
2616 if (ret) {
2617 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
2618 res = 1;
2619 }
2620 }
2621
2622 if (temp != NULL) {
2623 unlink(temp);
2624 free(temp);
2625 }
2626 xmlFreeDoc(xpathDocument);
2627
2628 if (err != NULL) {
2629 ret = compareFileMem(err, testErrors, testErrorsSize);
2630 if (ret != 0) {
2631 fprintf(stderr, "Error for %s failed\n", filename);
2632 res = 1;
2633 }
2634 }
2635 return(res);
2636 }
2637
2638 #endif /* LIBXML_DEBUG_ENABLED */
2639 #endif /* XPATH */
2640 /************************************************************************
2641 * *
2642 * URI based tests *
2643 * *
2644 ************************************************************************/
2645
2646 static void
handleURI(const char * str,const char * base,FILE * o)2647 handleURI(const char *str, const char *base, FILE *o) {
2648 int ret;
2649 xmlURIPtr uri;
2650 xmlChar *res = NULL;
2651
2652 uri = xmlCreateURI();
2653
2654 if (base == NULL) {
2655 ret = xmlParseURIReference(uri, str);
2656 if (ret != 0)
2657 fprintf(o, "%s : error %d\n", str, ret);
2658 else {
2659 xmlNormalizeURIPath(uri->path);
2660 xmlPrintURI(o, uri);
2661 fprintf(o, "\n");
2662 }
2663 } else {
2664 res = xmlBuildURI((xmlChar *)str, (xmlChar *) base);
2665 if (res != NULL) {
2666 fprintf(o, "%s\n", (char *) res);
2667 }
2668 else
2669 fprintf(o, "::ERROR::\n");
2670 }
2671 if (res != NULL)
2672 xmlFree(res);
2673 xmlFreeURI(uri);
2674 }
2675
2676 /**
2677 * uriCommonTest:
2678 * @filename: the file to parse
2679 * @result: the file with expected result
2680 * @err: the file with error messages
2681 *
2682 * Parse a file containing URI and check for errors
2683 *
2684 * Returns 0 in case of success, an error code otherwise
2685 */
2686 static int
uriCommonTest(const char * filename,const char * result,const char * err,const char * base)2687 uriCommonTest(const char *filename,
2688 const char *result,
2689 const char *err,
2690 const char *base) {
2691 char *temp;
2692 FILE *o, *f;
2693 char str[1024];
2694 int res = 0, i, ret;
2695
2696 temp = resultFilename(filename, "", ".res");
2697 if (temp == NULL) {
2698 fprintf(stderr, "Out of memory\n");
2699 fatalError();
2700 }
2701 o = fopen(temp, "wb");
2702 if (o == NULL) {
2703 fprintf(stderr, "failed to open output file %s\n", temp);
2704 free(temp);
2705 return(-1);
2706 }
2707 f = fopen(filename, "rb");
2708 if (f == NULL) {
2709 fprintf(stderr, "failed to open input file %s\n", filename);
2710 fclose(o);
2711 if (temp != NULL) {
2712 unlink(temp);
2713 free(temp);
2714 }
2715 return(-1);
2716 }
2717
2718 while (1) {
2719 /*
2720 * read one line in string buffer.
2721 */
2722 if (fgets (&str[0], sizeof (str) - 1, f) == NULL)
2723 break;
2724
2725 /*
2726 * remove the ending spaces
2727 */
2728 i = strlen(str);
2729 while ((i > 0) &&
2730 ((str[i - 1] == '\n') || (str[i - 1] == '\r') ||
2731 (str[i - 1] == ' ') || (str[i - 1] == '\t'))) {
2732 i--;
2733 str[i] = 0;
2734 }
2735 nb_tests++;
2736 handleURI(str, base, o);
2737 }
2738
2739 fclose(f);
2740 fclose(o);
2741
2742 if (result != NULL) {
2743 ret = compareFiles(temp, result);
2744 if (ret) {
2745 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
2746 res = 1;
2747 }
2748 }
2749 if (err != NULL) {
2750 ret = compareFileMem(err, testErrors, testErrorsSize);
2751 if (ret != 0) {
2752 fprintf(stderr, "Error for %s failed\n", filename);
2753 res = 1;
2754 }
2755 }
2756
2757 if (temp != NULL) {
2758 unlink(temp);
2759 free(temp);
2760 }
2761 return(res);
2762 }
2763
2764 /**
2765 * uriParseTest:
2766 * @filename: the file to parse
2767 * @result: the file with expected result
2768 * @err: the file with error messages
2769 *
2770 * Parse a file containing URI and check for errors
2771 *
2772 * Returns 0 in case of success, an error code otherwise
2773 */
2774 static int
uriParseTest(const char * filename,const char * result,const char * err,int options ATTRIBUTE_UNUSED)2775 uriParseTest(const char *filename,
2776 const char *result,
2777 const char *err,
2778 int options ATTRIBUTE_UNUSED) {
2779 return(uriCommonTest(filename, result, err, NULL));
2780 }
2781
2782 /**
2783 * uriBaseTest:
2784 * @filename: the file to parse
2785 * @result: the file with expected result
2786 * @err: the file with error messages
2787 *
2788 * Parse a file containing URI, compose them against a fixed base and
2789 * check for errors
2790 *
2791 * Returns 0 in case of success, an error code otherwise
2792 */
2793 static int
uriBaseTest(const char * filename,const char * result,const char * err,int options ATTRIBUTE_UNUSED)2794 uriBaseTest(const char *filename,
2795 const char *result,
2796 const char *err,
2797 int options ATTRIBUTE_UNUSED) {
2798 return(uriCommonTest(filename, result, err,
2799 "http://foo.com/path/to/index.html?orig#help"));
2800 }
2801
2802 static int urip_success = 1;
2803 static int urip_current = 0;
2804 static const char *urip_testURLs[] = {
2805 "urip://example.com/a b.html",
2806 "urip://example.com/a%20b.html",
2807 "file:///path/to/a b.html",
2808 "file:///path/to/a%20b.html",
2809 "/path/to/a b.html",
2810 "/path/to/a%20b.html",
2811 "urip://example.com/r" "\xe9" "sum" "\xe9" ".html",
2812 "urip://example.com/test?a=1&b=2%263&c=4#foo",
2813 NULL
2814 };
2815 static const char *urip_rcvsURLs[] = {
2816 /* it is an URI the strings must be escaped */
2817 "urip://example.com/a%20b.html",
2818 /* check that % escaping is not broken */
2819 "urip://example.com/a%20b.html",
2820 /* it's an URI path the strings must be escaped */
2821 "file:///path/to/a%20b.html",
2822 /* check that % escaping is not broken */
2823 "file:///path/to/a%20b.html",
2824 /* this is not an URI, this is a path, so this should not be escaped */
2825 "/path/to/a b.html",
2826 /* check that paths with % are not broken */
2827 "/path/to/a%20b.html",
2828 /* out of context the encoding can't be guessed byte by byte conversion */
2829 "urip://example.com/r%E9sum%E9.html",
2830 /* verify we don't destroy URIs especially the query part */
2831 "urip://example.com/test?a=1&b=2%263&c=4#foo",
2832 NULL
2833 };
2834 static const char *urip_res = "<list/>";
2835 static const char *urip_cur = NULL;
2836 static int urip_rlen;
2837
2838 /**
2839 * uripMatch:
2840 * @URI: an URI to test
2841 *
2842 * Check for an urip: query
2843 *
2844 * Returns 1 if yes and 0 if another Input module should be used
2845 */
2846 static int
uripMatch(const char * URI)2847 uripMatch(const char * URI) {
2848 if ((URI == NULL) || (!strcmp(URI, "file:///etc/xml/catalog")))
2849 return(0);
2850 /* Verify we received the escaped URL */
2851 if (strcmp(urip_rcvsURLs[urip_current], URI))
2852 urip_success = 0;
2853 return(1);
2854 }
2855
2856 /**
2857 * uripOpen:
2858 * @URI: an URI to test
2859 *
2860 * Return a pointer to the urip: query handler, in this example simply
2861 * the urip_current pointer...
2862 *
2863 * Returns an Input context or NULL in case or error
2864 */
2865 static void *
uripOpen(const char * URI)2866 uripOpen(const char * URI) {
2867 if ((URI == NULL) || (!strcmp(URI, "file:///etc/xml/catalog")))
2868 return(NULL);
2869 /* Verify we received the escaped URL */
2870 if (strcmp(urip_rcvsURLs[urip_current], URI))
2871 urip_success = 0;
2872 urip_cur = urip_res;
2873 urip_rlen = strlen(urip_res);
2874 return((void *) urip_cur);
2875 }
2876
2877 /**
2878 * uripClose:
2879 * @context: the read context
2880 *
2881 * Close the urip: query handler
2882 *
2883 * Returns 0 or -1 in case of error
2884 */
2885 static int
uripClose(void * context)2886 uripClose(void * context) {
2887 if (context == NULL) return(-1);
2888 urip_cur = NULL;
2889 urip_rlen = 0;
2890 return(0);
2891 }
2892
2893 /**
2894 * uripRead:
2895 * @context: the read context
2896 * @buffer: where to store data
2897 * @len: number of bytes to read
2898 *
2899 * Implement an urip: query read.
2900 *
2901 * Returns the number of bytes read or -1 in case of error
2902 */
2903 static int
uripRead(void * context,char * buffer,int len)2904 uripRead(void * context, char * buffer, int len) {
2905 const char *ptr = (const char *) context;
2906
2907 if ((context == NULL) || (buffer == NULL) || (len < 0))
2908 return(-1);
2909
2910 if (len > urip_rlen) len = urip_rlen;
2911 memcpy(buffer, ptr, len);
2912 urip_rlen -= len;
2913 return(len);
2914 }
2915
2916 static int
urip_checkURL(const char * URL)2917 urip_checkURL(const char *URL) {
2918 xmlDocPtr doc;
2919
2920 doc = xmlReadFile(URL, NULL, 0);
2921 if (doc == NULL)
2922 return(-1);
2923 xmlFreeDoc(doc);
2924 return(1);
2925 }
2926
2927 /**
2928 * uriPathTest:
2929 * @filename: ignored
2930 * @result: ignored
2931 * @err: ignored
2932 *
2933 * Run a set of tests to check how Path and URI are handled before
2934 * being passed to the I/O layer
2935 *
2936 * Returns 0 in case of success, an error code otherwise
2937 */
2938 static int
uriPathTest(const char * filename ATTRIBUTE_UNUSED,const char * result ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)2939 uriPathTest(const char *filename ATTRIBUTE_UNUSED,
2940 const char *result ATTRIBUTE_UNUSED,
2941 const char *err ATTRIBUTE_UNUSED,
2942 int options ATTRIBUTE_UNUSED) {
2943 int parsed;
2944 int failures = 0;
2945
2946 /*
2947 * register the new I/O handlers
2948 */
2949 if (xmlRegisterInputCallbacks(uripMatch, uripOpen, uripRead, uripClose) < 0)
2950 {
2951 fprintf(stderr, "failed to register HTTP handler\n");
2952 return(-1);
2953 }
2954
2955 for (urip_current = 0;urip_testURLs[urip_current] != NULL;urip_current++) {
2956 urip_success = 1;
2957 parsed = urip_checkURL(urip_testURLs[urip_current]);
2958 if (urip_success != 1) {
2959 fprintf(stderr, "failed the URL passing test for %s",
2960 urip_testURLs[urip_current]);
2961 failures++;
2962 } else if (parsed != 1) {
2963 fprintf(stderr, "failed the parsing test for %s",
2964 urip_testURLs[urip_current]);
2965 failures++;
2966 }
2967 nb_tests++;
2968 }
2969
2970 xmlPopInputCallbacks();
2971 return(failures);
2972 }
2973
2974 #ifdef LIBXML_SCHEMAS_ENABLED
2975 /************************************************************************
2976 * *
2977 * Schemas tests *
2978 * *
2979 ************************************************************************/
2980 static int
schemasOneTest(const char * sch,const char * filename,const char * result,const char * err,int options,xmlSchemaPtr schemas)2981 schemasOneTest(const char *sch,
2982 const char *filename,
2983 const char *result,
2984 const char *err,
2985 int options,
2986 xmlSchemaPtr schemas) {
2987 xmlDocPtr doc;
2988 xmlSchemaValidCtxtPtr ctxt;
2989 int ret = 0;
2990 int validResult = 0;
2991 char *temp;
2992 FILE *schemasOutput;
2993
2994 doc = xmlReadFile(filename, NULL, options);
2995 if (doc == NULL) {
2996 fprintf(stderr, "failed to parse instance %s for %s\n", filename, sch);
2997 return(-1);
2998 }
2999
3000 temp = resultFilename(result, "", ".res");
3001 if (temp == NULL) {
3002 fprintf(stderr, "Out of memory\n");
3003 fatalError();
3004 }
3005 schemasOutput = fopen(temp, "wb");
3006 if (schemasOutput == NULL) {
3007 fprintf(stderr, "failed to open output file %s\n", temp);
3008 xmlFreeDoc(doc);
3009 free(temp);
3010 return(-1);
3011 }
3012
3013 ctxt = xmlSchemaNewValidCtxt(schemas);
3014 xmlSchemaSetValidErrors(ctxt,
3015 (xmlSchemaValidityErrorFunc) testErrorHandler,
3016 (xmlSchemaValidityWarningFunc) testErrorHandler,
3017 ctxt);
3018 validResult = xmlSchemaValidateDoc(ctxt, doc);
3019 if (validResult == 0) {
3020 fprintf(schemasOutput, "%s validates\n", filename);
3021 } else if (validResult > 0) {
3022 fprintf(schemasOutput, "%s fails to validate\n", filename);
3023 } else {
3024 fprintf(schemasOutput, "%s validation generated an internal error\n",
3025 filename);
3026 }
3027 fclose(schemasOutput);
3028 if (result) {
3029 if (compareFiles(temp, result)) {
3030 fprintf(stderr, "Result for %s on %s failed\n", filename, sch);
3031 ret = 1;
3032 }
3033 }
3034 if (temp != NULL) {
3035 unlink(temp);
3036 free(temp);
3037 }
3038
3039 if ((validResult != 0) && (err != NULL)) {
3040 if (compareFileMem(err, testErrors, testErrorsSize)) {
3041 fprintf(stderr, "Error for %s on %s failed\n", filename, sch);
3042 ret = 1;
3043 }
3044 }
3045
3046 xmlSchemaFreeValidCtxt(ctxt);
3047 xmlFreeDoc(doc);
3048 return(ret);
3049 }
3050 /**
3051 * schemasTest:
3052 * @filename: the schemas file
3053 * @result: the file with expected result
3054 * @err: the file with error messages
3055 *
3056 * Parse a file containing URI, compose them against a fixed base and
3057 * check for errors
3058 *
3059 * Returns 0 in case of success, an error code otherwise
3060 */
3061 static int
schemasTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * errr ATTRIBUTE_UNUSED,int options)3062 schemasTest(const char *filename,
3063 const char *resul ATTRIBUTE_UNUSED,
3064 const char *errr ATTRIBUTE_UNUSED,
3065 int options) {
3066 const char *base = baseFilename(filename);
3067 const char *base2;
3068 const char *instance;
3069 xmlSchemaParserCtxtPtr ctxt;
3070 xmlSchemaPtr schemas;
3071 int res = 0, len, ret;
3072 char pattern[500];
3073 char prefix[500];
3074 char result[500];
3075 char err[500];
3076 glob_t globbuf;
3077 size_t i;
3078 char count = 0;
3079
3080 /* first compile the schemas if possible */
3081 ctxt = xmlSchemaNewParserCtxt(filename);
3082 xmlSchemaSetParserErrors(ctxt,
3083 (xmlSchemaValidityErrorFunc) testErrorHandler,
3084 (xmlSchemaValidityWarningFunc) testErrorHandler,
3085 ctxt);
3086 schemas = xmlSchemaParse(ctxt);
3087 xmlSchemaFreeParserCtxt(ctxt);
3088
3089 /*
3090 * most of the mess is about the output filenames generated by the Makefile
3091 */
3092 len = strlen(base);
3093 if ((len > 499) || (len < 5)) {
3094 xmlSchemaFree(schemas);
3095 return(-1);
3096 }
3097 len -= 4; /* remove trailing .xsd */
3098 if (base[len - 2] == '_') {
3099 len -= 2; /* remove subtest number */
3100 }
3101 if (base[len - 2] == '_') {
3102 len -= 2; /* remove subtest number */
3103 }
3104 memcpy(prefix, base, len);
3105 prefix[len] = 0;
3106
3107 snprintf(pattern, 499, "./test/schemas/%s_?.xml", prefix);
3108 pattern[499] = 0;
3109
3110 if (base[len] == '_') {
3111 len += 2;
3112 memcpy(prefix, base, len);
3113 prefix[len] = 0;
3114 }
3115
3116 globbuf.gl_offs = 0;
3117 glob(pattern, GLOB_DOOFFS, NULL, &globbuf);
3118 for (i = 0;i < globbuf.gl_pathc;i++) {
3119 testErrorsSize = 0;
3120 testErrors[0] = 0;
3121 instance = globbuf.gl_pathv[i];
3122 base2 = baseFilename(instance);
3123 len = strlen(base2);
3124 if ((len > 6) && (base2[len - 6] == '_')) {
3125 count = base2[len - 5];
3126 snprintf(result, 499, "result/schemas/%s_%c",
3127 prefix, count);
3128 result[499] = 0;
3129 snprintf(err, 499, "result/schemas/%s_%c.err",
3130 prefix, count);
3131 err[499] = 0;
3132 } else {
3133 fprintf(stderr, "don't know how to process %s\n", instance);
3134 continue;
3135 }
3136 if (schemas == NULL) {
3137 } else {
3138 nb_tests++;
3139 ret = schemasOneTest(filename, instance, result, err,
3140 options, schemas);
3141 if (ret != 0)
3142 res = ret;
3143 }
3144 }
3145 globfree(&globbuf);
3146 xmlSchemaFree(schemas);
3147
3148 return(res);
3149 }
3150
3151 /************************************************************************
3152 * *
3153 * Schemas tests *
3154 * *
3155 ************************************************************************/
3156 static int
rngOneTest(const char * sch,const char * filename,const char * result,const char * err,int options,xmlRelaxNGPtr schemas)3157 rngOneTest(const char *sch,
3158 const char *filename,
3159 const char *result,
3160 const char *err,
3161 int options,
3162 xmlRelaxNGPtr schemas) {
3163 xmlDocPtr doc;
3164 xmlRelaxNGValidCtxtPtr ctxt;
3165 int ret = 0;
3166 char *temp;
3167 FILE *schemasOutput;
3168
3169 doc = xmlReadFile(filename, NULL, options);
3170 if (doc == NULL) {
3171 fprintf(stderr, "failed to parse instance %s for %s\n", filename, sch);
3172 return(-1);
3173 }
3174
3175 temp = resultFilename(result, "", ".res");
3176 if (temp == NULL) {
3177 fprintf(stderr, "Out of memory\n");
3178 fatalError();
3179 }
3180 schemasOutput = fopen(temp, "wb");
3181 if (schemasOutput == NULL) {
3182 fprintf(stderr, "failed to open output file %s\n", temp);
3183 xmlFreeDoc(doc);
3184 free(temp);
3185 return(-1);
3186 }
3187
3188 ctxt = xmlRelaxNGNewValidCtxt(schemas);
3189 xmlRelaxNGSetValidErrors(ctxt,
3190 (xmlRelaxNGValidityErrorFunc) testErrorHandler,
3191 (xmlRelaxNGValidityWarningFunc) testErrorHandler,
3192 ctxt);
3193 ret = xmlRelaxNGValidateDoc(ctxt, doc);
3194 if (ret == 0) {
3195 testErrorHandler(NULL, "%s validates\n", filename);
3196 } else if (ret > 0) {
3197 testErrorHandler(NULL, "%s fails to validate\n", filename);
3198 } else {
3199 testErrorHandler(NULL, "%s validation generated an internal error\n",
3200 filename);
3201 }
3202 fclose(schemasOutput);
3203 ret = 0;
3204 if (result) {
3205 if (compareFiles(temp, result)) {
3206 fprintf(stderr, "Result for %s on %s failed\n", filename, sch);
3207 ret = 1;
3208 }
3209 }
3210 if (temp != NULL) {
3211 unlink(temp);
3212 free(temp);
3213 }
3214
3215 if (err != NULL) {
3216 if (compareFileMem(err, testErrors, testErrorsSize)) {
3217 fprintf(stderr, "Error for %s on %s failed\n", filename, sch);
3218 ret = 1;
3219 printf("%s", testErrors);
3220 }
3221 }
3222
3223
3224 xmlRelaxNGFreeValidCtxt(ctxt);
3225 xmlFreeDoc(doc);
3226 return(ret);
3227 }
3228 /**
3229 * rngTest:
3230 * @filename: the schemas file
3231 * @result: the file with expected result
3232 * @err: the file with error messages
3233 *
3234 * Parse an RNG schemas and then apply it to the related .xml
3235 *
3236 * Returns 0 in case of success, an error code otherwise
3237 */
3238 static int
rngTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * errr ATTRIBUTE_UNUSED,int options)3239 rngTest(const char *filename,
3240 const char *resul ATTRIBUTE_UNUSED,
3241 const char *errr ATTRIBUTE_UNUSED,
3242 int options) {
3243 const char *base = baseFilename(filename);
3244 const char *base2;
3245 const char *instance;
3246 xmlRelaxNGParserCtxtPtr ctxt;
3247 xmlRelaxNGPtr schemas;
3248 int res = 0, len, ret = 0;
3249 char pattern[500];
3250 char prefix[500];
3251 char result[500];
3252 char err[500];
3253 glob_t globbuf;
3254 size_t i;
3255 char count = 0;
3256
3257 /* first compile the schemas if possible */
3258 ctxt = xmlRelaxNGNewParserCtxt(filename);
3259 xmlRelaxNGSetParserErrors(ctxt,
3260 (xmlRelaxNGValidityErrorFunc) testErrorHandler,
3261 (xmlRelaxNGValidityWarningFunc) testErrorHandler,
3262 ctxt);
3263 schemas = xmlRelaxNGParse(ctxt);
3264 xmlRelaxNGFreeParserCtxt(ctxt);
3265
3266 /*
3267 * most of the mess is about the output filenames generated by the Makefile
3268 */
3269 len = strlen(base);
3270 if ((len > 499) || (len < 5)) {
3271 xmlRelaxNGFree(schemas);
3272 return(-1);
3273 }
3274 len -= 4; /* remove trailing .rng */
3275 memcpy(prefix, base, len);
3276 prefix[len] = 0;
3277
3278 snprintf(pattern, 499, "./test/relaxng/%s_?.xml", prefix);
3279 pattern[499] = 0;
3280
3281 globbuf.gl_offs = 0;
3282 glob(pattern, GLOB_DOOFFS, NULL, &globbuf);
3283 for (i = 0;i < globbuf.gl_pathc;i++) {
3284 testErrorsSize = 0;
3285 testErrors[0] = 0;
3286 instance = globbuf.gl_pathv[i];
3287 base2 = baseFilename(instance);
3288 len = strlen(base2);
3289 if ((len > 6) && (base2[len - 6] == '_')) {
3290 count = base2[len - 5];
3291 snprintf(result, 499, "result/relaxng/%s_%c",
3292 prefix, count);
3293 result[499] = 0;
3294 snprintf(err, 499, "result/relaxng/%s_%c.err",
3295 prefix, count);
3296 err[499] = 0;
3297 } else {
3298 fprintf(stderr, "don't know how to process %s\n", instance);
3299 continue;
3300 }
3301 if (schemas == NULL) {
3302 } else {
3303 nb_tests++;
3304 ret = rngOneTest(filename, instance, result, err,
3305 options, schemas);
3306 if (res != 0)
3307 ret = res;
3308 }
3309 }
3310 globfree(&globbuf);
3311 xmlRelaxNGFree(schemas);
3312
3313 return(ret);
3314 }
3315
3316 #ifdef LIBXML_READER_ENABLED
3317 /**
3318 * rngStreamTest:
3319 * @filename: the schemas file
3320 * @result: the file with expected result
3321 * @err: the file with error messages
3322 *
3323 * Parse a set of files with streaming, applying an RNG schemas
3324 *
3325 * Returns 0 in case of success, an error code otherwise
3326 */
3327 static int
rngStreamTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * errr ATTRIBUTE_UNUSED,int options)3328 rngStreamTest(const char *filename,
3329 const char *resul ATTRIBUTE_UNUSED,
3330 const char *errr ATTRIBUTE_UNUSED,
3331 int options) {
3332 const char *base = baseFilename(filename);
3333 const char *base2;
3334 const char *instance;
3335 int res = 0, len, ret;
3336 char pattern[500];
3337 char prefix[500];
3338 char result[500];
3339 char err[500];
3340 glob_t globbuf;
3341 size_t i;
3342 char count = 0;
3343 xmlTextReaderPtr reader;
3344 int disable_err = 0;
3345
3346 /*
3347 * most of the mess is about the output filenames generated by the Makefile
3348 */
3349 len = strlen(base);
3350 if ((len > 499) || (len < 5)) {
3351 fprintf(stderr, "len(base) == %d !\n", len);
3352 return(-1);
3353 }
3354 len -= 4; /* remove trailing .rng */
3355 memcpy(prefix, base, len);
3356 prefix[len] = 0;
3357
3358 /*
3359 * strictly unifying the error messages is nearly impossible this
3360 * hack is also done in the Makefile
3361 */
3362 if ((!strcmp(prefix, "tutor10_1")) || (!strcmp(prefix, "tutor10_2")) ||
3363 (!strcmp(prefix, "tutor3_2")) || (!strcmp(prefix, "307377")) ||
3364 (!strcmp(prefix, "tutor8_2")))
3365 disable_err = 1;
3366
3367 snprintf(pattern, 499, "./test/relaxng/%s_?.xml", prefix);
3368 pattern[499] = 0;
3369
3370 globbuf.gl_offs = 0;
3371 glob(pattern, GLOB_DOOFFS, NULL, &globbuf);
3372 for (i = 0;i < globbuf.gl_pathc;i++) {
3373 testErrorsSize = 0;
3374 testErrors[0] = 0;
3375 instance = globbuf.gl_pathv[i];
3376 base2 = baseFilename(instance);
3377 len = strlen(base2);
3378 if ((len > 6) && (base2[len - 6] == '_')) {
3379 count = base2[len - 5];
3380 snprintf(result, 499, "result/relaxng/%s_%c",
3381 prefix, count);
3382 result[499] = 0;
3383 snprintf(err, 499, "result/relaxng/%s_%c.err",
3384 prefix, count);
3385 err[499] = 0;
3386 } else {
3387 fprintf(stderr, "don't know how to process %s\n", instance);
3388 continue;
3389 }
3390 reader = xmlReaderForFile(instance, NULL, options);
3391 if (reader == NULL) {
3392 fprintf(stderr, "Failed to build reder for %s\n", instance);
3393 }
3394 if (disable_err == 1)
3395 ret = streamProcessTest(instance, result, NULL, reader, filename,
3396 options);
3397 else
3398 ret = streamProcessTest(instance, result, err, reader, filename,
3399 options);
3400 xmlFreeTextReader(reader);
3401 if (ret != 0) {
3402 fprintf(stderr, "instance %s failed\n", instance);
3403 res = ret;
3404 }
3405 }
3406 globfree(&globbuf);
3407
3408 return(res);
3409 }
3410 #endif /* READER */
3411
3412 #endif
3413
3414 #ifdef LIBXML_PATTERN_ENABLED
3415 #ifdef LIBXML_READER_ENABLED
3416 /************************************************************************
3417 * *
3418 * Patterns tests *
3419 * *
3420 ************************************************************************/
patternNode(FILE * out,xmlTextReaderPtr reader,const char * pattern,xmlPatternPtr patternc,xmlStreamCtxtPtr patstream)3421 static void patternNode(FILE *out, xmlTextReaderPtr reader,
3422 const char *pattern, xmlPatternPtr patternc,
3423 xmlStreamCtxtPtr patstream) {
3424 xmlChar *path = NULL;
3425 int match = -1;
3426 int type, empty;
3427
3428 type = xmlTextReaderNodeType(reader);
3429 empty = xmlTextReaderIsEmptyElement(reader);
3430
3431 if (type == XML_READER_TYPE_ELEMENT) {
3432 /* do the check only on element start */
3433 match = xmlPatternMatch(patternc, xmlTextReaderCurrentNode(reader));
3434
3435 if (match) {
3436 path = xmlGetNodePath(xmlTextReaderCurrentNode(reader));
3437 fprintf(out, "Node %s matches pattern %s\n", path, pattern);
3438 }
3439 }
3440 if (patstream != NULL) {
3441 int ret;
3442
3443 if (type == XML_READER_TYPE_ELEMENT) {
3444 ret = xmlStreamPush(patstream,
3445 xmlTextReaderConstLocalName(reader),
3446 xmlTextReaderConstNamespaceUri(reader));
3447 if (ret < 0) {
3448 fprintf(out, "xmlStreamPush() failure\n");
3449 xmlFreeStreamCtxt(patstream);
3450 patstream = NULL;
3451 } else if (ret != match) {
3452 if (path == NULL) {
3453 path = xmlGetNodePath(
3454 xmlTextReaderCurrentNode(reader));
3455 }
3456 fprintf(out,
3457 "xmlPatternMatch and xmlStreamPush disagree\n");
3458 fprintf(out,
3459 " pattern %s node %s\n",
3460 pattern, path);
3461 }
3462
3463
3464 }
3465 if ((type == XML_READER_TYPE_END_ELEMENT) ||
3466 ((type == XML_READER_TYPE_ELEMENT) && (empty))) {
3467 ret = xmlStreamPop(patstream);
3468 if (ret < 0) {
3469 fprintf(out, "xmlStreamPop() failure\n");
3470 xmlFreeStreamCtxt(patstream);
3471 patstream = NULL;
3472 }
3473 }
3474 }
3475 if (path != NULL)
3476 xmlFree(path);
3477 }
3478
3479 /**
3480 * patternTest:
3481 * @filename: the schemas file
3482 * @result: the file with expected result
3483 * @err: the file with error messages
3484 *
3485 * Parse a set of files with streaming, applying an RNG schemas
3486 *
3487 * Returns 0 in case of success, an error code otherwise
3488 */
3489 static int
patternTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options)3490 patternTest(const char *filename,
3491 const char *resul ATTRIBUTE_UNUSED,
3492 const char *err ATTRIBUTE_UNUSED,
3493 int options) {
3494 xmlPatternPtr patternc = NULL;
3495 xmlStreamCtxtPtr patstream = NULL;
3496 FILE *o, *f;
3497 char str[1024];
3498 char xml[500];
3499 char result[500];
3500 int len, i;
3501 int ret = 0, res;
3502 char *temp;
3503 xmlTextReaderPtr reader;
3504 xmlDocPtr doc;
3505
3506 len = strlen(filename);
3507 len -= 4;
3508 memcpy(xml, filename, len);
3509 xml[len] = 0;
3510 snprintf(result, 499, "result/pattern/%s", baseFilename(xml));
3511 result[499] = 0;
3512 memcpy(xml + len, ".xml", 5);
3513
3514 if (!checkTestFile(xml) && !update_results) {
3515 fprintf(stderr, "Missing xml file %s\n", xml);
3516 return(-1);
3517 }
3518 if (!checkTestFile(result) && !update_results) {
3519 fprintf(stderr, "Missing result file %s\n", result);
3520 return(-1);
3521 }
3522 f = fopen(filename, "rb");
3523 if (f == NULL) {
3524 fprintf(stderr, "Failed to open %s\n", filename);
3525 return(-1);
3526 }
3527 temp = resultFilename(filename, "", ".res");
3528 if (temp == NULL) {
3529 fprintf(stderr, "Out of memory\n");
3530 fatalError();
3531 }
3532 o = fopen(temp, "wb");
3533 if (o == NULL) {
3534 fprintf(stderr, "failed to open output file %s\n", temp);
3535 fclose(f);
3536 free(temp);
3537 return(-1);
3538 }
3539 while (1) {
3540 /*
3541 * read one line in string buffer.
3542 */
3543 if (fgets (&str[0], sizeof (str) - 1, f) == NULL)
3544 break;
3545
3546 /*
3547 * remove the ending spaces
3548 */
3549 i = strlen(str);
3550 while ((i > 0) &&
3551 ((str[i - 1] == '\n') || (str[i - 1] == '\r') ||
3552 (str[i - 1] == ' ') || (str[i - 1] == '\t'))) {
3553 i--;
3554 str[i] = 0;
3555 }
3556 doc = xmlReadFile(xml, NULL, options);
3557 if (doc == NULL) {
3558 fprintf(stderr, "Failed to parse %s\n", xml);
3559 ret = 1;
3560 } else {
3561 xmlNodePtr root;
3562 const xmlChar *namespaces[22];
3563 int j;
3564 xmlNsPtr ns;
3565
3566 root = xmlDocGetRootElement(doc);
3567 for (ns = root->nsDef, j = 0;ns != NULL && j < 20;ns=ns->next) {
3568 namespaces[j++] = ns->href;
3569 namespaces[j++] = ns->prefix;
3570 }
3571 namespaces[j++] = NULL;
3572 namespaces[j] = NULL;
3573
3574 patternc = xmlPatterncompile((const xmlChar *) str, doc->dict,
3575 0, &namespaces[0]);
3576 if (patternc == NULL) {
3577 testErrorHandler(NULL,
3578 "Pattern %s failed to compile\n", str);
3579 xmlFreeDoc(doc);
3580 ret = 1;
3581 continue;
3582 }
3583 patstream = xmlPatternGetStreamCtxt(patternc);
3584 if (patstream != NULL) {
3585 ret = xmlStreamPush(patstream, NULL, NULL);
3586 if (ret < 0) {
3587 fprintf(stderr, "xmlStreamPush() failure\n");
3588 xmlFreeStreamCtxt(patstream);
3589 patstream = NULL;
3590 }
3591 }
3592 nb_tests++;
3593
3594 reader = xmlReaderWalker(doc);
3595 res = xmlTextReaderRead(reader);
3596 while (res == 1) {
3597 patternNode(o, reader, str, patternc, patstream);
3598 res = xmlTextReaderRead(reader);
3599 }
3600 if (res != 0) {
3601 fprintf(o, "%s : failed to parse\n", filename);
3602 }
3603 xmlFreeTextReader(reader);
3604 xmlFreeDoc(doc);
3605 xmlFreeStreamCtxt(patstream);
3606 patstream = NULL;
3607 xmlFreePattern(patternc);
3608
3609 }
3610 }
3611
3612 fclose(f);
3613 fclose(o);
3614
3615 ret = compareFiles(temp, result);
3616 if (ret) {
3617 fprintf(stderr, "Result for %s failed in %s\n", filename, result);
3618 ret = 1;
3619 }
3620 if (temp != NULL) {
3621 unlink(temp);
3622 free(temp);
3623 }
3624 return(ret);
3625 }
3626 #endif /* READER */
3627 #endif /* PATTERN */
3628 #ifdef LIBXML_C14N_ENABLED
3629 /************************************************************************
3630 * *
3631 * Canonicalization tests *
3632 * *
3633 ************************************************************************/
3634 static xmlXPathObjectPtr
load_xpath_expr(xmlDocPtr parent_doc,const char * filename)3635 load_xpath_expr (xmlDocPtr parent_doc, const char* filename) {
3636 xmlXPathObjectPtr xpath;
3637 xmlDocPtr doc;
3638 xmlChar *expr;
3639 xmlXPathContextPtr ctx;
3640 xmlNodePtr node;
3641 xmlNsPtr ns;
3642
3643 /*
3644 * load XPath expr as a file
3645 */
3646 xmlLoadExtDtdDefaultValue = XML_DETECT_IDS | XML_COMPLETE_ATTRS;
3647 xmlSubstituteEntitiesDefault(1);
3648
3649 doc = xmlReadFile(filename, NULL, XML_PARSE_DTDATTR | XML_PARSE_NOENT);
3650 if (doc == NULL) {
3651 fprintf(stderr, "Error: unable to parse file \"%s\"\n", filename);
3652 return(NULL);
3653 }
3654
3655 /*
3656 * Check the document is of the right kind
3657 */
3658 if(xmlDocGetRootElement(doc) == NULL) {
3659 fprintf(stderr,"Error: empty document for file \"%s\"\n", filename);
3660 xmlFreeDoc(doc);
3661 return(NULL);
3662 }
3663
3664 node = doc->children;
3665 while(node != NULL && !xmlStrEqual(node->name, (const xmlChar *)"XPath")) {
3666 node = node->next;
3667 }
3668
3669 if(node == NULL) {
3670 fprintf(stderr,"Error: XPath element expected in the file \"%s\"\n", filename);
3671 xmlFreeDoc(doc);
3672 return(NULL);
3673 }
3674
3675 expr = xmlNodeGetContent(node);
3676 if(expr == NULL) {
3677 fprintf(stderr,"Error: XPath content element is NULL \"%s\"\n", filename);
3678 xmlFreeDoc(doc);
3679 return(NULL);
3680 }
3681
3682 ctx = xmlXPathNewContext(parent_doc);
3683 if(ctx == NULL) {
3684 fprintf(stderr,"Error: unable to create new context\n");
3685 xmlFree(expr);
3686 xmlFreeDoc(doc);
3687 return(NULL);
3688 }
3689
3690 /*
3691 * Register namespaces
3692 */
3693 ns = node->nsDef;
3694 while(ns != NULL) {
3695 if(xmlXPathRegisterNs(ctx, ns->prefix, ns->href) != 0) {
3696 fprintf(stderr,"Error: unable to register NS with prefix=\"%s\" and href=\"%s\"\n", ns->prefix, ns->href);
3697 xmlFree(expr);
3698 xmlXPathFreeContext(ctx);
3699 xmlFreeDoc(doc);
3700 return(NULL);
3701 }
3702 ns = ns->next;
3703 }
3704
3705 /*
3706 * Evaluate xpath
3707 */
3708 xpath = xmlXPathEvalExpression(expr, ctx);
3709 if(xpath == NULL) {
3710 fprintf(stderr,"Error: unable to evaluate xpath expression\n");
3711 xmlFree(expr);
3712 xmlXPathFreeContext(ctx);
3713 xmlFreeDoc(doc);
3714 return(NULL);
3715 }
3716
3717 /* print_xpath_nodes(xpath->nodesetval); */
3718
3719 xmlFree(expr);
3720 xmlXPathFreeContext(ctx);
3721 xmlFreeDoc(doc);
3722 return(xpath);
3723 }
3724
3725 /*
3726 * Macro used to grow the current buffer.
3727 */
3728 #define xxx_growBufferReentrant() { \
3729 buffer_size *= 2; \
3730 buffer = (xmlChar **) \
3731 xmlRealloc(buffer, buffer_size * sizeof(xmlChar*)); \
3732 if (buffer == NULL) { \
3733 perror("realloc failed"); \
3734 return(NULL); \
3735 } \
3736 }
3737
3738 static xmlChar **
parse_list(xmlChar * str)3739 parse_list(xmlChar *str) {
3740 xmlChar **buffer;
3741 xmlChar **out = NULL;
3742 int buffer_size = 0;
3743 int len;
3744
3745 if(str == NULL) {
3746 return(NULL);
3747 }
3748
3749 len = xmlStrlen(str);
3750 if((str[0] == '\'') && (str[len - 1] == '\'')) {
3751 str[len - 1] = '\0';
3752 str++;
3753 }
3754 /*
3755 * allocate an translation buffer.
3756 */
3757 buffer_size = 1000;
3758 buffer = (xmlChar **) xmlMalloc(buffer_size * sizeof(xmlChar*));
3759 if (buffer == NULL) {
3760 perror("malloc failed");
3761 return(NULL);
3762 }
3763 out = buffer;
3764
3765 while(*str != '\0') {
3766 if (out - buffer > buffer_size - 10) {
3767 int indx = out - buffer;
3768
3769 xxx_growBufferReentrant();
3770 out = &buffer[indx];
3771 }
3772 (*out++) = str;
3773 while(*str != ',' && *str != '\0') ++str;
3774 if(*str == ',') *(str++) = '\0';
3775 }
3776 (*out) = NULL;
3777 return buffer;
3778 }
3779
3780 static int
c14nRunTest(const char * xml_filename,int with_comments,int mode,const char * xpath_filename,const char * ns_filename,const char * result_file)3781 c14nRunTest(const char* xml_filename, int with_comments, int mode,
3782 const char* xpath_filename, const char *ns_filename,
3783 const char* result_file) {
3784 xmlDocPtr doc;
3785 xmlXPathObjectPtr xpath = NULL;
3786 xmlChar *result = NULL;
3787 int ret;
3788 xmlChar **inclusive_namespaces = NULL;
3789 const char *nslist = NULL;
3790 int nssize;
3791
3792
3793 /*
3794 * build an XML tree from a the file; we need to add default
3795 * attributes and resolve all character and entities references
3796 */
3797 xmlLoadExtDtdDefaultValue = XML_DETECT_IDS | XML_COMPLETE_ATTRS;
3798 xmlSubstituteEntitiesDefault(1);
3799
3800 doc = xmlReadFile(xml_filename, NULL, XML_PARSE_DTDATTR | XML_PARSE_NOENT);
3801 if (doc == NULL) {
3802 fprintf(stderr, "Error: unable to parse file \"%s\"\n", xml_filename);
3803 return(-1);
3804 }
3805
3806 /*
3807 * Check the document is of the right kind
3808 */
3809 if(xmlDocGetRootElement(doc) == NULL) {
3810 fprintf(stderr,"Error: empty document for file \"%s\"\n", xml_filename);
3811 xmlFreeDoc(doc);
3812 return(-1);
3813 }
3814
3815 /*
3816 * load xpath file if specified
3817 */
3818 if(xpath_filename) {
3819 xpath = load_xpath_expr(doc, xpath_filename);
3820 if(xpath == NULL) {
3821 fprintf(stderr,"Error: unable to evaluate xpath expression\n");
3822 xmlFreeDoc(doc);
3823 return(-1);
3824 }
3825 }
3826
3827 if (ns_filename != NULL) {
3828 if (loadMem(ns_filename, &nslist, &nssize)) {
3829 fprintf(stderr,"Error: unable to evaluate xpath expression\n");
3830 if(xpath != NULL) xmlXPathFreeObject(xpath);
3831 xmlFreeDoc(doc);
3832 return(-1);
3833 }
3834 inclusive_namespaces = parse_list((xmlChar *) nslist);
3835 }
3836
3837 /*
3838 * Canonical form
3839 */
3840 /* fprintf(stderr,"File \"%s\" loaded: start canonization\n", xml_filename); */
3841 ret = xmlC14NDocDumpMemory(doc,
3842 (xpath) ? xpath->nodesetval : NULL,
3843 mode, inclusive_namespaces,
3844 with_comments, &result);
3845 if (ret >= 0) {
3846 if(result != NULL) {
3847 if (compareFileMem(result_file, (const char *) result, ret)) {
3848 fprintf(stderr, "Result mismatch for %s\n", xml_filename);
3849 fprintf(stderr, "RESULT:\n%s\n", (const char*)result);
3850 ret = -1;
3851 }
3852 }
3853 } else {
3854 fprintf(stderr,"Error: failed to canonicalize XML file \"%s\" (ret=%d)\n", xml_filename, ret);
3855 ret = -1;
3856 }
3857
3858 /*
3859 * Cleanup
3860 */
3861 if (result != NULL) xmlFree(result);
3862 if(xpath != NULL) xmlXPathFreeObject(xpath);
3863 if (inclusive_namespaces != NULL) xmlFree(inclusive_namespaces);
3864 if (nslist != NULL) free((char *) nslist);
3865 xmlFreeDoc(doc);
3866
3867 return(ret);
3868 }
3869
3870 static int
c14nCommonTest(const char * filename,int with_comments,int mode,const char * subdir)3871 c14nCommonTest(const char *filename, int with_comments, int mode,
3872 const char *subdir) {
3873 char buf[500];
3874 char prefix[500];
3875 const char *base;
3876 int len;
3877 char *result = NULL;
3878 char *xpath = NULL;
3879 char *ns = NULL;
3880 int ret = 0;
3881
3882 base = baseFilename(filename);
3883 len = strlen(base);
3884 len -= 4;
3885 memcpy(prefix, base, len);
3886 prefix[len] = 0;
3887
3888 snprintf(buf, 499, "result/c14n/%s/%s", subdir,prefix);
3889 if (!checkTestFile(buf) && !update_results) {
3890 fprintf(stderr, "Missing result file %s", buf);
3891 return(-1);
3892 }
3893 result = strdup(buf);
3894 snprintf(buf, 499, "test/c14n/%s/%s.xpath", subdir,prefix);
3895 if (checkTestFile(buf)) {
3896 xpath = strdup(buf);
3897 }
3898 snprintf(buf, 499, "test/c14n/%s/%s.ns", subdir,prefix);
3899 if (checkTestFile(buf)) {
3900 ns = strdup(buf);
3901 }
3902
3903 nb_tests++;
3904 if (c14nRunTest(filename, with_comments, mode,
3905 xpath, ns, result) < 0)
3906 ret = 1;
3907
3908 if (result != NULL) free(result);
3909 if (xpath != NULL) free(xpath);
3910 if (ns != NULL) free(ns);
3911 return(ret);
3912 }
3913
3914 static int
c14nWithCommentTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)3915 c14nWithCommentTest(const char *filename,
3916 const char *resul ATTRIBUTE_UNUSED,
3917 const char *err ATTRIBUTE_UNUSED,
3918 int options ATTRIBUTE_UNUSED) {
3919 return(c14nCommonTest(filename, 1, XML_C14N_1_0, "with-comments"));
3920 }
3921 static int
c14nWithoutCommentTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)3922 c14nWithoutCommentTest(const char *filename,
3923 const char *resul ATTRIBUTE_UNUSED,
3924 const char *err ATTRIBUTE_UNUSED,
3925 int options ATTRIBUTE_UNUSED) {
3926 return(c14nCommonTest(filename, 0, XML_C14N_1_0, "without-comments"));
3927 }
3928 static int
c14nExcWithoutCommentTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)3929 c14nExcWithoutCommentTest(const char *filename,
3930 const char *resul ATTRIBUTE_UNUSED,
3931 const char *err ATTRIBUTE_UNUSED,
3932 int options ATTRIBUTE_UNUSED) {
3933 return(c14nCommonTest(filename, 0, XML_C14N_EXCLUSIVE_1_0, "exc-without-comments"));
3934 }
3935 static int
c14n11WithoutCommentTest(const char * filename,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)3936 c14n11WithoutCommentTest(const char *filename,
3937 const char *resul ATTRIBUTE_UNUSED,
3938 const char *err ATTRIBUTE_UNUSED,
3939 int options ATTRIBUTE_UNUSED) {
3940 return(c14nCommonTest(filename, 0, XML_C14N_1_1, "1-1-without-comments"));
3941 }
3942 #endif
3943 #if defined(LIBXML_THREAD_ENABLED) && defined(LIBXML_CATALOG_ENABLED)
3944 /************************************************************************
3945 * *
3946 * Catalog and threads test *
3947 * *
3948 ************************************************************************/
3949
3950 /*
3951 * mostly a cut and paste from testThreads.c
3952 */
3953 #define MAX_ARGC 20
3954
3955 typedef struct {
3956 const char *filename;
3957 int okay;
3958 } xmlThreadParams;
3959
3960 static const char *catalog = "test/threads/complex.xml";
3961 static xmlThreadParams threadParams[] = {
3962 { "test/threads/abc.xml", 0 },
3963 { "test/threads/acb.xml", 0 },
3964 { "test/threads/bac.xml", 0 },
3965 { "test/threads/bca.xml", 0 },
3966 { "test/threads/cab.xml", 0 },
3967 { "test/threads/cba.xml", 0 },
3968 { "test/threads/invalid.xml", 0 }
3969 };
3970 static const unsigned int num_threads = sizeof(threadParams) /
3971 sizeof(threadParams[0]);
3972
3973 #ifndef xmlDoValidityCheckingDefaultValue
3974 #error xmlDoValidityCheckingDefaultValue is not a macro
3975 #endif
3976 #ifndef xmlGenericErrorContext
3977 #error xmlGenericErrorContext is not a macro
3978 #endif
3979
3980 static void *
thread_specific_data(void * private_data)3981 thread_specific_data(void *private_data)
3982 {
3983 xmlDocPtr myDoc;
3984 xmlThreadParams *params = (xmlThreadParams *) private_data;
3985 const char *filename = params->filename;
3986 int okay = 1;
3987
3988 if (!strcmp(filename, "test/threads/invalid.xml")) {
3989 xmlDoValidityCheckingDefaultValue = 0;
3990 xmlGenericErrorContext = stdout;
3991 } else {
3992 xmlDoValidityCheckingDefaultValue = 1;
3993 xmlGenericErrorContext = stderr;
3994 }
3995 #ifdef LIBXML_SAX1_ENABLED
3996 myDoc = xmlParseFile(filename);
3997 #else
3998 myDoc = xmlReadFile(filename, NULL, XML_WITH_CATALOG);
3999 #endif
4000 if (myDoc) {
4001 xmlFreeDoc(myDoc);
4002 } else {
4003 printf("parse failed\n");
4004 okay = 0;
4005 }
4006 if (!strcmp(filename, "test/threads/invalid.xml")) {
4007 if (xmlDoValidityCheckingDefaultValue != 0) {
4008 printf("ValidityCheckingDefaultValue override failed\n");
4009 okay = 0;
4010 }
4011 if (xmlGenericErrorContext != stdout) {
4012 printf("xmlGenericErrorContext override failed\n");
4013 okay = 0;
4014 }
4015 } else {
4016 if (xmlDoValidityCheckingDefaultValue != 1) {
4017 printf("ValidityCheckingDefaultValue override failed\n");
4018 okay = 0;
4019 }
4020 if (xmlGenericErrorContext != stderr) {
4021 printf("xmlGenericErrorContext override failed\n");
4022 okay = 0;
4023 }
4024 }
4025 params->okay = okay;
4026 return(NULL);
4027 }
4028
4029 #if defined(_WIN32) && !defined(__CYGWIN__)
4030 #include <windows.h>
4031 #include <string.h>
4032
4033 #define TEST_REPEAT_COUNT 500
4034
4035 static HANDLE tid[MAX_ARGC];
4036
4037 static DWORD WINAPI
win32_thread_specific_data(void * private_data)4038 win32_thread_specific_data(void *private_data)
4039 {
4040 thread_specific_data(private_data);
4041 return(0);
4042 }
4043
4044 static int
testThread(void)4045 testThread(void)
4046 {
4047 unsigned int i, repeat;
4048 BOOL ret;
4049 int res = 0;
4050
4051 xmlInitParser();
4052 for (repeat = 0; repeat < TEST_REPEAT_COUNT; repeat++) {
4053 xmlLoadCatalog(catalog);
4054 nb_tests++;
4055
4056 for (i = 0; i < num_threads; i++) {
4057 tid[i] = (HANDLE) - 1;
4058 }
4059
4060 for (i = 0; i < num_threads; i++) {
4061 DWORD useless;
4062
4063 tid[i] = CreateThread(NULL, 0,
4064 win32_thread_specific_data,
4065 (void *) &threadParams[i], 0,
4066 &useless);
4067 if (tid[i] == NULL) {
4068 fprintf(stderr, "CreateThread failed\n");
4069 return(1);
4070 }
4071 }
4072
4073 if (WaitForMultipleObjects(num_threads, tid, TRUE, INFINITE) ==
4074 WAIT_FAILED) {
4075 fprintf(stderr, "WaitForMultipleObjects failed\n");
4076 return(1);
4077 }
4078
4079 for (i = 0; i < num_threads; i++) {
4080 DWORD exitCode;
4081 ret = GetExitCodeThread(tid[i], &exitCode);
4082 if (ret == 0) {
4083 fprintf(stderr, "GetExitCodeThread failed\n");
4084 return(1);
4085 }
4086 CloseHandle(tid[i]);
4087 }
4088
4089 xmlCatalogCleanup();
4090 for (i = 0; i < num_threads; i++) {
4091 if (threadParams[i].okay == 0) {
4092 fprintf(stderr, "Thread %d handling %s failed\n",
4093 i, threadParams[i].filename);
4094 res = 1;
4095 }
4096 }
4097 }
4098
4099 return (res);
4100 }
4101
4102 #elif defined __BEOS__
4103 #include <OS.h>
4104
4105 static thread_id tid[MAX_ARGC];
4106
4107 static int
testThread(void)4108 testThread(void)
4109 {
4110 unsigned int i, repeat;
4111 status_t ret;
4112 int res = 0;
4113
4114 xmlInitParser();
4115 for (repeat = 0; repeat < 500; repeat++) {
4116 xmlLoadCatalog(catalog);
4117 for (i = 0; i < num_threads; i++) {
4118 tid[i] = (thread_id) - 1;
4119 }
4120 for (i = 0; i < num_threads; i++) {
4121 tid[i] =
4122 spawn_thread(thread_specific_data, "xmlTestThread",
4123 B_NORMAL_PRIORITY, (void *) &threadParams[i]);
4124 if (tid[i] < B_OK) {
4125 fprintf(stderr, "beos_thread_create failed\n");
4126 return (1);
4127 }
4128 printf("beos_thread_create %d -> %d\n", i, tid[i]);
4129 }
4130 for (i = 0; i < num_threads; i++) {
4131 void *result;
4132 ret = wait_for_thread(tid[i], &result);
4133 printf("beos_thread_wait %d -> %d\n", i, ret);
4134 if (ret != B_OK) {
4135 fprintf(stderr, "beos_thread_wait failed\n");
4136 return (1);
4137 }
4138 }
4139
4140 xmlCatalogCleanup();
4141 ret = B_OK;
4142 for (i = 0; i < num_threads; i++)
4143 if (threadParams[i].okay == 0) {
4144 printf("Thread %d handling %s failed\n", i,
4145 threadParams[i].filename);
4146 ret = B_ERROR;
4147 }
4148 }
4149 if (ret != B_OK)
4150 return(1);
4151 return (0);
4152 }
4153
4154 #elif defined HAVE_PTHREAD_H
4155 #include <pthread.h>
4156
4157 static pthread_t tid[MAX_ARGC];
4158
4159 static int
testThread(void)4160 testThread(void)
4161 {
4162 unsigned int i, repeat;
4163 int ret;
4164 int res = 0;
4165
4166 xmlInitParser();
4167
4168 for (repeat = 0; repeat < 500; repeat++) {
4169 xmlLoadCatalog(catalog);
4170 nb_tests++;
4171
4172 for (i = 0; i < num_threads; i++) {
4173 tid[i] = (pthread_t) - 1;
4174 }
4175
4176 for (i = 0; i < num_threads; i++) {
4177 ret = pthread_create(&tid[i], 0, thread_specific_data,
4178 (void *) &threadParams[i]);
4179 if (ret != 0) {
4180 fprintf(stderr, "pthread_create failed\n");
4181 return (1);
4182 }
4183 }
4184 for (i = 0; i < num_threads; i++) {
4185 void *result;
4186 ret = pthread_join(tid[i], &result);
4187 if (ret != 0) {
4188 fprintf(stderr, "pthread_join failed\n");
4189 return (1);
4190 }
4191 }
4192
4193 xmlCatalogCleanup();
4194 for (i = 0; i < num_threads; i++)
4195 if (threadParams[i].okay == 0) {
4196 fprintf(stderr, "Thread %d handling %s failed\n",
4197 i, threadParams[i].filename);
4198 res = 1;
4199 }
4200 }
4201 return (res);
4202 }
4203
4204 #else
4205 static int
testThread(void)4206 testThread(void)
4207 {
4208 fprintf(stderr,
4209 "Specific platform thread support not detected\n");
4210 return (-1);
4211 }
4212 #endif
4213 static int
threadsTest(const char * filename ATTRIBUTE_UNUSED,const char * resul ATTRIBUTE_UNUSED,const char * err ATTRIBUTE_UNUSED,int options ATTRIBUTE_UNUSED)4214 threadsTest(const char *filename ATTRIBUTE_UNUSED,
4215 const char *resul ATTRIBUTE_UNUSED,
4216 const char *err ATTRIBUTE_UNUSED,
4217 int options ATTRIBUTE_UNUSED) {
4218 return(testThread());
4219 }
4220 #endif
4221 /************************************************************************
4222 * *
4223 * Tests Descriptions *
4224 * *
4225 ************************************************************************/
4226
4227 static
4228 testDesc testDescriptions[] = {
4229 { "XML regression tests" ,
4230 oldParseTest, "./test/*", "result/", "", NULL,
4231 0 },
4232 { "XML regression tests on memory" ,
4233 memParseTest, "./test/*", "result/", "", NULL,
4234 0 },
4235 { "XML entity subst regression tests" ,
4236 noentParseTest, "./test/*", "result/noent/", "", NULL,
4237 XML_PARSE_NOENT },
4238 { "XML Namespaces regression tests",
4239 errParseTest, "./test/namespaces/*", "result/namespaces/", "", ".err",
4240 0 },
4241 { "Error cases regression tests",
4242 errParseTest, "./test/errors/*.xml", "result/errors/", "", ".err",
4243 0 },
4244 { "Error cases regression tests (old 1.0)",
4245 errParseTest, "./test/errors10/*.xml", "result/errors10/", "", ".err",
4246 XML_PARSE_OLD10 },
4247 #ifdef LIBXML_READER_ENABLED
4248 { "Error cases stream regression tests",
4249 streamParseTest, "./test/errors/*.xml", "result/errors/", NULL, ".str",
4250 0 },
4251 { "Reader regression tests",
4252 streamParseTest, "./test/*", "result/", ".rdr", NULL,
4253 0 },
4254 { "Reader entities substitution regression tests",
4255 streamParseTest, "./test/*", "result/", ".rde", NULL,
4256 XML_PARSE_NOENT },
4257 { "Reader on memory regression tests",
4258 streamMemParseTest, "./test/*", "result/", ".rdr", NULL,
4259 0 },
4260 { "Walker regression tests",
4261 walkerParseTest, "./test/*", "result/", ".rdr", NULL,
4262 0 },
4263 #endif
4264 #ifdef LIBXML_SAX1_ENABLED
4265 { "SAX1 callbacks regression tests" ,
4266 saxParseTest, "./test/*", "result/", ".sax", NULL,
4267 XML_PARSE_SAX1 },
4268 #endif
4269 { "SAX2 callbacks regression tests" ,
4270 saxParseTest, "./test/*", "result/", ".sax2", NULL,
4271 0 },
4272 { "SAX2 callbacks regression tests with entity substitution" ,
4273 saxParseTest, "./test/*", "result/noent/", ".sax2", NULL,
4274 XML_PARSE_NOENT },
4275 #ifdef LIBXML_PUSH_ENABLED
4276 { "XML push regression tests" ,
4277 pushParseTest, "./test/*", "result/", "", NULL,
4278 0 },
4279 #endif
4280 #ifdef LIBXML_HTML_ENABLED
4281 { "HTML regression tests" ,
4282 errParseTest, "./test/HTML/*", "result/HTML/", "", ".err",
4283 XML_PARSE_HTML },
4284 #ifdef LIBXML_PUSH_ENABLED
4285 { "Push HTML regression tests" ,
4286 pushParseTest, "./test/HTML/*", "result/HTML/", "", ".err",
4287 XML_PARSE_HTML },
4288 #endif
4289 { "HTML SAX regression tests" ,
4290 saxParseTest, "./test/HTML/*", "result/HTML/", ".sax", NULL,
4291 XML_PARSE_HTML },
4292 #endif
4293 #ifdef LIBXML_VALID_ENABLED
4294 { "Valid documents regression tests" ,
4295 errParseTest, "./test/VCM/*", NULL, NULL, NULL,
4296 XML_PARSE_DTDVALID },
4297 { "Validity checking regression tests" ,
4298 errParseTest, "./test/VC/*", "result/VC/", NULL, "",
4299 XML_PARSE_DTDVALID },
4300 #ifdef LIBXML_READER_ENABLED
4301 { "Streaming validity checking regression tests" ,
4302 streamParseTest, "./test/valid/*.xml", "result/valid/", NULL, ".err.rdr",
4303 XML_PARSE_DTDVALID },
4304 { "Streaming validity error checking regression tests" ,
4305 streamParseTest, "./test/VC/*", "result/VC/", NULL, ".rdr",
4306 XML_PARSE_DTDVALID },
4307 #endif
4308 { "General documents valid regression tests" ,
4309 errParseTest, "./test/valid/*", "result/valid/", "", ".err",
4310 XML_PARSE_DTDVALID },
4311 #endif
4312 #ifdef LIBXML_XINCLUDE_ENABLED
4313 { "XInclude regression tests" ,
4314 errParseTest, "./test/XInclude/docs/*", "result/XInclude/", "", NULL,
4315 /* Ignore errors at this point ".err", */
4316 XML_PARSE_XINCLUDE },
4317 #ifdef LIBXML_READER_ENABLED
4318 { "XInclude xmlReader regression tests",
4319 streamParseTest, "./test/XInclude/docs/*", "result/XInclude/", ".rdr",
4320 /* Ignore errors at this point ".err", */
4321 NULL, XML_PARSE_XINCLUDE },
4322 #endif
4323 { "XInclude regression tests stripping include nodes" ,
4324 errParseTest, "./test/XInclude/docs/*", "result/XInclude/", "", NULL,
4325 /* Ignore errors at this point ".err", */
4326 XML_PARSE_XINCLUDE | XML_PARSE_NOXINCNODE },
4327 #ifdef LIBXML_READER_ENABLED
4328 { "XInclude xmlReader regression tests stripping include nodes",
4329 streamParseTest, "./test/XInclude/docs/*", "result/XInclude/", ".rdr",
4330 /* Ignore errors at this point ".err", */
4331 NULL, XML_PARSE_XINCLUDE | XML_PARSE_NOXINCNODE },
4332 #endif
4333 #endif
4334 #ifdef LIBXML_XPATH_ENABLED
4335 #ifdef LIBXML_DEBUG_ENABLED
4336 { "XPath expressions regression tests" ,
4337 xpathExprTest, "./test/XPath/expr/*", "result/XPath/expr/", "", NULL,
4338 0 },
4339 { "XPath document queries regression tests" ,
4340 xpathDocTest, "./test/XPath/docs/*", NULL, NULL, NULL,
4341 0 },
4342 #ifdef LIBXML_XPTR_ENABLED
4343 { "XPointer document queries regression tests" ,
4344 xptrDocTest, "./test/XPath/docs/*", NULL, NULL, NULL,
4345 0 },
4346 #endif
4347 { "xml:id regression tests" ,
4348 xmlidDocTest, "./test/xmlid/*", "result/xmlid/", "", ".err",
4349 0 },
4350 #endif
4351 #endif
4352 { "URI parsing tests" ,
4353 uriParseTest, "./test/URI/*.uri", "result/URI/", "", NULL,
4354 0 },
4355 { "URI base composition tests" ,
4356 uriBaseTest, "./test/URI/*.data", "result/URI/", "", NULL,
4357 0 },
4358 { "Path URI conversion tests" ,
4359 uriPathTest, NULL, NULL, NULL, NULL,
4360 0 },
4361 #ifdef LIBXML_SCHEMAS_ENABLED
4362 { "Schemas regression tests" ,
4363 schemasTest, "./test/schemas/*_*.xsd", NULL, NULL, NULL,
4364 0 },
4365 { "Relax-NG regression tests" ,
4366 rngTest, "./test/relaxng/*.rng", NULL, NULL, NULL,
4367 XML_PARSE_DTDATTR | XML_PARSE_NOENT },
4368 #ifdef LIBXML_READER_ENABLED
4369 { "Relax-NG streaming regression tests" ,
4370 rngStreamTest, "./test/relaxng/*.rng", NULL, NULL, NULL,
4371 XML_PARSE_DTDATTR | XML_PARSE_NOENT },
4372 #endif
4373 #endif
4374 #ifdef LIBXML_PATTERN_ENABLED
4375 #ifdef LIBXML_READER_ENABLED
4376 { "Pattern regression tests" ,
4377 patternTest, "./test/pattern/*.pat", "result/pattern/", NULL, NULL,
4378 0 },
4379 #endif
4380 #endif
4381 #ifdef LIBXML_C14N_ENABLED
4382 { "C14N with comments regression tests" ,
4383 c14nWithCommentTest, "./test/c14n/with-comments/*.xml", NULL, NULL, NULL,
4384 0 },
4385 { "C14N without comments regression tests" ,
4386 c14nWithoutCommentTest, "./test/c14n/without-comments/*.xml", NULL, NULL, NULL,
4387 0 },
4388 { "C14N exclusive without comments regression tests" ,
4389 c14nExcWithoutCommentTest, "./test/c14n/exc-without-comments/*.xml", NULL, NULL, NULL,
4390 0 },
4391 { "C14N 1.1 without comments regression tests" ,
4392 c14n11WithoutCommentTest, "./test/c14n/1-1-without-comments/*.xml", NULL, NULL, NULL,
4393 0 },
4394 #endif
4395 #if defined(LIBXML_THREAD_ENABLED) && defined(LIBXML_CATALOG_ENABLED)
4396 { "Catalog and Threads regression tests" ,
4397 threadsTest, NULL, NULL, NULL, NULL,
4398 0 },
4399 #endif
4400 {NULL, NULL, NULL, NULL, NULL, NULL, 0}
4401 };
4402
4403 /************************************************************************
4404 * *
4405 * The main code driving the tests *
4406 * *
4407 ************************************************************************/
4408
4409 static int
launchTests(testDescPtr tst)4410 launchTests(testDescPtr tst) {
4411 int res = 0, err = 0;
4412 size_t i;
4413 char *result;
4414 char *error;
4415 int mem;
4416 xmlCharEncodingHandlerPtr ebcdicHandler, eucJpHandler;
4417
4418 ebcdicHandler = xmlGetCharEncodingHandler(XML_CHAR_ENCODING_EBCDIC);
4419 eucJpHandler = xmlGetCharEncodingHandler(XML_CHAR_ENCODING_EUC_JP);
4420
4421 if (tst == NULL) return(-1);
4422 if (tst->in != NULL) {
4423 glob_t globbuf;
4424
4425 globbuf.gl_offs = 0;
4426 glob(tst->in, GLOB_DOOFFS, NULL, &globbuf);
4427 for (i = 0;i < globbuf.gl_pathc;i++) {
4428 if (!checkTestFile(globbuf.gl_pathv[i]))
4429 continue;
4430 if (((ebcdicHandler == NULL) &&
4431 (strstr(globbuf.gl_pathv[i], "ebcdic") != NULL)) ||
4432 ((eucJpHandler == NULL) &&
4433 (strstr(globbuf.gl_pathv[i], "icu_parse_test") != NULL)))
4434 continue;
4435 if (tst->suffix != NULL) {
4436 result = resultFilename(globbuf.gl_pathv[i], tst->out,
4437 tst->suffix);
4438 if (result == NULL) {
4439 fprintf(stderr, "Out of memory !\n");
4440 fatalError();
4441 }
4442 } else {
4443 result = NULL;
4444 }
4445 if (tst->err != NULL) {
4446 error = resultFilename(globbuf.gl_pathv[i], tst->out,
4447 tst->err);
4448 if (error == NULL) {
4449 fprintf(stderr, "Out of memory !\n");
4450 fatalError();
4451 }
4452 } else {
4453 error = NULL;
4454 }
4455 if ((result) &&(!checkTestFile(result)) && !update_results) {
4456 fprintf(stderr, "Missing result file %s\n", result);
4457 } else if ((error) &&(!checkTestFile(error)) && !update_results) {
4458 fprintf(stderr, "Missing error file %s\n", error);
4459 } else {
4460 mem = xmlMemUsed();
4461 extraMemoryFromResolver = 0;
4462 testErrorsSize = 0;
4463 testErrors[0] = 0;
4464 res = tst->func(globbuf.gl_pathv[i], result, error,
4465 tst->options | XML_PARSE_COMPACT);
4466 xmlResetLastError();
4467 if (res != 0) {
4468 fprintf(stderr, "File %s generated an error\n",
4469 globbuf.gl_pathv[i]);
4470 nb_errors++;
4471 err++;
4472 }
4473 else if (xmlMemUsed() != mem) {
4474 if ((xmlMemUsed() != mem) &&
4475 (extraMemoryFromResolver == 0)) {
4476 fprintf(stderr, "File %s leaked %d bytes\n",
4477 globbuf.gl_pathv[i], xmlMemUsed() - mem);
4478 nb_leaks++;
4479 err++;
4480 }
4481 }
4482 testErrorsSize = 0;
4483 }
4484 if (result)
4485 free(result);
4486 if (error)
4487 free(error);
4488 }
4489 globfree(&globbuf);
4490 } else {
4491 testErrorsSize = 0;
4492 testErrors[0] = 0;
4493 extraMemoryFromResolver = 0;
4494 res = tst->func(NULL, NULL, NULL, tst->options);
4495 if (res != 0) {
4496 nb_errors++;
4497 err++;
4498 }
4499 }
4500
4501 xmlCharEncCloseFunc(ebcdicHandler);
4502 xmlCharEncCloseFunc(eucJpHandler);
4503
4504 return(err);
4505 }
4506
4507 static int verbose = 0;
4508 static int tests_quiet = 0;
4509
4510 static int
runtest(int i)4511 runtest(int i) {
4512 int ret = 0, res;
4513 int old_errors, old_tests, old_leaks;
4514
4515 old_errors = nb_errors;
4516 old_tests = nb_tests;
4517 old_leaks = nb_leaks;
4518 if ((tests_quiet == 0) && (testDescriptions[i].desc != NULL))
4519 printf("## %s\n", testDescriptions[i].desc);
4520 res = launchTests(&testDescriptions[i]);
4521 if (res != 0)
4522 ret++;
4523 if (verbose) {
4524 if ((nb_errors == old_errors) && (nb_leaks == old_leaks))
4525 printf("Ran %d tests, no errors\n", nb_tests - old_tests);
4526 else
4527 printf("Ran %d tests, %d errors, %d leaks\n",
4528 nb_tests - old_tests,
4529 nb_errors - old_errors,
4530 nb_leaks - old_leaks);
4531 }
4532 return(ret);
4533 }
4534
4535 int
main(int argc ATTRIBUTE_UNUSED,char ** argv ATTRIBUTE_UNUSED)4536 main(int argc ATTRIBUTE_UNUSED, char **argv ATTRIBUTE_UNUSED) {
4537 int i, a, ret = 0;
4538 int subset = 0;
4539
4540 #if defined(_WIN32) && !defined(__CYGWIN__)
4541 setvbuf(stdout, NULL, _IONBF, 0);
4542 setvbuf(stderr, NULL, _IONBF, 0);
4543 #endif
4544
4545 initializeLibxml2();
4546
4547 for (a = 1; a < argc;a++) {
4548 if (!strcmp(argv[a], "-v"))
4549 verbose = 1;
4550 else if (!strcmp(argv[a], "-u"))
4551 update_results = 1;
4552 else if (!strcmp(argv[a], "-quiet"))
4553 tests_quiet = 1;
4554 else {
4555 for (i = 0; testDescriptions[i].func != NULL; i++) {
4556 if (strstr(testDescriptions[i].desc, argv[a])) {
4557 ret += runtest(i);
4558 subset++;
4559 }
4560 }
4561 }
4562 }
4563 if (subset == 0) {
4564 for (i = 0; testDescriptions[i].func != NULL; i++) {
4565 ret += runtest(i);
4566 }
4567 }
4568 if ((nb_errors == 0) && (nb_leaks == 0)) {
4569 ret = 0;
4570 printf("Total %d tests, no errors\n",
4571 nb_tests);
4572 } else {
4573 ret = 1;
4574 printf("Total %d tests, %d errors, %d leaks\n",
4575 nb_tests, nb_errors, nb_leaks);
4576 }
4577 xmlCleanupParser();
4578 xmlMemoryDump();
4579
4580 return(ret);
4581 }
4582
4583 #else /* ! LIBXML_OUTPUT_ENABLED */
4584 int
main(int argc ATTRIBUTE_UNUSED,char ** argv ATTRIBUTE_UNUSED)4585 main(int argc ATTRIBUTE_UNUSED, char **argv ATTRIBUTE_UNUSED) {
4586 fprintf(stderr, "runtest requires output to be enabled in libxml2\n");
4587 return(1);
4588 }
4589 #endif
4590