Coverage Report

Created: 2026-05-30 06:20

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libxml2/xinclude.c
Line
Count
Source
1
/*
2
 * xinclude.c : Code to implement XInclude processing
3
 *
4
 * World Wide Web Consortium W3C Last Call Working Draft 10 November 2003
5
 * http://www.w3.org/TR/2003/WD-xinclude-20031110
6
 *
7
 * See Copyright for the status of this software.
8
 *
9
 * Author: Daniel Veillard
10
 */
11
12
#define IN_LIBXML
13
#include "libxml.h"
14
15
#include <string.h>
16
#include <libxml/xmlmemory.h>
17
#include <libxml/tree.h>
18
#include <libxml/parser.h>
19
#include <libxml/uri.h>
20
#include <libxml/xpath.h>
21
#include <libxml/xpointer.h>
22
#include <libxml/parserInternals.h>
23
#include <libxml/xmlerror.h>
24
#include <libxml/encoding.h>
25
26
#ifdef LIBXML_XINCLUDE_ENABLED
27
#include <libxml/xinclude.h>
28
29
#include "private/buf.h"
30
#include "private/error.h"
31
#include "private/memory.h"
32
#include "private/parser.h"
33
#include "private/tree.h"
34
#include "private/xinclude.h"
35
36
0
#define XINCLUDE_MAX_DEPTH 40
37
38
/************************************************************************
39
 *                  *
40
 *      XInclude context handling     *
41
 *                  *
42
 ************************************************************************/
43
44
/*
45
 * An XInclude context
46
 */
47
typedef xmlChar *xmlURL;
48
49
typedef struct _xmlXIncludeRef xmlXIncludeRef;
50
typedef xmlXIncludeRef *xmlXIncludeRefPtr;
51
struct _xmlXIncludeRef {
52
    xmlChar              *URI; /* the fully resolved resource URL */
53
    xmlChar         *fragment; /* the fragment in the URI */
54
    xmlChar             *base; /* base URI of xi:include element */
55
    xmlNodePtr           elem; /* the xi:include element */
56
    xmlNodePtr            inc; /* the included copy */
57
    int                   xml; /* xml or txt */
58
    int              fallback; /* fallback was loaded */
59
    int       expanding; /* flag to detect inclusion loops */
60
    int         replace; /* should the node be replaced? */
61
};
62
63
typedef struct _xmlXIncludeDoc xmlXIncludeDoc;
64
typedef xmlXIncludeDoc *xmlXIncludeDocPtr;
65
struct _xmlXIncludeDoc {
66
    xmlDocPtr             doc; /* the parsed document */
67
    xmlChar              *url; /* the URL */
68
    int             expanding; /* flag to detect inclusion loops */
69
};
70
71
typedef struct _xmlXIncludeTxt xmlXIncludeTxt;
72
typedef xmlXIncludeTxt *xmlXIncludeTxtPtr;
73
struct _xmlXIncludeTxt {
74
    xmlChar   *text; /* text string */
75
    xmlChar              *url; /* the URL */
76
};
77
78
struct _xmlXIncludeCtxt {
79
    xmlDocPtr             doc; /* the source document */
80
    int                 incNr; /* number of includes */
81
    int                incMax; /* size of includes tab */
82
    xmlXIncludeRefPtr *incTab; /* array of included references */
83
84
    int                 txtNr; /* number of unparsed documents */
85
    int                txtMax; /* size of unparsed documents tab */
86
    xmlXIncludeTxt    *txtTab; /* array of unparsed documents */
87
88
    int                 urlNr; /* number of documents stacked */
89
    int                urlMax; /* size of document stack */
90
    xmlXIncludeDoc    *urlTab; /* document stack */
91
92
    int              nbErrors; /* the number of errors detected */
93
    int              fatalErr; /* abort processing */
94
    int                 errNo; /* error code */
95
    int                legacy; /* using XINCLUDE_OLD_NS */
96
    int            parseFlags; /* the flags used for parsing XML documents */
97
98
    void            *_private; /* application data */
99
100
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
101
    unsigned long    incTotal; /* total number of processed inclusions */
102
#endif
103
    int     depth; /* recursion depth */
104
    int        isStream; /* streaming mode */
105
106
#ifdef LIBXML_XPTR_ENABLED
107
    xmlXPathContextPtr xpctxt;
108
#endif
109
110
    xmlStructuredErrorFunc errorHandler;
111
    void *errorCtxt;
112
113
    xmlResourceLoader resourceLoader;
114
    void *resourceCtxt;
115
};
116
117
static xmlXIncludeRefPtr
118
xmlXIncludeExpandNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node);
119
120
static int
121
xmlXIncludeLoadNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref);
122
123
static int
124
xmlXIncludeDoProcess(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree);
125
126
127
/************************************************************************
128
 *                  *
129
 *      XInclude error handler        *
130
 *                  *
131
 ************************************************************************/
132
133
/**
134
 * Handle an out of memory condition
135
 *
136
 * @param ctxt  an XInclude context
137
 */
138
static void
139
xmlXIncludeErrMemory(xmlXIncludeCtxtPtr ctxt)
140
0
{
141
0
    ctxt->errNo = XML_ERR_NO_MEMORY;
142
0
    ctxt->fatalErr = 1;
143
0
    ctxt->nbErrors++;
144
145
0
    xmlRaiseMemoryError(ctxt->errorHandler, NULL, ctxt->errorCtxt,
146
0
                        XML_FROM_XINCLUDE, NULL);
147
0
}
148
149
/**
150
 * Handle an XInclude error
151
 *
152
 * @param ctxt  the XInclude context
153
 * @param node  the context node
154
 * @param error  the error code
155
 * @param msg  the error message
156
 * @param extra  extra information
157
 */
158
static void LIBXML_ATTR_FORMAT(4,0)
159
xmlXIncludeErr(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node, int error,
160
               const char *msg, const xmlChar *extra)
161
0
{
162
0
    xmlStructuredErrorFunc schannel = NULL;
163
0
    xmlGenericErrorFunc channel = NULL;
164
0
    void *data = NULL;
165
0
    int res;
166
167
0
    if (error == XML_ERR_NO_MEMORY) {
168
0
        xmlXIncludeErrMemory(ctxt);
169
0
        return;
170
0
    }
171
172
0
    if (ctxt->fatalErr != 0)
173
0
        return;
174
0
    ctxt->nbErrors++;
175
176
0
    schannel = ctxt->errorHandler;
177
0
    data = ctxt->errorCtxt;
178
179
0
    if (schannel == NULL) {
180
0
        channel = xmlGenericError;
181
0
        data = xmlGenericErrorContext;
182
0
    }
183
184
0
    res = xmlRaiseError(schannel, channel, data, ctxt, node,
185
0
                        XML_FROM_XINCLUDE, error, XML_ERR_ERROR,
186
0
                        NULL, 0, (const char *) extra, NULL, NULL, 0, 0,
187
0
                        msg, (const char *) extra);
188
0
    if (res < 0) {
189
0
        ctxt->errNo = XML_ERR_NO_MEMORY;
190
0
        ctxt->fatalErr = 1;
191
0
    } else {
192
0
        ctxt->errNo = error;
193
        /*
194
         * Note that we treat IO errors except ENOENT as fatal
195
         * although the XInclude spec could be interpreted in a
196
         * way that at least some IO errors should be handled
197
         * gracefully.
198
         */
199
0
        if (xmlIsCatastrophicError(XML_ERR_FATAL, error))
200
0
            ctxt->fatalErr = 1;
201
0
    }
202
0
}
203
204
/**
205
 * Get an XInclude attribute
206
 *
207
 * @param ctxt  the XInclude context
208
 * @param cur  the node
209
 * @param name  the attribute name
210
 * @returns the value (to be freed) or NULL if not found
211
 */
212
static xmlChar *
213
xmlXIncludeGetProp(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur,
214
0
                   const xmlChar *name) {
215
0
    xmlChar *ret;
216
217
0
    if (xmlNodeGetAttrValue(cur, name, XINCLUDE_NS, &ret) < 0)
218
0
        xmlXIncludeErrMemory(ctxt);
219
0
    if (ret != NULL)
220
0
        return(ret);
221
222
0
    if (ctxt->legacy != 0) {
223
0
        if (xmlNodeGetAttrValue(cur, name, XINCLUDE_OLD_NS, &ret) < 0)
224
0
            xmlXIncludeErrMemory(ctxt);
225
0
        if (ret != NULL)
226
0
            return(ret);
227
0
    }
228
229
0
    if (xmlNodeGetAttrValue(cur, name, NULL, &ret) < 0)
230
0
        xmlXIncludeErrMemory(ctxt);
231
0
    return(ret);
232
0
}
233
/**
234
 * Free an XInclude reference
235
 *
236
 * @param ref  the XInclude reference
237
 */
238
static void
239
0
xmlXIncludeFreeRef(xmlXIncludeRefPtr ref) {
240
0
    if (ref == NULL)
241
0
  return;
242
0
    if (ref->URI != NULL)
243
0
  xmlFree(ref->URI);
244
0
    if (ref->fragment != NULL)
245
0
  xmlFree(ref->fragment);
246
0
    if (ref->base != NULL)
247
0
  xmlFree(ref->base);
248
0
    xmlFree(ref);
249
0
}
250
251
/**
252
 * Creates a new XInclude context
253
 *
254
 * @param doc  an XML Document
255
 * @returns the new set
256
 */
257
xmlXIncludeCtxt *
258
0
xmlXIncludeNewContext(xmlDoc *doc) {
259
0
    xmlXIncludeCtxtPtr ret;
260
261
0
    if (doc == NULL)
262
0
  return(NULL);
263
0
    ret = (xmlXIncludeCtxtPtr) xmlMalloc(sizeof(xmlXIncludeCtxt));
264
0
    if (ret == NULL)
265
0
  return(NULL);
266
0
    memset(ret, 0, sizeof(xmlXIncludeCtxt));
267
0
    ret->doc = doc;
268
0
    ret->incNr = 0;
269
0
    ret->incMax = 0;
270
0
    ret->incTab = NULL;
271
0
    ret->nbErrors = 0;
272
0
    return(ret);
273
0
}
274
275
/**
276
 * Free an XInclude context
277
 *
278
 * @param ctxt  the XInclude context
279
 */
