Annotation of embedaddon/libxml2/catalog.c, revision 1.1.1.1
1.1 misho 1: /**
2: * catalog.c: set of generic Catalog related routines
3: *
4: * Reference: SGML Open Technical Resolution TR9401:1997.
5: * http://www.jclark.com/sp/catalog.htm
6: *
7: * XML Catalogs Working Draft 06 August 2001
8: * http://www.oasis-open.org/committees/entity/spec-2001-08-06.html
9: *
10: * See Copyright for the status of this software.
11: *
12: * Daniel.Veillard@imag.fr
13: */
14:
15: #define IN_LIBXML
16: #include "libxml.h"
17:
18: #ifdef LIBXML_CATALOG_ENABLED
19: #ifdef HAVE_SYS_TYPES_H
20: #include <sys/types.h>
21: #endif
22: #ifdef HAVE_SYS_STAT_H
23: #include <sys/stat.h>
24: #endif
25: #ifdef HAVE_UNISTD_H
26: #include <unistd.h>
27: #endif
28: #ifdef HAVE_FCNTL_H
29: #include <fcntl.h>
30: #endif
31: #ifdef HAVE_STDLIB_H
32: #include <stdlib.h>
33: #endif
34: #include <string.h>
35: #include <libxml/xmlmemory.h>
36: #include <libxml/hash.h>
37: #include <libxml/uri.h>
38: #include <libxml/parserInternals.h>
39: #include <libxml/catalog.h>
40: #include <libxml/xmlerror.h>
41: #include <libxml/threads.h>
42: #include <libxml/globals.h>
43:
44: #define MAX_DELEGATE 50
45: #define MAX_CATAL_DEPTH 50
46:
47: #ifdef _WIN32
48: # define PATH_SEAPARATOR ';'
49: #else
50: # define PATH_SEAPARATOR ':'
51: #endif
52:
53: /**
54: * TODO:
55: *
56: * macro to flag unimplemented blocks
57: * XML_CATALOG_PREFER user env to select between system/public prefered
58: * option. C.f. Richard Tobin <richard@cogsci.ed.ac.uk>
59: *> Just FYI, I am using an environment variable XML_CATALOG_PREFER with
60: *> values "system" and "public". I have made the default be "system" to
61: *> match yours.
62: */
63: #define TODO \
64: xmlGenericError(xmlGenericErrorContext, \
65: "Unimplemented block at %s:%d\n", \
66: __FILE__, __LINE__);
67:
68: #define XML_URN_PUBID "urn:publicid:"
69: #define XML_CATAL_BREAK ((xmlChar *) -1)
70: #ifndef XML_XML_DEFAULT_CATALOG
71: #define XML_XML_DEFAULT_CATALOG "file:///etc/xml/catalog"
72: #endif
73: #ifndef XML_SGML_DEFAULT_CATALOG
74: #define XML_SGML_DEFAULT_CATALOG "file:///etc/sgml/catalog"
75: #endif
76:
77: #if defined(_WIN32) && defined(_MSC_VER)
78: #undef XML_XML_DEFAULT_CATALOG
79: static char XML_XML_DEFAULT_CATALOG[256] = "file:///etc/xml/catalog";
80: #if defined(_WIN32_WCE)
81: /* Windows CE don't have a A variant */
82: #define GetModuleHandleA GetModuleHandle
83: #define GetModuleFileNameA GetModuleFileName
84: #else
85: void* __stdcall GetModuleHandleA(const char*);
86: unsigned long __stdcall GetModuleFileNameA(void*, char*, unsigned long);
87: #endif
88: #endif
89:
90: static xmlChar *xmlCatalogNormalizePublic(const xmlChar *pubID);
91: static int xmlExpandCatalog(xmlCatalogPtr catal, const char *filename);
92:
93: /************************************************************************
94: * *
95: * Types, all private *
96: * *
97: ************************************************************************/
98:
99: typedef enum {
100: XML_CATA_REMOVED = -1,
101: XML_CATA_NONE = 0,
102: XML_CATA_CATALOG,
103: XML_CATA_BROKEN_CATALOG,
104: XML_CATA_NEXT_CATALOG,
105: XML_CATA_GROUP,
106: XML_CATA_PUBLIC,
107: XML_CATA_SYSTEM,
108: XML_CATA_REWRITE_SYSTEM,
109: XML_CATA_DELEGATE_PUBLIC,
110: XML_CATA_DELEGATE_SYSTEM,
111: XML_CATA_URI,
112: XML_CATA_REWRITE_URI,
113: XML_CATA_DELEGATE_URI,
114: SGML_CATA_SYSTEM,
115: SGML_CATA_PUBLIC,
116: SGML_CATA_ENTITY,
117: SGML_CATA_PENTITY,
118: SGML_CATA_DOCTYPE,
119: SGML_CATA_LINKTYPE,
120: SGML_CATA_NOTATION,
121: SGML_CATA_DELEGATE,
122: SGML_CATA_BASE,
123: SGML_CATA_CATALOG,
124: SGML_CATA_DOCUMENT,
125: SGML_CATA_SGMLDECL
126: } xmlCatalogEntryType;
127:
128: typedef struct _xmlCatalogEntry xmlCatalogEntry;
129: typedef xmlCatalogEntry *xmlCatalogEntryPtr;
130: struct _xmlCatalogEntry {
131: struct _xmlCatalogEntry *next;
132: struct _xmlCatalogEntry *parent;
133: struct _xmlCatalogEntry *children;
134: xmlCatalogEntryType type;
135: xmlChar *name;
136: xmlChar *value;
137: xmlChar *URL; /* The expanded URL using the base */
138: xmlCatalogPrefer prefer;
139: int dealloc;
140: int depth;
141: struct _xmlCatalogEntry *group;
142: };
143:
144: typedef enum {
145: XML_XML_CATALOG_TYPE = 1,
146: XML_SGML_CATALOG_TYPE
147: } xmlCatalogType;
148:
149: #define XML_MAX_SGML_CATA_DEPTH 10
150: struct _xmlCatalog {
151: xmlCatalogType type; /* either XML or SGML */
152:
153: /*
154: * SGML Catalogs are stored as a simple hash table of catalog entries
155: * Catalog stack to check against overflows when building the
156: * SGML catalog
157: */
158: char *catalTab[XML_MAX_SGML_CATA_DEPTH]; /* stack of catals */
159: int catalNr; /* Number of current catal streams */
160: int catalMax; /* Max number of catal streams */
161: xmlHashTablePtr sgml;
162:
163: /*
164: * XML Catalogs are stored as a tree of Catalog entries
165: */
166: xmlCatalogPrefer prefer;
167: xmlCatalogEntryPtr xml;
168: };
169:
170: /************************************************************************
171: * *
172: * Global variables *
173: * *
174: ************************************************************************/
175:
176: /*
177: * Those are preferences
178: */
179: static int xmlDebugCatalogs = 0; /* used for debugging */
180: static xmlCatalogAllow xmlCatalogDefaultAllow = XML_CATA_ALLOW_ALL;
181: static xmlCatalogPrefer xmlCatalogDefaultPrefer = XML_CATA_PREFER_PUBLIC;
182:
183: /*
184: * Hash table containing all the trees of XML catalogs parsed by
185: * the application.
186: */
187: static xmlHashTablePtr xmlCatalogXMLFiles = NULL;
188:
189: /*
190: * The default catalog in use by the application
191: */
192: static xmlCatalogPtr xmlDefaultCatalog = NULL;
193:
194: /*
195: * A mutex for modifying the shared global catalog(s)
196: * xmlDefaultCatalog tree.
197: * It also protects xmlCatalogXMLFiles
198: * The core of this readers/writer scheme is in xmlFetchXMLCatalogFile()
199: */
200: static xmlRMutexPtr xmlCatalogMutex = NULL;
201:
202: /*
203: * Whether the catalog support was initialized.
204: */
205: static int xmlCatalogInitialized = 0;
206:
207: /************************************************************************
208: * *
209: * Catalog error handlers *
210: * *
211: ************************************************************************/
212:
213: /**
214: * xmlCatalogErrMemory:
215: * @extra: extra informations
216: *
217: * Handle an out of memory condition
218: */
219: static void
220: xmlCatalogErrMemory(const char *extra)
221: {
222: __xmlRaiseError(NULL, NULL, NULL, NULL, NULL, XML_FROM_CATALOG,
223: XML_ERR_NO_MEMORY, XML_ERR_ERROR, NULL, 0,
224: extra, NULL, NULL, 0, 0,
225: "Memory allocation failed : %s\n", extra);
226: }
227:
228: /**
229: * xmlCatalogErr:
230: * @catal: the Catalog entry
231: * @node: the context node
232: * @msg: the error message
233: * @extra: extra informations
234: *
235: * Handle a catalog error
236: */
237: static void
238: xmlCatalogErr(xmlCatalogEntryPtr catal, xmlNodePtr node, int error,
239: const char *msg, const xmlChar *str1, const xmlChar *str2,
240: const xmlChar *str3)
241: {
242: __xmlRaiseError(NULL, NULL, NULL, catal, node, XML_FROM_CATALOG,
243: error, XML_ERR_ERROR, NULL, 0,
244: (const char *) str1, (const char *) str2,
245: (const char *) str3, 0, 0,
246: msg, str1, str2, str3);
247: }
248:
249:
250: /************************************************************************
251: * *
252: * Allocation and Freeing *
253: * *
254: ************************************************************************/
255:
256: /**
257: * xmlNewCatalogEntry:
258: * @type: type of entry
259: * @name: name of the entry
260: * @value: value of the entry
261: * @prefer: the PUBLIC vs. SYSTEM current preference value
262: * @group: for members of a group, the group entry
263: *
264: * create a new Catalog entry, this type is shared both by XML and
265: * SGML catalogs, but the acceptable types values differs.
266: *
267: * Returns the xmlCatalogEntryPtr or NULL in case of error
268: */
269: static xmlCatalogEntryPtr
270: xmlNewCatalogEntry(xmlCatalogEntryType type, const xmlChar *name,
271: const xmlChar *value, const xmlChar *URL, xmlCatalogPrefer prefer,
272: xmlCatalogEntryPtr group) {
273: xmlCatalogEntryPtr ret;
274: xmlChar *normid = NULL;
275:
276: ret = (xmlCatalogEntryPtr) xmlMalloc(sizeof(xmlCatalogEntry));
277: if (ret == NULL) {
278: xmlCatalogErrMemory("allocating catalog entry");
279: return(NULL);
280: }
281: ret->next = NULL;
282: ret->parent = NULL;
283: ret->children = NULL;
284: ret->type = type;
285: if (type == XML_CATA_PUBLIC || type == XML_CATA_DELEGATE_PUBLIC) {
286: normid = xmlCatalogNormalizePublic(name);
287: if (normid != NULL)
288: name = (*normid != 0 ? normid : NULL);
289: }
290: if (name != NULL)
291: ret->name = xmlStrdup(name);
292: else
293: ret->name = NULL;
294: if (normid != NULL)
295: xmlFree(normid);
296: if (value != NULL)
297: ret->value = xmlStrdup(value);
298: else
299: ret->value = NULL;
300: if (URL == NULL)
301: URL = value;
302: if (URL != NULL)
303: ret->URL = xmlStrdup(URL);
304: else
305: ret->URL = NULL;
306: ret->prefer = prefer;
307: ret->dealloc = 0;
308: ret->depth = 0;
309: ret->group = group;
310: return(ret);
311: }
312:
313: static void
314: xmlFreeCatalogEntryList(xmlCatalogEntryPtr ret);
315:
316: /**
317: * xmlFreeCatalogEntry:
318: * @ret: a Catalog entry
319: *
320: * Free the memory allocated to a Catalog entry
321: */
322: static void
323: xmlFreeCatalogEntry(xmlCatalogEntryPtr ret) {
324: if (ret == NULL)
325: return;
326: /*
327: * Entries stored in the file hash must be deallocated
328: * only by the file hash cleaner !
329: */
330: if (ret->dealloc == 1)
331: return;
332:
333: if (xmlDebugCatalogs) {
334: if (ret->name != NULL)
335: xmlGenericError(xmlGenericErrorContext,
336: "Free catalog entry %s\n", ret->name);
337: else if (ret->value != NULL)
338: xmlGenericError(xmlGenericErrorContext,
339: "Free catalog entry %s\n", ret->value);
340: else
341: xmlGenericError(xmlGenericErrorContext,
342: "Free catalog entry\n");
343: }
344:
345: if (ret->name != NULL)
346: xmlFree(ret->name);
347: if (ret->value != NULL)
348: xmlFree(ret->value);
349: if (ret->URL != NULL)
350: xmlFree(ret->URL);
351: xmlFree(ret);
352: }
353:
354: /**
355: * xmlFreeCatalogEntryList:
356: * @ret: a Catalog entry list
357: *
358: * Free the memory allocated to a full chained list of Catalog entries
359: */
360: static void
361: xmlFreeCatalogEntryList(xmlCatalogEntryPtr ret) {
362: xmlCatalogEntryPtr next;
363:
364: while (ret != NULL) {
365: next = ret->next;
366: xmlFreeCatalogEntry(ret);
367: ret = next;
368: }
369: }
370:
371: /**
372: * xmlFreeCatalogHashEntryList:
373: * @ret: a Catalog entry list
374: *
375: * Free the memory allocated to list of Catalog entries from the
376: * catalog file hash.
377: */
378: static void
379: xmlFreeCatalogHashEntryList(xmlCatalogEntryPtr catal) {
380: xmlCatalogEntryPtr children, next;
381:
382: if (catal == NULL)
383: return;
384:
385: children = catal->children;
386: while (children != NULL) {
387: next = children->next;
388: children->dealloc = 0;
389: children->children = NULL;
390: xmlFreeCatalogEntry(children);
391: children = next;
392: }
393: catal->dealloc = 0;
394: xmlFreeCatalogEntry(catal);
395: }
396:
397: /**
398: * xmlCreateNewCatalog:
399: * @type: type of catalog
400: * @prefer: the PUBLIC vs. SYSTEM current preference value
401: *
402: * create a new Catalog, this type is shared both by XML and
403: * SGML catalogs, but the acceptable types values differs.
404: *
405: * Returns the xmlCatalogPtr or NULL in case of error
406: */
407: static xmlCatalogPtr
408: xmlCreateNewCatalog(xmlCatalogType type, xmlCatalogPrefer prefer) {
409: xmlCatalogPtr ret;
410:
411: ret = (xmlCatalogPtr) xmlMalloc(sizeof(xmlCatalog));
412: if (ret == NULL) {
413: xmlCatalogErrMemory("allocating catalog");
414: return(NULL);
415: }
416: memset(ret, 0, sizeof(xmlCatalog));
417: ret->type = type;
418: ret->catalNr = 0;
419: ret->catalMax = XML_MAX_SGML_CATA_DEPTH;
420: ret->prefer = prefer;
421: if (ret->type == XML_SGML_CATALOG_TYPE)
422: ret->sgml = xmlHashCreate(10);
423: return(ret);
424: }
425:
426: /**
427: * xmlFreeCatalog:
428: * @catal: a Catalog
429: *
430: * Free the memory allocated to a Catalog
431: */
432: void
433: xmlFreeCatalog(xmlCatalogPtr catal) {
434: if (catal == NULL)
435: return;
436: if (catal->xml != NULL)
437: xmlFreeCatalogEntryList(catal->xml);
438: if (catal->sgml != NULL)
439: xmlHashFree(catal->sgml,
440: (xmlHashDeallocator) xmlFreeCatalogEntry);
441: xmlFree(catal);
442: }
443:
444: /************************************************************************
445: * *
446: * Serializing Catalogs *
447: * *
448: ************************************************************************/
449:
450: #ifdef LIBXML_OUTPUT_ENABLED
451: /**
452: * xmlCatalogDumpEntry:
453: * @entry: the catalog entry
454: * @out: the file.
455: *
456: * Serialize an SGML Catalog entry
457: */
458: static void
459: xmlCatalogDumpEntry(xmlCatalogEntryPtr entry, FILE *out) {
460: if ((entry == NULL) || (out == NULL))
461: return;
462: switch (entry->type) {
463: case SGML_CATA_ENTITY:
464: fprintf(out, "ENTITY "); break;
465: case SGML_CATA_PENTITY:
466: fprintf(out, "ENTITY %%"); break;
467: case SGML_CATA_DOCTYPE:
468: fprintf(out, "DOCTYPE "); break;
469: case SGML_CATA_LINKTYPE:
470: fprintf(out, "LINKTYPE "); break;
471: case SGML_CATA_NOTATION:
472: fprintf(out, "NOTATION "); break;
473: case SGML_CATA_PUBLIC:
474: fprintf(out, "PUBLIC "); break;
475: case SGML_CATA_SYSTEM:
476: fprintf(out, "SYSTEM "); break;
477: case SGML_CATA_DELEGATE:
478: fprintf(out, "DELEGATE "); break;
479: case SGML_CATA_BASE:
480: fprintf(out, "BASE "); break;
481: case SGML_CATA_CATALOG:
482: fprintf(out, "CATALOG "); break;
483: case SGML_CATA_DOCUMENT:
484: fprintf(out, "DOCUMENT "); break;
485: case SGML_CATA_SGMLDECL:
486: fprintf(out, "SGMLDECL "); break;
487: default:
488: return;
489: }
490: switch (entry->type) {
491: case SGML_CATA_ENTITY:
492: case SGML_CATA_PENTITY:
493: case SGML_CATA_DOCTYPE:
494: case SGML_CATA_LINKTYPE:
495: case SGML_CATA_NOTATION:
496: fprintf(out, "%s", (const char *) entry->name); break;
497: case SGML_CATA_PUBLIC:
498: case SGML_CATA_SYSTEM:
499: case SGML_CATA_SGMLDECL:
500: case SGML_CATA_DOCUMENT:
501: case SGML_CATA_CATALOG:
502: case SGML_CATA_BASE:
503: case SGML_CATA_DELEGATE:
504: fprintf(out, "\"%s\"", entry->name); break;
505: default:
506: break;
507: }
508: switch (entry->type) {
509: case SGML_CATA_ENTITY:
510: case SGML_CATA_PENTITY:
511: case SGML_CATA_DOCTYPE:
512: case SGML_CATA_LINKTYPE:
513: case SGML_CATA_NOTATION:
514: case SGML_CATA_PUBLIC:
515: case SGML_CATA_SYSTEM:
516: case SGML_CATA_DELEGATE:
517: fprintf(out, " \"%s\"", entry->value); break;
518: default:
519: break;
520: }
521: fprintf(out, "\n");
522: }
523:
524: /**
525: * xmlDumpXMLCatalogNode:
526: * @catal: top catalog entry
527: * @catalog: pointer to the xml tree
528: * @doc: the containing document
529: * @ns: the current namespace
530: * @cgroup: group node for group members
531: *
532: * Serializes a Catalog entry, called by xmlDumpXMLCatalog and recursively
533: * for group entries
534: */
535: static void xmlDumpXMLCatalogNode(xmlCatalogEntryPtr catal, xmlNodePtr catalog,
536: xmlDocPtr doc, xmlNsPtr ns, xmlCatalogEntryPtr cgroup) {
537: xmlNodePtr node;
538: xmlCatalogEntryPtr cur;
539: /*
540: * add all the catalog entries
541: */
542: cur = catal;
543: while (cur != NULL) {
544: if (cur->group == cgroup) {
545: switch (cur->type) {
546: case XML_CATA_REMOVED:
547: break;
548: case XML_CATA_BROKEN_CATALOG:
549: case XML_CATA_CATALOG:
550: if (cur == catal) {
551: cur = cur->children;
552: continue;
553: }
554: break;
555: case XML_CATA_NEXT_CATALOG:
556: node = xmlNewDocNode(doc, ns, BAD_CAST "nextCatalog", NULL);
557: xmlSetProp(node, BAD_CAST "catalog", cur->value);
558: xmlAddChild(catalog, node);
559: break;
560: case XML_CATA_NONE:
561: break;
562: case XML_CATA_GROUP:
563: node = xmlNewDocNode(doc, ns, BAD_CAST "group", NULL);
564: xmlSetProp(node, BAD_CAST "id", cur->name);
565: if (cur->value != NULL) {
566: xmlNsPtr xns;
567: xns = xmlSearchNsByHref(doc, node, XML_XML_NAMESPACE);
568: if (xns != NULL)
569: xmlSetNsProp(node, xns, BAD_CAST "base",
570: cur->value);
571: }
572: switch (cur->prefer) {
573: case XML_CATA_PREFER_NONE:
574: break;
575: case XML_CATA_PREFER_PUBLIC:
576: xmlSetProp(node, BAD_CAST "prefer", BAD_CAST "public");
577: break;
578: case XML_CATA_PREFER_SYSTEM:
579: xmlSetProp(node, BAD_CAST "prefer", BAD_CAST "system");
580: break;
581: }
582: xmlDumpXMLCatalogNode(cur->next, node, doc, ns, cur);
583: xmlAddChild(catalog, node);
584: break;
585: case XML_CATA_PUBLIC:
586: node = xmlNewDocNode(doc, ns, BAD_CAST "public", NULL);
587: xmlSetProp(node, BAD_CAST "publicId", cur->name);
588: xmlSetProp(node, BAD_CAST "uri", cur->value);
589: xmlAddChild(catalog, node);
590: break;
591: case XML_CATA_SYSTEM:
592: node = xmlNewDocNode(doc, ns, BAD_CAST "system", NULL);
593: xmlSetProp(node, BAD_CAST "systemId", cur->name);
594: xmlSetProp(node, BAD_CAST "uri", cur->value);
595: xmlAddChild(catalog, node);
596: break;
597: case XML_CATA_REWRITE_SYSTEM:
598: node = xmlNewDocNode(doc, ns, BAD_CAST "rewriteSystem", NULL);
599: xmlSetProp(node, BAD_CAST "systemIdStartString", cur->name);
600: xmlSetProp(node, BAD_CAST "rewritePrefix", cur->value);
601: xmlAddChild(catalog, node);
602: break;
603: case XML_CATA_DELEGATE_PUBLIC:
604: node = xmlNewDocNode(doc, ns, BAD_CAST "delegatePublic", NULL);
605: xmlSetProp(node, BAD_CAST "publicIdStartString", cur->name);
606: xmlSetProp(node, BAD_CAST "catalog", cur->value);
607: xmlAddChild(catalog, node);
608: break;
609: case XML_CATA_DELEGATE_SYSTEM:
610: node = xmlNewDocNode(doc, ns, BAD_CAST "delegateSystem", NULL);
611: xmlSetProp(node, BAD_CAST "systemIdStartString", cur->name);
612: xmlSetProp(node, BAD_CAST "catalog", cur->value);
613: xmlAddChild(catalog, node);
614: break;
615: case XML_CATA_URI:
616: node = xmlNewDocNode(doc, ns, BAD_CAST "uri", NULL);
617: xmlSetProp(node, BAD_CAST "name", cur->name);
618: xmlSetProp(node, BAD_CAST "uri", cur->value);
619: xmlAddChild(catalog, node);
620: break;
621: case XML_CATA_REWRITE_URI:
622: node = xmlNewDocNode(doc, ns, BAD_CAST "rewriteURI", NULL);
623: xmlSetProp(node, BAD_CAST "uriStartString", cur->name);
624: xmlSetProp(node, BAD_CAST "rewritePrefix", cur->value);
625: xmlAddChild(catalog, node);
626: break;
627: case XML_CATA_DELEGATE_URI:
628: node = xmlNewDocNode(doc, ns, BAD_CAST "delegateURI", NULL);
629: xmlSetProp(node, BAD_CAST "uriStartString", cur->name);
630: xmlSetProp(node, BAD_CAST "catalog", cur->value);
631: xmlAddChild(catalog, node);
632: break;
633: case SGML_CATA_SYSTEM:
634: case SGML_CATA_PUBLIC:
635: case SGML_CATA_ENTITY:
636: case SGML_CATA_PENTITY:
637: case SGML_CATA_DOCTYPE:
638: case SGML_CATA_LINKTYPE:
639: case SGML_CATA_NOTATION:
640: case SGML_CATA_DELEGATE:
641: case SGML_CATA_BASE:
642: case SGML_CATA_CATALOG:
643: case SGML_CATA_DOCUMENT:
644: case SGML_CATA_SGMLDECL:
645: break;
646: }
647: }
648: cur = cur->next;
649: }
650: }
651:
652: static int
653: xmlDumpXMLCatalog(FILE *out, xmlCatalogEntryPtr catal) {
654: int ret;
655: xmlDocPtr doc;
656: xmlNsPtr ns;
657: xmlDtdPtr dtd;
658: xmlNodePtr catalog;
659: xmlOutputBufferPtr buf;
660:
661: /*
662: * Rebuild a catalog
663: */
664: doc = xmlNewDoc(NULL);
665: if (doc == NULL)
666: return(-1);
667: dtd = xmlNewDtd(doc, BAD_CAST "catalog",
668: BAD_CAST "-//OASIS//DTD Entity Resolution XML Catalog V1.0//EN",
669: BAD_CAST "http://www.oasis-open.org/committees/entity/release/1.0/catalog.dtd");
670:
671: xmlAddChild((xmlNodePtr) doc, (xmlNodePtr) dtd);
672:
673: ns = xmlNewNs(NULL, XML_CATALOGS_NAMESPACE, NULL);
674: if (ns == NULL) {
675: xmlFreeDoc(doc);
676: return(-1);
677: }
678: catalog = xmlNewDocNode(doc, ns, BAD_CAST "catalog", NULL);
679: if (catalog == NULL) {
680: xmlFreeNs(ns);
681: xmlFreeDoc(doc);
682: return(-1);
683: }
684: catalog->nsDef = ns;
685: xmlAddChild((xmlNodePtr) doc, catalog);
686:
687: xmlDumpXMLCatalogNode(catal, catalog, doc, ns, NULL);
688:
689: /*
690: * reserialize it
691: */
692: buf = xmlOutputBufferCreateFile(out, NULL);
693: if (buf == NULL) {
694: xmlFreeDoc(doc);
695: return(-1);
696: }
697: ret = xmlSaveFormatFileTo(buf, doc, NULL, 1);
698:
699: /*
700: * Free it
701: */
702: xmlFreeDoc(doc);
703:
704: return(ret);
705: }
706: #endif /* LIBXML_OUTPUT_ENABLED */
707:
708: /************************************************************************
709: * *
710: * Converting SGML Catalogs to XML *
711: * *
712: ************************************************************************/
713:
714: /**
715: * xmlCatalogConvertEntry:
716: * @entry: the entry
717: * @catal: pointer to the catalog being converted
718: *
719: * Convert one entry from the catalog
720: */
721: static void
722: xmlCatalogConvertEntry(xmlCatalogEntryPtr entry, xmlCatalogPtr catal) {
723: if ((entry == NULL) || (catal == NULL) || (catal->sgml == NULL) ||
724: (catal->xml == NULL))
725: return;
726: switch (entry->type) {
727: case SGML_CATA_ENTITY:
728: entry->type = XML_CATA_PUBLIC;
729: break;
730: case SGML_CATA_PENTITY:
731: entry->type = XML_CATA_PUBLIC;
732: break;
733: case SGML_CATA_DOCTYPE:
734: entry->type = XML_CATA_PUBLIC;
735: break;
736: case SGML_CATA_LINKTYPE:
737: entry->type = XML_CATA_PUBLIC;
738: break;
739: case SGML_CATA_NOTATION:
740: entry->type = XML_CATA_PUBLIC;
741: break;
742: case SGML_CATA_PUBLIC:
743: entry->type = XML_CATA_PUBLIC;
744: break;
745: case SGML_CATA_SYSTEM:
746: entry->type = XML_CATA_SYSTEM;
747: break;
748: case SGML_CATA_DELEGATE:
749: entry->type = XML_CATA_DELEGATE_PUBLIC;
750: break;
751: case SGML_CATA_CATALOG:
752: entry->type = XML_CATA_CATALOG;
753: break;
754: default:
755: xmlHashRemoveEntry(catal->sgml, entry->name,
756: (xmlHashDeallocator) xmlFreeCatalogEntry);
757: return;
758: }
759: /*
760: * Conversion successful, remove from the SGML catalog
761: * and add it to the default XML one
762: */
763: xmlHashRemoveEntry(catal->sgml, entry->name, NULL);
764: entry->parent = catal->xml;
765: entry->next = NULL;
766: if (catal->xml->children == NULL)
767: catal->xml->children = entry;
768: else {
769: xmlCatalogEntryPtr prev;
770:
771: prev = catal->xml->children;
772: while (prev->next != NULL)
773: prev = prev->next;
774: prev->next = entry;
775: }
776: }
777:
778: /**
779: * xmlConvertSGMLCatalog:
780: * @catal: the catalog
781: *
782: * Convert all the SGML catalog entries as XML ones
783: *
784: * Returns the number of entries converted if successful, -1 otherwise
785: */
786: int
787: xmlConvertSGMLCatalog(xmlCatalogPtr catal) {
788:
789: if ((catal == NULL) || (catal->type != XML_SGML_CATALOG_TYPE))
790: return(-1);
791:
792: if (xmlDebugCatalogs) {
793: xmlGenericError(xmlGenericErrorContext,
794: "Converting SGML catalog to XML\n");
795: }
796: xmlHashScan(catal->sgml,
797: (xmlHashScanner) xmlCatalogConvertEntry,
798: &catal);
799: return(0);
800: }
801:
802: /************************************************************************
803: * *
804: * Helper function *
805: * *
806: ************************************************************************/
807:
808: /**
809: * xmlCatalogUnWrapURN:
810: * @urn: an "urn:publicid:" to unwrap
811: *
812: * Expand the URN into the equivalent Public Identifier
813: *
814: * Returns the new identifier or NULL, the string must be deallocated
815: * by the caller.
816: */
817: static xmlChar *
818: xmlCatalogUnWrapURN(const xmlChar *urn) {
819: xmlChar result[2000];
820: unsigned int i = 0;
821:
822: if (xmlStrncmp(urn, BAD_CAST XML_URN_PUBID, sizeof(XML_URN_PUBID) - 1))
823: return(NULL);
824: urn += sizeof(XML_URN_PUBID) - 1;
825:
826: while (*urn != 0) {
827: if (i > sizeof(result) - 4)
828: break;
829: if (*urn == '+') {
830: result[i++] = ' ';
831: urn++;
832: } else if (*urn == ':') {
833: result[i++] = '/';
834: result[i++] = '/';
835: urn++;
836: } else if (*urn == ';') {
837: result[i++] = ':';
838: result[i++] = ':';
839: urn++;
840: } else if (*urn == '%') {
841: if ((urn[1] == '2') && (urn[2] == 'B'))
842: result[i++] = '+';
843: else if ((urn[1] == '3') && (urn[2] == 'A'))
844: result[i++] = ':';
845: else if ((urn[1] == '2') && (urn[2] == 'F'))
846: result[i++] = '/';
847: else if ((urn[1] == '3') && (urn[2] == 'B'))
848: result[i++] = ';';
849: else if ((urn[1] == '2') && (urn[2] == '7'))
850: result[i++] = '\'';
851: else if ((urn[1] == '3') && (urn[2] == 'F'))
852: result[i++] = '?';
853: else if ((urn[1] == '2') && (urn[2] == '3'))
854: result[i++] = '#';
855: else if ((urn[1] == '2') && (urn[2] == '5'))
856: result[i++] = '%';
857: else {
858: result[i++] = *urn;
859: urn++;
860: continue;
861: }
862: urn += 3;
863: } else {
864: result[i++] = *urn;
865: urn++;
866: }
867: }
868: result[i] = 0;
869:
870: return(xmlStrdup(result));
871: }
872:
873: /**
874: * xmlParseCatalogFile:
875: * @filename: the filename
876: *
877: * parse an XML file and build a tree. It's like xmlParseFile()
878: * except it bypass all catalog lookups.
879: *
880: * Returns the resulting document tree or NULL in case of error
881: */
882:
883: xmlDocPtr
884: xmlParseCatalogFile(const char *filename) {
885: xmlDocPtr ret;
886: xmlParserCtxtPtr ctxt;
887: char *directory = NULL;
888: xmlParserInputPtr inputStream;
889: xmlParserInputBufferPtr buf;
890:
891: ctxt = xmlNewParserCtxt();
892: if (ctxt == NULL) {
893: #ifdef LIBXML_SAX1_ENABLED
894: if (xmlDefaultSAXHandler.error != NULL) {
895: xmlDefaultSAXHandler.error(NULL, "out of memory\n");
896: }
897: #endif
898: return(NULL);
899: }
900:
901: buf = xmlParserInputBufferCreateFilename(filename, XML_CHAR_ENCODING_NONE);
902: if (buf == NULL) {
903: xmlFreeParserCtxt(ctxt);
904: return(NULL);
905: }
906:
907: inputStream = xmlNewInputStream(ctxt);
908: if (inputStream == NULL) {
909: xmlFreeParserCtxt(ctxt);
910: return(NULL);
911: }
912:
913: inputStream->filename = (char *) xmlCanonicPath((const xmlChar *)filename);
914: inputStream->buf = buf;
915: inputStream->base = inputStream->buf->buffer->content;
916: inputStream->cur = inputStream->buf->buffer->content;
917: inputStream->end =
918: &inputStream->buf->buffer->content[inputStream->buf->buffer->use];
919:
920: inputPush(ctxt, inputStream);
921: if ((ctxt->directory == NULL) && (directory == NULL))
922: directory = xmlParserGetDirectory(filename);
923: if ((ctxt->directory == NULL) && (directory != NULL))
924: ctxt->directory = directory;
925: ctxt->valid = 0;
926: ctxt->validate = 0;
927: ctxt->loadsubset = 0;
928: ctxt->pedantic = 0;
929: ctxt->dictNames = 1;
930:
931: xmlParseDocument(ctxt);
932:
933: if (ctxt->wellFormed)
934: ret = ctxt->myDoc;
935: else {
936: ret = NULL;
937: xmlFreeDoc(ctxt->myDoc);
938: ctxt->myDoc = NULL;
939: }
940: xmlFreeParserCtxt(ctxt);
941:
942: return(ret);
943: }
944:
945: /**
946: * xmlLoadFileContent:
947: * @filename: a file path
948: *
949: * Load a file content into memory.
950: *
951: * Returns a pointer to the 0 terminated string or NULL in case of error
952: */
953: static xmlChar *
954: xmlLoadFileContent(const char *filename)
955: {
956: #ifdef HAVE_STAT
957: int fd;
958: #else
959: FILE *fd;
960: #endif
961: int len;
962: long size;
963:
964: #ifdef HAVE_STAT
965: struct stat info;
966: #endif
967: xmlChar *content;
968:
969: if (filename == NULL)
970: return (NULL);
971:
972: #ifdef HAVE_STAT
973: if (stat(filename, &info) < 0)
974: return (NULL);
975: #endif
976:
977: #ifdef HAVE_STAT
978: if ((fd = open(filename, O_RDONLY)) < 0)
979: #else
980: if ((fd = fopen(filename, "rb")) == NULL)
981: #endif
982: {
983: return (NULL);
984: }
985: #ifdef HAVE_STAT
986: size = info.st_size;
987: #else
988: if (fseek(fd, 0, SEEK_END) || (size = ftell(fd)) == EOF || fseek(fd, 0, SEEK_SET)) { /* File operations denied? ok, just close and return failure */
989: fclose(fd);
990: return (NULL);
991: }
992: #endif
993: content = xmlMallocAtomic(size + 10);
994: if (content == NULL) {
995: xmlCatalogErrMemory("allocating catalog data");
996: return (NULL);
997: }
998: #ifdef HAVE_STAT
999: len = read(fd, content, size);
1000: close(fd);
1001: #else
1002: len = fread(content, 1, size, fd);
1003: fclose(fd);
1004: #endif
1005: if (len < 0) {
1006: xmlFree(content);
1007: return (NULL);
1008: }
1009: content[len] = 0;
1010:
1011: return(content);
1012: }
1013:
1014: /**
1015: * xmlCatalogNormalizePublic:
1016: * @pubID: the public ID string
1017: *
1018: * Normalizes the Public Identifier
1019: *
1020: * Implements 6.2. Public Identifier Normalization
1021: * from http://www.oasis-open.org/committees/entity/spec-2001-08-06.html
1022: *
1023: * Returns the new string or NULL, the string must be deallocated
1024: * by the caller.
1025: */
1026: static xmlChar *
1027: xmlCatalogNormalizePublic(const xmlChar *pubID)
1028: {
1029: int ok = 1;
1030: int white;
1031: const xmlChar *p;
1032: xmlChar *ret;
1033: xmlChar *q;
1034:
1035: if (pubID == NULL)
1036: return(NULL);
1037:
1038: white = 1;
1039: for (p = pubID;*p != 0 && ok;p++) {
1040: if (!xmlIsBlank_ch(*p))
1041: white = 0;
1042: else if (*p == 0x20 && !white)
1043: white = 1;
1044: else
1045: ok = 0;
1046: }
1047: if (ok && !white) /* is normalized */
1048: return(NULL);
1049:
1050: ret = xmlStrdup(pubID);
1051: q = ret;
1052: white = 0;
1053: for (p = pubID;*p != 0;p++) {
1054: if (xmlIsBlank_ch(*p)) {
1055: if (q != ret)
1056: white = 1;
1057: } else {
1058: if (white) {
1059: *(q++) = 0x20;
1060: white = 0;
1061: }
1062: *(q++) = *p;
1063: }
1064: }
1065: *q = 0;
1066: return(ret);
1067: }
1068:
1069: /************************************************************************
1070: * *
1071: * The XML Catalog parser *
1072: * *
1073: ************************************************************************/
1074:
1075: static xmlCatalogEntryPtr
1076: xmlParseXMLCatalogFile(xmlCatalogPrefer prefer, const xmlChar *filename);
1077: static void
1078: xmlParseXMLCatalogNodeList(xmlNodePtr cur, xmlCatalogPrefer prefer,
1079: xmlCatalogEntryPtr parent, xmlCatalogEntryPtr cgroup);
1080: static xmlChar *
1081: xmlCatalogListXMLResolve(xmlCatalogEntryPtr catal, const xmlChar *pubID,
1082: const xmlChar *sysID);
1083: static xmlChar *
1084: xmlCatalogListXMLResolveURI(xmlCatalogEntryPtr catal, const xmlChar *URI);
1085:
1086:
1087: /**
1088: * xmlGetXMLCatalogEntryType:
1089: * @name: the name
1090: *
1091: * lookup the internal type associated to an XML catalog entry name
1092: *
1093: * Returns the type associated with that name
1094: */
1095: static xmlCatalogEntryType
1096: xmlGetXMLCatalogEntryType(const xmlChar *name) {
1097: xmlCatalogEntryType type = XML_CATA_NONE;
1098: if (xmlStrEqual(name, (const xmlChar *) "system"))
1099: type = XML_CATA_SYSTEM;
1100: else if (xmlStrEqual(name, (const xmlChar *) "public"))
1101: type = XML_CATA_PUBLIC;
1102: else if (xmlStrEqual(name, (const xmlChar *) "rewriteSystem"))
1103: type = XML_CATA_REWRITE_SYSTEM;
1104: else if (xmlStrEqual(name, (const xmlChar *) "delegatePublic"))
1105: type = XML_CATA_DELEGATE_PUBLIC;
1106: else if (xmlStrEqual(name, (const xmlChar *) "delegateSystem"))
1107: type = XML_CATA_DELEGATE_SYSTEM;
1108: else if (xmlStrEqual(name, (const xmlChar *) "uri"))
1109: type = XML_CATA_URI;
1110: else if (xmlStrEqual(name, (const xmlChar *) "rewriteURI"))
1111: type = XML_CATA_REWRITE_URI;
1112: else if (xmlStrEqual(name, (const xmlChar *) "delegateURI"))
1113: type = XML_CATA_DELEGATE_URI;
1114: else if (xmlStrEqual(name, (const xmlChar *) "nextCatalog"))
1115: type = XML_CATA_NEXT_CATALOG;
1116: else if (xmlStrEqual(name, (const xmlChar *) "catalog"))
1117: type = XML_CATA_CATALOG;
1118: return(type);
1119: }
1120:
1121: /**
1122: * xmlParseXMLCatalogOneNode:
1123: * @cur: the XML node
1124: * @type: the type of Catalog entry
1125: * @name: the name of the node
1126: * @attrName: the attribute holding the value
1127: * @uriAttrName: the attribute holding the URI-Reference
1128: * @prefer: the PUBLIC vs. SYSTEM current preference value
1129: * @cgroup: the group which includes this node
1130: *
1131: * Finishes the examination of an XML tree node of a catalog and build
1132: * a Catalog entry from it.
1133: *
1134: * Returns the new Catalog entry node or NULL in case of error.
1135: */
1136: static xmlCatalogEntryPtr
1137: xmlParseXMLCatalogOneNode(xmlNodePtr cur, xmlCatalogEntryType type,
1138: const xmlChar *name, const xmlChar *attrName,
1139: const xmlChar *uriAttrName, xmlCatalogPrefer prefer,
1140: xmlCatalogEntryPtr cgroup) {
1141: int ok = 1;
1142: xmlChar *uriValue;
1143: xmlChar *nameValue = NULL;
1144: xmlChar *base = NULL;
1145: xmlChar *URL = NULL;
1146: xmlCatalogEntryPtr ret = NULL;
1147:
1148: if (attrName != NULL) {
1149: nameValue = xmlGetProp(cur, attrName);
1150: if (nameValue == NULL) {
1151: xmlCatalogErr(ret, cur, XML_CATALOG_MISSING_ATTR,
1152: "%s entry lacks '%s'\n", name, attrName, NULL);
1153: ok = 0;
1154: }
1155: }
1156: uriValue = xmlGetProp(cur, uriAttrName);
1157: if (uriValue == NULL) {
1158: xmlCatalogErr(ret, cur, XML_CATALOG_MISSING_ATTR,
1159: "%s entry lacks '%s'\n", name, uriAttrName, NULL);
1160: ok = 0;
1161: }
1162: if (!ok) {
1163: if (nameValue != NULL)
1164: xmlFree(nameValue);
1165: if (uriValue != NULL)
1166: xmlFree(uriValue);
1167: return(NULL);
1168: }
1169:
1170: base = xmlNodeGetBase(cur->doc, cur);
1171: URL = xmlBuildURI(uriValue, base);
1172: if (URL != NULL) {
1173: if (xmlDebugCatalogs > 1) {
1174: if (nameValue != NULL)
1175: xmlGenericError(xmlGenericErrorContext,
1176: "Found %s: '%s' '%s'\n", name, nameValue, URL);
1177: else
1178: xmlGenericError(xmlGenericErrorContext,
1179: "Found %s: '%s'\n", name, URL);
1180: }
1181: ret = xmlNewCatalogEntry(type, nameValue, uriValue, URL, prefer, cgroup);
1182: } else {
1183: xmlCatalogErr(ret, cur, XML_CATALOG_ENTRY_BROKEN,
1184: "%s entry '%s' broken ?: %s\n", name, uriAttrName, uriValue);
1185: }
1186: if (nameValue != NULL)
1187: xmlFree(nameValue);
1188: if (uriValue != NULL)
1189: xmlFree(uriValue);
1190: if (base != NULL)
1191: xmlFree(base);
1192: if (URL != NULL)
1193: xmlFree(URL);
1194: return(ret);
1195: }
1196:
1197: /**
1198: * xmlParseXMLCatalogNode:
1199: * @cur: the XML node
1200: * @prefer: the PUBLIC vs. SYSTEM current preference value
1201: * @parent: the parent Catalog entry
1202: * @cgroup: the group which includes this node
1203: *
1204: * Examines an XML tree node of a catalog and build
1205: * a Catalog entry from it adding it to its parent. The examination can
1206: * be recursive.
1207: */
1208: static void
1209: xmlParseXMLCatalogNode(xmlNodePtr cur, xmlCatalogPrefer prefer,
1210: xmlCatalogEntryPtr parent, xmlCatalogEntryPtr cgroup)
1211: {
1212: xmlChar *base = NULL;
1213: xmlCatalogEntryPtr entry = NULL;
1214:
1215: if (cur == NULL)
1216: return;
1217: if (xmlStrEqual(cur->name, BAD_CAST "group")) {
1218: xmlChar *prop;
1219: xmlCatalogPrefer pref = XML_CATA_PREFER_NONE;
1220:
1221: prop = xmlGetProp(cur, BAD_CAST "prefer");
1222: if (prop != NULL) {
1223: if (xmlStrEqual(prop, BAD_CAST "system")) {
1224: prefer = XML_CATA_PREFER_SYSTEM;
1225: } else if (xmlStrEqual(prop, BAD_CAST "public")) {
1226: prefer = XML_CATA_PREFER_PUBLIC;
1227: } else {
1228: xmlCatalogErr(parent, cur, XML_CATALOG_PREFER_VALUE,
1229: "Invalid value for prefer: '%s'\n",
1230: prop, NULL, NULL);
1231: }
1232: xmlFree(prop);
1233: pref = prefer;
1234: }
1235: prop = xmlGetProp(cur, BAD_CAST "id");
1236: base = xmlGetNsProp(cur, BAD_CAST "base", XML_XML_NAMESPACE);
1237: entry = xmlNewCatalogEntry(XML_CATA_GROUP, prop, base, NULL, pref, cgroup);
1238: xmlFree(prop);
1239: } else if (xmlStrEqual(cur->name, BAD_CAST "public")) {
1240: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_PUBLIC,
1241: BAD_CAST "public", BAD_CAST "publicId", BAD_CAST "uri", prefer, cgroup);
1242: } else if (xmlStrEqual(cur->name, BAD_CAST "system")) {
1243: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_SYSTEM,
1244: BAD_CAST "system", BAD_CAST "systemId", BAD_CAST "uri", prefer, cgroup);
1245: } else if (xmlStrEqual(cur->name, BAD_CAST "rewriteSystem")) {
1246: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_REWRITE_SYSTEM,
1247: BAD_CAST "rewriteSystem", BAD_CAST "systemIdStartString",
1248: BAD_CAST "rewritePrefix", prefer, cgroup);
1249: } else if (xmlStrEqual(cur->name, BAD_CAST "delegatePublic")) {
1250: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_DELEGATE_PUBLIC,
1251: BAD_CAST "delegatePublic", BAD_CAST "publicIdStartString",
1252: BAD_CAST "catalog", prefer, cgroup);
1253: } else if (xmlStrEqual(cur->name, BAD_CAST "delegateSystem")) {
1254: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_DELEGATE_SYSTEM,
1255: BAD_CAST "delegateSystem", BAD_CAST "systemIdStartString",
1256: BAD_CAST "catalog", prefer, cgroup);
1257: } else if (xmlStrEqual(cur->name, BAD_CAST "uri")) {
1258: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_URI,
1259: BAD_CAST "uri", BAD_CAST "name",
1260: BAD_CAST "uri", prefer, cgroup);
1261: } else if (xmlStrEqual(cur->name, BAD_CAST "rewriteURI")) {
1262: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_REWRITE_URI,
1263: BAD_CAST "rewriteURI", BAD_CAST "uriStartString",
1264: BAD_CAST "rewritePrefix", prefer, cgroup);
1265: } else if (xmlStrEqual(cur->name, BAD_CAST "delegateURI")) {
1266: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_DELEGATE_URI,
1267: BAD_CAST "delegateURI", BAD_CAST "uriStartString",
1268: BAD_CAST "catalog", prefer, cgroup);
1269: } else if (xmlStrEqual(cur->name, BAD_CAST "nextCatalog")) {
1270: entry = xmlParseXMLCatalogOneNode(cur, XML_CATA_NEXT_CATALOG,
1271: BAD_CAST "nextCatalog", NULL,
1272: BAD_CAST "catalog", prefer, cgroup);
1273: }
1274: if (entry != NULL) {
1275: if (parent != NULL) {
1276: entry->parent = parent;
1277: if (parent->children == NULL)
1278: parent->children = entry;
1279: else {
1280: xmlCatalogEntryPtr prev;
1281:
1282: prev = parent->children;
1283: while (prev->next != NULL)
1284: prev = prev->next;
1285: prev->next = entry;
1286: }
1287: }
1288: if (entry->type == XML_CATA_GROUP) {
1289: /*
1290: * Recurse to propagate prefer to the subtree
1291: * (xml:base handling is automated)
1292: */
1293: xmlParseXMLCatalogNodeList(cur->children, prefer, parent, entry);
1294: }
1295: }
1296: if (base != NULL)
1297: xmlFree(base);
1298: }
1299:
1300: /**
1301: * xmlParseXMLCatalogNodeList:
1302: * @cur: the XML node list of siblings
1303: * @prefer: the PUBLIC vs. SYSTEM current preference value
1304: * @parent: the parent Catalog entry
1305: * @cgroup: the group which includes this list
1306: *
1307: * Examines a list of XML sibling nodes of a catalog and build
1308: * a list of Catalog entry from it adding it to the parent.
1309: * The examination will recurse to examine node subtrees.
1310: */
1311: static void
1312: xmlParseXMLCatalogNodeList(xmlNodePtr cur, xmlCatalogPrefer prefer,
1313: xmlCatalogEntryPtr parent, xmlCatalogEntryPtr cgroup) {
1314: while (cur != NULL) {
1315: if ((cur->ns != NULL) && (cur->ns->href != NULL) &&
1316: (xmlStrEqual(cur->ns->href, XML_CATALOGS_NAMESPACE))) {
1317: xmlParseXMLCatalogNode(cur, prefer, parent, cgroup);
1318: }
1319: cur = cur->next;
1320: }
1321: /* TODO: sort the list according to REWRITE lengths and prefer value */
1322: }
1323:
1324: /**
1325: * xmlParseXMLCatalogFile:
1326: * @prefer: the PUBLIC vs. SYSTEM current preference value
1327: * @filename: the filename for the catalog
1328: *
1329: * Parses the catalog file to extract the XML tree and then analyze the
1330: * tree to build a list of Catalog entries corresponding to this catalog
1331: *
1332: * Returns the resulting Catalog entries list
1333: */
1334: static xmlCatalogEntryPtr
1335: xmlParseXMLCatalogFile(xmlCatalogPrefer prefer, const xmlChar *filename) {
1336: xmlDocPtr doc;
1337: xmlNodePtr cur;
1338: xmlChar *prop;
1339: xmlCatalogEntryPtr parent = NULL;
1340:
1341: if (filename == NULL)
1342: return(NULL);
1343:
1344: doc = xmlParseCatalogFile((const char *) filename);
1345: if (doc == NULL) {
1346: if (xmlDebugCatalogs)
1347: xmlGenericError(xmlGenericErrorContext,
1348: "Failed to parse catalog %s\n", filename);
1349: return(NULL);
1350: }
1351:
1352: if (xmlDebugCatalogs)
1353: xmlGenericError(xmlGenericErrorContext,
1354: "%d Parsing catalog %s\n", xmlGetThreadId(), filename);
1355:
1356: cur = xmlDocGetRootElement(doc);
1357: if ((cur != NULL) && (xmlStrEqual(cur->name, BAD_CAST "catalog")) &&
1358: (cur->ns != NULL) && (cur->ns->href != NULL) &&
1359: (xmlStrEqual(cur->ns->href, XML_CATALOGS_NAMESPACE))) {
1360:
1361: parent = xmlNewCatalogEntry(XML_CATA_CATALOG, NULL,
1362: (const xmlChar *)filename, NULL, prefer, NULL);
1363: if (parent == NULL) {
1364: xmlFreeDoc(doc);
1365: return(NULL);
1366: }
1367:
1368: prop = xmlGetProp(cur, BAD_CAST "prefer");
1369: if (prop != NULL) {
1370: if (xmlStrEqual(prop, BAD_CAST "system")) {
1371: prefer = XML_CATA_PREFER_SYSTEM;
1372: } else if (xmlStrEqual(prop, BAD_CAST "public")) {
1373: prefer = XML_CATA_PREFER_PUBLIC;
1374: } else {
1375: xmlCatalogErr(NULL, cur, XML_CATALOG_PREFER_VALUE,
1376: "Invalid value for prefer: '%s'\n",
1377: prop, NULL, NULL);
1378: }
1379: xmlFree(prop);
1380: }
1381: cur = cur->children;
1382: xmlParseXMLCatalogNodeList(cur, prefer, parent, NULL);
1383: } else {
1384: xmlCatalogErr(NULL, (xmlNodePtr) doc, XML_CATALOG_NOT_CATALOG,
1385: "File %s is not an XML Catalog\n",
1386: filename, NULL, NULL);
1387: xmlFreeDoc(doc);
1388: return(NULL);
1389: }
1390: xmlFreeDoc(doc);
1391: return(parent);
1392: }
1393:
1394: /**
1395: * xmlFetchXMLCatalogFile:
1396: * @catal: an existing but incomplete catalog entry
1397: *
1398: * Fetch and parse the subcatalog referenced by an entry
1399: *
1400: * Returns 0 in case of success, -1 otherwise
1401: */
1402: static int
1403: xmlFetchXMLCatalogFile(xmlCatalogEntryPtr catal) {
1404: xmlCatalogEntryPtr doc;
1405:
1406: if (catal == NULL)
1407: return(-1);
1408: if (catal->URL == NULL)
1409: return(-1);
1410: if (catal->children != NULL)
1411: return(-1);
1412:
1413: /*
1414: * lock the whole catalog for modification
1415: */
1416: xmlRMutexLock(xmlCatalogMutex);
1417: if (catal->children != NULL) {
1418: /* Okay someone else did it in the meantime */
1419: xmlRMutexUnlock(xmlCatalogMutex);
1420: return(0);
1421: }
1422:
1423: if (xmlCatalogXMLFiles != NULL) {
1424: doc = (xmlCatalogEntryPtr)
1425: xmlHashLookup(xmlCatalogXMLFiles, catal->URL);
1426: if (doc != NULL) {
1427: if (xmlDebugCatalogs)
1428: xmlGenericError(xmlGenericErrorContext,
1429: "Found %s in file hash\n", catal->URL);
1430:
1431: if (catal->type == XML_CATA_CATALOG)
1432: catal->children = doc->children;
1433: else
1434: catal->children = doc;
1435: catal->dealloc = 0;
1436: xmlRMutexUnlock(xmlCatalogMutex);
1437: return(0);
1438: }
1439: if (xmlDebugCatalogs)
1440: xmlGenericError(xmlGenericErrorContext,
1441: "%s not found in file hash\n", catal->URL);
1442: }
1443:
1444: /*
1445: * Fetch and parse. Note that xmlParseXMLCatalogFile does not
1446: * use the existing catalog, there is no recursion allowed at
1447: * that level.
1448: */
1449: doc = xmlParseXMLCatalogFile(catal->prefer, catal->URL);
1450: if (doc == NULL) {
1451: catal->type = XML_CATA_BROKEN_CATALOG;
1452: xmlRMutexUnlock(xmlCatalogMutex);
1453: return(-1);
1454: }
1455:
1456: if (catal->type == XML_CATA_CATALOG)
1457: catal->children = doc->children;
1458: else
1459: catal->children = doc;
1460:
1461: doc->dealloc = 1;
1462:
1463: if (xmlCatalogXMLFiles == NULL)
1464: xmlCatalogXMLFiles = xmlHashCreate(10);
1465: if (xmlCatalogXMLFiles != NULL) {
1466: if (xmlDebugCatalogs)
1467: xmlGenericError(xmlGenericErrorContext,
1468: "%s added to file hash\n", catal->URL);
1469: xmlHashAddEntry(xmlCatalogXMLFiles, catal->URL, doc);
1470: }
1471: xmlRMutexUnlock(xmlCatalogMutex);
1472: return(0);
1473: }
1474:
1475: /************************************************************************
1476: * *
1477: * XML Catalog handling *
1478: * *
1479: ************************************************************************/
1480:
1481: /**
1482: * xmlAddXMLCatalog:
1483: * @catal: top of an XML catalog
1484: * @type: the type of record to add to the catalog
1485: * @orig: the system, public or prefix to match (or NULL)
1486: * @replace: the replacement value for the match
1487: *
1488: * Add an entry in the XML catalog, it may overwrite existing but
1489: * different entries.
1490: *
1491: * Returns 0 if successful, -1 otherwise
1492: */
1493: static int
1494: xmlAddXMLCatalog(xmlCatalogEntryPtr catal, const xmlChar *type,
1495: const xmlChar *orig, const xmlChar *replace) {
1496: xmlCatalogEntryPtr cur;
1497: xmlCatalogEntryType typ;
1498: int doregister = 0;
1499:
1500: if ((catal == NULL) ||
1501: ((catal->type != XML_CATA_CATALOG) &&
1502: (catal->type != XML_CATA_BROKEN_CATALOG)))
1503: return(-1);
1504: if (catal->children == NULL) {
1505: xmlFetchXMLCatalogFile(catal);
1506: }
1507: if (catal->children == NULL)
1508: doregister = 1;
1509:
1510: typ = xmlGetXMLCatalogEntryType(type);
1511: if (typ == XML_CATA_NONE) {
1512: if (xmlDebugCatalogs)
1513: xmlGenericError(xmlGenericErrorContext,
1514: "Failed to add unknown element %s to catalog\n", type);
1515: return(-1);
1516: }
1517:
1518: cur = catal->children;
1519: /*
1520: * Might be a simple "update in place"
1521: */
1522: if (cur != NULL) {
1523: while (cur != NULL) {
1524: if ((orig != NULL) && (cur->type == typ) &&
1525: (xmlStrEqual(orig, cur->name))) {
1526: if (xmlDebugCatalogs)
1527: xmlGenericError(xmlGenericErrorContext,
1528: "Updating element %s to catalog\n", type);
1529: if (cur->value != NULL)
1530: xmlFree(cur->value);
1531: if (cur->URL != NULL)
1532: xmlFree(cur->URL);
1533: cur->value = xmlStrdup(replace);
1534: cur->URL = xmlStrdup(replace);
1535: return(0);
1536: }
1537: if (cur->next == NULL)
1538: break;
1539: cur = cur->next;
1540: }
1541: }
1542: if (xmlDebugCatalogs)
1543: xmlGenericError(xmlGenericErrorContext,
1544: "Adding element %s to catalog\n", type);
1545: if (cur == NULL)
1546: catal->children = xmlNewCatalogEntry(typ, orig, replace,
1547: NULL, catal->prefer, NULL);
1548: else
1549: cur->next = xmlNewCatalogEntry(typ, orig, replace,
1550: NULL, catal->prefer, NULL);
1551: if (doregister) {
1552: catal->type = XML_CATA_CATALOG;
1553: cur = xmlHashLookup(xmlCatalogXMLFiles, catal->URL);
1554: if (cur != NULL)
1555: cur->children = catal->children;
1556: }
1557:
1558: return(0);
1559: }
1560:
1561: /**
1562: * xmlDelXMLCatalog:
1563: * @catal: top of an XML catalog
1564: * @value: the value to remove from the catalog
1565: *
1566: * Remove entries in the XML catalog where the value or the URI
1567: * is equal to @value
1568: *
1569: * Returns the number of entries removed if successful, -1 otherwise
1570: */
1571: static int
1572: xmlDelXMLCatalog(xmlCatalogEntryPtr catal, const xmlChar *value) {
1573: xmlCatalogEntryPtr cur;
1574: int ret = 0;
1575:
1576: if ((catal == NULL) ||
1577: ((catal->type != XML_CATA_CATALOG) &&
1578: (catal->type != XML_CATA_BROKEN_CATALOG)))
1579: return(-1);
1580: if (value == NULL)
1581: return(-1);
1582: if (catal->children == NULL) {
1583: xmlFetchXMLCatalogFile(catal);
1584: }
1585:
1586: /*
1587: * Scan the children
1588: */
1589: cur = catal->children;
1590: while (cur != NULL) {
1591: if (((cur->name != NULL) && (xmlStrEqual(value, cur->name))) ||
1592: (xmlStrEqual(value, cur->value))) {
1593: if (xmlDebugCatalogs) {
1594: if (cur->name != NULL)
1595: xmlGenericError(xmlGenericErrorContext,
1596: "Removing element %s from catalog\n", cur->name);
1597: else
1598: xmlGenericError(xmlGenericErrorContext,
1599: "Removing element %s from catalog\n", cur->value);
1600: }
1601: cur->type = XML_CATA_REMOVED;
1602: }
1603: cur = cur->next;
1604: }
1605: return(ret);
1606: }
1607:
1608: /**
1609: * xmlCatalogXMLResolve:
1610: * @catal: a catalog list
1611: * @pubID: the public ID string
1612: * @sysID: the system ID string
1613: *
1614: * Do a complete resolution lookup of an External Identifier for a
1615: * list of catalog entries.
1616: *
1617: * Implements (or tries to) 7.1. External Identifier Resolution
1618: * from http://www.oasis-open.org/committees/entity/spec-2001-08-06.html
1619: *
1620: * Returns the URI of the resource or NULL if not found
1621: */
1622: static xmlChar *
1623: xmlCatalogXMLResolve(xmlCatalogEntryPtr catal, const xmlChar *pubID,
1624: const xmlChar *sysID) {
1625: xmlChar *ret = NULL;
1626: xmlCatalogEntryPtr cur;
1627: int haveDelegate = 0;
1628: int haveNext = 0;
1629:
1630: /*
1631: * protection against loops
1632: */
1633: if (catal->depth > MAX_CATAL_DEPTH) {
1634: xmlCatalogErr(catal, NULL, XML_CATALOG_RECURSION,
1635: "Detected recursion in catalog %s\n",
1636: catal->name, NULL, NULL);
1637: return(NULL);
1638: }
1639: catal->depth++;
1640:
1641: /*
1642: * First tries steps 2/ 3/ 4/ if a system ID is provided.
1643: */
1644: if (sysID != NULL) {
1645: xmlCatalogEntryPtr rewrite = NULL;
1646: int lenrewrite = 0, len;
1647: cur = catal;
1648: haveDelegate = 0;
1649: while (cur != NULL) {
1650: switch (cur->type) {
1651: case XML_CATA_SYSTEM:
1652: if (xmlStrEqual(sysID, cur->name)) {
1653: if (xmlDebugCatalogs)
1654: xmlGenericError(xmlGenericErrorContext,
1655: "Found system match %s, using %s\n",
1656: cur->name, cur->URL);
1657: catal->depth--;
1658: return(xmlStrdup(cur->URL));
1659: }
1660: break;
1661: case XML_CATA_REWRITE_SYSTEM:
1662: len = xmlStrlen(cur->name);
1663: if ((len > lenrewrite) &&
1664: (!xmlStrncmp(sysID, cur->name, len))) {
1665: lenrewrite = len;
1666: rewrite = cur;
1667: }
1668: break;
1669: case XML_CATA_DELEGATE_SYSTEM:
1670: if (!xmlStrncmp(sysID, cur->name, xmlStrlen(cur->name)))
1671: haveDelegate++;
1672: break;
1673: case XML_CATA_NEXT_CATALOG:
1674: haveNext++;
1675: break;
1676: default:
1677: break;
1678: }
1679: cur = cur->next;
1680: }
1681: if (rewrite != NULL) {
1682: if (xmlDebugCatalogs)
1683: xmlGenericError(xmlGenericErrorContext,
1684: "Using rewriting rule %s\n", rewrite->name);
1685: ret = xmlStrdup(rewrite->URL);
1686: if (ret != NULL)
1687: ret = xmlStrcat(ret, &sysID[lenrewrite]);
1688: catal->depth--;
1689: return(ret);
1690: }
1691: if (haveDelegate) {
1692: const xmlChar *delegates[MAX_DELEGATE];
1693: int nbList = 0, i;
1694:
1695: /*
1696: * Assume the entries have been sorted by decreasing substring
1697: * matches when the list was produced.
1698: */
1699: cur = catal;
1700: while (cur != NULL) {
1701: if ((cur->type == XML_CATA_DELEGATE_SYSTEM) &&
1702: (!xmlStrncmp(sysID, cur->name, xmlStrlen(cur->name)))) {
1703: for (i = 0;i < nbList;i++)
1704: if (xmlStrEqual(cur->URL, delegates[i]))
1705: break;
1706: if (i < nbList) {
1707: cur = cur->next;
1708: continue;
1709: }
1710: if (nbList < MAX_DELEGATE)
1711: delegates[nbList++] = cur->URL;
1712:
1713: if (cur->children == NULL) {
1714: xmlFetchXMLCatalogFile(cur);
1715: }
1716: if (cur->children != NULL) {
1717: if (xmlDebugCatalogs)
1718: xmlGenericError(xmlGenericErrorContext,
1719: "Trying system delegate %s\n", cur->URL);
1720: ret = xmlCatalogListXMLResolve(
1721: cur->children, NULL, sysID);
1722: if (ret != NULL) {
1723: catal->depth--;
1724: return(ret);
1725: }
1726: }
1727: }
1728: cur = cur->next;
1729: }
1730: /*
1731: * Apply the cut algorithm explained in 4/
1732: */
1733: catal->depth--;
1734: return(XML_CATAL_BREAK);
1735: }
1736: }
1737: /*
1738: * Then tries 5/ 6/ if a public ID is provided
1739: */
1740: if (pubID != NULL) {
1741: cur = catal;
1742: haveDelegate = 0;
1743: while (cur != NULL) {
1744: switch (cur->type) {
1745: case XML_CATA_PUBLIC:
1746: if (xmlStrEqual(pubID, cur->name)) {
1747: if (xmlDebugCatalogs)
1748: xmlGenericError(xmlGenericErrorContext,
1749: "Found public match %s\n", cur->name);
1750: catal->depth--;
1751: return(xmlStrdup(cur->URL));
1752: }
1753: break;
1754: case XML_CATA_DELEGATE_PUBLIC:
1755: if (!xmlStrncmp(pubID, cur->name, xmlStrlen(cur->name)) &&
1756: (cur->prefer == XML_CATA_PREFER_PUBLIC))
1757: haveDelegate++;
1758: break;
1759: case XML_CATA_NEXT_CATALOG:
1760: if (sysID == NULL)
1761: haveNext++;
1762: break;
1763: default:
1764: break;
1765: }
1766: cur = cur->next;
1767: }
1768: if (haveDelegate) {
1769: const xmlChar *delegates[MAX_DELEGATE];
1770: int nbList = 0, i;
1771:
1772: /*
1773: * Assume the entries have been sorted by decreasing substring
1774: * matches when the list was produced.
1775: */
1776: cur = catal;
1777: while (cur != NULL) {
1778: if ((cur->type == XML_CATA_DELEGATE_PUBLIC) &&
1779: (cur->prefer == XML_CATA_PREFER_PUBLIC) &&
1780: (!xmlStrncmp(pubID, cur->name, xmlStrlen(cur->name)))) {
1781:
1782: for (i = 0;i < nbList;i++)
1783: if (xmlStrEqual(cur->URL, delegates[i]))
1784: break;
1785: if (i < nbList) {
1786: cur = cur->next;
1787: continue;
1788: }
1789: if (nbList < MAX_DELEGATE)
1790: delegates[nbList++] = cur->URL;
1791:
1792: if (cur->children == NULL) {
1793: xmlFetchXMLCatalogFile(cur);
1794: }
1795: if (cur->children != NULL) {
1796: if (xmlDebugCatalogs)
1797: xmlGenericError(xmlGenericErrorContext,
1798: "Trying public delegate %s\n", cur->URL);
1799: ret = xmlCatalogListXMLResolve(
1800: cur->children, pubID, NULL);
1801: if (ret != NULL) {
1802: catal->depth--;
1803: return(ret);
1804: }
1805: }
1806: }
1807: cur = cur->next;
1808: }
1809: /*
1810: * Apply the cut algorithm explained in 4/
1811: */
1812: catal->depth--;
1813: return(XML_CATAL_BREAK);
1814: }
1815: }
1816: if (haveNext) {
1817: cur = catal;
1818: while (cur != NULL) {
1819: if (cur->type == XML_CATA_NEXT_CATALOG) {
1820: if (cur->children == NULL) {
1821: xmlFetchXMLCatalogFile(cur);
1822: }
1823: if (cur->children != NULL) {
1824: ret = xmlCatalogListXMLResolve(cur->children, pubID, sysID);
1825: if (ret != NULL) {
1826: catal->depth--;
1827: return(ret);
1828: } else if (catal->depth > MAX_CATAL_DEPTH) {
1829: return(NULL);
1830: }
1831: }
1832: }
1833: cur = cur->next;
1834: }
1835: }
1836:
1837: catal->depth--;
1838: return(NULL);
1839: }
1840:
1841: /**
1842: * xmlCatalogXMLResolveURI:
1843: * @catal: a catalog list
1844: * @URI: the URI
1845: * @sysID: the system ID string
1846: *
1847: * Do a complete resolution lookup of an External Identifier for a
1848: * list of catalog entries.
1849: *
1850: * Implements (or tries to) 7.2.2. URI Resolution
1851: * from http://www.oasis-open.org/committees/entity/spec-2001-08-06.html
1852: *
1853: * Returns the URI of the resource or NULL if not found
1854: */
1855: static xmlChar *
1856: xmlCatalogXMLResolveURI(xmlCatalogEntryPtr catal, const xmlChar *URI) {
1857: xmlChar *ret = NULL;
1858: xmlCatalogEntryPtr cur;
1859: int haveDelegate = 0;
1860: int haveNext = 0;
1861: xmlCatalogEntryPtr rewrite = NULL;
1862: int lenrewrite = 0, len;
1863:
1864: if (catal == NULL)
1865: return(NULL);
1866:
1867: if (URI == NULL)
1868: return(NULL);
1869:
1870: if (catal->depth > MAX_CATAL_DEPTH) {
1871: xmlCatalogErr(catal, NULL, XML_CATALOG_RECURSION,
1872: "Detected recursion in catalog %s\n",
1873: catal->name, NULL, NULL);
1874: return(NULL);
1875: }
1876:
1877: /*
1878: * First tries steps 2/ 3/ 4/ if a system ID is provided.
1879: */
1880: cur = catal;
1881: haveDelegate = 0;
1882: while (cur != NULL) {
1883: switch (cur->type) {
1884: case XML_CATA_URI:
1885: if (xmlStrEqual(URI, cur->name)) {
1886: if (xmlDebugCatalogs)
1887: xmlGenericError(xmlGenericErrorContext,
1888: "Found URI match %s\n", cur->name);
1889: return(xmlStrdup(cur->URL));
1890: }
1891: break;
1892: case XML_CATA_REWRITE_URI:
1893: len = xmlStrlen(cur->name);
1894: if ((len > lenrewrite) &&
1895: (!xmlStrncmp(URI, cur->name, len))) {
1896: lenrewrite = len;
1897: rewrite = cur;
1898: }
1899: break;
1900: case XML_CATA_DELEGATE_URI:
1901: if (!xmlStrncmp(URI, cur->name, xmlStrlen(cur->name)))
1902: haveDelegate++;
1903: break;
1904: case XML_CATA_NEXT_CATALOG:
1905: haveNext++;
1906: break;
1907: default:
1908: break;
1909: }
1910: cur = cur->next;
1911: }
1912: if (rewrite != NULL) {
1913: if (xmlDebugCatalogs)
1914: xmlGenericError(xmlGenericErrorContext,
1915: "Using rewriting rule %s\n", rewrite->name);
1916: ret = xmlStrdup(rewrite->URL);
1917: if (ret != NULL)
1918: ret = xmlStrcat(ret, &URI[lenrewrite]);
1919: return(ret);
1920: }
1921: if (haveDelegate) {
1922: const xmlChar *delegates[MAX_DELEGATE];
1923: int nbList = 0, i;
1924:
1925: /*
1926: * Assume the entries have been sorted by decreasing substring
1927: * matches when the list was produced.
1928: */
1929: cur = catal;
1930: while (cur != NULL) {
1931: if (((cur->type == XML_CATA_DELEGATE_SYSTEM) ||
1932: (cur->type == XML_CATA_DELEGATE_URI)) &&
1933: (!xmlStrncmp(URI, cur->name, xmlStrlen(cur->name)))) {
1934: for (i = 0;i < nbList;i++)
1935: if (xmlStrEqual(cur->URL, delegates[i]))
1936: break;
1937: if (i < nbList) {
1938: cur = cur->next;
1939: continue;
1940: }
1941: if (nbList < MAX_DELEGATE)
1942: delegates[nbList++] = cur->URL;
1943:
1944: if (cur->children == NULL) {
1945: xmlFetchXMLCatalogFile(cur);
1946: }
1947: if (cur->children != NULL) {
1948: if (xmlDebugCatalogs)
1949: xmlGenericError(xmlGenericErrorContext,
1950: "Trying URI delegate %s\n", cur->URL);
1951: ret = xmlCatalogListXMLResolveURI(
1952: cur->children, URI);
1953: if (ret != NULL)
1954: return(ret);
1955: }
1956: }
1957: cur = cur->next;
1958: }
1959: /*
1960: * Apply the cut algorithm explained in 4/
1961: */
1962: return(XML_CATAL_BREAK);
1963: }
1964: if (haveNext) {
1965: cur = catal;
1966: while (cur != NULL) {
1967: if (cur->type == XML_CATA_NEXT_CATALOG) {
1968: if (cur->children == NULL) {
1969: xmlFetchXMLCatalogFile(cur);
1970: }
1971: if (cur->children != NULL) {
1972: ret = xmlCatalogListXMLResolveURI(cur->children, URI);
1973: if (ret != NULL)
1974: return(ret);
1975: }
1976: }
1977: cur = cur->next;
1978: }
1979: }
1980:
1981: return(NULL);
1982: }
1983:
1984: /**
1985: * xmlCatalogListXMLResolve:
1986: * @catal: a catalog list
1987: * @pubID: the public ID string
1988: * @sysID: the system ID string
1989: *
1990: * Do a complete resolution lookup of an External Identifier for a
1991: * list of catalogs
1992: *
1993: * Implements (or tries to) 7.1. External Identifier Resolution
1994: * from http://www.oasis-open.org/committees/entity/spec-2001-08-06.html
1995: *
1996: * Returns the URI of the resource or NULL if not found
1997: */
1998: static xmlChar *
1999: xmlCatalogListXMLResolve(xmlCatalogEntryPtr catal, const xmlChar *pubID,
2000: const xmlChar *sysID) {
2001: xmlChar *ret = NULL;
2002: xmlChar *urnID = NULL;
2003: xmlChar *normid;
2004:
2005: if (catal == NULL)
2006: return(NULL);
2007: if ((pubID == NULL) && (sysID == NULL))
2008: return(NULL);
2009:
2010: normid = xmlCatalogNormalizePublic(pubID);
2011: if (normid != NULL)
2012: pubID = (*normid != 0 ? normid : NULL);
2013:
2014: if (!xmlStrncmp(pubID, BAD_CAST XML_URN_PUBID, sizeof(XML_URN_PUBID) - 1)) {
2015: urnID = xmlCatalogUnWrapURN(pubID);
2016: if (xmlDebugCatalogs) {
2017: if (urnID == NULL)
2018: xmlGenericError(xmlGenericErrorContext,
2019: "Public URN ID %s expanded to NULL\n", pubID);
2020: else
2021: xmlGenericError(xmlGenericErrorContext,
2022: "Public URN ID expanded to %s\n", urnID);
2023: }
2024: ret = xmlCatalogListXMLResolve(catal, urnID, sysID);
2025: if (urnID != NULL)
2026: xmlFree(urnID);
2027: if (normid != NULL)
2028: xmlFree(normid);
2029: return(ret);
2030: }
2031: if (!xmlStrncmp(sysID, BAD_CAST XML_URN_PUBID, sizeof(XML_URN_PUBID) - 1)) {
2032: urnID = xmlCatalogUnWrapURN(sysID);
2033: if (xmlDebugCatalogs) {
2034: if (urnID == NULL)
2035: xmlGenericError(xmlGenericErrorContext,
2036: "System URN ID %s expanded to NULL\n", sysID);
2037: else
2038: xmlGenericError(xmlGenericErrorContext,
2039: "System URN ID expanded to %s\n", urnID);
2040: }
2041: if (pubID == NULL)
2042: ret = xmlCatalogListXMLResolve(catal, urnID, NULL);
2043: else if (xmlStrEqual(pubID, urnID))
2044: ret = xmlCatalogListXMLResolve(catal, pubID, NULL);
2045: else {
2046: ret = xmlCatalogListXMLResolve(catal, pubID, urnID);
2047: }
2048: if (urnID != NULL)
2049: xmlFree(urnID);
2050: if (normid != NULL)
2051: xmlFree(normid);
2052: return(ret);
2053: }
2054: while (catal != NULL) {
2055: if (catal->type == XML_CATA_CATALOG) {
2056: if (catal->children == NULL) {
2057: xmlFetchXMLCatalogFile(catal);
2058: }
2059: if (catal->children != NULL) {
2060: ret = xmlCatalogXMLResolve(catal->children, pubID, sysID);
2061: if (ret != NULL) {
2062: break;
2063: } else if ((catal->children != NULL) &&
2064: (catal->children->depth > MAX_CATAL_DEPTH)) {
2065: ret = NULL;
2066: break;
2067: }
2068: }
2069: }
2070: catal = catal->next;
2071: }
2072: if (normid != NULL)
2073: xmlFree(normid);
2074: return(ret);
2075: }
2076:
2077: /**
2078: * xmlCatalogListXMLResolveURI:
2079: * @catal: a catalog list
2080: * @URI: the URI
2081: *
2082: * Do a complete resolution lookup of an URI for a list of catalogs
2083: *
2084: * Implements (or tries to) 7.2. URI Resolution
2085: * from http://www.oasis-open.org/committees/entity/spec-2001-08-06.html
2086: *
2087: * Returns the URI of the resource or NULL if not found
2088: */
2089: static xmlChar *
2090: xmlCatalogListXMLResolveURI(xmlCatalogEntryPtr catal, const xmlChar *URI) {
2091: xmlChar *ret = NULL;
2092: xmlChar *urnID = NULL;
2093:
2094: if (catal == NULL)
2095: return(NULL);
2096: if (URI == NULL)
2097: return(NULL);
2098:
2099: if (!xmlStrncmp(URI, BAD_CAST XML_URN_PUBID, sizeof(XML_URN_PUBID) - 1)) {
2100: urnID = xmlCatalogUnWrapURN(URI);
2101: if (xmlDebugCatalogs) {
2102: if (urnID == NULL)
2103: xmlGenericError(xmlGenericErrorContext,
2104: "URN ID %s expanded to NULL\n", URI);
2105: else
2106: xmlGenericError(xmlGenericErrorContext,
2107: "URN ID expanded to %s\n", urnID);
2108: }
2109: ret = xmlCatalogListXMLResolve(catal, urnID, NULL);
2110: if (urnID != NULL)
2111: xmlFree(urnID);
2112: return(ret);
2113: }
2114: while (catal != NULL) {
2115: if (catal->type == XML_CATA_CATALOG) {
2116: if (catal->children == NULL) {
2117: xmlFetchXMLCatalogFile(catal);
2118: }
2119: if (catal->children != NULL) {
2120: ret = xmlCatalogXMLResolveURI(catal->children, URI);
2121: if (ret != NULL)
2122: return(ret);
2123: }
2124: }
2125: catal = catal->next;
2126: }
2127: return(ret);
2128: }
2129:
2130: /************************************************************************
2131: * *
2132: * The SGML Catalog parser *
2133: * *
2134: ************************************************************************/
2135:
2136:
2137: #define RAW *cur
2138: #define NEXT cur++;
2139: #define SKIP(x) cur += x;
2140:
2141: #define SKIP_BLANKS while (IS_BLANK_CH(*cur)) NEXT;
2142:
2143: /**
2144: * xmlParseSGMLCatalogComment:
2145: * @cur: the current character
2146: *
2147: * Skip a comment in an SGML catalog
2148: *
2149: * Returns new current character
2150: */
2151: static const xmlChar *
2152: xmlParseSGMLCatalogComment(const xmlChar *cur) {
2153: if ((cur[0] != '-') || (cur[1] != '-'))
2154: return(cur);
2155: SKIP(2);
2156: while ((cur[0] != 0) && ((cur[0] != '-') || ((cur[1] != '-'))))
2157: NEXT;
2158: if (cur[0] == 0) {
2159: return(NULL);
2160: }
2161: return(cur + 2);
2162: }
2163:
2164: /**
2165: * xmlParseSGMLCatalogPubid:
2166: * @cur: the current character
2167: * @id: the return location
2168: *
2169: * Parse an SGML catalog ID
2170: *
2171: * Returns new current character and store the value in @id
2172: */
2173: static const xmlChar *
2174: xmlParseSGMLCatalogPubid(const xmlChar *cur, xmlChar **id) {
2175: xmlChar *buf = NULL, *tmp;
2176: int len = 0;
2177: int size = 50;
2178: xmlChar stop;
2179: int count = 0;
2180:
2181: *id = NULL;
2182:
2183: if (RAW == '"') {
2184: NEXT;
2185: stop = '"';
2186: } else if (RAW == '\'') {
2187: NEXT;
2188: stop = '\'';
2189: } else {
2190: stop = ' ';
2191: }
2192: buf = (xmlChar *) xmlMallocAtomic(size * sizeof(xmlChar));
2193: if (buf == NULL) {
2194: xmlCatalogErrMemory("allocating public ID");
2195: return(NULL);
2196: }
2197: while (IS_PUBIDCHAR_CH(*cur) || (*cur == '?')) {
2198: if ((*cur == stop) && (stop != ' '))
2199: break;
2200: if ((stop == ' ') && (IS_BLANK_CH(*cur)))
2201: break;
2202: if (len + 1 >= size) {
2203: size *= 2;
2204: tmp = (xmlChar *) xmlRealloc(buf, size * sizeof(xmlChar));
2205: if (tmp == NULL) {
2206: xmlCatalogErrMemory("allocating public ID");
2207: xmlFree(buf);
2208: return(NULL);
2209: }
2210: buf = tmp;
2211: }
2212: buf[len++] = *cur;
2213: count++;
2214: NEXT;
2215: }
2216: buf[len] = 0;
2217: if (stop == ' ') {
2218: if (!IS_BLANK_CH(*cur)) {
2219: xmlFree(buf);
2220: return(NULL);
2221: }
2222: } else {
2223: if (*cur != stop) {
2224: xmlFree(buf);
2225: return(NULL);
2226: }
2227: NEXT;
2228: }
2229: *id = buf;
2230: return(cur);
2231: }
2232:
2233: /**
2234: * xmlParseSGMLCatalogName:
2235: * @cur: the current character
2236: * @name: the return location
2237: *
2238: * Parse an SGML catalog name
2239: *
2240: * Returns new current character and store the value in @name
2241: */
2242: static const xmlChar *
2243: xmlParseSGMLCatalogName(const xmlChar *cur, xmlChar **name) {
2244: xmlChar buf[XML_MAX_NAMELEN + 5];
2245: int len = 0;
2246: int c;
2247:
2248: *name = NULL;
2249:
2250: /*
2251: * Handler for more complex cases
2252: */
2253: c = *cur;
2254: if ((!IS_LETTER(c) && (c != '_') && (c != ':'))) {
2255: return(NULL);
2256: }
2257:
2258: while (((IS_LETTER(c)) || (IS_DIGIT(c)) ||
2259: (c == '.') || (c == '-') ||
2260: (c == '_') || (c == ':'))) {
2261: buf[len++] = c;
2262: cur++;
2263: c = *cur;
2264: if (len >= XML_MAX_NAMELEN)
2265: return(NULL);
2266: }
2267: *name = xmlStrndup(buf, len);
2268: return(cur);
2269: }
2270:
2271: /**
2272: * xmlGetSGMLCatalogEntryType:
2273: * @name: the entry name
2274: *
2275: * Get the Catalog entry type for a given SGML Catalog name
2276: *
2277: * Returns Catalog entry type
2278: */
2279: static xmlCatalogEntryType
2280: xmlGetSGMLCatalogEntryType(const xmlChar *name) {
2281: xmlCatalogEntryType type = XML_CATA_NONE;
2282: if (xmlStrEqual(name, (const xmlChar *) "SYSTEM"))
2283: type = SGML_CATA_SYSTEM;
2284: else if (xmlStrEqual(name, (const xmlChar *) "PUBLIC"))
2285: type = SGML_CATA_PUBLIC;
2286: else if (xmlStrEqual(name, (const xmlChar *) "DELEGATE"))
2287: type = SGML_CATA_DELEGATE;
2288: else if (xmlStrEqual(name, (const xmlChar *) "ENTITY"))
2289: type = SGML_CATA_ENTITY;
2290: else if (xmlStrEqual(name, (const xmlChar *) "DOCTYPE"))
2291: type = SGML_CATA_DOCTYPE;
2292: else if (xmlStrEqual(name, (const xmlChar *) "LINKTYPE"))
2293: type = SGML_CATA_LINKTYPE;
2294: else if (xmlStrEqual(name, (const xmlChar *) "NOTATION"))
2295: type = SGML_CATA_NOTATION;
2296: else if (xmlStrEqual(name, (const xmlChar *) "SGMLDECL"))
2297: type = SGML_CATA_SGMLDECL;
2298: else if (xmlStrEqual(name, (const xmlChar *) "DOCUMENT"))
2299: type = SGML_CATA_DOCUMENT;
2300: else if (xmlStrEqual(name, (const xmlChar *) "CATALOG"))
2301: type = SGML_CATA_CATALOG;
2302: else if (xmlStrEqual(name, (const xmlChar *) "BASE"))
2303: type = SGML_CATA_BASE;
2304: return(type);
2305: }
2306:
2307: /**
2308: * xmlParseSGMLCatalog:
2309: * @catal: the SGML Catalog
2310: * @value: the content of the SGML Catalog serialization
2311: * @file: the filepath for the catalog
2312: * @super: should this be handled as a Super Catalog in which case
2313: * parsing is not recursive
2314: *
2315: * Parse an SGML catalog content and fill up the @catal hash table with
2316: * the new entries found.
2317: *
2318: * Returns 0 in case of success, -1 in case of error.
2319: */
2320: static int
2321: xmlParseSGMLCatalog(xmlCatalogPtr catal, const xmlChar *value,
2322: const char *file, int super) {
2323: const xmlChar *cur = value;
2324: xmlChar *base = NULL;
2325: int res;
2326:
2327: if ((cur == NULL) || (file == NULL))
2328: return(-1);
2329: base = xmlStrdup((const xmlChar *) file);
2330:
2331: while ((cur != NULL) && (cur[0] != 0)) {
2332: SKIP_BLANKS;
2333: if (cur[0] == 0)
2334: break;
2335: if ((cur[0] == '-') && (cur[1] == '-')) {
2336: cur = xmlParseSGMLCatalogComment(cur);
2337: if (cur == NULL) {
2338: /* error */
2339: break;
2340: }
2341: } else {
2342: xmlChar *sysid = NULL;
2343: xmlChar *name = NULL;
2344: xmlCatalogEntryType type = XML_CATA_NONE;
2345:
2346: cur = xmlParseSGMLCatalogName(cur, &name);
2347: if (name == NULL) {
2348: /* error */
2349: break;
2350: }
2351: if (!IS_BLANK_CH(*cur)) {
2352: /* error */
2353: break;
2354: }
2355: SKIP_BLANKS;
2356: if (xmlStrEqual(name, (const xmlChar *) "SYSTEM"))
2357: type = SGML_CATA_SYSTEM;
2358: else if (xmlStrEqual(name, (const xmlChar *) "PUBLIC"))
2359: type = SGML_CATA_PUBLIC;
2360: else if (xmlStrEqual(name, (const xmlChar *) "DELEGATE"))
2361: type = SGML_CATA_DELEGATE;
2362: else if (xmlStrEqual(name, (const xmlChar *) "ENTITY"))
2363: type = SGML_CATA_ENTITY;
2364: else if (xmlStrEqual(name, (const xmlChar *) "DOCTYPE"))
2365: type = SGML_CATA_DOCTYPE;
2366: else if (xmlStrEqual(name, (const xmlChar *) "LINKTYPE"))
2367: type = SGML_CATA_LINKTYPE;
2368: else if (xmlStrEqual(name, (const xmlChar *) "NOTATION"))
2369: type = SGML_CATA_NOTATION;
2370: else if (xmlStrEqual(name, (const xmlChar *) "SGMLDECL"))
2371: type = SGML_CATA_SGMLDECL;
2372: else if (xmlStrEqual(name, (const xmlChar *) "DOCUMENT"))
2373: type = SGML_CATA_DOCUMENT;
2374: else if (xmlStrEqual(name, (const xmlChar *) "CATALOG"))
2375: type = SGML_CATA_CATALOG;
2376: else if (xmlStrEqual(name, (const xmlChar *) "BASE"))
2377: type = SGML_CATA_BASE;
2378: else if (xmlStrEqual(name, (const xmlChar *) "OVERRIDE")) {
2379: xmlFree(name);
2380: cur = xmlParseSGMLCatalogName(cur, &name);
2381: if (name == NULL) {
2382: /* error */
2383: break;
2384: }
2385: xmlFree(name);
2386: continue;
2387: }
2388: xmlFree(name);
2389: name = NULL;
2390:
2391: switch(type) {
2392: case SGML_CATA_ENTITY:
2393: if (*cur == '%')
2394: type = SGML_CATA_PENTITY;
2395: case SGML_CATA_PENTITY:
2396: case SGML_CATA_DOCTYPE:
2397: case SGML_CATA_LINKTYPE:
2398: case SGML_CATA_NOTATION:
2399: cur = xmlParseSGMLCatalogName(cur, &name);
2400: if (cur == NULL) {
2401: /* error */
2402: break;
2403: }
2404: if (!IS_BLANK_CH(*cur)) {
2405: /* error */
2406: break;
2407: }
2408: SKIP_BLANKS;
2409: cur = xmlParseSGMLCatalogPubid(cur, &sysid);
2410: if (cur == NULL) {
2411: /* error */
2412: break;
2413: }
2414: break;
2415: case SGML_CATA_PUBLIC:
2416: case SGML_CATA_SYSTEM:
2417: case SGML_CATA_DELEGATE:
2418: cur = xmlParseSGMLCatalogPubid(cur, &name);
2419: if (cur == NULL) {
2420: /* error */
2421: break;
2422: }
2423: if (type != SGML_CATA_SYSTEM) {
2424: xmlChar *normid;
2425:
2426: normid = xmlCatalogNormalizePublic(name);
2427: if (normid != NULL) {
2428: if (name != NULL)
2429: xmlFree(name);
2430: if (*normid != 0)
2431: name = normid;
2432: else {
2433: xmlFree(normid);
2434: name = NULL;
2435: }
2436: }
2437: }
2438: if (!IS_BLANK_CH(*cur)) {
2439: /* error */
2440: break;
2441: }
2442: SKIP_BLANKS;
2443: cur = xmlParseSGMLCatalogPubid(cur, &sysid);
2444: if (cur == NULL) {
2445: /* error */
2446: break;
2447: }
2448: break;
2449: case SGML_CATA_BASE:
2450: case SGML_CATA_CATALOG:
2451: case SGML_CATA_DOCUMENT:
2452: case SGML_CATA_SGMLDECL:
2453: cur = xmlParseSGMLCatalogPubid(cur, &sysid);
2454: if (cur == NULL) {
2455: /* error */
2456: break;
2457: }
2458: break;
2459: default:
2460: break;
2461: }
2462: if (cur == NULL) {
2463: if (name != NULL)
2464: xmlFree(name);
2465: if (sysid != NULL)
2466: xmlFree(sysid);
2467: break;
2468: } else if (type == SGML_CATA_BASE) {
2469: if (base != NULL)
2470: xmlFree(base);
2471: base = xmlStrdup(sysid);
2472: } else if ((type == SGML_CATA_PUBLIC) ||
2473: (type == SGML_CATA_SYSTEM)) {
2474: xmlChar *filename;
2475:
2476: filename = xmlBuildURI(sysid, base);
2477: if (filename != NULL) {
2478: xmlCatalogEntryPtr entry;
2479:
2480: entry = xmlNewCatalogEntry(type, name, filename,
2481: NULL, XML_CATA_PREFER_NONE, NULL);
2482: res = xmlHashAddEntry(catal->sgml, name, entry);
2483: if (res < 0) {
2484: xmlFreeCatalogEntry(entry);
2485: }
2486: xmlFree(filename);
2487: }
2488:
2489: } else if (type == SGML_CATA_CATALOG) {
2490: if (super) {
2491: xmlCatalogEntryPtr entry;
2492:
2493: entry = xmlNewCatalogEntry(type, sysid, NULL, NULL,
2494: XML_CATA_PREFER_NONE, NULL);
2495: res = xmlHashAddEntry(catal->sgml, sysid, entry);
2496: if (res < 0) {
2497: xmlFreeCatalogEntry(entry);
2498: }
2499: } else {
2500: xmlChar *filename;
2501:
2502: filename = xmlBuildURI(sysid, base);
2503: if (filename != NULL) {
2504: xmlExpandCatalog(catal, (const char *)filename);
2505: xmlFree(filename);
2506: }
2507: }
2508: }
2509: /*
2510: * drop anything else we won't handle it
2511: */
2512: if (name != NULL)
2513: xmlFree(name);
2514: if (sysid != NULL)
2515: xmlFree(sysid);
2516: }
2517: }
2518: if (base != NULL)
2519: xmlFree(base);
2520: if (cur == NULL)
2521: return(-1);
2522: return(0);
2523: }
2524:
2525: /************************************************************************
2526: * *
2527: * SGML Catalog handling *
2528: * *
2529: ************************************************************************/
2530:
2531: /**
2532: * xmlCatalogGetSGMLPublic:
2533: * @catal: an SGML catalog hash
2534: * @pubID: the public ID string
2535: *
2536: * Try to lookup the catalog local reference associated to a public ID
2537: *
2538: * Returns the local resource if found or NULL otherwise.
2539: */
2540: static const xmlChar *
2541: xmlCatalogGetSGMLPublic(xmlHashTablePtr catal, const xmlChar *pubID) {
2542: xmlCatalogEntryPtr entry;
2543: xmlChar *normid;
2544:
2545: if (catal == NULL)
2546: return(NULL);
2547:
2548: normid = xmlCatalogNormalizePublic(pubID);
2549: if (normid != NULL)
2550: pubID = (*normid != 0 ? normid : NULL);
2551:
2552: entry = (xmlCatalogEntryPtr) xmlHashLookup(catal, pubID);
2553: if (entry == NULL) {
2554: if (normid != NULL)
2555: xmlFree(normid);
2556: return(NULL);
2557: }
2558: if (entry->type == SGML_CATA_PUBLIC) {
2559: if (normid != NULL)
2560: xmlFree(normid);
2561: return(entry->URL);
2562: }
2563: if (normid != NULL)
2564: xmlFree(normid);
2565: return(NULL);
2566: }
2567:
2568: /**
2569: * xmlCatalogGetSGMLSystem:
2570: * @catal: an SGML catalog hash
2571: * @sysID: the system ID string
2572: *
2573: * Try to lookup the catalog local reference for a system ID
2574: *
2575: * Returns the local resource if found or NULL otherwise.
2576: */
2577: static const xmlChar *
2578: xmlCatalogGetSGMLSystem(xmlHashTablePtr catal, const xmlChar *sysID) {
2579: xmlCatalogEntryPtr entry;
2580:
2581: if (catal == NULL)
2582: return(NULL);
2583:
2584: entry = (xmlCatalogEntryPtr) xmlHashLookup(catal, sysID);
2585: if (entry == NULL)
2586: return(NULL);
2587: if (entry->type == SGML_CATA_SYSTEM)
2588: return(entry->URL);
2589: return(NULL);
2590: }
2591:
2592: /**
2593: * xmlCatalogSGMLResolve:
2594: * @catal: the SGML catalog
2595: * @pubID: the public ID string
2596: * @sysID: the system ID string
2597: *
2598: * Do a complete resolution lookup of an External Identifier
2599: *
2600: * Returns the URI of the resource or NULL if not found
2601: */
2602: static const xmlChar *
2603: xmlCatalogSGMLResolve(xmlCatalogPtr catal, const xmlChar *pubID,
2604: const xmlChar *sysID) {
2605: const xmlChar *ret = NULL;
2606:
2607: if (catal->sgml == NULL)
2608: return(NULL);
2609:
2610: if (pubID != NULL)
2611: ret = xmlCatalogGetSGMLPublic(catal->sgml, pubID);
2612: if (ret != NULL)
2613: return(ret);
2614: if (sysID != NULL)
2615: ret = xmlCatalogGetSGMLSystem(catal->sgml, sysID);
2616: if (ret != NULL)
2617: return(ret);
2618: return(NULL);
2619: }
2620:
2621: /************************************************************************
2622: * *
2623: * Specific Public interfaces *
2624: * *
2625: ************************************************************************/
2626:
2627: /**
2628: * xmlLoadSGMLSuperCatalog:
2629: * @filename: a file path
2630: *
2631: * Load an SGML super catalog. It won't expand CATALOG or DELEGATE
2632: * references. This is only needed for manipulating SGML Super Catalogs
2633: * like adding and removing CATALOG or DELEGATE entries.
2634: *
2635: * Returns the catalog parsed or NULL in case of error
2636: */
2637: xmlCatalogPtr
2638: xmlLoadSGMLSuperCatalog(const char *filename)
2639: {
2640: xmlChar *content;
2641: xmlCatalogPtr catal;
2642: int ret;
2643:
2644: content = xmlLoadFileContent(filename);
2645: if (content == NULL)
2646: return(NULL);
2647:
2648: catal = xmlCreateNewCatalog(XML_SGML_CATALOG_TYPE, xmlCatalogDefaultPrefer);
2649: if (catal == NULL) {
2650: xmlFree(content);
2651: return(NULL);
2652: }
2653:
2654: ret = xmlParseSGMLCatalog(catal, content, filename, 1);
2655: xmlFree(content);
2656: if (ret < 0) {
2657: xmlFreeCatalog(catal);
2658: return(NULL);
2659: }
2660: return (catal);
2661: }
2662:
2663: /**
2664: * xmlLoadACatalog:
2665: * @filename: a file path
2666: *
2667: * Load the catalog and build the associated data structures.
2668: * This can be either an XML Catalog or an SGML Catalog
2669: * It will recurse in SGML CATALOG entries. On the other hand XML
2670: * Catalogs are not handled recursively.
2671: *
2672: * Returns the catalog parsed or NULL in case of error
2673: */
2674: xmlCatalogPtr
2675: xmlLoadACatalog(const char *filename)
2676: {
2677: xmlChar *content;
2678: xmlChar *first;
2679: xmlCatalogPtr catal;
2680: int ret;
2681:
2682: content = xmlLoadFileContent(filename);
2683: if (content == NULL)
2684: return(NULL);
2685:
2686:
2687: first = content;
2688:
2689: while ((*first != 0) && (*first != '-') && (*first != '<') &&
2690: (!(((*first >= 'A') && (*first <= 'Z')) ||
2691: ((*first >= 'a') && (*first <= 'z')))))
2692: first++;
2693:
2694: if (*first != '<') {
2695: catal = xmlCreateNewCatalog(XML_SGML_CATALOG_TYPE, xmlCatalogDefaultPrefer);
2696: if (catal == NULL) {
2697: xmlFree(content);
2698: return(NULL);
2699: }
2700: ret = xmlParseSGMLCatalog(catal, content, filename, 0);
2701: if (ret < 0) {
2702: xmlFreeCatalog(catal);
2703: xmlFree(content);
2704: return(NULL);
2705: }
2706: } else {
2707: catal = xmlCreateNewCatalog(XML_XML_CATALOG_TYPE, xmlCatalogDefaultPrefer);
2708: if (catal == NULL) {
2709: xmlFree(content);
2710: return(NULL);
2711: }
2712: catal->xml = xmlNewCatalogEntry(XML_CATA_CATALOG, NULL,
2713: NULL, BAD_CAST filename, xmlCatalogDefaultPrefer, NULL);
2714: }
2715: xmlFree(content);
2716: return (catal);
2717: }
2718:
2719: /**
2720: * xmlExpandCatalog:
2721: * @catal: a catalog
2722: * @filename: a file path
2723: *
2724: * Load the catalog and expand the existing catal structure.
2725: * This can be either an XML Catalog or an SGML Catalog
2726: *
2727: * Returns 0 in case of success, -1 in case of error
2728: */
2729: static int
2730: xmlExpandCatalog(xmlCatalogPtr catal, const char *filename)
2731: {
2732: int ret;
2733:
2734: if ((catal == NULL) || (filename == NULL))
2735: return(-1);
2736:
2737:
2738: if (catal->type == XML_SGML_CATALOG_TYPE) {
2739: xmlChar *content;
2740:
2741: content = xmlLoadFileContent(filename);
2742: if (content == NULL)
2743: return(-1);
2744:
2745: ret = xmlParseSGMLCatalog(catal, content, filename, 0);
2746: if (ret < 0) {
2747: xmlFree(content);
2748: return(-1);
2749: }
2750: xmlFree(content);
2751: } else {
2752: xmlCatalogEntryPtr tmp, cur;
2753: tmp = xmlNewCatalogEntry(XML_CATA_CATALOG, NULL,
2754: NULL, BAD_CAST filename, xmlCatalogDefaultPrefer, NULL);
2755:
2756: cur = catal->xml;
2757: if (cur == NULL) {
2758: catal->xml = tmp;
2759: } else {
2760: while (cur->next != NULL) cur = cur->next;
2761: cur->next = tmp;
2762: }
2763: }
2764: return (0);
2765: }
2766:
2767: /**
2768: * xmlACatalogResolveSystem:
2769: * @catal: a Catalog
2770: * @sysID: the system ID string
2771: *
2772: * Try to lookup the catalog resource for a system ID
2773: *
2774: * Returns the resource if found or NULL otherwise, the value returned
2775: * must be freed by the caller.
2776: */
2777: xmlChar *
2778: xmlACatalogResolveSystem(xmlCatalogPtr catal, const xmlChar *sysID) {
2779: xmlChar *ret = NULL;
2780:
2781: if ((sysID == NULL) || (catal == NULL))
2782: return(NULL);
2783:
2784: if (xmlDebugCatalogs)
2785: xmlGenericError(xmlGenericErrorContext,
2786: "Resolve sysID %s\n", sysID);
2787:
2788: if (catal->type == XML_XML_CATALOG_TYPE) {
2789: ret = xmlCatalogListXMLResolve(catal->xml, NULL, sysID);
2790: if (ret == XML_CATAL_BREAK)
2791: ret = NULL;
2792: } else {
2793: const xmlChar *sgml;
2794:
2795: sgml = xmlCatalogGetSGMLSystem(catal->sgml, sysID);
2796: if (sgml != NULL)
2797: ret = xmlStrdup(sgml);
2798: }
2799: return(ret);
2800: }
2801:
2802: /**
2803: * xmlACatalogResolvePublic:
2804: * @catal: a Catalog
2805: * @pubID: the public ID string
2806: *
2807: * Try to lookup the catalog local reference associated to a public ID in that catalog
2808: *
2809: * Returns the local resource if found or NULL otherwise, the value returned
2810: * must be freed by the caller.
2811: */
2812: xmlChar *
2813: xmlACatalogResolvePublic(xmlCatalogPtr catal, const xmlChar *pubID) {
2814: xmlChar *ret = NULL;
2815:
2816: if ((pubID == NULL) || (catal == NULL))
2817: return(NULL);
2818:
2819: if (xmlDebugCatalogs)
2820: xmlGenericError(xmlGenericErrorContext,
2821: "Resolve pubID %s\n", pubID);
2822:
2823: if (catal->type == XML_XML_CATALOG_TYPE) {
2824: ret = xmlCatalogListXMLResolve(catal->xml, pubID, NULL);
2825: if (ret == XML_CATAL_BREAK)
2826: ret = NULL;
2827: } else {
2828: const xmlChar *sgml;
2829:
2830: sgml = xmlCatalogGetSGMLPublic(catal->sgml, pubID);
2831: if (sgml != NULL)
2832: ret = xmlStrdup(sgml);
2833: }
2834: return(ret);
2835: }
2836:
2837: /**
2838: * xmlACatalogResolve:
2839: * @catal: a Catalog
2840: * @pubID: the public ID string
2841: * @sysID: the system ID string
2842: *
2843: * Do a complete resolution lookup of an External Identifier
2844: *
2845: * Returns the URI of the resource or NULL if not found, it must be freed
2846: * by the caller.
2847: */
2848: xmlChar *
2849: xmlACatalogResolve(xmlCatalogPtr catal, const xmlChar * pubID,
2850: const xmlChar * sysID)
2851: {
2852: xmlChar *ret = NULL;
2853:
2854: if (((pubID == NULL) && (sysID == NULL)) || (catal == NULL))
2855: return (NULL);
2856:
2857: if (xmlDebugCatalogs) {
2858: if ((pubID != NULL) && (sysID != NULL)) {
2859: xmlGenericError(xmlGenericErrorContext,
2860: "Resolve: pubID %s sysID %s\n", pubID, sysID);
2861: } else if (pubID != NULL) {
2862: xmlGenericError(xmlGenericErrorContext,
2863: "Resolve: pubID %s\n", pubID);
2864: } else {
2865: xmlGenericError(xmlGenericErrorContext,
2866: "Resolve: sysID %s\n", sysID);
2867: }
2868: }
2869:
2870: if (catal->type == XML_XML_CATALOG_TYPE) {
2871: ret = xmlCatalogListXMLResolve(catal->xml, pubID, sysID);
2872: if (ret == XML_CATAL_BREAK)
2873: ret = NULL;
2874: } else {
2875: const xmlChar *sgml;
2876:
2877: sgml = xmlCatalogSGMLResolve(catal, pubID, sysID);
2878: if (sgml != NULL)
2879: ret = xmlStrdup(sgml);
2880: }
2881: return (ret);
2882: }
2883:
2884: /**
2885: * xmlACatalogResolveURI:
2886: * @catal: a Catalog
2887: * @URI: the URI
2888: *
2889: * Do a complete resolution lookup of an URI
2890: *
2891: * Returns the URI of the resource or NULL if not found, it must be freed
2892: * by the caller.
2893: */
2894: xmlChar *
2895: xmlACatalogResolveURI(xmlCatalogPtr catal, const xmlChar *URI) {
2896: xmlChar *ret = NULL;
2897:
2898: if ((URI == NULL) || (catal == NULL))
2899: return(NULL);
2900:
2901: if (xmlDebugCatalogs)
2902: xmlGenericError(xmlGenericErrorContext,
2903: "Resolve URI %s\n", URI);
2904:
2905: if (catal->type == XML_XML_CATALOG_TYPE) {
2906: ret = xmlCatalogListXMLResolveURI(catal->xml, URI);
2907: if (ret == XML_CATAL_BREAK)
2908: ret = NULL;
2909: } else {
2910: const xmlChar *sgml;
2911:
2912: sgml = xmlCatalogSGMLResolve(catal, NULL, URI);
2913: if (sgml != NULL)
2914: ret = xmlStrdup(sgml);
2915: }
2916: return(ret);
2917: }
2918:
2919: #ifdef LIBXML_OUTPUT_ENABLED
2920: /**
2921: * xmlACatalogDump:
2922: * @catal: a Catalog
2923: * @out: the file.
2924: *
2925: * Dump the given catalog to the given file.
2926: */
2927: void
2928: xmlACatalogDump(xmlCatalogPtr catal, FILE *out) {
2929: if ((out == NULL) || (catal == NULL))
2930: return;
2931:
2932: if (catal->type == XML_XML_CATALOG_TYPE) {
2933: xmlDumpXMLCatalog(out, catal->xml);
2934: } else {
2935: xmlHashScan(catal->sgml,
2936: (xmlHashScanner) xmlCatalogDumpEntry, out);
2937: }
2938: }
2939: #endif /* LIBXML_OUTPUT_ENABLED */
2940:
2941: /**
2942: * xmlACatalogAdd:
2943: * @catal: a Catalog
2944: * @type: the type of record to add to the catalog
2945: * @orig: the system, public or prefix to match
2946: * @replace: the replacement value for the match
2947: *
2948: * Add an entry in the catalog, it may overwrite existing but
2949: * different entries.
2950: *
2951: * Returns 0 if successful, -1 otherwise
2952: */
2953: int
2954: xmlACatalogAdd(xmlCatalogPtr catal, const xmlChar * type,
2955: const xmlChar * orig, const xmlChar * replace)
2956: {
2957: int res = -1;
2958:
2959: if (catal == NULL)
2960: return(-1);
2961:
2962: if (catal->type == XML_XML_CATALOG_TYPE) {
2963: res = xmlAddXMLCatalog(catal->xml, type, orig, replace);
2964: } else {
2965: xmlCatalogEntryType cattype;
2966:
2967: cattype = xmlGetSGMLCatalogEntryType(type);
2968: if (cattype != XML_CATA_NONE) {
2969: xmlCatalogEntryPtr entry;
2970:
2971: entry = xmlNewCatalogEntry(cattype, orig, replace, NULL,
2972: XML_CATA_PREFER_NONE, NULL);
2973: if (catal->sgml == NULL)
2974: catal->sgml = xmlHashCreate(10);
2975: res = xmlHashAddEntry(catal->sgml, orig, entry);
2976: }
2977: }
2978: return (res);
2979: }
2980:
2981: /**
2982: * xmlACatalogRemove:
2983: * @catal: a Catalog
2984: * @value: the value to remove
2985: *
2986: * Remove an entry from the catalog
2987: *
2988: * Returns the number of entries removed if successful, -1 otherwise
2989: */
2990: int
2991: xmlACatalogRemove(xmlCatalogPtr catal, const xmlChar *value) {
2992: int res = -1;
2993:
2994: if ((catal == NULL) || (value == NULL))
2995: return(-1);
2996:
2997: if (catal->type == XML_XML_CATALOG_TYPE) {
2998: res = xmlDelXMLCatalog(catal->xml, value);
2999: } else {
3000: res = xmlHashRemoveEntry(catal->sgml, value,
3001: (xmlHashDeallocator) xmlFreeCatalogEntry);
3002: if (res == 0)
3003: res = 1;
3004: }
3005: return(res);
3006: }
3007:
3008: /**
3009: * xmlNewCatalog:
3010: * @sgml: should this create an SGML catalog
3011: *
3012: * create a new Catalog.
3013: *
3014: * Returns the xmlCatalogPtr or NULL in case of error
3015: */
3016: xmlCatalogPtr
3017: xmlNewCatalog(int sgml) {
3018: xmlCatalogPtr catal = NULL;
3019:
3020: if (sgml) {
3021: catal = xmlCreateNewCatalog(XML_SGML_CATALOG_TYPE,
3022: xmlCatalogDefaultPrefer);
3023: if ((catal != NULL) && (catal->sgml == NULL))
3024: catal->sgml = xmlHashCreate(10);
3025: } else
3026: catal = xmlCreateNewCatalog(XML_XML_CATALOG_TYPE,
3027: xmlCatalogDefaultPrefer);
3028: return(catal);
3029: }
3030:
3031: /**
3032: * xmlCatalogIsEmpty:
3033: * @catal: should this create an SGML catalog
3034: *
3035: * Check is a catalog is empty
3036: *
3037: * Returns 1 if the catalog is empty, 0 if not, amd -1 in case of error.
3038: */
3039: int
3040: xmlCatalogIsEmpty(xmlCatalogPtr catal) {
3041: if (catal == NULL)
3042: return(-1);
3043:
3044: if (catal->type == XML_XML_CATALOG_TYPE) {
3045: if (catal->xml == NULL)
3046: return(1);
3047: if ((catal->xml->type != XML_CATA_CATALOG) &&
3048: (catal->xml->type != XML_CATA_BROKEN_CATALOG))
3049: return(-1);
3050: if (catal->xml->children == NULL)
3051: return(1);
3052: return(0);
3053: } else {
3054: int res;
3055:
3056: if (catal->sgml == NULL)
3057: return(1);
3058: res = xmlHashSize(catal->sgml);
3059: if (res == 0)
3060: return(1);
3061: if (res < 0)
3062: return(-1);
3063: }
3064: return(0);
3065: }
3066:
3067: /************************************************************************
3068: * *
3069: * Public interfaces manipulating the global shared default catalog *
3070: * *
3071: ************************************************************************/
3072:
3073: /**
3074: * xmlInitializeCatalogData:
3075: *
3076: * Do the catalog initialization only of global data, doesn't try to load
3077: * any catalog actually.
3078: * this function is not thread safe, catalog initialization should
3079: * preferably be done once at startup
3080: */
3081: static void
3082: xmlInitializeCatalogData(void) {
3083: if (xmlCatalogInitialized != 0)
3084: return;
3085:
3086: if (getenv("XML_DEBUG_CATALOG"))
3087: xmlDebugCatalogs = 1;
3088: xmlCatalogMutex = xmlNewRMutex();
3089:
3090: xmlCatalogInitialized = 1;
3091: }
3092: /**
3093: * xmlInitializeCatalog:
3094: *
3095: * Do the catalog initialization.
3096: * this function is not thread safe, catalog initialization should
3097: * preferably be done once at startup
3098: */
3099: void
3100: xmlInitializeCatalog(void) {
3101: if (xmlCatalogInitialized != 0)
3102: return;
3103:
3104: xmlInitializeCatalogData();
3105: xmlRMutexLock(xmlCatalogMutex);
3106:
3107: if (getenv("XML_DEBUG_CATALOG"))
3108: xmlDebugCatalogs = 1;
3109:
3110: if (xmlDefaultCatalog == NULL) {
3111: const char *catalogs;
3112: char *path;
3113: const char *cur, *paths;
3114: xmlCatalogPtr catal;
3115: xmlCatalogEntryPtr *nextent;
3116:
3117: catalogs = (const char *) getenv("XML_CATALOG_FILES");
3118: if (catalogs == NULL)
3119: #if defined(_WIN32) && defined(_MSC_VER)
3120: {
3121: void* hmodule;
3122: hmodule = GetModuleHandleA("libxml2.dll");
3123: if (hmodule == NULL)
3124: hmodule = GetModuleHandleA(NULL);
3125: if (hmodule != NULL) {
3126: char buf[256];
3127: unsigned long len = GetModuleFileNameA(hmodule, buf, 255);
3128: if (len != 0) {
3129: char* p = &(buf[len]);
3130: while (*p != '\\' && p > buf)
3131: p--;
3132: if (p != buf) {
3133: xmlChar* uri;
3134: strncpy(p, "\\..\\etc\\catalog", 255 - (p - buf));
3135: uri = xmlCanonicPath(buf);
3136: if (uri != NULL) {
3137: strncpy(XML_XML_DEFAULT_CATALOG, uri, 255);
3138: xmlFree(uri);
3139: }
3140: }
3141: }
3142: }
3143: catalogs = XML_XML_DEFAULT_CATALOG;
3144: }
3145: #else
3146: catalogs = XML_XML_DEFAULT_CATALOG;
3147: #endif
3148:
3149: catal = xmlCreateNewCatalog(XML_XML_CATALOG_TYPE,
3150: xmlCatalogDefaultPrefer);
3151: if (catal != NULL) {
3152: /* the XML_CATALOG_FILES envvar is allowed to contain a
3153: space-separated list of entries. */
3154: cur = catalogs;
3155: nextent = &catal->xml;
3156: while (*cur != '\0') {
3157: while (xmlIsBlank_ch(*cur))
3158: cur++;
3159: if (*cur != 0) {
3160: paths = cur;
3161: while ((*cur != 0) && (!xmlIsBlank_ch(*cur)))
3162: cur++;
3163: path = (char *) xmlStrndup((const xmlChar *)paths, cur - paths);
3164: if (path != NULL) {
3165: *nextent = xmlNewCatalogEntry(XML_CATA_CATALOG, NULL,
3166: NULL, BAD_CAST path, xmlCatalogDefaultPrefer, NULL);
3167: if (*nextent != NULL)
3168: nextent = &((*nextent)->next);
3169: xmlFree(path);
3170: }
3171: }
3172: }
3173: xmlDefaultCatalog = catal;
3174: }
3175: }
3176:
3177: xmlRMutexUnlock(xmlCatalogMutex);
3178: }
3179:
3180:
3181: /**
3182: * xmlLoadCatalog:
3183: * @filename: a file path
3184: *
3185: * Load the catalog and makes its definitions effective for the default
3186: * external entity loader. It will recurse in SGML CATALOG entries.
3187: * this function is not thread safe, catalog initialization should
3188: * preferably be done once at startup
3189: *
3190: * Returns 0 in case of success -1 in case of error
3191: */
3192: int
3193: xmlLoadCatalog(const char *filename)
3194: {
3195: int ret;
3196: xmlCatalogPtr catal;
3197:
3198: if (!xmlCatalogInitialized)
3199: xmlInitializeCatalogData();
3200:
3201: xmlRMutexLock(xmlCatalogMutex);
3202:
3203: if (xmlDefaultCatalog == NULL) {
3204: catal = xmlLoadACatalog(filename);
3205: if (catal == NULL) {
3206: xmlRMutexUnlock(xmlCatalogMutex);
3207: return(-1);
3208: }
3209:
3210: xmlDefaultCatalog = catal;
3211: xmlRMutexUnlock(xmlCatalogMutex);
3212: return(0);
3213: }
3214:
3215: ret = xmlExpandCatalog(xmlDefaultCatalog, filename);
3216: xmlRMutexUnlock(xmlCatalogMutex);
3217: return(ret);
3218: }
3219:
3220: /**
3221: * xmlLoadCatalogs:
3222: * @pathss: a list of directories separated by a colon or a space.
3223: *
3224: * Load the catalogs and makes their definitions effective for the default
3225: * external entity loader.
3226: * this function is not thread safe, catalog initialization should
3227: * preferably be done once at startup
3228: */
3229: void
3230: xmlLoadCatalogs(const char *pathss) {
3231: const char *cur;
3232: const char *paths;
3233: xmlChar *path;
3234: #ifdef _WIN32
3235: int i, iLen;
3236: #endif
3237:
3238: if (pathss == NULL)
3239: return;
3240:
3241: cur = pathss;
3242: while (*cur != 0) {
3243: while (xmlIsBlank_ch(*cur)) cur++;
3244: if (*cur != 0) {
3245: paths = cur;
3246: while ((*cur != 0) && (*cur != PATH_SEAPARATOR) && (!xmlIsBlank_ch(*cur)))
3247: cur++;
3248: path = xmlStrndup((const xmlChar *)paths, cur - paths);
3249: #ifdef _WIN32
3250: iLen = strlen(path);
3251: for(i = 0; i < iLen; i++) {
3252: if(path[i] == '\\') {
3253: path[i] = '/';
3254: }
3255: }
3256: #endif
3257: if (path != NULL) {
3258: xmlLoadCatalog((const char *) path);
3259: xmlFree(path);
3260: }
3261: }
3262: while (*cur == PATH_SEAPARATOR)
3263: cur++;
3264: }
3265: }
3266:
3267: /**
3268: * xmlCatalogCleanup:
3269: *
3270: * Free up all the memory associated with catalogs
3271: */
3272: void
3273: xmlCatalogCleanup(void) {
3274: if (xmlCatalogInitialized == 0)
3275: return;
3276:
3277: xmlRMutexLock(xmlCatalogMutex);
3278: if (xmlDebugCatalogs)
3279: xmlGenericError(xmlGenericErrorContext,
3280: "Catalogs cleanup\n");
3281: if (xmlCatalogXMLFiles != NULL)
3282: xmlHashFree(xmlCatalogXMLFiles,
3283: (xmlHashDeallocator)xmlFreeCatalogHashEntryList);
3284: xmlCatalogXMLFiles = NULL;
3285: if (xmlDefaultCatalog != NULL)
3286: xmlFreeCatalog(xmlDefaultCatalog);
3287: xmlDefaultCatalog = NULL;
3288: xmlDebugCatalogs = 0;
3289: xmlCatalogInitialized = 0;
3290: xmlRMutexUnlock(xmlCatalogMutex);
3291: xmlFreeRMutex(xmlCatalogMutex);
3292: }
3293:
3294: /**
3295: * xmlCatalogResolveSystem:
3296: * @sysID: the system ID string
3297: *
3298: * Try to lookup the catalog resource for a system ID
3299: *
3300: * Returns the resource if found or NULL otherwise, the value returned
3301: * must be freed by the caller.
3302: */
3303: xmlChar *
3304: xmlCatalogResolveSystem(const xmlChar *sysID) {
3305: xmlChar *ret;
3306:
3307: if (!xmlCatalogInitialized)
3308: xmlInitializeCatalog();
3309:
3310: ret = xmlACatalogResolveSystem(xmlDefaultCatalog, sysID);
3311: return(ret);
3312: }
3313:
3314: /**
3315: * xmlCatalogResolvePublic:
3316: * @pubID: the public ID string
3317: *
3318: * Try to lookup the catalog reference associated to a public ID
3319: *
3320: * Returns the resource if found or NULL otherwise, the value returned
3321: * must be freed by the caller.
3322: */
3323: xmlChar *
3324: xmlCatalogResolvePublic(const xmlChar *pubID) {
3325: xmlChar *ret;
3326:
3327: if (!xmlCatalogInitialized)
3328: xmlInitializeCatalog();
3329:
3330: ret = xmlACatalogResolvePublic(xmlDefaultCatalog, pubID);
3331: return(ret);
3332: }
3333:
3334: /**
3335: * xmlCatalogResolve:
3336: * @pubID: the public ID string
3337: * @sysID: the system ID string
3338: *
3339: * Do a complete resolution lookup of an External Identifier
3340: *
3341: * Returns the URI of the resource or NULL if not found, it must be freed
3342: * by the caller.
3343: */
3344: xmlChar *
3345: xmlCatalogResolve(const xmlChar *pubID, const xmlChar *sysID) {
3346: xmlChar *ret;
3347:
3348: if (!xmlCatalogInitialized)
3349: xmlInitializeCatalog();
3350:
3351: ret = xmlACatalogResolve(xmlDefaultCatalog, pubID, sysID);
3352: return(ret);
3353: }
3354:
3355: /**
3356: * xmlCatalogResolveURI:
3357: * @URI: the URI
3358: *
3359: * Do a complete resolution lookup of an URI
3360: *
3361: * Returns the URI of the resource or NULL if not found, it must be freed
3362: * by the caller.
3363: */
3364: xmlChar *
3365: xmlCatalogResolveURI(const xmlChar *URI) {
3366: xmlChar *ret;
3367:
3368: if (!xmlCatalogInitialized)
3369: xmlInitializeCatalog();
3370:
3371: ret = xmlACatalogResolveURI(xmlDefaultCatalog, URI);
3372: return(ret);
3373: }
3374:
3375: #ifdef LIBXML_OUTPUT_ENABLED
3376: /**
3377: * xmlCatalogDump:
3378: * @out: the file.
3379: *
3380: * Dump all the global catalog content to the given file.
3381: */
3382: void
3383: xmlCatalogDump(FILE *out) {
3384: if (out == NULL)
3385: return;
3386:
3387: if (!xmlCatalogInitialized)
3388: xmlInitializeCatalog();
3389:
3390: xmlACatalogDump(xmlDefaultCatalog, out);
3391: }
3392: #endif /* LIBXML_OUTPUT_ENABLED */
3393:
3394: /**
3395: * xmlCatalogAdd:
3396: * @type: the type of record to add to the catalog
3397: * @orig: the system, public or prefix to match
3398: * @replace: the replacement value for the match
3399: *
3400: * Add an entry in the catalog, it may overwrite existing but
3401: * different entries.
3402: * If called before any other catalog routine, allows to override the
3403: * default shared catalog put in place by xmlInitializeCatalog();
3404: *
3405: * Returns 0 if successful, -1 otherwise
3406: */
3407: int
3408: xmlCatalogAdd(const xmlChar *type, const xmlChar *orig, const xmlChar *replace) {
3409: int res = -1;
3410:
3411: if (!xmlCatalogInitialized)
3412: xmlInitializeCatalogData();
3413:
3414: xmlRMutexLock(xmlCatalogMutex);
3415: /*
3416: * Specific case where one want to override the default catalog
3417: * put in place by xmlInitializeCatalog();
3418: */
3419: if ((xmlDefaultCatalog == NULL) &&
3420: (xmlStrEqual(type, BAD_CAST "catalog"))) {
3421: xmlDefaultCatalog = xmlCreateNewCatalog(XML_XML_CATALOG_TYPE,
3422: xmlCatalogDefaultPrefer);
3423: xmlDefaultCatalog->xml = xmlNewCatalogEntry(XML_CATA_CATALOG, NULL,
3424: orig, NULL, xmlCatalogDefaultPrefer, NULL);
3425:
3426: xmlRMutexUnlock(xmlCatalogMutex);
3427: return(0);
3428: }
3429:
3430: res = xmlACatalogAdd(xmlDefaultCatalog, type, orig, replace);
3431: xmlRMutexUnlock(xmlCatalogMutex);
3432: return(res);
3433: }
3434:
3435: /**
3436: * xmlCatalogRemove:
3437: * @value: the value to remove
3438: *
3439: * Remove an entry from the catalog
3440: *
3441: * Returns the number of entries removed if successful, -1 otherwise
3442: */
3443: int
3444: xmlCatalogRemove(const xmlChar *value) {
3445: int res;
3446:
3447: if (!xmlCatalogInitialized)
3448: xmlInitializeCatalog();
3449:
3450: xmlRMutexLock(xmlCatalogMutex);
3451: res = xmlACatalogRemove(xmlDefaultCatalog, value);
3452: xmlRMutexUnlock(xmlCatalogMutex);
3453: return(res);
3454: }
3455:
3456: /**
3457: * xmlCatalogConvert:
3458: *
3459: * Convert all the SGML catalog entries as XML ones
3460: *
3461: * Returns the number of entries converted if successful, -1 otherwise
3462: */
3463: int
3464: xmlCatalogConvert(void) {
3465: int res = -1;
3466:
3467: if (!xmlCatalogInitialized)
3468: xmlInitializeCatalog();
3469:
3470: xmlRMutexLock(xmlCatalogMutex);
3471: res = xmlConvertSGMLCatalog(xmlDefaultCatalog);
3472: xmlRMutexUnlock(xmlCatalogMutex);
3473: return(res);
3474: }
3475:
3476: /************************************************************************
3477: * *
3478: * Public interface manipulating the common preferences *
3479: * *
3480: ************************************************************************/
3481:
3482: /**
3483: * xmlCatalogGetDefaults:
3484: *
3485: * Used to get the user preference w.r.t. to what catalogs should
3486: * be accepted
3487: *
3488: * Returns the current xmlCatalogAllow value
3489: */
3490: xmlCatalogAllow
3491: xmlCatalogGetDefaults(void) {
3492: return(xmlCatalogDefaultAllow);
3493: }
3494:
3495: /**
3496: * xmlCatalogSetDefaults:
3497: * @allow: what catalogs should be accepted
3498: *
3499: * Used to set the user preference w.r.t. to what catalogs should
3500: * be accepted
3501: */
3502: void
3503: xmlCatalogSetDefaults(xmlCatalogAllow allow) {
3504: if (xmlDebugCatalogs) {
3505: switch (allow) {
3506: case XML_CATA_ALLOW_NONE:
3507: xmlGenericError(xmlGenericErrorContext,
3508: "Disabling catalog usage\n");
3509: break;
3510: case XML_CATA_ALLOW_GLOBAL:
3511: xmlGenericError(xmlGenericErrorContext,
3512: "Allowing only global catalogs\n");
3513: break;
3514: case XML_CATA_ALLOW_DOCUMENT:
3515: xmlGenericError(xmlGenericErrorContext,
3516: "Allowing only catalogs from the document\n");
3517: break;
3518: case XML_CATA_ALLOW_ALL:
3519: xmlGenericError(xmlGenericErrorContext,
3520: "Allowing all catalogs\n");
3521: break;
3522: }
3523: }
3524: xmlCatalogDefaultAllow = allow;
3525: }
3526:
3527: /**
3528: * xmlCatalogSetDefaultPrefer:
3529: * @prefer: the default preference for delegation
3530: *
3531: * Allows to set the preference between public and system for deletion
3532: * in XML Catalog resolution. C.f. section 4.1.1 of the spec
3533: * Values accepted are XML_CATA_PREFER_PUBLIC or XML_CATA_PREFER_SYSTEM
3534: *
3535: * Returns the previous value of the default preference for delegation
3536: */
3537: xmlCatalogPrefer
3538: xmlCatalogSetDefaultPrefer(xmlCatalogPrefer prefer) {
3539: xmlCatalogPrefer ret = xmlCatalogDefaultPrefer;
3540:
3541: if (prefer == XML_CATA_PREFER_NONE)
3542: return(ret);
3543:
3544: if (xmlDebugCatalogs) {
3545: switch (prefer) {
3546: case XML_CATA_PREFER_PUBLIC:
3547: xmlGenericError(xmlGenericErrorContext,
3548: "Setting catalog preference to PUBLIC\n");
3549: break;
3550: case XML_CATA_PREFER_SYSTEM:
3551: xmlGenericError(xmlGenericErrorContext,
3552: "Setting catalog preference to SYSTEM\n");
3553: break;
3554: case XML_CATA_PREFER_NONE:
3555: break;
3556: }
3557: }
3558: xmlCatalogDefaultPrefer = prefer;
3559: return(ret);
3560: }
3561:
3562: /**
3563: * xmlCatalogSetDebug:
3564: * @level: the debug level of catalogs required
3565: *
3566: * Used to set the debug level for catalog operation, 0 disable
3567: * debugging, 1 enable it
3568: *
3569: * Returns the previous value of the catalog debugging level
3570: */
3571: int
3572: xmlCatalogSetDebug(int level) {
3573: int ret = xmlDebugCatalogs;
3574:
3575: if (level <= 0)
3576: xmlDebugCatalogs = 0;
3577: else
3578: xmlDebugCatalogs = level;
3579: return(ret);
3580: }
3581:
3582: /************************************************************************
3583: * *
3584: * Minimal interfaces used for per-document catalogs by the parser *
3585: * *
3586: ************************************************************************/
3587:
3588: /**
3589: * xmlCatalogFreeLocal:
3590: * @catalogs: a document's list of catalogs
3591: *
3592: * Free up the memory associated to the catalog list
3593: */
3594: void
3595: xmlCatalogFreeLocal(void *catalogs) {
3596: xmlCatalogEntryPtr catal;
3597:
3598: if (!xmlCatalogInitialized)
3599: xmlInitializeCatalog();
3600:
3601: catal = (xmlCatalogEntryPtr) catalogs;
3602: if (catal != NULL)
3603: xmlFreeCatalogEntryList(catal);
3604: }
3605:
3606:
3607: /**
3608: * xmlCatalogAddLocal:
3609: * @catalogs: a document's list of catalogs
3610: * @URL: the URL to a new local catalog
3611: *
3612: * Add the new entry to the catalog list
3613: *
3614: * Returns the updated list
3615: */
3616: void *
3617: xmlCatalogAddLocal(void *catalogs, const xmlChar *URL) {
3618: xmlCatalogEntryPtr catal, add;
3619:
3620: if (!xmlCatalogInitialized)
3621: xmlInitializeCatalog();
3622:
3623: if (URL == NULL)
3624: return(catalogs);
3625:
3626: if (xmlDebugCatalogs)
3627: xmlGenericError(xmlGenericErrorContext,
3628: "Adding document catalog %s\n", URL);
3629:
3630: add = xmlNewCatalogEntry(XML_CATA_CATALOG, NULL, URL, NULL,
3631: xmlCatalogDefaultPrefer, NULL);
3632: if (add == NULL)
3633: return(catalogs);
3634:
3635: catal = (xmlCatalogEntryPtr) catalogs;
3636: if (catal == NULL)
3637: return((void *) add);
3638:
3639: while (catal->next != NULL)
3640: catal = catal->next;
3641: catal->next = add;
3642: return(catalogs);
3643: }
3644:
3645: /**
3646: * xmlCatalogLocalResolve:
3647: * @catalogs: a document's list of catalogs
3648: * @pubID: the public ID string
3649: * @sysID: the system ID string
3650: *
3651: * Do a complete resolution lookup of an External Identifier using a
3652: * document's private catalog list
3653: *
3654: * Returns the URI of the resource or NULL if not found, it must be freed
3655: * by the caller.
3656: */
3657: xmlChar *
3658: xmlCatalogLocalResolve(void *catalogs, const xmlChar *pubID,
3659: const xmlChar *sysID) {
3660: xmlCatalogEntryPtr catal;
3661: xmlChar *ret;
3662:
3663: if (!xmlCatalogInitialized)
3664: xmlInitializeCatalog();
3665:
3666: if ((pubID == NULL) && (sysID == NULL))
3667: return(NULL);
3668:
3669: if (xmlDebugCatalogs) {
3670: if ((pubID != NULL) && (sysID != NULL)) {
3671: xmlGenericError(xmlGenericErrorContext,
3672: "Local Resolve: pubID %s sysID %s\n", pubID, sysID);
3673: } else if (pubID != NULL) {
3674: xmlGenericError(xmlGenericErrorContext,
3675: "Local Resolve: pubID %s\n", pubID);
3676: } else {
3677: xmlGenericError(xmlGenericErrorContext,
3678: "Local Resolve: sysID %s\n", sysID);
3679: }
3680: }
3681:
3682: catal = (xmlCatalogEntryPtr) catalogs;
3683: if (catal == NULL)
3684: return(NULL);
3685: ret = xmlCatalogListXMLResolve(catal, pubID, sysID);
3686: if ((ret != NULL) && (ret != XML_CATAL_BREAK))
3687: return(ret);
3688: return(NULL);
3689: }
3690:
3691: /**
3692: * xmlCatalogLocalResolveURI:
3693: * @catalogs: a document's list of catalogs
3694: * @URI: the URI
3695: *
3696: * Do a complete resolution lookup of an URI using a
3697: * document's private catalog list
3698: *
3699: * Returns the URI of the resource or NULL if not found, it must be freed
3700: * by the caller.
3701: */
3702: xmlChar *
3703: xmlCatalogLocalResolveURI(void *catalogs, const xmlChar *URI) {
3704: xmlCatalogEntryPtr catal;
3705: xmlChar *ret;
3706:
3707: if (!xmlCatalogInitialized)
3708: xmlInitializeCatalog();
3709:
3710: if (URI == NULL)
3711: return(NULL);
3712:
3713: if (xmlDebugCatalogs)
3714: xmlGenericError(xmlGenericErrorContext,
3715: "Resolve URI %s\n", URI);
3716:
3717: catal = (xmlCatalogEntryPtr) catalogs;
3718: if (catal == NULL)
3719: return(NULL);
3720: ret = xmlCatalogListXMLResolveURI(catal, URI);
3721: if ((ret != NULL) && (ret != XML_CATAL_BREAK))
3722: return(ret);
3723: return(NULL);
3724: }
3725:
3726: /************************************************************************
3727: * *
3728: * Deprecated interfaces *
3729: * *
3730: ************************************************************************/
3731: /**
3732: * xmlCatalogGetSystem:
3733: * @sysID: the system ID string
3734: *
3735: * Try to lookup the catalog reference associated to a system ID
3736: * DEPRECATED, use xmlCatalogResolveSystem()
3737: *
3738: * Returns the resource if found or NULL otherwise.
3739: */
3740: const xmlChar *
3741: xmlCatalogGetSystem(const xmlChar *sysID) {
3742: xmlChar *ret;
3743: static xmlChar result[1000];
3744: static int msg = 0;
3745:
3746: if (!xmlCatalogInitialized)
3747: xmlInitializeCatalog();
3748:
3749: if (msg == 0) {
3750: xmlGenericError(xmlGenericErrorContext,
3751: "Use of deprecated xmlCatalogGetSystem() call\n");
3752: msg++;
3753: }
3754:
3755: if (sysID == NULL)
3756: return(NULL);
3757:
3758: /*
3759: * Check first the XML catalogs
3760: */
3761: if (xmlDefaultCatalog != NULL) {
3762: ret = xmlCatalogListXMLResolve(xmlDefaultCatalog->xml, NULL, sysID);
3763: if ((ret != NULL) && (ret != XML_CATAL_BREAK)) {
3764: snprintf((char *) result, sizeof(result) - 1, "%s", (char *) ret);
3765: result[sizeof(result) - 1] = 0;
3766: return(result);
3767: }
3768: }
3769:
3770: if (xmlDefaultCatalog != NULL)
3771: return(xmlCatalogGetSGMLSystem(xmlDefaultCatalog->sgml, sysID));
3772: return(NULL);
3773: }
3774:
3775: /**
3776: * xmlCatalogGetPublic:
3777: * @pubID: the public ID string
3778: *
3779: * Try to lookup the catalog reference associated to a public ID
3780: * DEPRECATED, use xmlCatalogResolvePublic()
3781: *
3782: * Returns the resource if found or NULL otherwise.
3783: */
3784: const xmlChar *
3785: xmlCatalogGetPublic(const xmlChar *pubID) {
3786: xmlChar *ret;
3787: static xmlChar result[1000];
3788: static int msg = 0;
3789:
3790: if (!xmlCatalogInitialized)
3791: xmlInitializeCatalog();
3792:
3793: if (msg == 0) {
3794: xmlGenericError(xmlGenericErrorContext,
3795: "Use of deprecated xmlCatalogGetPublic() call\n");
3796: msg++;
3797: }
3798:
3799: if (pubID == NULL)
3800: return(NULL);
3801:
3802: /*
3803: * Check first the XML catalogs
3804: */
3805: if (xmlDefaultCatalog != NULL) {
3806: ret = xmlCatalogListXMLResolve(xmlDefaultCatalog->xml, pubID, NULL);
3807: if ((ret != NULL) && (ret != XML_CATAL_BREAK)) {
3808: snprintf((char *) result, sizeof(result) - 1, "%s", (char *) ret);
3809: result[sizeof(result) - 1] = 0;
3810: return(result);
3811: }
3812: }
3813:
3814: if (xmlDefaultCatalog != NULL)
3815: return(xmlCatalogGetSGMLPublic(xmlDefaultCatalog->sgml, pubID));
3816: return(NULL);
3817: }
3818:
3819: #define bottom_catalog
3820: #include "elfgcchack.h"
3821: #endif /* LIBXML_CATALOG_ENABLED */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>