280
void
281
0
xmlXIncludeFreeContext(xmlXIncludeCtxt *ctxt) {
282
0
    int i;
283
284
0
    if (ctxt == NULL)
285
0
  return;
286
0
    if (ctxt->urlTab != NULL) {
287
0
  for (i = 0; i < ctxt->urlNr; i++) {
288
0
      xmlFreeDoc(ctxt->urlTab[i].doc);
289
0
      xmlFree(ctxt->urlTab[i].url);
290
0
  }
291
0
  xmlFree(ctxt->urlTab);
292
0
    }
293
0
    for (i = 0;i < ctxt->incNr;i++) {
294
0
  if (ctxt->incTab[i] != NULL)
295
0
      xmlXIncludeFreeRef(ctxt->incTab[i]);
296
0
    }
297
0
    if (ctxt->incTab != NULL)
298
0
  xmlFree(ctxt->incTab);
299
0
    if (ctxt->txtTab != NULL) {
300
0
  for (i = 0;i < ctxt->txtNr;i++) {
301
0
      xmlFree(ctxt->txtTab[i].text);
302
0
      xmlFree(ctxt->txtTab[i].url);
303
0
  }
304
0
  xmlFree(ctxt->txtTab);
305
0
    }
306
0
#ifdef LIBXML_XPTR_ENABLED
307
0
    if (ctxt->xpctxt != NULL)
308
0
  xmlXPathFreeContext(ctxt->xpctxt);
309
0
#endif
310
0
    xmlFree(ctxt);
311
0
}
312
313
/**
314
 * parse a document for XInclude
315
 *
316
 * @param ctxt  the XInclude context
317
 * @param URL  the URL or file path
318
 */
319
static xmlDocPtr
320
0
xmlXIncludeParseFile(xmlXIncludeCtxtPtr ctxt, const char *URL) {
321
0
    xmlDocPtr ret = NULL;
322
0
    xmlParserCtxtPtr pctxt;
323
0
    xmlParserInputPtr inputStream;
324
325
0
    xmlInitParser();
326
327
0
    pctxt = xmlNewParserCtxt();
328
0
    if (pctxt == NULL) {
329
0
  xmlXIncludeErrMemory(ctxt);
330
0
  return(NULL);
331
0
    }
332
0
    if (ctxt->errorHandler != NULL)
333
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
334
0
    if (ctxt->resourceLoader != NULL)
335
0
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
336
0
                                 ctxt->resourceCtxt);
337
338
    /*
339
     * pass in the application data to the parser context.
340
     */
341
0
    pctxt->_private = ctxt->_private;
342
343
    /*
344
     * try to ensure that new documents included are actually
345
     * built with the same dictionary as the including document.
346
     */
347
0
    if ((ctxt->doc != NULL) && (ctxt->doc->dict != NULL)) {
348
0
       if (pctxt->dict != NULL)
349
0
            xmlDictFree(pctxt->dict);
350
0
  pctxt->dict = ctxt->doc->dict;
351
0
  xmlDictReference(pctxt->dict);
352
0
    }
353
354
    /*
355
     * We set DTDLOAD to make sure that ID attributes declared in
356
     * external DTDs are detected.
357
     */
358
0
    xmlCtxtUseOptions(pctxt, ctxt->parseFlags | XML_PARSE_DTDLOAD);
359
360
0
    inputStream = xmlLoadResource(pctxt, URL, NULL, XML_RESOURCE_XINCLUDE);
361
0
    if (inputStream == NULL)
362
0
        goto error;
363
364
0
    if (xmlCtxtPushInput(pctxt, inputStream) < 0) {
365
0
        xmlFreeInputStream(inputStream);
366
0
        goto error;
367
0
    }
368
369
0
    xmlParseDocument(pctxt);
370
371
0
    if (pctxt->wellFormed) {
372
0
        ret = pctxt->myDoc;
373
0
    }
374
0
    else {
375
0
        ret = NULL;
376
0
  if (pctxt->myDoc != NULL)
377
0
      xmlFreeDoc(pctxt->myDoc);
378
0
        pctxt->myDoc = NULL;
379
0
    }
380
381
0
error:
382
0
    if (xmlCtxtIsCatastrophicError(pctxt))
383
0
        xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "parser error", NULL);
384
0
    xmlFreeParserCtxt(pctxt);
385
386
0
    return(ret);
387
0
}
388
389
/**
390
 * Add a new node to process to an XInclude context
391
 *
392
 * @param ctxt  the XInclude context
393
 * @param cur  the new node
394
 */
395
static xmlXIncludeRefPtr
396
0
xmlXIncludeAddNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur) {
397
0
    xmlXIncludeRefPtr ref = NULL;
398
0
    xmlXIncludeRefPtr ret = NULL;
399
0
    xmlURIPtr uri = NULL;
400
0
    xmlChar *href = NULL;
401
0
    xmlChar *parse = NULL;
402
0
    xmlChar *fragment = NULL;
403
0
    xmlChar *base = NULL;
404
0
    xmlChar *tmp;
405
0
    int xml = 1;
406
0
    int local = 0;
407
0
    int res;
408
409
0
    if (ctxt == NULL)
410
0
  return(NULL);
411
0
    if (cur == NULL)
412
0
  return(NULL);
413
414
    /*
415
     * read the attributes
416
     */
417
418
0
    fragment = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE_XPOINTER);
419
420
0
    href = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_HREF);
421
0
    if (href == NULL) {
422
0
        if (fragment == NULL) {
423
0
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_HREF,
424
0
                     "href or xpointer must be present\n", parse);
425
0
      goto error;
426
0
        }
427
428
0
  href = xmlStrdup(BAD_CAST ""); /* @@@@ href is now optional */
429
0
  if (href == NULL) {
430
0
            xmlXIncludeErrMemory(ctxt);
431
0
      goto error;
432
0
        }
433
0
    } else if (xmlStrlen(href) > XML_MAX_URI_LENGTH) {
434
0
        xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI, "URI too long\n",
435
0
                       NULL);
436
0
        goto error;
437
0
    }
438
439
0
    parse = xmlXIncludeGetProp(ctxt, cur, XINCLUDE_PARSE);
440
0
    if (parse != NULL) {
441
0
  if (xmlStrEqual(parse, XINCLUDE_PARSE_XML))
442
0
      xml = 1;
443
0
  else if (xmlStrEqual(parse, XINCLUDE_PARSE_TEXT))
444
0
      xml = 0;
445
0
  else {
446
0
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_PARSE_VALUE,
447
0
                     "invalid value %s for 'parse'\n", parse);
448
0
      goto error;
449
0
  }
450
0
    }
451
452
    /*
453
     * Check the URL and remove any fragment identifier
454
     */
455
0
    res = xmlParseURISafe((const char *)href, &uri);
456
0
    if (uri == NULL) {
457
0
        if (res < 0)
458
0
            xmlXIncludeErrMemory(ctxt);
459
0
        else
460
0
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
461
0
                           "invalid value href %s\n", href);
462
0
        goto error;
463
0
    }
464
465
0
    if (uri->fragment != NULL) {
466
0
        if (ctxt->legacy != 0) {
467
0
      if (fragment == NULL) {
468
0
    fragment = (xmlChar *) uri->fragment;
469
0
      } else {
470
0
    xmlFree(uri->fragment);
471
0
      }
472
0
  } else {
473
0
      xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_FRAGMENT_ID,
474
0
       "Invalid fragment identifier in URI %s use the xpointer attribute\n",
475
0
                           href);
476
0
      goto error;
477
0
  }
478
0
  uri->fragment = NULL;
479
0
    }
480
0
    tmp = xmlSaveUri(uri);
481
0
    if (tmp == NULL) {
482
0
  xmlXIncludeErrMemory(ctxt);
483
0
  goto error;
484
0
    }
485
0
    xmlFree(href);
486
0
    href = tmp;
487
488
    /*
489
     * Resolve URI
490
     */
491
492
0
    if (xmlNodeGetBaseSafe(ctxt->doc, cur, &base) < 0) {
493
0
        xmlXIncludeErrMemory(ctxt);
494
0
        goto error;
495
0
    }
496
497
0
    if (href[0] != 0) {
498
0
        if (xmlBuildURISafe(href, base, &tmp) < 0) {
499
0
            xmlXIncludeErrMemory(ctxt);
500
0
            goto error;
501
0
        }
502
0
        if (tmp == NULL) {
503
0
            xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_HREF_URI,
504
0
                           "failed build URL\n", NULL);
505
0
            goto error;
506
0
        }
507
0
        xmlFree(href);
508
0
        href = tmp;
509
510
0
        if (xmlStrEqual(href, ctxt->doc->URL))
511
0
            local = 1;
512
0
    } else {
513
0
        local = 1;
514
0
    }
515
516
    /*
517
     * If local and xml then we need a fragment
518
     */
519
0
    if ((local == 1) && (xml == 1) &&
520
0
        ((fragment == NULL) || (fragment[0] == 0))) {
521
0
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_RECURSION,
522
0
                 "detected a local recursion with no xpointer in %s\n",
523
0
           href);
524
0
  goto error;
525
0
    }
526
527
0
    ref = (xmlXIncludeRefPtr) xmlMalloc(sizeof(xmlXIncludeRef));
528
0
    if (ref == NULL) {
529
0
        xmlXIncludeErrMemory(ctxt);
530
0
        goto error;
531
0
    }
532
0
    memset(ref, 0, sizeof(xmlXIncludeRef));
533
534
0
    ref->elem = cur;
535
0
    ref->xml = xml;
536
0
    ref->URI = href;
537
0
    href = NULL;
538
0
    ref->fragment = fragment;
539
0
    fragment = NULL;
540
541
    /*
542
     * xml:base fixup
543
     */
544
0
    if (((ctxt->parseFlags & XML_PARSE_NOBASEFIX) == 0) &&
545
0
        (cur->doc != NULL) &&
546
0
        ((cur->doc->parseFlags & XML_PARSE_NOBASEFIX) == 0)) {
547
0
        if (base != NULL) {
548
0
            ref->base = base;
549
0
            base = NULL;
550
0
        } else {
551
0
            ref->base = xmlStrdup(BAD_CAST "");
552
0
            if (ref->base == NULL) {
553
0
          xmlXIncludeErrMemory(ctxt);
554
0
                goto error;
555
0
            }
556
0
        }
557
0
    }
558
559
0
    if (ctxt->incNr >= ctxt->incMax) {
560
0
        xmlXIncludeRefPtr *table;
561
0
        int newSize;
562
563
0
        newSize = xmlGrowCapacity(ctxt->incMax, sizeof(table[0]),
564
0
                                  4, XML_MAX_ITEMS);
565
0
        if (newSize < 0) {
566
0
      xmlXIncludeErrMemory(ctxt);
567
0
      goto error;
568
0
  }
569
0
        table = xmlRealloc(ctxt->incTab, newSize * sizeof(table[0]));
570
0
        if (table == NULL) {
571
0
      xmlXIncludeErrMemory(ctxt);
572
0
      goto error;
573
0
  }
574
0
        ctxt->incTab = table;
575
0
        ctxt->incMax = newSize;
576
0
    }
577
0
    ctxt->incTab[ctxt->incNr++] = ref;
578
579
0
    ret = ref;
580
0
    ref = NULL;
581
582
0
error:
583
0
    xmlXIncludeFreeRef(ref);
584
0
    xmlFreeURI(uri);
585
0
    xmlFree(href);
586
0
    xmlFree(parse);
587
0
    xmlFree(fragment);
588
0
    xmlFree(base);
589
0
    return(ret);
590
0
}
591
592
/**
593
 * The XInclude recursive nature is handled at this point.
594
 *
595
 * @param ctxt  the XInclude context
596
 * @param doc  the new document
597
 */
598
static void
599
0
xmlXIncludeRecurseDoc(xmlXIncludeCtxtPtr ctxt, xmlDocPtr doc) {
600
0
    xmlDocPtr oldDoc;
601
0
    xmlXIncludeRefPtr *oldIncTab;
602
0
    int oldIncMax, oldIncNr, oldIsStream;
603
0
    int i;
604
605
0
    oldDoc = ctxt->doc;
606
0
    oldIncMax = ctxt->incMax;
607
0
    oldIncNr = ctxt->incNr;
608
0
    oldIncTab = ctxt->incTab;
609
0
    oldIsStream = ctxt->isStream;
610
0
    ctxt->doc = doc;
611
0
    ctxt->incMax = 0;
612
0
    ctxt->incNr = 0;
613
0
    ctxt->incTab = NULL;
614
0
    ctxt->isStream = 0;
615
616
0
    xmlXIncludeDoProcess(ctxt, xmlDocGetRootElement(doc));
617
618
0
    if (ctxt->incTab != NULL) {
619
0
        for (i = 0; i < ctxt->incNr; i++)
620
0
            xmlXIncludeFreeRef(ctxt->incTab[i]);
621
0
        xmlFree(ctxt->incTab);
622
0
    }
623
624
0
    ctxt->doc = oldDoc;
625
0
    ctxt->incMax = oldIncMax;
626
0
    ctxt->incNr = oldIncNr;
627
0
    ctxt->incTab = oldIncTab;
628
0
    ctxt->isStream = oldIsStream;
629
0
}
630
631
/************************************************************************
632
 *                  *
633
 *      Node copy with specific semantic    *
634
 *                  *
635
 ************************************************************************/
636
637
static void
638
xmlXIncludeBaseFixup(xmlXIncludeCtxtPtr ctxt, xmlNodePtr cur, xmlNodePtr copy,
639
0
                     const xmlChar *targetBase) {
640
0
    xmlChar *base = NULL;
641
0
    xmlChar *relBase = NULL;
642
0
    xmlNs ns;
643
0
    int res;
644
645
0
    if (cur->type != XML_ELEMENT_NODE)
646
0
        return;
647
648
0
    if (xmlNodeGetBaseSafe(cur->doc, cur, &base) < 0)
649
0
        xmlXIncludeErrMemory(ctxt);
650
651
0
    if ((base != NULL) && !xmlStrEqual(base, targetBase)) {
652
0
        if ((xmlStrlen(base) > XML_MAX_URI_LENGTH) ||
653
0
            (xmlStrlen(targetBase) > XML_MAX_URI_LENGTH)) {
654
0
            relBase = xmlStrdup(base);
655
0
            if (relBase == NULL) {
656
0
                xmlXIncludeErrMemory(ctxt);
657
0
                goto done;
658
0
            }
659
0
        } else if (xmlBuildRelativeURISafe(base, targetBase, &relBase) < 0) {
660
0
            xmlXIncludeErrMemory(ctxt);
661
0
            goto done;
662
0
        }
663
0
        if (relBase == NULL) {
664
0
            xmlXIncludeErr(ctxt, cur,
665
0
                    XML_XINCLUDE_HREF_URI,
666
0
                    "Building relative URI failed: %s\n",
667
0
                    base);
668
0
            goto done;
669
0
        }
670
671
        /*
672
         * If the new base doesn't contain a slash, it can be omitted.
673
         */
674
0
        if (xmlStrchr(relBase, '/') != NULL) {
675
0
            res = xmlNodeSetBase(copy, relBase);
676
0
            if (res < 0)
677
0
                xmlXIncludeErrMemory(ctxt);
678
0
            goto done;
679
0
        }
680
0
    }
681
682
    /*
683
     * Delete existing xml:base if bases are equal
684
     */
685
0
    memset(&ns, 0, sizeof(ns));
686
0
    ns.href = XML_XML_NAMESPACE;
687
0
    xmlUnsetNsProp(copy, &ns, BAD_CAST "base");
688
689
0
done:
690
0
    xmlFree(base);
691
0
    xmlFree(relBase);
692
0
}
693
694
/**
695
 * Make a copy of the node while expanding nested XIncludes.
696
 *
697
 * @param ctxt  the XInclude context
698
 * @param elem  the element
699
 * @param copyChildren  copy children instead of node if true
700
 * @param targetBase  the xml:base of the target node
701
 * @returns a node list, not a single node.
702
 */
703
static xmlNodePtr
704
xmlXIncludeCopyNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr elem,
705
0
                    int copyChildren, const xmlChar *targetBase) {
706
0
    xmlNodePtr result = NULL;
707
0
    xmlNodePtr insertParent = NULL;
708
0
    xmlNodePtr insertLast = NULL;
709
0
    xmlNodePtr cur;
710
0
    xmlNodePtr item;
711
0
    int depth = 0;
712
713
0
    if (copyChildren) {
714
0
        cur = elem->children;
715
0
        if (cur == NULL)
716
0
            return(NULL);
717
0
    } else {
718
0
        cur = elem;
719
0
    }
720
721
0
    while (1) {
722
0
        xmlNodePtr copy = NULL;
723
0
        int recurse = 0;
724
725
0
        if ((cur->type == XML_DOCUMENT_NODE) ||
726
0
            (cur->type == XML_DTD_NODE)) {
727
0
            ;
728
0
        } else if ((cur->type == XML_ELEMENT_NODE) &&
729
0
                   (cur->ns != NULL) &&
730
0
                   (xmlStrEqual(cur->name, XINCLUDE_NODE)) &&
731
0
                   ((xmlStrEqual(cur->ns->href, XINCLUDE_NS)) ||
732
0
                    (xmlStrEqual(cur->ns->href, XINCLUDE_OLD_NS)))) {
733
0
            xmlXIncludeRefPtr ref = xmlXIncludeExpandNode(ctxt, cur);
734
735
0
            if (ref == NULL)
736
0
                goto error;
737
            /*
738
             * TODO: Insert XML_XINCLUDE_START and XML_XINCLUDE_END nodes
739
             */
740
0
            for (item = ref->inc; item != NULL; item = item->next) {
741
0
                copy = xmlStaticCopyNode(item, ctxt->doc, insertParent, 1);
742
0
                if (copy == NULL) {
743
0
                    xmlXIncludeErrMemory(ctxt);
744
0
                    goto error;
745
0
                }
746
747
0
                if (result == NULL)
748
0
                    result = copy;
749
0
                if (insertLast != NULL) {
750
0
                    insertLast->next = copy;
751
0
                    copy->prev = insertLast;
752
0
                } else if (insertParent != NULL) {
753
0
                    insertParent->children = copy;
754
0
                }
755
0
                insertLast = copy;
756
757
0
                if ((depth == 0) && (targetBase != NULL))
758
0
                    xmlXIncludeBaseFixup(ctxt, item, copy, targetBase);
759
0
            }
760
0
        } else {
761
0
            copy = xmlStaticCopyNode(cur, ctxt->doc, insertParent, 2);
762
0
            if (copy == NULL) {
763
0
                xmlXIncludeErrMemory(ctxt);
764
0
                goto error;
765
0
            }
766
767
0
            if (result == NULL)
768
0
                result = copy;
769
0
            if (insertLast != NULL) {
770
0
                insertLast->next = copy;
771
0
                copy->prev = insertLast;
772
0
            } else if (insertParent != NULL) {
773
0
                insertParent->children = copy;
774
0
            }
775
0
            insertLast = copy;
776
777
0
            if ((depth == 0) && (targetBase != NULL))
778
0
                xmlXIncludeBaseFixup(ctxt, cur, copy, targetBase);
779
780
0
            recurse = (cur->type != XML_ENTITY_REF_NODE) &&
781
0
                      (cur->children != NULL);
782
0
        }
783
784
0
        if (recurse) {
785
0
            cur = cur->children;
786
0
            insertParent = insertLast;
787
0
            insertLast = NULL;
788
0
            depth += 1;
789
0
            continue;
790
0
        }
791
792
0
        if (cur == elem)
793
0
            return(result);
794
795
0
        while (cur->next == NULL) {
796
0
            if (insertParent != NULL)
797
0
                insertParent->last = insertLast;
798
0
            cur = cur->parent;
799
0
            if (cur == elem)
800
0
                return(result);
801
0
            insertLast = insertParent;
802
0
            insertParent = insertParent->parent;
803
0
            depth -= 1;
804
0
        }
805
806
0
        cur = cur->next;
807
0
    }
808
809
0
error:
810
0
    xmlFreeNodeList(result);
811
0
    return(NULL);
812
0
}
813
814
#ifdef LIBXML_XPTR_ENABLED
815
/**
816
 * Build a node list tree copy of the XPointer result.
817
 * This will drop Attributes and Namespace declarations.
818
 *
819
 * @param ctxt  the XInclude context
820
 * @param obj  the XPointer result from the evaluation.
821
 * @param targetBase  the xml:base of the target node
822
 * @returns an xmlNode list or NULL.
823
 *         the caller has to free the node tree.
824
 */
825
static xmlNodePtr
826
xmlXIncludeCopyXPointer(xmlXIncludeCtxtPtr ctxt, xmlXPathObjectPtr obj,
827
0
                        const xmlChar *targetBase) {
828
0
    xmlNodePtr list = NULL, last = NULL, copy;
829
0
    int i;
830
831
0
    if ((ctxt == NULL) || (obj == NULL))
832
0
  return(NULL);
833
0
    switch (obj->type) {
834
0
        case XPATH_NODESET: {
835
0
      xmlNodeSetPtr set = obj->nodesetval;
836
0
      if (set == NULL)
837
0
    break;
838
0
      for (i = 0;i < set->nodeNr;i++) {
839
0
                xmlNodePtr node;
840
841
0
    if (set->nodeTab[i] == NULL)
842
0
        continue;
843
0
    switch (set->nodeTab[i]->type) {
844
0
        case XML_DOCUMENT_NODE:
845
0
        case XML_HTML_DOCUMENT_NODE:
846
0
                        node = xmlDocGetRootElement(
847
0
                                (xmlDocPtr) set->nodeTab[i]);
848
0
                        if (node == NULL) {
849
0
                            xmlXIncludeErr(ctxt, set->nodeTab[i],
850
0
                                           XML_ERR_INTERNAL_ERROR,
851
0
                                          "document without root\n", NULL);
852
0
                            continue;
853
0
                        }
854
0
                        break;
855
0
                    case XML_TEXT_NODE:
856
0
        case XML_CDATA_SECTION_NODE:
857
0
        case XML_ELEMENT_NODE:
858
0
        case XML_PI_NODE:
859
0
        case XML_COMMENT_NODE:
860
0
                        node = set->nodeTab[i];
861
0
      break;
862
0
                    default:
863
0
                        xmlXIncludeErr(ctxt, set->nodeTab[i],
864
0
                                       XML_XINCLUDE_XPTR_RESULT,
865
0
                                       "invalid node type in XPtr result\n",
866
0
                                       NULL);
867
0
      continue; /* for */
868
0
    }
869
                /*
870
                 * OPTIMIZE TODO: External documents should already be
871
                 * expanded, so xmlDocCopyNode should work as well.
872
                 * xmlXIncludeCopyNode is only required for the initial
873
                 * document.
874
                 */
875
0
    copy = xmlXIncludeCopyNode(ctxt, node, 0, targetBase);
876
0
                if (copy == NULL) {
877
0
                    xmlFreeNodeList(list);
878
0
                    return(NULL);
879
0
                }
880
0
    if (last == NULL) {
881
0
                    list = copy;
882
0
                } else {
883
0
                    while (last->next != NULL)
884
0
                        last = last->next;
885
0
                    copy->prev = last;
886
0
                    last->next = copy;
887
0
    }
888
0
                last = copy;
889
0
      }
890
0
      break;
891
0
  }
892
0
  default:
893
0
      break;
894
0
    }
895
0
    return(list);
896
0
}
897
#endif
898
899
/************************************************************************
900
 *                  *
901
 *      XInclude I/O handling       *
902
 *                  *
903
 ************************************************************************/
904
905
typedef struct _xmlXIncludeMergeData xmlXIncludeMergeData;
906
typedef xmlXIncludeMergeData *xmlXIncludeMergeDataPtr;
907
struct _xmlXIncludeMergeData {
908
    xmlDocPtr doc;
909
    xmlXIncludeCtxtPtr ctxt;
910
};
911
912
/**
913
 * Implements the merge of one entity
914
 *
915
 * @param payload  the entity
916
 * @param vdata  the merge data
917
 * @param name  unused
918
 */
919
static void
920
xmlXIncludeMergeEntity(void *payload, void *vdata,
921
0
                 const xmlChar *name ATTRIBUTE_UNUSED) {
922
0
    xmlEntityPtr ent = (xmlEntityPtr) payload;
923
0
    xmlXIncludeMergeDataPtr data = (xmlXIncludeMergeDataPtr) vdata;
924
0
    xmlEntityPtr ret, prev;
925
0
    xmlDocPtr doc;
926
0
    xmlXIncludeCtxtPtr ctxt;
927
928
0
    if ((ent == NULL) || (data == NULL))
929
0
  return;
930
0
    ctxt = data->ctxt;
931
0
    doc = data->doc;
932
0
    if ((ctxt == NULL) || (doc == NULL))
933
0
  return;
934
0
    switch (ent->etype) {
935
0
        case XML_INTERNAL_PARAMETER_ENTITY:
936
0
        case XML_EXTERNAL_PARAMETER_ENTITY:
937
0
        case XML_INTERNAL_PREDEFINED_ENTITY:
938
0
      return;
939
0
        case XML_INTERNAL_GENERAL_ENTITY:
940
0
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
941
0
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
942
0
      break;
943
0
    }
944
0
    prev = xmlGetDocEntity(doc, ent->name);
945
0
    if (prev == NULL) {
946
0
        ret = xmlAddDocEntity(doc, ent->name, ent->etype, ent->ExternalID,
947
0
                              ent->SystemID, ent->content);
948
0
        if (ret == NULL) {
949
0
            xmlXIncludeErrMemory(ctxt);
950
0
            return;
951
0
        }
952
0
  if (ent->URI != NULL) {
953
0
      ret->URI = xmlStrdup(ent->URI);
954
0
            if (ret->URI == 0)
955
0
                xmlXIncludeErrMemory(ctxt);
956
0
        }
957
0
    } else {
958
0
        if (ent->etype != prev->etype)
959
0
            goto error;
960
961
0
        if ((ent->SystemID != NULL) && (prev->SystemID != NULL)) {
962
0
            if (!xmlStrEqual(ent->SystemID, prev->SystemID))
963
0
                goto error;
964
0
        } else if ((ent->ExternalID != NULL) &&
965
0
                   (prev->ExternalID != NULL)) {
966
0
            if (!xmlStrEqual(ent->ExternalID, prev->ExternalID))
967
0
                goto error;
968
0
        } else if ((ent->content != NULL) && (prev->content != NULL)) {
969
0
            if (!xmlStrEqual(ent->content, prev->content))
970
0
                goto error;
971
0
        } else {
972
0
            goto error;
973
0
        }
974
0
    }
975
0
    return;
976
0
error:
977
0
    switch (ent->etype) {
978
0
        case XML_INTERNAL_PARAMETER_ENTITY:
979
0
        case XML_EXTERNAL_PARAMETER_ENTITY:
980
0
        case XML_INTERNAL_PREDEFINED_ENTITY:
981
0
        case XML_INTERNAL_GENERAL_ENTITY:
982
0
        case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
983
0
      return;
984
0
        case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
985
0
      break;
986
0
    }
987
0
    xmlXIncludeErr(ctxt, (xmlNodePtr) ent, XML_XINCLUDE_ENTITY_DEF_MISMATCH,
988
0
                   "mismatch in redefinition of entity %s\n",
989
0
       ent->name);
990
0
}
991
992
/**
993
 * Implements the entity merge
994
 *
995
 * @param ctxt  an XInclude context
996
 * @param doc  the including doc
997
 * @param from  the included doc
998
 * @returns 0 if merge succeeded, -1 if some processing failed
999
 */
1000
static int
1001
xmlXIncludeMergeEntities(xmlXIncludeCtxtPtr ctxt, xmlDocPtr doc,
1002
0
                   xmlDocPtr from) {
1003
0
    xmlNodePtr cur;
1004
0
    xmlDtdPtr target, source;
1005
1006
0
    if (ctxt == NULL)
1007
0
  return(-1);
1008
1009
0
    if ((from == NULL) || (from->intSubset == NULL))
1010
0
  return(0);
1011
1012
0
    target = doc->intSubset;
1013
0
    if (target == NULL) {
1014
0
  cur = xmlDocGetRootElement(doc);
1015
0
  if (cur == NULL)
1016
0
      return(-1);
1017
0
        target = xmlCreateIntSubset(doc, cur->name, NULL, NULL);
1018
0
  if (target == NULL) {
1019
0
            xmlXIncludeErrMemory(ctxt);
1020
0
      return(-1);
1021
0
        }
1022
0
    }
1023
1024
0
    source = from->intSubset;
1025
0
    if ((source != NULL) && (source->entities != NULL)) {
1026
0
  xmlXIncludeMergeData data;
1027
1028
0
  data.ctxt = ctxt;
1029
0
  data.doc = doc;
1030
1031
0
  xmlHashScan((xmlHashTablePtr) source->entities,
1032
0
        xmlXIncludeMergeEntity, &data);
1033
0
    }
1034
0
    source = from->extSubset;
1035
0
    if ((source != NULL) && (source->entities != NULL)) {
1036
0
  xmlXIncludeMergeData data;
1037
1038
0
  data.ctxt = ctxt;
1039
0
  data.doc = doc;
1040
1041
  /*
1042
   * don't duplicate existing stuff when external subsets are the same
1043
   */
1044
0
  if ((!xmlStrEqual(target->ExternalID, source->ExternalID)) &&
1045
0
      (!xmlStrEqual(target->SystemID, source->SystemID))) {
1046
0
      xmlHashScan((xmlHashTablePtr) source->entities,
1047
0
      xmlXIncludeMergeEntity, &data);
1048
0
  }
1049
0
    }
1050
0
    return(0);
1051
0
}
1052
1053
/**
1054
 * Load the document, and store the result in the XInclude context
1055
 *
1056
 * @param ctxt  the XInclude context
1057
 * @param ref  an XMLXincludeRefPtr
1058
 * @returns 0 in case of success, -1 in case of failure
1059
 */
1060
static int
1061
0
xmlXIncludeLoadDoc(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1062
0
    xmlXIncludeDocPtr cache;
1063
0
    xmlDocPtr doc;
1064
0
    const xmlChar *url = ref->URI;
1065
0
    const xmlChar *fragment = ref->fragment;
1066
0
    int i = 0;
1067
0
    int ret = -1;
1068
0
    int cacheNr;
1069
0
#ifdef LIBXML_XPTR_ENABLED
1070
0
    int saveFlags;
1071
0
#endif
1072
1073
    /*
1074
     * Handling of references to the local document are done
1075
     * directly through ctxt->doc.
1076
     */
1077
0
    if ((url[0] == 0) || (url[0] == '#') ||
1078
0
  ((ctxt->doc != NULL) && (xmlStrEqual(url, ctxt->doc->URL)))) {
1079
0
  doc = ctxt->doc;
1080
0
        goto loaded;
1081
0
    }
1082
1083
    /*
1084
     * Prevent reloading the document twice.
1085
     */
1086
0
    for (i = 0; i < ctxt->urlNr; i++) {
1087
0
  if (xmlStrEqual(url, ctxt->urlTab[i].url)) {
1088
0
            if (ctxt->urlTab[i].expanding) {
1089
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_RECURSION,
1090
0
                               "inclusion loop detected\n", NULL);
1091
0
                goto error;
1092
0
            }
1093
0
      doc = ctxt->urlTab[i].doc;
1094
0
            if (doc == NULL)
1095
0
                goto error;
1096
0
      goto loaded;
1097
0
  }
1098
0
    }
1099
1100
    /*
1101
     * Load it.
1102
     */
1103
0
#ifdef LIBXML_XPTR_ENABLED
1104
    /*
1105
     * If this is an XPointer evaluation, we want to assure that
1106
     * all entities have been resolved prior to processing the
1107
     * referenced document
1108
     */
1109
0
    saveFlags = ctxt->parseFlags;
1110
0
    if (fragment != NULL) { /* if this is an XPointer eval */
1111
0
  ctxt->parseFlags |= XML_PARSE_NOENT;
1112
0
    }
1113
0
#endif
1114
1115
0
    doc = xmlXIncludeParseFile(ctxt, (const char *)url);
1116
0
#ifdef LIBXML_XPTR_ENABLED
1117
0
    ctxt->parseFlags = saveFlags;
1118
0
#endif
1119
1120
    /* Also cache NULL docs */
1121
0
    if (ctxt->urlNr >= ctxt->urlMax) {
1122
0
        xmlXIncludeDoc *tmp;
1123
0
        int newSize;
1124
1125
0
        newSize = xmlGrowCapacity(ctxt->urlMax, sizeof(tmp[0]),
1126
0
                                  8, XML_MAX_ITEMS);
1127
0
        if (newSize < 0) {
1128
0
            xmlXIncludeErrMemory(ctxt);
1129
0
            xmlFreeDoc(doc);
1130
0
            goto error;
1131
0
        }
1132
0
        tmp = xmlRealloc(ctxt->urlTab, newSize * sizeof(tmp[0]));
1133
0
        if (tmp == NULL) {
1134
0
            xmlXIncludeErrMemory(ctxt);
1135
0
            xmlFreeDoc(doc);
1136
0
            goto error;
1137
0
        }
1138
0
        ctxt->urlMax = newSize;
1139
0
        ctxt->urlTab = tmp;
1140
0
    }
1141
0
    cache = &ctxt->urlTab[ctxt->urlNr];
1142
0
    cache->doc = doc;
1143
0
    cache->url = xmlStrdup(url);
1144
0
    if (cache->url == NULL) {
1145
0
        xmlXIncludeErrMemory(ctxt);
1146
0
        xmlFreeDoc(doc);
1147
0
        goto error;
1148
0
    }
1149
0
    cache->expanding = 0;
1150
0
    cacheNr = ctxt->urlNr++;
1151
1152
0
    if (doc == NULL)
1153
0
        goto error;
1154
    /*
1155
     * It's possible that the requested URL has been mapped to a
1156
     * completely different location (e.g. through a catalog entry).
1157
     * To check for this, we compare the URL with that of the doc
1158
     * and change it if they disagree (bug 146988).
1159
     */
1160
0
    if ((doc->URL != NULL) && (!xmlStrEqual(url, doc->URL)))
1161
0
        url = doc->URL;
1162
1163
    /*
1164
     * Make sure we have all entities fixed up
1165
     */
1166
0
    xmlXIncludeMergeEntities(ctxt, ctxt->doc, doc);
1167
1168
    /*
1169
     * We don't need the DTD anymore, free up space
1170
    if (doc->intSubset != NULL) {
1171
  xmlUnlinkNode((xmlNodePtr) doc->intSubset);
1172
  xmlFreeNode((xmlNodePtr) doc->intSubset);
1173
  doc->intSubset = NULL;
1174
    }
1175
    if (doc->extSubset != NULL) {
1176
  xmlUnlinkNode((xmlNodePtr) doc->extSubset);
1177
  xmlFreeNode((xmlNodePtr) doc->extSubset);
1178
  doc->extSubset = NULL;
1179
    }
1180
     */
1181
0
    cache->expanding = 1;
1182
0
    xmlXIncludeRecurseDoc(ctxt, doc);
1183
    /* urlTab might be reallocated. */
1184
0
    cache = &ctxt->urlTab[cacheNr];
1185
0
    cache->expanding = 0;
1186
1187
0
loaded:
1188
0
    if (fragment == NULL) {
1189
0
        xmlNodePtr root;
1190
1191
0
        root = xmlDocGetRootElement(doc);
1192
0
        if (root == NULL) {
1193
0
            xmlXIncludeErr(ctxt, ref->elem, XML_ERR_INTERNAL_ERROR,
1194
0
                           "document without root\n", NULL);
1195
0
            goto error;
1196
0
        }
1197
1198
0
        ref->inc = xmlDocCopyNode(root, ctxt->doc, 1);
1199
0
        if (ref->inc == NULL) {
1200
0
            xmlXIncludeErrMemory(ctxt);
1201
0
            goto error;
1202
0
        }
1203
1204
0
        if (ref->base != NULL)
1205
0
            xmlXIncludeBaseFixup(ctxt, root, ref->inc, ref->base);
1206
0
    }
1207
0
#ifdef LIBXML_XPTR_ENABLED
1208
0
    else {
1209
  /*
1210
   * Computes the XPointer expression and make a copy used
1211
   * as the replacement copy.
1212
   */
1213
0
  xmlXPathObjectPtr xptr;
1214
0
  xmlNodeSetPtr set;
1215
1216
0
        if (ctxt->isStream && doc == ctxt->doc) {
1217
0
      xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_FAILED,
1218
0
         "XPointer expressions not allowed in streaming"
1219
0
                           " mode\n", NULL);
1220
0
            goto error;
1221
0
        }
1222
1223
0
        if (ctxt->xpctxt == NULL) {
1224
0
            ctxt->xpctxt = xmlXPathNewContext(doc);
1225
0
            if (ctxt->xpctxt == NULL) {
1226
0
                xmlXIncludeErrMemory(ctxt);
1227
0
                goto error;
1228
0
            }
1229
0
            if (ctxt->errorHandler != NULL)
1230
0
                xmlXPathSetErrorHandler(ctxt->xpctxt, ctxt->errorHandler,
1231
0
                                        ctxt->errorCtxt);
1232
0
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1233
0
            ctxt->xpctxt->opLimit = 100000;
1234
0
#endif
1235
0
        } else {
1236
0
            ctxt->xpctxt->doc = doc;
1237
0
        }
1238
0
  xptr = xmlXPtrEval(fragment, ctxt->xpctxt);
1239
0
  if (ctxt->xpctxt->lastError.code != XML_ERR_OK) {
1240
0
            if (ctxt->xpctxt->lastError.code == XML_ERR_NO_MEMORY)
1241
0
                xmlXIncludeErrMemory(ctxt);
1242
0
            else
1243
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_FAILED,
1244
0
                               "XPointer evaluation failed: #%s\n",
1245
0
                               fragment);
1246
0
            goto error;
1247
0
  }
1248
0
        if (xptr == NULL)
1249
0
            goto done;
1250
0
  switch (xptr->type) {
1251
0
      case XPATH_UNDEFINED:
1252
0
      case XPATH_BOOLEAN:
1253
0
      case XPATH_NUMBER:
1254
0
      case XPATH_STRING:
1255
0
      case XPATH_USERS:
1256
0
      case XPATH_XSLT_TREE:
1257
0
    xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_XPTR_RESULT,
1258
0
             "XPointer is not a range: #%s\n",
1259
0
             fragment);
1260
0
                xmlXPathFreeObject(xptr);
1261
0
                goto error;
1262
0
      case XPATH_NODESET:
1263
0
                break;
1264
1265
0
  }
1266
0
  set = xptr->nodesetval;
1267
0
  if (set != NULL) {
1268
0
      for (i = 0;i < set->nodeNr;i++) {
1269
0
    if (set->nodeTab[i] == NULL) /* shouldn't happen */
1270
0
        continue;
1271
0
    switch (set->nodeTab[i]->type) {
1272
0
        case XML_ELEMENT_NODE:
1273
0
        case XML_TEXT_NODE:
1274
0
        case XML_CDATA_SECTION_NODE:
1275
0
        case XML_ENTITY_REF_NODE:
1276
0
        case XML_ENTITY_NODE:
1277
0
        case XML_PI_NODE:
1278
0
        case XML_COMMENT_NODE:
1279
0
        case XML_DOCUMENT_NODE:
1280
0
        case XML_HTML_DOCUMENT_NODE:
1281
0
      continue;
1282
1283
0
        case XML_ATTRIBUTE_NODE:
1284
0
      xmlXIncludeErr(ctxt, ref->elem,
1285
0
                     XML_XINCLUDE_XPTR_RESULT,
1286
0
               "XPointer selects an attribute: #%s\n",
1287
0
               fragment);
1288
0
      goto xptr_error;
1289
0
        case XML_NAMESPACE_DECL:
1290
0
      xmlXIncludeErr(ctxt, ref->elem,
1291
0
                     XML_XINCLUDE_XPTR_RESULT,
1292
0
               "XPointer selects a namespace: #%s\n",
1293
0
               fragment);
1294
0
      goto xptr_error;
1295
0
        case XML_DOCUMENT_TYPE_NODE:
1296
0
        case XML_DOCUMENT_FRAG_NODE:
1297
0
        case XML_NOTATION_NODE:
1298
0
        case XML_DTD_NODE:
1299
0
        case XML_ELEMENT_DECL:
1300
0
        case XML_ATTRIBUTE_DECL:
1301
0
        case XML_ENTITY_DECL:
1302
0
        case XML_XINCLUDE_START:
1303
0
        case XML_XINCLUDE_END:
1304
                        /* shouldn't happen */
1305
0
      xmlXIncludeErr(ctxt, ref->elem,
1306
0
                     XML_XINCLUDE_XPTR_RESULT,
1307
0
           "XPointer selects unexpected nodes: #%s\n",
1308
0
               fragment);
1309
0
      goto xptr_error;
1310
0
    }
1311
0
      }
1312
0
  }
1313
0
        ref->inc = xmlXIncludeCopyXPointer(ctxt, xptr, ref->base);
1314
0
xptr_error:
1315
0
        xmlXPathFreeObject(xptr);
1316
0
    }
1317
1318
0
done:
1319
0
#endif
1320
1321
0
    ret = 0;
1322
1323
0
error:
1324
0
    return(ret);
1325
0
}
1326
1327
/**
1328
 * Load the content, and store the result in the XInclude context
1329
 *
1330
 * @param ctxt  the XInclude context
1331
 * @param ref  an XMLXincludeRefPtr
1332
 * @returns 0 in case of success, -1 in case of failure
1333
 */
1334
static int
1335
0
xmlXIncludeLoadTxt(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1336
0
    xmlParserInputBufferPtr buf;
1337
0
    xmlNodePtr node = NULL;
1338
0
    const xmlChar *url = ref->URI;
1339
0
    int i;
1340
0
    int ret = -1;
1341
0
    xmlChar *encoding = NULL;
1342
0
    xmlCharEncodingHandlerPtr handler = NULL;
1343
0
    xmlParserCtxtPtr pctxt = NULL;
1344
0
    xmlParserInputPtr inputStream = NULL;
1345
0
    int len;
1346
0
    int res;
1347
0
    const xmlChar *content;
1348
1349
    /*
1350
     * Handling of references to the local document are done
1351
     * directly through ctxt->doc.
1352
     */
1353
0
    if (url[0] == 0) {
1354
0
  xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_TEXT_DOCUMENT,
1355
0
           "text serialization of document not available\n", NULL);
1356
0
  goto error;
1357
0
    }
1358
1359
    /*
1360
     * Prevent reloading the document twice.
1361
     */
1362
0
    for (i = 0; i < ctxt->txtNr; i++) {
1363
0
  if (xmlStrEqual(url, ctxt->txtTab[i].url)) {
1364
0
            node = xmlNewDocText(ctxt->doc, ctxt->txtTab[i].text);
1365
0
            if (node == NULL)
1366
0
                xmlXIncludeErrMemory(ctxt);
1367
0
      goto loaded;
1368
0
  }
1369
0
    }
1370
1371
    /*
1372
     * Try to get the encoding if available
1373
     */
1374
0
    if (ref->elem != NULL) {
1375
0
  encoding = xmlXIncludeGetProp(ctxt, ref->elem, XINCLUDE_PARSE_ENCODING);
1376
0
    }
1377
0
    if (encoding != NULL) {
1378
0
        xmlParserErrors code;
1379
1380
0
        code = xmlOpenCharEncodingHandler((const char *) encoding,
1381
0
                                          /* output */ 0, &handler);
1382
1383
0
        if (code != XML_ERR_OK) {
1384
0
            if (code == XML_ERR_NO_MEMORY) {
1385
0
                xmlXIncludeErrMemory(ctxt);
1386
0
            } else if (code == XML_ERR_UNSUPPORTED_ENCODING) {
1387
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_UNKNOWN_ENCODING,
1388
0
                               "encoding %s not supported\n", encoding);
1389
0
                goto error;
1390
0
            } else {
1391
0
                xmlXIncludeErr(ctxt, ref->elem, code,
1392
0
                               "unexpected error from iconv or ICU\n", NULL);
1393
0
                goto error;
1394
0
            }
1395
0
        }
1396
0
    }
1397
1398
    /*
1399
     * Load it.
1400
     */
1401
0
    pctxt = xmlNewParserCtxt();
1402
0
    if (pctxt == NULL) {
1403
0
        xmlXIncludeErrMemory(ctxt);
1404
0
        goto error;
1405
0
    }
1406
0
    if (ctxt->errorHandler != NULL)
1407
0
        xmlCtxtSetErrorHandler(pctxt, ctxt->errorHandler, ctxt->errorCtxt);
1408
0
    if (ctxt->resourceLoader != NULL)
1409
0
        xmlCtxtSetResourceLoader(pctxt, ctxt->resourceLoader,
1410
0
                                 ctxt->resourceCtxt);
1411
1412
0
    xmlCtxtUseOptions(pctxt, ctxt->parseFlags);
1413
1414
0
    inputStream = xmlLoadResource(pctxt, (const char*) url, NULL,
1415
0
                                  XML_RESOURCE_XINCLUDE_TEXT);
1416
0
    if (inputStream == NULL) {
1417
        /*
1418
         * ENOENT only produces a warning which isn't reflected in errNo.
1419
         */
1420
0
        if (pctxt->errNo == XML_ERR_NO_MEMORY)
1421
0
            xmlXIncludeErrMemory(ctxt);
1422
0
        else if ((pctxt->errNo != XML_ERR_OK) &&
1423
0
                 (pctxt->errNo != XML_IO_ENOENT) &&
1424
0
                 (pctxt->errNo != XML_IO_UNKNOWN) &&
1425
0
                 (pctxt->errNo != XML_IO_NETWORK_ATTEMPT))
1426
0
            xmlXIncludeErr(ctxt, NULL, pctxt->errNo, "load error", NULL);
1427
0
  goto error;
1428
0
    }
1429
0
    buf = inputStream->buf;
1430
0
    if (buf == NULL)
1431
0
  goto error;
1432
0
    if (buf->encoder)
1433
0
  xmlCharEncCloseFunc(buf->encoder);
1434
0
    buf->encoder = handler;
1435
0
    handler = NULL;
1436
1437
0
    node = xmlNewDocText(ctxt->doc, NULL);
1438
0
    if (node == NULL) {
1439
0
        xmlXIncludeErrMemory(ctxt);
1440
0
  goto error;
1441
0
    }
1442
1443
    /*
1444
     * Scan all chars from the resource and add the to the node
1445
     */
1446
0
    do {
1447
0
        res = xmlParserInputBufferRead(buf, 4096);
1448
0
    } while (res > 0);
1449
0
    if (res < 0) {
1450
0
        if (buf->error == XML_ERR_NO_MEMORY)
1451
0
            xmlXIncludeErrMemory(ctxt);
1452
0
        else
1453
0
            xmlXIncludeErr(ctxt, NULL, buf->error, "read error", NULL);
1454
0
        goto error;
1455
0
    }
1456
1457
0
    content = xmlBufContent(buf->buffer);
1458
0
    len = xmlBufUse(buf->buffer);
1459
0
    for (i = 0; i < len;) {
1460
0
        int cur;
1461
0
        int l;
1462
1463
0
        l = len - i;
1464
0
        cur = xmlGetUTF8Char(&content[i], &l);
1465
0
        if ((cur < 0) || (!IS_CHAR(cur))) {
1466
0
            xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_INVALID_CHAR,
1467
0
                           "%s contains invalid char\n", url);
1468
0
            goto error;
1469
0
        }
1470
1471
0
        i += l;
1472
0
    }
1473
1474
0
    if (xmlNodeAddContentLen(node, content, len) < 0)
1475
0
        xmlXIncludeErrMemory(ctxt);
1476
1477
0
    if (ctxt->txtNr >= ctxt->txtMax) {
1478
0
        xmlXIncludeTxt *tmp;
1479
0
        int newSize;
1480
1481
0
        newSize = xmlGrowCapacity(ctxt->txtMax, sizeof(tmp[0]),
1482
0
                                  8, XML_MAX_ITEMS);
1483
0
        if (newSize < 0) {
1484
0
            xmlXIncludeErrMemory(ctxt);
1485
0
      goto error;
1486
0
        }
1487
0
        tmp = xmlRealloc(ctxt->txtTab, newSize * sizeof(tmp[0]));
1488
0
        if (tmp == NULL) {
1489
0
            xmlXIncludeErrMemory(ctxt);
1490
0
      goto error;
1491
0
        }
1492
0
        ctxt->txtMax = newSize;
1493
0
        ctxt->txtTab = tmp;
1494
0
    }
1495
0
    ctxt->txtTab[ctxt->txtNr].text = xmlStrdup(node->content);
1496
0
    if ((node->content != NULL) &&
1497
0
        (ctxt->txtTab[ctxt->txtNr].text == NULL)) {
1498
0
        xmlXIncludeErrMemory(ctxt);
1499
0
        goto error;
1500
0
    }
1501
0
    ctxt->txtTab[ctxt->txtNr].url = xmlStrdup(url);
1502
0
    if (ctxt->txtTab[ctxt->txtNr].url == NULL) {
1503
0
        xmlXIncludeErrMemory(ctxt);
1504
0
        xmlFree(ctxt->txtTab[ctxt->txtNr].text);
1505
0
        goto error;
1506
0
    }
1507
0
    ctxt->txtNr++;
1508
1509
0
loaded:
1510
    /*
1511
     * Add the element as the replacement copy.
1512
     */
1513
0
    ref->inc = node;
1514
0
    node = NULL;
1515
0
    ret = 0;
1516
1517
0
error:
1518
0
    xmlFreeNode(node);
1519
0
    xmlFreeInputStream(inputStream);
1520
0
    xmlFreeParserCtxt(pctxt);
1521
0
    xmlCharEncCloseFunc(handler);
1522
0
    xmlFree(encoding);
1523
0
    return(ret);
1524
0
}
1525
1526
/**
1527
 * Load the content of the fallback node, and store the result
1528
 * in the XInclude context
1529
 *
1530
 * @param ctxt  the XInclude context
1531
 * @param fallback  the fallback node
1532
 * @param ref  an XMLXincludeRefPtr
1533
 * @returns 0 in case of success, -1 in case of failure
1534
 */
1535
static int
1536
xmlXIncludeLoadFallback(xmlXIncludeCtxtPtr ctxt, xmlNodePtr fallback,
1537
0
                        xmlXIncludeRefPtr ref) {
1538
0
    int ret = 0;
1539
0
    int oldNbErrors;
1540
1541
0
    if ((fallback == NULL) || (fallback->type == XML_NAMESPACE_DECL) ||
1542
0
        (ctxt == NULL))
1543
0
  return(-1);
1544
0
    if (fallback->children != NULL) {
1545
  /*
1546
   * It's possible that the fallback also has 'includes'
1547
   * (Bug 129969), so we re-process the fallback just in case
1548
   */
1549
0
        oldNbErrors = ctxt->nbErrors;
1550
0
  ref->inc = xmlXIncludeCopyNode(ctxt, fallback, 1, ref->base);
1551
0
  if (ctxt->nbErrors > oldNbErrors)
1552
0
      ret = -1;
1553
0
    } else {
1554
0
        ref->inc = NULL;
1555
0
    }
1556
0
    ref->fallback = 1;
1557
0
    return(ret);
1558
0
}
1559
1560
/************************************************************************
1561
 *                  *
1562
 *      XInclude Processing       *
1563
 *                  *
1564
 ************************************************************************/
1565
1566
/**
1567
 * If the XInclude node wasn't processed yet, create a new RefPtr,
1568
 * add it to ctxt->incTab and load the included items.
1569
 *
1570
 * @param ctxt  an XInclude context
1571
 * @param node  an XInclude node
1572
 * @returns the new or existing xmlXIncludeRef, or NULL in case of error.
1573
 */
1574
static xmlXIncludeRefPtr
1575
0
xmlXIncludeExpandNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1576
0
    xmlXIncludeRefPtr ref;
1577
0
    int i;
1578
1579
0
    if (ctxt->fatalErr)
1580
0
        return(NULL);
1581
0
    if (ctxt->depth >= XINCLUDE_MAX_DEPTH) {
1582
0
        xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1583
0
                       "maximum recursion depth exceeded\n", NULL);
1584
0
        ctxt->fatalErr = 1;
1585
0
        return(NULL);
1586
0
    }
1587
1588
0
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
1589
    /*
1590
     * The XInclude engine offers no protection against exponential
1591
     * expansion attacks similar to "billion laughs". Avoid timeouts by
1592
     * limiting the total number of replacements when fuzzing.
1593
     *
1594
     * Unfortuately, a single XInclude can already result in quadratic
1595
     * behavior:
1596
     *
1597
     *     <doc xmlns:xi="http://www.w3.org/2001/XInclude">
1598
     *       <xi:include xpointer="xpointer(//e)"/>
1599
     *       <e>
1600
     *         <e>
1601
     *           <e>
1602
     *             <!-- more nested elements -->
1603
     *           </e>
1604
     *         </e>
1605
     *       </e>
1606
     *     </doc>
1607
     */
1608
0
    if (ctxt->incTotal >= 20)
1609
0
        return(NULL);
1610
0
    ctxt->incTotal++;
1611
0
#endif
1612
1613
0
    for (i = 0; i < ctxt->incNr; i++) {
1614
0
        if (ctxt->incTab[i]->elem == node) {
1615
0
            if (ctxt->incTab[i]->expanding) {
1616
0
                xmlXIncludeErr(ctxt, node, XML_XINCLUDE_RECURSION,
1617
0
                               "inclusion loop detected\n", NULL);
1618
0
                return(NULL);
1619
0
            }
1620
0
            return(ctxt->incTab[i]);
1621
0
        }
1622
0
    }
1623
1624
0
    ref = xmlXIncludeAddNode(ctxt, node);
1625
0
    if (ref == NULL)
1626
0
        return(NULL);
1627
0
    ref->expanding = 1;
1628
0
    ctxt->depth++;
1629
0
    xmlXIncludeLoadNode(ctxt, ref);
1630
0
    ctxt->depth--;
1631
0
    ref->expanding = 0;
1632
1633
0
    return(ref);
1634
0
}
1635
1636
/**
1637
 * Find and load the infoset replacement for the given node.
1638
 *
1639
 * @param ctxt  an XInclude context
1640
 * @param ref  an xmlXIncludeRef
1641
 * @returns 0 if substitution succeeded, -1 if some processing failed
1642
 */
1643
static int
1644
0
xmlXIncludeLoadNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1645
0
    xmlNodePtr cur;
1646
0
    int ret;
1647
1648
0
    if ((ctxt == NULL) || (ref == NULL))
1649
0
  return(-1);
1650
0
    cur = ref->elem;
1651
0
    if (cur == NULL)
1652
0
  return(-1);
1653
1654
0
    if (ref->xml) {
1655
0
  ret = xmlXIncludeLoadDoc(ctxt, ref);
1656
  /* xmlXIncludeGetFragment(ctxt, cur, URI); */
1657
0
    } else {
1658
0
  ret = xmlXIncludeLoadTxt(ctxt, ref);
1659
0
    }
1660
1661
0
    if (ret < 0) {
1662
0
  xmlNodePtr children;
1663
1664
  /*
1665
   * Time to try a fallback if available
1666
   */
1667
0
  children = cur->children;
1668
0
  while (children != NULL) {
1669
0
      if ((children->type == XML_ELEMENT_NODE) &&
1670
0
    (children->ns != NULL) &&
1671
0
    (xmlStrEqual(children->name, XINCLUDE_FALLBACK)) &&
1672
0
    ((xmlStrEqual(children->ns->href, XINCLUDE_NS)) ||
1673
0
     (xmlStrEqual(children->ns->href, XINCLUDE_OLD_NS)))) {
1674
0
    ret = xmlXIncludeLoadFallback(ctxt, children, ref);
1675
0
    break;
1676
0
      }
1677
0
      children = children->next;
1678
0
  }
1679
0
    }
1680
0
    if (ret < 0) {
1681
0
  xmlXIncludeErr(ctxt, cur, XML_XINCLUDE_NO_FALLBACK,
1682
0
           "could not load %s, and no fallback was found\n",
1683
0
           ref->URI);
1684
0
    }
1685
1686
0
    return(0);
1687
0
}
1688
1689
/**
1690
 * Implement the infoset replacement for the given node
1691
 *
1692
 * @param ctxt  an XInclude context
1693
 * @param ref  an xmlXIncludeRef
1694
 * @returns 0 if substitution succeeded, -1 if some processing failed
1695
 */
1696
static int
1697
0
xmlXIncludeIncludeNode(xmlXIncludeCtxtPtr ctxt, xmlXIncludeRefPtr ref) {
1698
0
    xmlNodePtr cur, end, list, tmp;
1699
1700
0
    if ((ctxt == NULL) || (ref == NULL))
1701
0
  return(-1);
1702
0
    cur = ref->elem;
1703
0
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL))
1704
0
  return(-1);
1705
1706
0
    list = ref->inc;
1707
0
    ref->inc = NULL;
1708
1709
    /*
1710
     * Check against the risk of generating a multi-rooted document
1711
     */
1712
0
    if ((cur->parent != NULL) &&
1713
0
  (cur->parent->type != XML_ELEMENT_NODE)) {
1714
0
  int nb_elem = 0;
1715
1716
0
  tmp = list;
1717
0
  while (tmp != NULL) {
1718
0
      if (tmp->type == XML_ELEMENT_NODE)
1719
0
    nb_elem++;
1720
0
      tmp = tmp->next;
1721
0
  }
1722
0
        if (nb_elem != 1) {
1723
0
            if (nb_elem > 1)
1724
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1725
0
                               "XInclude error: would result in multiple root "
1726
0
                               "nodes\n", NULL);
1727
0
            else
1728
0
                xmlXIncludeErr(ctxt, ref->elem, XML_XINCLUDE_MULTIPLE_ROOT,
1729
0
                               "XInclude error: would result in no root "
1730
0
                               "node\n", NULL);
1731
0
            xmlFreeNodeList(list);
1732
0
      return(-1);
1733
0
  }
1734
0
    }
1735
1736
0
    if (ctxt->parseFlags & XML_PARSE_NOXINCNODE) {
1737
  /*
1738
   * Add the list of nodes
1739
         *
1740
         * TODO: Coalesce text nodes unless we are streaming mode.
1741
   */
1742
0
  while (list != NULL) {
1743
0
      end = list;
1744
0
      list = list->next;
1745
1746
0
      if (xmlAddPrevSibling(cur, end) == NULL) {
1747
0
                xmlUnlinkNode(end);
1748
0
                xmlFreeNode(end);
1749
0
                goto err_memory;
1750
0
            }
1751
0
  }
1752
0
  xmlUnlinkNode(cur);
1753
0
  xmlFreeNode(cur);
1754
0
    } else {
1755
0
        xmlNodePtr child, next;
1756
1757
  /*
1758
   * Change the current node as an XInclude start one, and add an
1759
   * XInclude end one
1760
   */
1761
0
        if (ref->fallback)
1762
0
            xmlUnsetProp(cur, BAD_CAST "href");
1763
0
  cur->type = XML_XINCLUDE_START;
1764
        /* Remove fallback children */
1765
0
        for (child = cur->children; child != NULL; child = next) {
1766
0
            next = child->next;
1767
0
            xmlUnlinkNode(child);
1768
0
            xmlFreeNode(child);
1769
0
        }
1770
0
  end = xmlNewDocNode(cur->doc, cur->ns, cur->name, NULL);
1771
0
  if (end == NULL)
1772
0
            goto err_memory;
1773
0
  end->type = XML_XINCLUDE_END;
1774
0
  if (xmlAddNextSibling(cur, end) == NULL) {
1775
0
            xmlFreeNode(end);
1776
0
            goto err_memory;
1777
0
        }
1778
1779
  /*
1780
   * Add the list of nodes
1781
   */
1782
0
  while (list != NULL) {
1783
0
      cur = list;
1784
0
      list = list->next;
1785
1786
0
      if (xmlAddPrevSibling(end, cur) == NULL) {
1787
0
                xmlUnlinkNode(cur);
1788
0
                xmlFreeNode(cur);
1789
0
                goto err_memory;
1790
0
            }
1791
0
  }
1792
0
    }
1793
1794
1795
0
    return(0);
1796
1797
0
err_memory:
1798
0
    xmlXIncludeErrMemory(ctxt);
1799
0
    xmlFreeNodeList(list);
1800
0
    return(-1);
1801
0
}
1802
1803
/**
1804
 * test if the node is an XInclude node
1805
 *
1806
 * @param ctxt  the XInclude processing context
1807
 * @param node  an XInclude node
1808
 * @returns 1 true, 0 otherwise
1809
 */
1810
static int
1811
0
xmlXIncludeTestNode(xmlXIncludeCtxtPtr ctxt, xmlNodePtr node) {
1812
0
    if (node == NULL)
1813
0
  return(0);
1814
0
    if (node->type != XML_ELEMENT_NODE)
1815
0
  return(0);
1816
0
    if (node->ns == NULL)
1817
0
  return(0);
1818
0
    if ((xmlStrEqual(node->ns->href, XINCLUDE_NS)) ||
1819
0
        (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS))) {
1820
0
  if (xmlStrEqual(node->ns->href, XINCLUDE_OLD_NS)) {
1821
0
      if (ctxt->legacy == 0) {
1822
0
          ctxt->legacy = 1;
1823
0
      }
1824
0
  }
1825
0
  if (xmlStrEqual(node->name, XINCLUDE_NODE)) {
1826
0
      xmlNodePtr child = node->children;
1827
0
      int nb_fallback = 0;
1828
1829
0
      while (child != NULL) {
1830
0
    if ((child->type == XML_ELEMENT_NODE) &&
1831
0
        (child->ns != NULL) &&
1832
0
        ((xmlStrEqual(child->ns->href, XINCLUDE_NS)) ||
1833
0
         (xmlStrEqual(child->ns->href, XINCLUDE_OLD_NS)))) {
1834
0
        if (xmlStrEqual(child->name, XINCLUDE_NODE)) {
1835
0
      xmlXIncludeErr(ctxt, node,
1836
0
                     XML_XINCLUDE_INCLUDE_IN_INCLUDE,
1837
0
               "%s has an 'include' child\n",
1838
0
               XINCLUDE_NODE);
1839
0
      return(0);
1840
0
        }
1841
0
        if (xmlStrEqual(child->name, XINCLUDE_FALLBACK)) {
1842
0
      nb_fallback++;
1843
0
        }
1844
0
    }
1845
0
    child = child->next;
1846
0
      }
1847
0
      if (nb_fallback > 1) {
1848
0
    xmlXIncludeErr(ctxt, node, XML_XINCLUDE_FALLBACKS_IN_INCLUDE,
1849
0
             "%s has multiple fallback children\n",
1850
0
                   XINCLUDE_NODE);
1851
0
    return(0);
1852
0
      }
1853
0
      return(1);
1854
0
  }
1855
0
  if (xmlStrEqual(node->name, XINCLUDE_FALLBACK)) {
1856
0
      if ((node->parent == NULL) ||
1857
0
    (node->parent->type != XML_ELEMENT_NODE) ||
1858
0
    (node->parent->ns == NULL) ||
1859
0
    ((!xmlStrEqual(node->parent->ns->href, XINCLUDE_NS)) &&
1860
0
     (!xmlStrEqual(node->parent->ns->href, XINCLUDE_OLD_NS))) ||
1861
0
    (!xmlStrEqual(node->parent->name, XINCLUDE_NODE))) {
1862
0
    xmlXIncludeErr(ctxt, node,
1863
0
                   XML_XINCLUDE_FALLBACK_NOT_IN_INCLUDE,
1864
0
             "%s is not the child of an 'include'\n",
1865
0
             XINCLUDE_FALLBACK);
1866
0
      }
1867
0
  }
1868
0
    }
1869
0
    return(0);
1870
0
}
1871
1872
/**
1873
 * Implement the XInclude substitution on the XML document `doc`
1874
 *
1875
 * @param ctxt  the XInclude processing context
1876
 * @param tree  the top of the tree to process
1877
 * @returns 0 if no substitution were done, -1 if some processing failed
1878
 *    or the number of substitutions done.
1879
 */
1880
static int
1881
0
xmlXIncludeDoProcess(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1882
0
    xmlXIncludeRefPtr ref;
1883
0
    xmlNodePtr cur;
1884
0
    int ret = 0;
1885
0
    int i, start;
1886
1887
    /*
1888
     * First phase: lookup the elements in the document
1889
     */
1890
0
    start = ctxt->incNr;
1891
0
    cur = tree;
1892
0
    do {
1893
  /* TODO: need to work on entities -> stack */
1894
0
        if (xmlXIncludeTestNode(ctxt, cur) == 1) {
1895
0
            ref = xmlXIncludeExpandNode(ctxt, cur);
1896
            /*
1897
             * Mark direct includes.
1898
             */
1899
0
            if (ref != NULL)
1900
0
                ref->replace = 1;
1901
0
        } else if ((cur->children != NULL) &&
1902
0
                   ((cur->type == XML_DOCUMENT_NODE) ||
1903
0
                    (cur->type == XML_ELEMENT_NODE))) {
1904
0
            cur = cur->children;
1905
0
            continue;
1906
0
        }
1907
0
        do {
1908
0
            if (cur == tree)
1909
0
                break;
1910
0
            if (cur->next != NULL) {
1911
0
                cur = cur->next;
1912
0
                break;
1913
0
            }
1914
0
            cur = cur->parent;
1915
0
        } while (cur != NULL);
1916
0
    } while ((cur != NULL) && (cur != tree));
1917
1918
    /*
1919
     * Second phase: extend the original document infoset.
1920
     */
1921
0
    for (i = start; i < ctxt->incNr; i++) {
1922
0
  if (ctxt->incTab[i]->replace != 0) {
1923
0
            xmlXIncludeIncludeNode(ctxt, ctxt->incTab[i]);
1924
0
            ctxt->incTab[i]->replace = 0;
1925
0
        } else {
1926
            /*
1927
             * Ignore includes which were added indirectly, for example
1928
             * inside xi:fallback elements.
1929
             */
1930
0
            if (ctxt->incTab[i]->inc != NULL) {
1931
0
                xmlFreeNodeList(ctxt->incTab[i]->inc);
1932
0
                ctxt->incTab[i]->inc = NULL;
1933
0
            }
1934
0
        }
1935
0
  ret++;
1936
0
    }
1937
1938
0
    if (ctxt->isStream) {
1939
        /*
1940
         * incTab references nodes which will eventually be deleted in
1941
         * streaming mode. The table is only required for XPointer
1942
         * expressions which aren't allowed in streaming mode.
1943
         */
1944
0
        for (i = 0;i < ctxt->incNr;i++) {
1945
0
            xmlXIncludeFreeRef(ctxt->incTab[i]);
1946
0
        }
1947
0
        ctxt->incNr = 0;
1948
0
    }
1949
1950
0
    return(ret);
1951
0
}
1952
1953
/**
1954
 * Implement the XInclude substitution on the XML document `doc`
1955
 *
1956
 * @param ctxt  the XInclude processing context
1957
 * @param tree  the top of the tree to process
1958
 * @returns 0 if no substitution were done, -1 if some processing failed
1959
 *    or the number of substitutions done.
1960
 */
1961
static int
1962
0
xmlXIncludeDoProcessRoot(xmlXIncludeCtxtPtr ctxt, xmlNodePtr tree) {
1963
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL))
1964
0
  return(-1);
1965
0
    if (ctxt == NULL)
1966
0
  return(-1);
1967
1968
0
    return(xmlXIncludeDoProcess(ctxt, tree));
1969
0
}
1970
1971
/**
1972
 * @since 2.13.0
1973
 *
1974
 * @param ctxt  an XInclude processing context
1975
 * @returns the last error code.
1976
 */
1977
int
1978
0
xmlXIncludeGetLastError(xmlXIncludeCtxt *ctxt) {
1979
0
    if (ctxt == NULL)
1980
0
        return(XML_ERR_ARGUMENT);
1981
0
    return(ctxt->errNo);
1982
0
}
1983
1984
/**
1985
 * Register a callback function that will be called on errors and
1986
 * warnings. If handler is NULL, the error handler will be deactivated.
1987
 *
1988
 * @since 2.13.0
1989
 * @param ctxt  an XInclude processing context
1990
 * @param handler  error handler
1991
 * @param data  user data which will be passed to the handler
1992
 */
1993
void
1994
xmlXIncludeSetErrorHandler(xmlXIncludeCtxt *ctxt,
1995
0
                           xmlStructuredErrorFunc handler, void *data) {
1996
0
    if (ctxt == NULL)
1997
0
        return;
1998
0
    ctxt->errorHandler = handler;
1999
0
    ctxt->errorCtxt = data;
2000
0
}
2001
2002
/**
2003
 * Register a callback function that will be called to load included
2004
 * documents.
2005
 *
2006
 * @since 2.14.0
2007
 * @param ctxt  an XInclude processing context
2008
 * @param loader  resource loader
2009
 * @param data  user data which will be passed to the loader
2010
 */
2011
void
2012
xmlXIncludeSetResourceLoader(xmlXIncludeCtxt *ctxt,
2013
0
                             xmlResourceLoader loader, void *data) {
2014
0
    if (ctxt == NULL)
2015
0
        return;
2016
0
    ctxt->resourceLoader = loader;
2017
0
    ctxt->resourceCtxt = data;
2018
0
}
2019
2020
/**
2021
 * Set the flags used for further processing of XML resources.
2022
 *
2023
 * @param ctxt  an XInclude processing context
2024
 * @param flags  a set of xmlParserOption used for parsing XML includes
2025
 * @returns 0 in case of success and -1 in case of error.
2026
 */
2027
int
2028
0
xmlXIncludeSetFlags(xmlXIncludeCtxt *ctxt, int flags) {
2029
0
    if (ctxt == NULL)
2030
0
        return(-1);
2031
0
    ctxt->parseFlags = flags;
2032
0
    return(0);
2033
0
}
2034
2035
/**
2036
 * In streaming mode, XPointer expressions aren't allowed.
2037
 *
2038
 * @param ctxt  an XInclude processing context
2039
 * @param mode  whether streaming mode should be enabled
2040
 * @returns 0 in case of success and -1 in case of error.
2041
 */
2042
int
2043
0
xmlXIncludeSetStreamingMode(xmlXIncludeCtxt *ctxt, int mode) {
2044
0
    if (ctxt == NULL)
2045
0
        return(-1);
2046
0
    ctxt->isStream = !!mode;
2047
0
    return(0);
2048
0
}
2049
2050
/**
2051
 * Implement the XInclude substitution on the XML node `tree`
2052
 *
2053
 * @param tree  an XML node
2054
 * @param flags  a set of xmlParserOption used for parsing XML includes
2055
 * @param data  application data that will be passed to the parser context
2056
 *        in the _private field of the parser context(s)
2057
 * @returns 0 if no substitution were done, -1 if some processing failed
2058
 *    or the number of substitutions done.
2059
 */
2060
2061
int
2062
0
xmlXIncludeProcessTreeFlagsData(xmlNode *tree, int flags, void *data) {
2063
0
    xmlXIncludeCtxtPtr ctxt;
2064
0
    int ret = 0;
2065
2066
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL) ||
2067
0
        (tree->doc == NULL))
2068
0
        return(-1);
2069
2070
0
    ctxt = xmlXIncludeNewContext(tree->doc);
2071
0
    if (ctxt == NULL)
2072
0
        return(-1);
2073
0
    ctxt->_private = data;
2074
0
    xmlXIncludeSetFlags(ctxt, flags);
2075
0
    ret = xmlXIncludeDoProcessRoot(ctxt, tree);
2076
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2077
0
        ret = -1;
2078
2079
0
    xmlXIncludeFreeContext(ctxt);
2080
0
    return(ret);
2081
0
}
2082
2083
/**
2084
 * Implement the XInclude substitution on the XML document `doc`
2085
 *
2086
 * @param doc  an XML document
2087
 * @param flags  a set of xmlParserOption used for parsing XML includes
2088
 * @param data  application data that will be passed to the parser context
2089
 *        in the _private field of the parser context(s)
2090
 * @returns 0 if no substitution were done, -1 if some processing failed
2091
 *    or the number of substitutions done.
2092
 */
2093
int
2094
0
xmlXIncludeProcessFlagsData(xmlDoc *doc, int flags, void *data) {
2095
0
    xmlNodePtr tree;
2096
2097
0
    if (doc == NULL)
2098
0
  return(-1);
2099
0
    tree = xmlDocGetRootElement(doc);
2100
0
    if (tree == NULL)
2101
0
  return(-1);
2102
0
    return(xmlXIncludeProcessTreeFlagsData(tree, flags, data));
2103
0
}
2104
2105
/**
2106
 * Implement the XInclude substitution on the XML document `doc`
2107
 *
2108
 * @param doc  an XML document
2109
 * @param flags  a set of xmlParserOption used for parsing XML includes
2110
 * @returns 0 if no substitution were done, -1 if some processing failed
2111
 *    or the number of substitutions done.
2112
 */
2113
int
2114
0
xmlXIncludeProcessFlags(xmlDoc *doc, int flags) {
2115
0
    return xmlXIncludeProcessFlagsData(doc, flags, NULL);
2116
0
}
2117
2118
/**
2119
 * Implement the XInclude substitution on the XML document `doc`
2120
 *
2121
 * @param doc  an XML document
2122
 * @returns 0 if no substitution were done, -1 if some processing failed
2123
 *    or the number of substitutions done.
2124
 */
2125
int
2126
0
xmlXIncludeProcess(xmlDoc *doc) {
2127
0
    return(xmlXIncludeProcessFlags(doc, doc ? doc->parseFlags : 0));
2128
0
}
2129
2130
/**
2131
 * Implement the XInclude substitution for the given subtree
2132
 *
2133
 * @param tree  a node in an XML document
2134
 * @param flags  a set of xmlParserOption used for parsing XML includes
2135
 * @returns 0 if no substitution were done, -1 if some processing failed
2136
 *    or the number of substitutions done.
2137
 */
2138
int
2139
0
xmlXIncludeProcessTreeFlags(xmlNode *tree, int flags) {
2140
0
    xmlXIncludeCtxtPtr ctxt;
2141
0
    int ret = 0;
2142
2143
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL) ||
2144
0
        (tree->doc == NULL))
2145
0
  return(-1);
2146
0
    ctxt = xmlXIncludeNewContext(tree->doc);
2147
0
    if (ctxt == NULL)
2148
0
  return(-1);
2149
0
    xmlXIncludeSetFlags(ctxt, flags);
2150
0
    ret = xmlXIncludeDoProcessRoot(ctxt, tree);
2151
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2152
0
  ret = -1;
2153
2154
0
    xmlXIncludeFreeContext(ctxt);
2155
0
    return(ret);
2156
0
}
2157
2158
/**
2159
 * Implement the XInclude substitution for the given subtree
2160
 *
2161
 * @param tree  a node in an XML document
2162
 * @returns 0 if no substitution were done, -1 if some processing failed
2163
 *    or the number of substitutions done.
2164
 */
2165
int
2166
0
xmlXIncludeProcessTree(xmlNode *tree) {
2167
0
    return(xmlXIncludeProcessTreeFlags(tree, (tree && tree->doc) ? tree->doc->parseFlags : 0));
2168
0
}
2169
2170
/**
2171
 * Implement the XInclude substitution for the given subtree reusing
2172
 * the information and data coming from the given context.
2173
 *
2174
 * @param ctxt  an existing XInclude context
2175
 * @param node  a node in an XML document
2176
 * @returns 0 if no substitution were done, -1 if some processing failed
2177
 *    or the number of substitutions done.
2178
 */
2179
int
2180
0
xmlXIncludeProcessNode(xmlXIncludeCtxt *ctxt, xmlNode *node) {
2181
0
    int ret = 0;
2182
2183
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL) ||
2184
0
        (node->doc == NULL) || (ctxt == NULL))
2185
0
  return(-1);
2186
0
    ret = xmlXIncludeDoProcessRoot(ctxt, node);
2187
0
    if ((ret >= 0) && (ctxt->nbErrors > 0))
2188
0
  ret = -1;
2189
0
    return(ret);
2190
0
}
2191
2192
#else /* !LIBXML_XINCLUDE_ENABLED */
2193
#endif