Coverage Report

Created: 2025-08-26 07:06

/src/libxml2/tree.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * tree.c : implementation of access function for an XML tree.
3
 *
4
 * References:
5
 *   XHTML 1.0 W3C REC: http://www.w3.org/TR/2002/REC-xhtml1-20020801/
6
 *
7
 * See Copyright for the status of this software.
8
 *
9
 * Author: Daniel Veillard
10
 *
11
 */
12
13
/* To avoid EBCDIC trouble when parsing on zOS */
14
#if defined(__MVS__)
15
#pragma convert("ISO8859-1")
16
#endif
17
18
#define IN_LIBXML
19
#include "libxml.h"
20
21
#include <string.h> /* for memset() only ! */
22
#include <stddef.h>
23
#include <limits.h>
24
#include <ctype.h>
25
#include <stdlib.h>
26
27
#ifdef LIBXML_ZLIB_ENABLED
28
#include <zlib.h>
29
#endif
30
31
#include <libxml/tree.h>
32
#include <libxml/xmlmemory.h>
33
#include <libxml/parser.h>
34
#include <libxml/uri.h>
35
#include <libxml/entities.h>
36
#include <libxml/xmlerror.h>
37
#include <libxml/parserInternals.h>
38
#ifdef LIBXML_HTML_ENABLED
39
#include <libxml/HTMLtree.h>
40
#endif
41
42
#include "private/buf.h"
43
#include "private/entities.h"
44
#include "private/error.h"
45
#include "private/memory.h"
46
#include "private/io.h"
47
#include "private/parser.h"
48
#include "private/tree.h"
49
50
#ifndef SIZE_MAX
51
  #define SIZE_MAX ((size_t) -1)
52
#endif
53
54
/*
55
 * Internal variable indicating whether a callback has been registered
56
 * for node creation/destruction. This avoids looking up thread-local
57
 * data if no callback was ever registered.
58
 */
59
int xmlRegisterCallbacks = 0;
60
61
/************************************************************************
62
 *                  *
63
 *    Forward declarations          *
64
 *                  *
65
 ************************************************************************/
66
67
static xmlNodePtr
68
xmlNewEntityRef(xmlDocPtr doc, xmlChar *name);
69
70
static xmlNsPtr
71
xmlNewReconciledNs(xmlNodePtr tree, xmlNsPtr ns);
72
73
static xmlAttrPtr
74
xmlGetPropNodeInternal(const xmlNode *node, const xmlChar *name,
75
           const xmlChar *nsName, int useDTD);
76
77
static xmlChar* xmlGetPropNodeValueInternal(const xmlAttr *prop);
78
79
static void
80
xmlBufGetChildContent(xmlBufPtr buf, const xmlNode *tree);
81
82
static void
83
xmlUnlinkNodeInternal(xmlNodePtr cur);
84
85
/************************************************************************
86
 *                  *
87
 *    A few static variables and macros     *
88
 *                  *
89
 ************************************************************************/
90
/* #undef xmlStringText */
91
const xmlChar xmlStringText[] = { 't', 'e', 'x', 't', 0 };
92
/* #undef xmlStringTextNoenc */
93
const xmlChar xmlStringTextNoenc[] =
94
              { 't', 'e', 'x', 't', 'n', 'o', 'e', 'n', 'c', 0 };
95
/* #undef xmlStringComment */
96
const xmlChar xmlStringComment[] = { 'c', 'o', 'm', 'm', 'e', 'n', 't', 0 };
97
98
static int xmlCompressMode = 0;
99
100
9.01k
#define IS_STR_XML(str) ((str != NULL) && (str[0] == 'x') && \
101
9.01k
  (str[1] == 'm') && (str[2] == 'l') && (str[3] == 0))
102
103
/************************************************************************
104
 *                  *
105
 *    Functions to move to entities.c once the    *
106
 *    API freeze is smoothen and they can be made public. *
107
 *                  *
108
 ************************************************************************/
109
#include <libxml/hash.h>
110
111
/**
112
 * Do an entity lookup in the DTD entity hash table and
113
 *
114
 * @param dtd  A pointer to the DTD to search
115
 * @param name  The entity name
116
 * @returns the corresponding entity, if found.
117
 */
118
static xmlEntityPtr
119
0
xmlGetEntityFromDtd(const xmlDtd *dtd, const xmlChar *name) {
120
0
    xmlEntitiesTablePtr table;
121
122
0
    if((dtd != NULL) && (dtd->entities != NULL)) {
123
0
  table = (xmlEntitiesTablePtr) dtd->entities;
124
0
  return((xmlEntityPtr) xmlHashLookup(table, name));
125
  /* return(xmlGetEntityFromTable(table, name)); */
126
0
    }
127
0
    return(NULL);
128
0
}
129
/**
130
 * Do an entity lookup in the DTD parameter entity hash table and
131
 *
132
 * @param dtd  A pointer to the DTD to search
133
 * @param name  The entity name
134
 * @returns the corresponding entity, if found.
135
 */
136
static xmlEntityPtr
137
0
xmlGetParameterEntityFromDtd(const xmlDtd *dtd, const xmlChar *name) {
138
0
    xmlEntitiesTablePtr table;
139
140
0
    if ((dtd != NULL) && (dtd->pentities != NULL)) {
141
0
  table = (xmlEntitiesTablePtr) dtd->pentities;
142
0
  return((xmlEntityPtr) xmlHashLookup(table, name));
143
  /* return(xmlGetEntityFromTable(table, name)); */
144
0
    }
145
0
    return(NULL);
146
0
}
147
148
/************************************************************************
149
 *                  *
150
 *      QName handling helper       *
151
 *                  *
152
 ************************************************************************/
153
154
/**
155
 * Build a QName from prefix and local name.
156
 *
157
 * Builds the QName `prefix:ncname` in `memory` if there is enough space
158
 * and prefix is not NULL nor empty, otherwise allocate a new string.
159
 * If prefix is NULL or empty it returns ncname.
160
 *
161
 * @param ncname  the Name
162
 * @param prefix  the prefix
163
 * @param memory  preallocated memory
164
 * @param len  preallocated memory length
165
 * @returns the new string which must be freed by the caller if different from
166
 *         `memory` and `ncname` or NULL in case of error
167
 */
168
xmlChar *
169
xmlBuildQName(const xmlChar *ncname, const xmlChar *prefix,
170
27.5k
        xmlChar *memory, int len) {
171
27.5k
    size_t lenn, lenp;
172
27.5k
    xmlChar *ret;
173
174
27.5k
    if ((ncname == NULL) || (len < 0)) return(NULL);
175
27.5k
    if (prefix == NULL) return((xmlChar *) ncname);
176
177
27.5k
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
178
    /* Make allocation more likely */
179
27.5k
    if (len > 8)
180
511
        len = 8;
181
27.5k
#endif
182
183
27.5k
    lenn = strlen((char *) ncname);
184
27.5k
    lenp = strlen((char *) prefix);
185
27.5k
    if (lenn >= SIZE_MAX - lenp - 1)
186
0
        return(NULL);
187
188
27.5k
    if ((memory == NULL) || ((size_t) len < lenn + lenp + 2)) {
189
27.2k
  ret = xmlMalloc(lenn + lenp + 2);
190
27.2k
  if (ret == NULL)
191
0
      return(NULL);
192
27.2k
    } else {
193
279
  ret = memory;
194
279
    }
195
27.5k
    memcpy(&ret[0], prefix, lenp);
196
27.5k
    ret[lenp] = ':';
197
27.5k
    memcpy(&ret[lenp + 1], ncname, lenn);
198
27.5k
    ret[lenn + lenp + 1] = 0;
199
27.5k
    return(ret);
200
27.5k
}
201
202
/**
203
 * Parse an XML qualified name.
204
 *
205
 * @deprecated This function doesn't report malloc failures.
206
 *
207
 *     [NS 5] QName ::= (Prefix ':')? LocalPart
208
 *
209
 *     [NS 6] Prefix ::= NCName
210
 *
211
 *     [NS 7] LocalPart ::= NCName
212
 *
213
 * @param name  the full QName
214
 * @param prefix  a xmlChar **
215
 * @returns NULL if the name doesn't have a prefix. Otherwise, returns the
216
 * local part, and prefix is updated to get the Prefix. Both the return value
217
 * and the prefix must be freed by the caller.
218
 */
219
xmlChar *
220
0
xmlSplitQName2(const xmlChar *name, xmlChar **prefix) {
221
0
    int len = 0;
222
0
    xmlChar *ret = NULL;
223
224
0
    if (prefix == NULL) return(NULL);
225
0
    *prefix = NULL;
226
0
    if (name == NULL) return(NULL);
227
228
    /* nasty but valid */
229
0
    if (name[0] == ':')
230
0
  return(NULL);
231
232
    /*
233
     * we are not trying to validate but just to cut, and yes it will
234
     * work even if this is as set of UTF-8 encoded chars
235
     */
236
0
    while ((name[len] != 0) && (name[len] != ':'))
237
0
  len++;
238
239
0
    if ((name[len] == 0) || (name[len+1] == 0))
240
0
  return(NULL);
241
242
0
    *prefix = xmlStrndup(name, len);
243
0
    if (*prefix == NULL)
244
0
  return(NULL);
245
0
    ret = xmlStrdup(&name[len + 1]);
246
0
    if (ret == NULL) {
247
0
  if (*prefix != NULL) {
248
0
      xmlFree(*prefix);
249
0
      *prefix = NULL;
250
0
  }
251
0
  return(NULL);
252
0
    }
253
254
0
    return(ret);
255
0
}
256
257
/**
258
 * Parse an XML qualified name.
259
 *
260
 * @param name  the full QName
261
 * @param len  an int *
262
 * @returns NULL if it is not a Qualified Name, otherwise, update len
263
 *         with the length in byte of the prefix and return a pointer
264
 *         to the start of the name without the prefix
265
 */
266
267
const xmlChar *
268
49.1k
xmlSplitQName3(const xmlChar *name, int *len) {
269
49.1k
    int l = 0;
270
271
49.1k
    if (name == NULL) return(NULL);
272
49.1k
    if (len == NULL) return(NULL);
273
274
    /* nasty but valid */
275
49.1k
    if (name[0] == ':')
276
3.85k
  return(NULL);
277
278
    /*
279
     * we are not trying to validate but just to cut, and yes it will
280
     * work even if this is as set of UTF-8 encoded chars
281
     */
282
8.70M
    while ((name[l] != 0) && (name[l] != ':'))
283
8.65M
  l++;
284
285
45.2k
    if ((name[l] == 0) || (name[l+1] == 0))
286
33.1k
  return(NULL);
287
288
12.1k
    *len = l;
289
290
12.1k
    return(&name[l+1]);
291
45.2k
}
292
293
/**
294
 * Parse a QName.
295
 *
296
 * The return value points to the start of the local
297
 * name in the input string. If the QName has a prefix, it will be
298
 * allocated and stored in `prefixPtr`. This string must be freed by
299
 * the caller. If there's no prefix, `prefixPtr` is set to NULL.
300
 *
301
 * @param name  the full QName
302
 * @param prefixPtr  pointer to resulting prefix
303
 * @returns the local name or NULL if a memory allocation failed.
304
 */
305
const xmlChar *
306
25.6k
xmlSplitQName4(const xmlChar *name, xmlChar **prefixPtr) {
307
25.6k
    xmlChar *prefix;
308
25.6k
    int l = 0;
309
310
25.6k
    if ((name == NULL) || (prefixPtr == NULL))
311
0
        return(NULL);
312
313
25.6k
    *prefixPtr = NULL;
314
315
    /* nasty but valid */
316
25.6k
    if (name[0] == ':')
317
2.65k
  return(name);
318
319
    /*
320
     * we are not trying to validate but just to cut, and yes it will
321
     * work even if this is as set of UTF-8 encoded chars
322
     */
323
76.6k
    while ((name[l] != 0) && (name[l] != ':'))
324
53.5k
  l++;
325
326
    /*
327
     * TODO: What about names with multiple colons?
328
     */
329
23.0k
    if ((name[l] == 0) || (name[l+1] == 0))
330
14.3k
  return(name);
331
332
8.66k
    prefix = xmlStrndup(name, l);
333
8.66k
    if (prefix == NULL)
334
0
        return(NULL);
335
336
8.66k
    *prefixPtr = prefix;
337
8.66k
    return(&name[l+1]);
338
8.66k
}
339
340
/************************************************************************
341
 *                  *
342
 *    Check Name, NCName and QName strings      *
343
 *                  *
344
 ************************************************************************/
345
346
/**
347
 * Check that a value conforms to the lexical space of NCName
348
 *
349
 * @param value  the value to check
350
 * @param space  allow spaces in front and end of the string
351
 * @returns 0 if this validates, a positive error code number otherwise
352
 *         and -1 in case of internal or API error.
353
 */
354
int
355
3.69k
xmlValidateNCName(const xmlChar *value, int space) {
356
3.69k
    const xmlChar *cur;
357
358
3.69k
    if (value == NULL)
359
0
        return(-1);
360
361
3.69k
    cur = value;
362
363
3.69k
    if (space) {
364
3.69k
  while (IS_BLANK_CH(*cur))
365
306k
            cur++;
366
3.69k
    }
367
368
3.69k
    value = cur;
369
3.69k
    cur = xmlScanName(value, SIZE_MAX, XML_SCAN_NC);
370
3.69k
    if ((cur == NULL) || (cur == value))
371
844
        return(1);
372
373
2.85k
    if (space) {
374
2.85k
  while (IS_BLANK_CH(*cur))
375
121k
            cur++;
376
2.85k
    }
377
378
2.85k
    return(*cur != 0);
379
3.69k
}
380
381
/**
382
 * Check that a value conforms to the lexical space of QName
383
 *
384
 * @param value  the value to check
385
 * @param space  allow spaces in front and end of the string
386
 * @returns 0 if this validates, a positive error code number otherwise
387
 *         and -1 in case of internal or API error.
388
 */
389
int
390
0
xmlValidateQName(const xmlChar *value, int space) {
391
0
    const xmlChar *cur;
392
393
0
    if (value == NULL)
394
0
        return(-1);
395
396
0
    cur = value;
397
398
0
    if (space) {
399
0
  while (IS_BLANK_CH(*cur))
400
0
            cur++;
401
0
    }
402
403
0
    value = cur;
404
0
    cur = xmlScanName(value, SIZE_MAX, XML_SCAN_NC);
405
0
    if ((cur == NULL) || (cur == value))
406
0
        return(1);
407
408
0
    if (*cur == ':') {
409
0
        cur += 1;
410
0
        value = cur;
411
0
        cur = xmlScanName(value, SIZE_MAX, XML_SCAN_NC);
412
0
        if ((cur == NULL) || (cur == value))
413
0
            return(1);
414
0
    }
415
416
0
    if (space) {
417
0
  while (IS_BLANK_CH(*cur))
418
0
            cur++;
419
0
    }
420
421
0
    return(*cur != 0);
422
0
}
423
424
/**
425
 * Check that a value conforms to the lexical space of Name
426
 *
427
 * @param value  the value to check
428
 * @param space  allow spaces in front and end of the string
429
 * @returns 0 if this validates, a positive error code number otherwise
430
 *         and -1 in case of internal or API error.
431
 */
432
int
433
0
xmlValidateName(const xmlChar *value, int space) {
434
0
    const xmlChar *cur;
435
436
0
    if (value == NULL)
437
0
        return(-1);
438
439
0
    cur = value;
440
441
0
    if (space) {
442
0
  while (IS_BLANK_CH(*cur))
443
0
            cur++;
444
0
    }
445
446
0
    value = cur;
447
0
    cur = xmlScanName(value, SIZE_MAX, 0);
448
0
    if ((cur == NULL) || (cur == value))
449
0
        return(1);
450
451
0
    if (space) {
452
0
  while (IS_BLANK_CH(*cur))
453
0
            cur++;
454
0
    }
455
456
0
    return(*cur != 0);
457
0
}
458
459
/**
460
 * Check that a value conforms to the lexical space of NMToken
461
 *
462
 * @param value  the value to check
463
 * @param space  allow spaces in front and end of the string
464
 * @returns 0 if this validates, a positive error code number otherwise
465
 *         and -1 in case of internal or API error.
466
 */
467
int
468
0
xmlValidateNMToken(const xmlChar *value, int space) {
469
0
    const xmlChar *cur;
470
471
0
    if (value == NULL)
472
0
        return(-1);
473
474
0
    cur = value;
475
476
0
    if (space) {
477
0
  while (IS_BLANK_CH(*cur))
478
0
            cur++;
479
0
    }
480
481
0
    value = cur;
482
0
    cur = xmlScanName(value, SIZE_MAX, XML_SCAN_NMTOKEN);
483
0
    if ((cur == NULL) || (cur == value))
484
0
        return(1);
485
486
0
    if (space) {
487
0
  while (IS_BLANK_CH(*cur))
488
0
            cur++;
489
0
    }
490
491
0
    return(*cur != 0);
492
0
}
493
494
/************************************************************************
495
 *                  *
496
 *    Allocation and deallocation of basic structures   *
497
 *                  *
498
 ************************************************************************/
499
500
/**
501
 * Create a new namespace. For a default namespace, `prefix` should be
502
 * NULL. The namespace URI in `href` is not checked. You should make sure
503
 * to pass a valid URI.
504
 *
505
 * If `node` is provided, it must be an element node. The namespace will
506
 * be appended to the node's namespace declarations. It is an error if
507
 * the node already has a definition for the prefix or default
508
 * namespace.
509
 *
510
 * @param node  the element carrying the namespace (optional)
511
 * @param href  the URI associated
512
 * @param prefix  the prefix for the namespace (optional)
513
 * @returns a new namespace pointer or NULL if arguments are invalid,
514
 * the prefix is already in use or a memory allocation failed.
515
 */
516
xmlNs *
517
142k
xmlNewNs(xmlNode *node, const xmlChar *href, const xmlChar *prefix) {
518
142k
    xmlNsPtr cur;
519
520
142k
    if ((node != NULL) && (node->type != XML_ELEMENT_NODE))
521
0
  return(NULL);
522
523
    /*
524
     * Allocate a new Namespace and fill the fields.
525
     */
526
142k
    cur = (xmlNsPtr) xmlMalloc(sizeof(xmlNs));
527
142k
    if (cur == NULL)
528
0
  return(NULL);
529
142k
    memset(cur, 0, sizeof(xmlNs));
530
142k
    cur->type = XML_LOCAL_NAMESPACE;
531
532
142k
    if (href != NULL) {
533
142k
  cur->href = xmlStrdup(href);
534
142k
        if (cur->href == NULL)
535
0
            goto error;
536
142k
    }
537
142k
    if (prefix != NULL) {
538
140k
  cur->prefix = xmlStrdup(prefix);
539
140k
        if (cur->prefix == NULL)
540
0
            goto error;
541
140k
    }
542
543
    /*
544
     * Add it at the end to preserve parsing order ...
545
     * and checks for existing use of the prefix
546
     */
547
142k
    if (node != NULL) {
548
0
  if (node->nsDef == NULL) {
549
0
      node->nsDef = cur;
550
0
  } else {
551
0
      xmlNsPtr prev = node->nsDef;
552
553
0
      if ((xmlStrEqual(prev->prefix, cur->prefix)) &&
554
0
                (prev->href != NULL))
555
0
                goto error;
556
0
      while (prev->next != NULL) {
557
0
          prev = prev->next;
558
0
    if ((xmlStrEqual(prev->prefix, cur->prefix)) &&
559
0
                    (prev->href != NULL))
560
0
                    goto error;
561
0
      }
562
0
      prev->next = cur;
563
0
  }
564
0
    }
565
142k
    return(cur);
566
567
0
error:
568
0
    xmlFreeNs(cur);
569
0
    return(NULL);
570
142k
}
571
572
/**
573
 * Set the namespace of an element or attribute node. Passing a NULL
574
 * namespace unsets the namespace.
575
 *
576
 * @param node  a node in the document
577
 * @param ns  a namespace pointer (optional)
578
 */
579
void
580
0
xmlSetNs(xmlNode *node, xmlNs *ns) {
581
0
    if (node == NULL) {
582
0
  return;
583
0
    }
584
0
    if ((node->type == XML_ELEMENT_NODE) ||
585
0
        (node->type == XML_ATTRIBUTE_NODE))
586
0
  node->ns = ns;
587
0
}
588
589
/**
590
 * Free an xmlNs object.
591
 *
592
 * @param cur  the namespace pointer
593
 */
594
void
595
143k
xmlFreeNs(xmlNs *cur) {
596
143k
    if (cur == NULL) {
597
0
  return;
598
0
    }
599
143k
    if (cur->href != NULL) xmlFree((char *) cur->href);
600
143k
    if (cur->prefix != NULL) xmlFree((char *) cur->prefix);
601
143k
    xmlFree(cur);
602
143k
}
603
604
/**
605
 * Free a list of xmlNs objects.
606
 *
607
 * @param cur  the first namespace pointer
608
 */
609
void
610
27.1k
xmlFreeNsList(xmlNs *cur) {
611
27.1k
    xmlNsPtr next;
612
27.1k
    if (cur == NULL) {
613
0
  return;
614
0
    }
615
170k
    while (cur != NULL) {
616
143k
        next = cur->next;
617
143k
        xmlFreeNs(cur);
618
143k
  cur = next;
619
143k
    }
620
27.1k
}
621
622
/**
623
 * Create a DTD node.
624
 *
625
 * If a document is provided, it is an error if it already has an
626
 * external subset. If the document has no external subset, it
627
 * will be set to the created DTD.
628
 *
629
 * To create an internal subset, use #xmlCreateIntSubset.
630
 *
631
 * @param doc  the document pointer (optional)
632
 * @param name  the DTD name (optional)
633
 * @param publicId  public identifier of the DTD (optional)
634
 * @param systemId  system identifier (URL) of the DTD (optional)
635
 * @returns a pointer to the new DTD object or NULL if arguments are
636
 * invalid or a memory allocation failed.
637
 */
638
xmlDtd *
639
xmlNewDtd(xmlDoc *doc, const xmlChar *name, const xmlChar *publicId,
640
1.62k
          const xmlChar *systemId) {
641
1.62k
    xmlDtdPtr cur;
642
643
1.62k
    if ((doc != NULL) && (doc->extSubset != NULL)) {
644
0
  return(NULL);
645
0
    }
646
647
    /*
648
     * Allocate a new DTD and fill the fields.
649
     */
650
1.62k
    cur = (xmlDtdPtr) xmlMalloc(sizeof(xmlDtd));
651
1.62k
    if (cur == NULL)
652
0
  return(NULL);
653
1.62k
    memset(cur, 0 , sizeof(xmlDtd));
654
1.62k
    cur->type = XML_DTD_NODE;
655
656
1.62k
    if (name != NULL) {
657
1.62k
  cur->name = xmlStrdup(name);
658
1.62k
        if (cur->name == NULL)
659
0
            goto error;
660
1.62k
    }
661
1.62k
    if (publicId != NULL) {
662
0
  cur->ExternalID = xmlStrdup(publicId);
663
0
        if (cur->ExternalID == NULL)
664
0
            goto error;
665
0
    }
666
1.62k
    if (systemId != NULL) {
667
0
  cur->SystemID = xmlStrdup(systemId);
668
0
        if (cur->SystemID == NULL)
669
0
            goto error;
670
0
    }
671
1.62k
    if (doc != NULL)
672
1.62k
  doc->extSubset = cur;
673
1.62k
    cur->doc = doc;
674
675
1.62k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
676
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)cur);
677
1.62k
    return(cur);
678
679
0
error:
680
0
    xmlFreeDtd(cur);
681
0
    return(NULL);
682
1.62k
}
683
684
/**
685
 * Get the internal subset of a document.
686
 *
687
 * @param doc  the document pointer
688
 * @returns a pointer to the DTD object or NULL if not found.
689
 */
690
xmlDtd *
691
7.48k
xmlGetIntSubset(const xmlDoc *doc) {
692
7.48k
    xmlNodePtr cur;
693
694
7.48k
    if (doc == NULL)
695
0
  return(NULL);
696
7.48k
    cur = doc->children;
697
8.25k
    while (cur != NULL) {
698
772
  if (cur->type == XML_DTD_NODE)
699
0
      return((xmlDtdPtr) cur);
700
772
  cur = cur->next;
701
772
    }
702
7.48k
    return((xmlDtdPtr) doc->intSubset);
703
7.48k
}
704
705
/**
706
 * Create a DTD node.
707
 *
708
 * If a document is provided and it already has an internal subset,
709
 * the existing DTD object is returned without creating a new object.
710
 * If the document has no internal subset, it will be set to the
711
 * created DTD.
712
 *
713
 * @param doc  the document pointer (optional)
714
 * @param name  the DTD name (optional)
715
 * @param publicId  public identifier of the DTD (optional)
716
 * @param systemId  system identifier (URL) of the DTD (optional)
717
 * @returns a pointer to the new or existing DTD object or NULL if
718
 * arguments are invalid or a memory allocation failed.
719
 */
720
xmlDtd *
721
xmlCreateIntSubset(xmlDoc *doc, const xmlChar *name, const xmlChar *publicId,
722
3.74k
                   const xmlChar *systemId) {
723
3.74k
    xmlDtdPtr cur;
724
725
3.74k
    if (doc != NULL) {
726
3.74k
        cur = xmlGetIntSubset(doc);
727
3.74k
        if (cur != NULL)
728
0
            return(cur);
729
3.74k
    }
730
731
    /*
732
     * Allocate a new DTD and fill the fields.
733
     */
734
3.74k
    cur = (xmlDtdPtr) xmlMalloc(sizeof(xmlDtd));
735
3.74k
    if (cur == NULL)
736
0
  return(NULL);
737
3.74k
    memset(cur, 0, sizeof(xmlDtd));
738
3.74k
    cur->type = XML_DTD_NODE;
739
740
3.74k
    if (name != NULL) {
741
3.74k
  cur->name = xmlStrdup(name);
742
3.74k
  if (cur->name == NULL)
743
0
            goto error;
744
3.74k
    }
745
3.74k
    if (publicId != NULL) {
746
1
  cur->ExternalID = xmlStrdup(publicId);
747
1
  if (cur->ExternalID  == NULL)
748
0
            goto error;
749
1
    }
750
3.74k
    if (systemId != NULL) {
751
192
  cur->SystemID = xmlStrdup(systemId);
752
192
  if (cur->SystemID == NULL)
753
0
            goto error;
754
192
    }
755
3.74k
    if (doc != NULL) {
756
3.74k
  doc->intSubset = cur;
757
3.74k
  cur->parent = doc;
758
3.74k
  cur->doc = doc;
759
3.74k
  if (doc->children == NULL) {
760
3.65k
      doc->children = (xmlNodePtr) cur;
761
3.65k
      doc->last = (xmlNodePtr) cur;
762
3.65k
  } else {
763
84
      if (doc->type == XML_HTML_DOCUMENT_NODE) {
764
0
    xmlNodePtr prev;
765
766
0
    prev = doc->children;
767
0
    prev->prev = (xmlNodePtr) cur;
768
0
    cur->next = prev;
769
0
    doc->children = (xmlNodePtr) cur;
770
84
      } else {
771
84
    xmlNodePtr next;
772
773
84
    next = doc->children;
774
470
    while ((next != NULL) && (next->type != XML_ELEMENT_NODE))
775
386
        next = next->next;
776
84
    if (next == NULL) {
777
84
        cur->prev = doc->last;
778
84
        cur->prev->next = (xmlNodePtr) cur;
779
84
        cur->next = NULL;
780
84
        doc->last = (xmlNodePtr) cur;
781
84
    } else {
782
0
        cur->next = next;
783
0
        cur->prev = next->prev;
784
0
        if (cur->prev == NULL)
785
0
      doc->children = (xmlNodePtr) cur;
786
0
        else
787
0
      cur->prev->next = (xmlNodePtr) cur;
788
0
        next->prev = (xmlNodePtr) cur;
789
0
    }
790
84
      }
791
84
  }
792
3.74k
    }
793
794
3.74k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
795
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)cur);
796
3.74k
    return(cur);
797
798
0
error:
799
0
    xmlFreeDtd(cur);
800
0
    return(NULL);
801
3.74k
}
802
803
/**
804
 * Free a string if it is not owned by the "dict" dictionary in the
805
 * current scope
806
 *
807
 * @param str  a string
808
 */
809
#define DICT_FREE(str)            \
810
205k
  if ((str) && ((!dict) ||       \
811
187k
      (xmlDictOwns(dict, (const xmlChar *)(str)) == 0))) \
812
187k
      xmlFree((char *)(str));
813
814
/**
815
 * Free a DTD structure.
816
 *
817
 * @param cur  the DTD structure to free up
818
 */
819
void
820
5.37k
xmlFreeDtd(xmlDtd *cur) {
821
5.37k
    xmlDictPtr dict = NULL;
822
823
5.37k
    if (cur == NULL) {
824
0
  return;
825
0
    }
826
5.37k
    if (cur->doc != NULL) dict = cur->doc->dict;
827
828
5.37k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
829
0
  xmlDeregisterNodeDefaultValue((xmlNodePtr)cur);
830
831
5.37k
    if (cur->children != NULL) {
832
4.85k
  xmlNodePtr next, c = cur->children;
833
834
  /*
835
   * Cleanup all nodes which are not part of the specific lists
836
   * of notations, elements, attributes and entities.
837
   */
838
21.2k
        while (c != NULL) {
839
16.3k
      next = c->next;
840
16.3k
      if ((c->type != XML_ELEMENT_DECL) &&
841
16.3k
    (c->type != XML_ATTRIBUTE_DECL) &&
842
16.3k
    (c->type != XML_ENTITY_DECL)) {
843
898
    xmlUnlinkNodeInternal(c);
844
898
    xmlFreeNode(c);
845
898
      }
846
16.3k
      c = next;
847
16.3k
  }
848
4.85k
    }
849
850
5.37k
    DICT_FREE(cur->name)
851
852
5.37k
    if (cur->SystemID != NULL)
853
192
        xmlFree(cur->SystemID);
854
5.37k
    if (cur->ExternalID != NULL)
855
1
        xmlFree(cur->ExternalID);
856
857
    /* TODO !!! */
858
5.37k
    if (cur->notations != NULL)
859
35
        xmlFreeNotationTable((xmlNotationTablePtr) cur->notations);
860
861
5.37k
    if (cur->elements != NULL)
862
990
        xmlFreeElementTable((xmlElementTablePtr) cur->elements);
863
5.37k
    if (cur->attributes != NULL)
864
884
        xmlFreeAttributeTable((xmlAttributeTablePtr) cur->attributes);
865
5.37k
    if (cur->entities != NULL)
866
2.39k
        xmlFreeEntitiesTable((xmlEntitiesTablePtr) cur->entities);
867
5.37k
    if (cur->pentities != NULL)
868
1.68k
        xmlFreeEntitiesTable((xmlEntitiesTablePtr) cur->pentities);
869
870
5.37k
    xmlFree(cur);
871
5.37k
}
872
873
/**
874
 * Creates a new XML document. If version is NULL, `"1.0"` is used.
875
 *
876
 * @param version  XML version string like `"1.0"` (optional)
877
 * @returns a new document or NULL if a memory allocation failed.
878
 */
879
xmlDoc *
880
16.6k
xmlNewDoc(const xmlChar *version) {
881
16.6k
    xmlDocPtr cur;
882
883
16.6k
    if (version == NULL)
884
0
  version = (const xmlChar *) "1.0";
885
886
    /*
887
     * Allocate a new document and fill the fields.
888
     */
889
16.6k
    cur = (xmlDocPtr) xmlMalloc(sizeof(xmlDoc));
890
16.6k
    if (cur == NULL)
891
0
  return(NULL);
892
16.6k
    memset(cur, 0, sizeof(xmlDoc));
893
16.6k
    cur->type = XML_DOCUMENT_NODE;
894
895
16.6k
    cur->version = xmlStrdup(version);
896
16.6k
    if (cur->version == NULL) {
897
0
  xmlFree(cur);
898
0
  return(NULL);
899
0
    }
900
16.6k
    cur->standalone = -1;
901
16.6k
    cur->compression = -1; /* not initialized */
902
16.6k
    cur->doc = cur;
903
16.6k
    cur->parseFlags = 0;
904
16.6k
    cur->properties = XML_DOC_USERBUILT;
905
    /*
906
     * The in memory encoding is always UTF8
907
     * This field will never change and would
908
     * be obsolete if not for binary compatibility.
909
     */
910
16.6k
    cur->charset = XML_CHAR_ENCODING_UTF8;
911
912
16.6k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
913
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)cur);
914
16.6k
    return(cur);
915
16.6k
}
916
917
/**
918
 * Free a document including all children and associated DTDs.
919
 *
920
 * @param cur  pointer to the document
921
 */
922
void
923
16.6k
xmlFreeDoc(xmlDoc *cur) {
924
16.6k
    xmlDtdPtr extSubset, intSubset;
925
16.6k
    xmlDictPtr dict = NULL;
926
927
16.6k
    if (cur == NULL) {
928
0
  return;
929
0
    }
930
931
16.6k
    dict = cur->dict;
932
933
16.6k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
934
0
  xmlDeregisterNodeDefaultValue((xmlNodePtr)cur);
935
936
    /*
937
     * Do this before freeing the children list to avoid ID lookups
938
     */
939
16.6k
    if (cur->ids != NULL) xmlFreeIDTable((xmlIDTablePtr) cur->ids);
940
16.6k
    cur->ids = NULL;
941
16.6k
    if (cur->refs != NULL) xmlFreeRefTable((xmlRefTablePtr) cur->refs);
942
16.6k
    cur->refs = NULL;
943
16.6k
    extSubset = cur->extSubset;
944
16.6k
    intSubset = cur->intSubset;
945
16.6k
    if (intSubset == extSubset)
946
12.8k
  extSubset = NULL;
947
16.6k
    if (extSubset != NULL) {
948
0
  xmlUnlinkNodeInternal((xmlNodePtr) cur->extSubset);
949
0
  cur->extSubset = NULL;
950
0
  xmlFreeDtd(extSubset);
951
0
    }
952
16.6k
    if (intSubset != NULL) {
953
5.37k
  xmlUnlinkNodeInternal((xmlNodePtr) cur->intSubset);
954
5.37k
  cur->intSubset = NULL;
955
5.37k
  xmlFreeDtd(intSubset);
956
5.37k
    }
957
958
16.6k
    if (cur->children != NULL) xmlFreeNodeList(cur->children);
959
16.6k
    if (cur->oldNs != NULL) xmlFreeNsList(cur->oldNs);
960
961
16.6k
    DICT_FREE(cur->name)
962
963
16.6k
    if (cur->version != NULL)
964
16.6k
        xmlFree(cur->version);
965
16.6k
    if (cur->encoding != NULL)
966
563
        xmlFree(cur->encoding);
967
16.6k
    if (cur->URL != NULL)
968
0
        xmlFree(cur->URL);
969
970
16.6k
    xmlFree(cur);
971
16.6k
    if (dict) xmlDictFree(dict);
972
16.6k
}
973
974
/**
975
 * Parse an attribute value and replace the node's children with
976
 * the resulting node list.
977
 *
978
 * `content` is expected to be a valid XML attribute value possibly
979
 * containing character and entity references. Syntax errors
980
 * and references to undeclared entities are ignored silently.
981
 * Only references are handled, nested elements, comments or PIs are
982
 * not.
983
 *
984
 * This function is used by the SAX parser to parse attribute
985
 * values a second time if entities aren't substituted. It's also
986
 * used to parse node content in the tree API, but this is a
987
 * historical mistake.
988
 *
989
 * @param doc  a document (optional)
990
 * @param attr  an attribute or element (optional)
991
 * @param value  an attribute value
992
 * @param len  maximum length of the attribute value
993
 * @param listPtr  pointer to the resulting node list (optional)
994
 * @returns 0 on success, -1 if a memory allocation failed.
995
 */
996
int
997
xmlNodeParseAttValue(const xmlDoc *doc, xmlAttr *attr,
998
0
                     const xmlChar *value, size_t len, xmlNode **listPtr) {
999
0
    xmlNodePtr head = NULL, last = NULL;
1000
0
    xmlNodePtr node;
1001
0
    xmlChar *val = NULL;
1002
0
    const xmlChar *cur;
1003
0
    const xmlChar *q;
1004
0
    xmlEntityPtr ent;
1005
0
    xmlBufPtr buf;
1006
0
    size_t remaining = len;
1007
1008
0
    if (listPtr != NULL)
1009
0
        *listPtr = NULL;
1010
1011
0
    if ((value == NULL) || (value[0] == 0))
1012
0
        goto done;
1013
1014
0
    cur = value;
1015
1016
0
    buf = xmlBufCreate(50);
1017
0
    if (buf == NULL)
1018
0
        return(-1);
1019
1020
0
    q = cur;
1021
0
    while ((remaining > 0) && (*cur != 0)) {
1022
0
  if (cur[0] == '&') {
1023
0
      int charval = 0;
1024
1025
      /*
1026
       * Save the current text.
1027
       */
1028
0
            if (cur != q) {
1029
0
    if (xmlBufAdd(buf, q, cur - q))
1030
0
        goto out;
1031
0
          q = cur;
1032
0
      }
1033
1034
0
      if ((remaining > 2) && (cur[1] == '#') && (cur[2] == 'x')) {
1035
0
          int tmp = 0;
1036
1037
0
    cur += 3;
1038
0
                remaining -= 3;
1039
0
    while ((remaining > 0) && ((tmp = *cur) != ';')) {
1040
0
        if ((tmp >= '0') && (tmp <= '9'))
1041
0
      charval = charval * 16 + (tmp - '0');
1042
0
        else if ((tmp >= 'a') && (tmp <= 'f'))
1043
0
      charval = charval * 16 + (tmp - 'a') + 10;
1044
0
        else if ((tmp >= 'A') && (tmp <= 'F'))
1045
0
      charval = charval * 16 + (tmp - 'A') + 10;
1046
0
        else {
1047
0
      charval = 0;
1048
0
      break;
1049
0
        }
1050
0
                    if (charval > 0x110000)
1051
0
                        charval = 0x110000;
1052
0
        cur++;
1053
0
                    remaining--;
1054
0
    }
1055
0
    if (tmp == ';') {
1056
0
        cur++;
1057
0
                    remaining--;
1058
0
                }
1059
0
    q = cur;
1060
0
      } else if ((remaining > 1) && (cur[1] == '#')) {
1061
0
          int tmp = 0;
1062
1063
0
    cur += 2;
1064
0
                remaining -= 2;
1065
0
    while ((remaining > 0) && ((tmp = *cur) != ';')) {
1066
0
        if ((tmp >= '0') && (tmp <= '9'))
1067
0
      charval = charval * 10 + (tmp - '0');
1068
0
        else {
1069
0
      charval = 0;
1070
0
      break;
1071
0
        }
1072
0
                    if (charval > 0x110000)
1073
0
                        charval = 0x110000;
1074
0
        cur++;
1075
0
                    remaining--;
1076
0
    }
1077
0
    if (tmp == ';') {
1078
0
        cur++;
1079
0
                    remaining--;
1080
0
                }
1081
0
    q = cur;
1082
0
      } else {
1083
    /*
1084
     * Read the entity string
1085
     */
1086
0
    cur++;
1087
0
                remaining--;
1088
0
    q = cur;
1089
0
    while ((remaining > 0) && (*cur != 0) && (*cur != ';')) {
1090
0
                    cur++;
1091
0
                    remaining--;
1092
0
                }
1093
0
    if ((remaining <= 0) || (*cur == 0))
1094
0
        break;
1095
0
    if (cur != q) {
1096
0
        val = xmlStrndup(q, cur - q);
1097
0
                    if (val == NULL)
1098
0
                        goto out;
1099
0
        ent = xmlGetDocEntity(doc, val);
1100
0
        if ((ent != NULL) &&
1101
0
      (ent->etype == XML_INTERNAL_PREDEFINED_ENTITY)) {
1102
                        /*
1103
                         * Predefined entities don't generate nodes
1104
                         */
1105
0
      if (xmlBufCat(buf, ent->content))
1106
0
          goto out;
1107
0
                    } else if (ent == NULL ||
1108
0
                               (ent->flags & XML_ENT_EXPANDING) == 0) {
1109
      /*
1110
       * Flush buffer so far
1111
       */
1112
0
      if (!xmlBufIsEmpty(buf)) {
1113
0
          node = xmlNewDocText(doc, NULL);
1114
0
          if (node == NULL)
1115
0
        goto out;
1116
0
          node->content = xmlBufDetach(buf);
1117
0
                            node->parent = (xmlNode *) attr;
1118
1119
0
          if (last == NULL) {
1120
0
        head = node;
1121
0
          } else {
1122
0
                                last->next = node;
1123
0
                                node->prev = last;
1124
0
          }
1125
0
                            last = node;
1126
0
      }
1127
1128
0
      if ((ent != NULL) &&
1129
0
                            ((ent->flags & XML_ENT_PARSED) == 0) &&
1130
0
                            (ent->content != NULL)) {
1131
0
                            int res;
1132
1133
0
                            ent->flags |= XML_ENT_EXPANDING;
1134
0
                            res = xmlNodeParseAttValue(doc, (xmlAttr *) ent,
1135
0
                                                       ent->content, SIZE_MAX,
1136
0
                                                       NULL);
1137
0
                            ent->flags &= ~XML_ENT_EXPANDING;
1138
0
                            if (res < 0)
1139
0
                                goto out;
1140
0
                            ent->flags |= XML_ENT_PARSED;
1141
0
      }
1142
1143
      /*
1144
       * Create a new REFERENCE_REF node
1145
       */
1146
0
      node = xmlNewEntityRef((xmlDocPtr) doc, val);
1147
0
                        val = NULL;
1148
0
      if (node == NULL)
1149
0
          goto out;
1150
0
                        node->parent = (xmlNode *) attr;
1151
0
                        node->last = (xmlNodePtr) ent;
1152
0
                        if (ent != NULL) {
1153
0
                            node->children = (xmlNodePtr) ent;
1154
0
                            node->content = ent->content;
1155
0
                        }
1156
1157
0
      if (last == NULL) {
1158
0
          head = node;
1159
0
      } else {
1160
0
                            last->next = node;
1161
0
                            node->prev = last;
1162
0
      }
1163
0
                        last = node;
1164
0
        }
1165
0
        xmlFree(val);
1166
0
                    val = NULL;
1167
0
    }
1168
0
    cur++;
1169
0
                remaining--;
1170
0
    q = cur;
1171
0
      }
1172
0
      if (charval != 0) {
1173
0
    xmlChar buffer[10];
1174
0
    int l;
1175
1176
0
                if (charval >= 0x110000)
1177
0
                    charval = 0xFFFD; /* replacement character */
1178
1179
0
    l = xmlCopyCharMultiByte(buffer, charval);
1180
0
    buffer[l] = 0;
1181
1182
0
    if (xmlBufCat(buf, buffer))
1183
0
        goto out;
1184
0
      }
1185
0
  } else {
1186
0
      cur++;
1187
0
            remaining--;
1188
0
        }
1189
0
    }
1190
1191
0
    if (cur != q) {
1192
        /*
1193
   * Handle the last piece of text.
1194
   */
1195
0
  if (xmlBufAdd(buf, q, cur - q))
1196
0
      goto out;
1197
0
    }
1198
1199
0
    if (!xmlBufIsEmpty(buf)) {
1200
0
  node = xmlNewDocText(doc, NULL);
1201
0
  if (node == NULL)
1202
0
            goto out;
1203
0
        node->parent = (xmlNode *) attr;
1204
0
  node->content = xmlBufDetach(buf);
1205
1206
0
  if (last == NULL) {
1207
0
      head = node;
1208
0
  } else {
1209
0
            last->next = node;
1210
0
            node->prev = last;
1211
0
  }
1212
0
        last = node;
1213
0
    } else if (head == NULL) {
1214
0
        head = xmlNewDocText(doc, BAD_CAST "");
1215
0
        if (head == NULL)
1216
0
            goto out;
1217
0
        head->parent = (xmlNode *) attr;
1218
0
        last = head;
1219
0
    }
1220
1221
0
    xmlBufFree(buf);
1222
1223
0
done:
1224
0
    if (attr != NULL) {
1225
0
        if (attr->children != NULL)
1226
0
            xmlFreeNodeList(attr->children);
1227
0
        attr->children = head;
1228
0
        attr->last = last;
1229
0
    }
1230
1231
0
    if (listPtr != NULL)
1232
0
        *listPtr = head;
1233
1234
0
    return(0);
1235
1236
0
out:
1237
0
    xmlBufFree(buf);
1238
0
    if (val != NULL)
1239
0
        xmlFree(val);
1240
0
    if (head != NULL)
1241
0
        xmlFreeNodeList(head);
1242
0
    return(-1);
1243
0
}
1244
1245
/**
1246
 * See #xmlStringGetNodeList.
1247
 *
1248
 * @deprecated Internal function, don't use.
1249
 *
1250
 * @param doc  a document (optional)
1251
 * @param value  an attribute value
1252
 * @param len  maximum length of the attribute value
1253
 * @returns a pointer to the first child or NULL if the value if empty
1254
 * or a memory allocation failed.
1255
 */
1256
xmlNode *
1257
0
xmlStringLenGetNodeList(const xmlDoc *doc, const xmlChar *value, int len) {
1258
0
    xmlNodePtr ret;
1259
0
    size_t maxSize = len < 0 ? SIZE_MAX : (size_t) len;
1260
1261
0
    xmlNodeParseAttValue(doc, NULL, value, maxSize, &ret);
1262
0
    return(ret);
1263
0
}
1264
1265
/**
1266
 * Parse an attribute value and build a node list with
1267
 * text and entity reference nodes. The resulting nodes will be
1268
 * associated with the document if provided. The document is also
1269
 * used to look up entities.
1270
 *
1271
 * @deprecated Internal function, don't use.
1272
 *
1273
 * The input is not validated. Syntax errors or references to
1274
 * undeclared entities will be ignored silently with unspecified
1275
 * results.
1276
 *
1277
 * @param doc  a document (optional)
1278
 * @param value  an attribute value
1279
 * @returns a pointer to the first child or NULL if the value if empty
1280
 * or a memory allocation failed.
1281
 */
1282
xmlNode *
1283
0
xmlStringGetNodeList(const xmlDoc *doc, const xmlChar *value) {
1284
0
    xmlNodePtr ret;
1285
1286
0
    xmlNodeParseAttValue(doc, NULL, value, SIZE_MAX, &ret);
1287
0
    return(ret);
1288
0
}
1289
1290
/**
1291
 * @param node  a node list
1292
 * @param escape  whether to escape characters and keep entity refs
1293
 * @param flags  escape flags
1294
 * @returns a pointer to the string.
1295
 */
1296
xmlChar *
1297
0
xmlNodeListGetStringInternal(const xmlNode *node, int escape, int flags) {
1298
0
    xmlBufPtr buf;
1299
0
    xmlChar *ret;
1300
1301
0
    if (node == NULL)
1302
0
        return(xmlStrdup(BAD_CAST ""));
1303
1304
0
    if ((escape == 0) &&
1305
0
        ((node->type == XML_TEXT_NODE) ||
1306
0
         (node->type == XML_CDATA_SECTION_NODE)) &&
1307
0
        (node->next == NULL)) {
1308
0
        if (node->content == NULL)
1309
0
            return(xmlStrdup(BAD_CAST ""));
1310
0
        return(xmlStrdup(node->content));
1311
0
    }
1312
1313
0
    buf = xmlBufCreate(50);
1314
0
    if (buf == NULL)
1315
0
        return(NULL);
1316
1317
0
    while (node != NULL) {
1318
0
        if ((node->type == XML_TEXT_NODE) ||
1319
0
            (node->type == XML_CDATA_SECTION_NODE)) {
1320
0
            if (node->content != NULL) {
1321
0
                if (escape == 0) {
1322
0
                    xmlBufCat(buf, node->content);
1323
0
                } else {
1324
0
                    xmlChar *encoded;
1325
1326
0
                    encoded = xmlEscapeText(node->content, flags);
1327
0
                    if (encoded == NULL)
1328
0
                        goto error;
1329
0
                    xmlBufCat(buf, encoded);
1330
0
                    xmlFree(encoded);
1331
0
                }
1332
0
            }
1333
0
        } else if (node->type == XML_ENTITY_REF_NODE) {
1334
0
            if (escape == 0) {
1335
0
                xmlBufGetNodeContent(buf, node);
1336
0
            } else {
1337
0
                xmlBufAdd(buf, BAD_CAST "&", 1);
1338
0
                xmlBufCat(buf, node->name);
1339
0
                xmlBufAdd(buf, BAD_CAST ";", 1);
1340
0
            }
1341
0
        }
1342
1343
0
        node = node->next;
1344
0
    }
1345
1346
0
    ret = xmlBufDetach(buf);
1347
0
    xmlBufFree(buf);
1348
0
    return(ret);
1349
1350
0
error:
1351
0
    xmlBufFree(buf);
1352
0
    return(NULL);
1353
0
}
1354
1355
/**
1356
 * Serializes attribute children (text and entity reference nodes)
1357
 * into a string.
1358
 *
1359
 * If `inLine` is true, entity references will be substituted.
1360
 * Otherwise, entity references will be kept and special characters
1361
 * like `&` as well as non-ASCII chars will be escaped. See
1362
 * #xmlEncodeEntitiesReentrant for details. If `list` is the child
1363
 * of an attribute, escaping rules apply are adjusted.
1364
 *
1365
 * See #xmlNodeListGetRawString for an alternative option.
1366
 *
1367
 * @param doc  a document (optional)
1368
 * @param list  a node list of attribute children
1369
 * @param inLine  whether entity references are substituted
1370
 * @returns a string or NULL if a memory allocation failed.
1371
 */
1372
xmlChar *
1373
xmlNodeListGetString(xmlDoc *doc, const xmlNode *list, int inLine)
1374
0
{
1375
0
    int flags = 0;
1376
0
    int escape = 0;
1377
1378
    /* backward compatibility */
1379
0
    if (list == NULL)
1380
0
        return(NULL);
1381
1382
0
    if (!inLine) {
1383
0
        escape = 1;
1384
1385
0
        if ((doc != NULL) && (doc->type == XML_HTML_DOCUMENT_NODE))
1386
0
            flags |= XML_ESCAPE_HTML;
1387
0
        else if ((doc == NULL) || (doc->encoding == NULL))
1388
0
            flags |= XML_ESCAPE_NON_ASCII;
1389
1390
0
        if ((list->parent != NULL) &&
1391
0
            (list->parent->type == XML_ATTRIBUTE_NODE))
1392
0
            flags |= XML_ESCAPE_ATTR;
1393
0
    }
1394
1395
0
    return(xmlNodeListGetStringInternal(list, escape, flags));
1396
0
}
1397
1398
/**
1399
 * Serializes attribute children (text and entity reference nodes)
1400
 * into a string.
1401
 *
1402
 * If `inLine` is true, entity references will be substituted.
1403
 * Otherwise, entity references will be kept and special characters
1404
 * like `&` will be escaped. See #xmlEncodeSpecialChars for
1405
 * details.
1406
 *
1407
 * @param doc  a document (unused)
1408
 * @param list  a node list of attribute children
1409
 * @param inLine  whether entity references are substituted
1410
 * @returns a string or NULL if a memory allocation failed.
1411
 */
1412
xmlChar *
1413
xmlNodeListGetRawString(const xmlDoc *doc ATTRIBUTE_UNUSED,
1414
                        const xmlNode *list, int inLine)
1415
0
{
1416
0
    int escape = 0;
1417
0
    int flags = 0;
1418
1419
    /* backward compatibility */
1420
0
    if (list == NULL)
1421
0
        return(NULL);
1422
1423
0
    if (!inLine) {
1424
0
        escape = 1;
1425
0
        flags = XML_ESCAPE_QUOT;
1426
0
    }
1427
1428
0
    return(xmlNodeListGetStringInternal(list, escape, flags));
1429
0
}
1430
1431
static xmlAttrPtr
1432
xmlNewPropInternal(xmlNodePtr node, xmlNsPtr ns,
1433
                   const xmlChar * name, const xmlChar * value,
1434
                   int eatname)
1435
0
{
1436
0
    xmlAttrPtr cur;
1437
0
    xmlDocPtr doc = NULL;
1438
1439
0
    if ((node != NULL) && (node->type != XML_ELEMENT_NODE)) {
1440
0
        if ((eatname == 1) &&
1441
0
      ((node->doc == NULL) || (node->doc->dict == NULL) ||
1442
0
       (!(xmlDictOwns(node->doc->dict, name)))))
1443
0
            xmlFree((xmlChar *) name);
1444
0
        return (NULL);
1445
0
    }
1446
1447
    /*
1448
     * Allocate a new property and fill the fields.
1449
     */
1450
0
    cur = (xmlAttrPtr) xmlMalloc(sizeof(xmlAttr));
1451
0
    if (cur == NULL) {
1452
0
        if ((eatname == 1) &&
1453
0
      ((node == NULL) || (node->doc == NULL) ||
1454
0
             (node->doc->dict == NULL) ||
1455
0
       (!(xmlDictOwns(node->doc->dict, name)))))
1456
0
            xmlFree((xmlChar *) name);
1457
0
        return (NULL);
1458
0
    }
1459
0
    memset(cur, 0, sizeof(xmlAttr));
1460
0
    cur->type = XML_ATTRIBUTE_NODE;
1461
1462
0
    cur->parent = node;
1463
0
    if (node != NULL) {
1464
0
        doc = node->doc;
1465
0
        cur->doc = doc;
1466
0
    }
1467
0
    cur->ns = ns;
1468
1469
0
    if (eatname == 0) {
1470
0
        if ((doc != NULL) && (doc->dict != NULL))
1471
0
            cur->name = (xmlChar *) xmlDictLookup(doc->dict, name, -1);
1472
0
        else
1473
0
            cur->name = xmlStrdup(name);
1474
0
        if (cur->name == NULL)
1475
0
            goto error;
1476
0
    } else
1477
0
        cur->name = name;
1478
1479
0
    if (value != NULL) {
1480
0
        xmlNodePtr tmp;
1481
1482
0
        cur->children = xmlNewDocText(doc, value);
1483
0
        if (cur->children == NULL)
1484
0
            goto error;
1485
0
        cur->last = NULL;
1486
0
        tmp = cur->children;
1487
0
        while (tmp != NULL) {
1488
0
            tmp->parent = (xmlNodePtr) cur;
1489
0
            if (tmp->next == NULL)
1490
0
                cur->last = tmp;
1491
0
            tmp = tmp->next;
1492
0
        }
1493
1494
0
        if (doc != NULL) {
1495
0
            int res = xmlIsID(doc, node, cur);
1496
1497
0
            if (res < 0)
1498
0
                goto error;
1499
0
            if ((res == 1) && (xmlAddIDSafe(cur, value) < 0))
1500
0
                goto error;
1501
0
        }
1502
0
    }
1503
1504
    /*
1505
     * Add it at the end to preserve parsing order ...
1506
     */
1507
0
    if (node != NULL) {
1508
0
        if (node->properties == NULL) {
1509
0
            node->properties = cur;
1510
0
        } else {
1511
0
            xmlAttrPtr prev = node->properties;
1512
1513
0
            while (prev->next != NULL)
1514
0
                prev = prev->next;
1515
0
            prev->next = cur;
1516
0
            cur->prev = prev;
1517
0
        }
1518
0
    }
1519
1520
0
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
1521
0
        xmlRegisterNodeDefaultValue((xmlNodePtr) cur);
1522
0
    return (cur);
1523
1524
0
error:
1525
0
    xmlFreeProp(cur);
1526
0
    return(NULL);
1527
0
}
1528
1529
/**
1530
 * Create an attribute node.
1531
 *
1532
 * If provided, `value` should be a raw, unescaped string.
1533
 *
1534
 * If `node` is provided, the created attribute will be appended without
1535
 * checking for duplicate names. It is an error if `node` is not an
1536
 * element.
1537
 *
1538
 * @param node  the parent node (optional)
1539
 * @param name  the name of the attribute
1540
 * @param value  the value of the attribute (optional)
1541
 * @returns a pointer to the attribute or NULL if arguments are invalid
1542
 * or a memory allocation failed.
1543
 */
1544
xmlAttr *
1545
0
xmlNewProp(xmlNode *node, const xmlChar *name, const xmlChar *value) {
1546
1547
0
    if (name == NULL) {
1548
0
  return(NULL);
1549
0
    }
1550
1551
0
  return xmlNewPropInternal(node, NULL, name, value, 0);
1552
0
}
1553
1554
/**
1555
 * Create an attribute node.
1556
 *
1557
 * If provided, `value` should be a raw, unescaped string.
1558
 *
1559
 * If `node` is provided, the created attribute will be appended without
1560
 * checking for duplicate names. It is an error if `node` is not an
1561
 * element.
1562
 *
1563
 * @param node  the parent node (optional)
1564
 * @param ns  the namespace (optional)
1565
 * @param name  the local name of the attribute
1566
 * @param value  the value of the attribute (optional)
1567
 * @returns a pointer to the attribute or NULL if arguments are invalid
1568
 * or a memory allocation failed.
1569
 */
1570
xmlAttr *
1571
xmlNewNsProp(xmlNode *node, xmlNs *ns, const xmlChar *name,
1572
0
           const xmlChar *value) {
1573
1574
0
    if (name == NULL) {
1575
0
  return(NULL);
1576
0
    }
1577
1578
0
    return xmlNewPropInternal(node, ns, name, value, 0);
1579
0
}
1580
1581
/**
1582
 * Create an attribute node.
1583
 *
1584
 * Like #xmlNewNsProp, but the `name` string will be used directly
1585
 * without making a copy. Takes ownership of `name` which will also
1586
 * be freed on error.
1587
 *
1588
 * @param node  the parent node (optional)
1589
 * @param ns  the namespace (optional)
1590
 * @param name  the local name of the attribute
1591
 * @param value  the value of the attribute (optional)
1592
 * @returns a pointer to the attribute or NULL if arguments are invalid
1593
 * or a memory allocation failed.
1594
 */
1595
xmlAttr *
1596
xmlNewNsPropEatName(xmlNode *node, xmlNs *ns, xmlChar *name,
1597
0
           const xmlChar *value) {
1598
1599
0
    if (name == NULL) {
1600
0
  return(NULL);
1601
0
    }
1602
1603
0
    return xmlNewPropInternal(node, ns, name, value, 1);
1604
0
}
1605
1606
/**
1607
 * Create an attribute node.
1608
 *
1609
 * If provided, `value` is expected to be a valid XML attribute value
1610
 * possibly containing character and entity references. Syntax errors
1611
 * and references to undeclared entities are ignored silently.
1612
 * If you want to pass a raw string, see #xmlNewProp.
1613
 *
1614
 * @param doc  the target document (optional)
1615
 * @param name  the name of the attribute
1616
 * @param value  attribute value with XML references (optional)
1617
 * @returns a pointer to the attribute or NULL if arguments are invalid
1618
 * or a memory allocation failed.
1619
 */
1620
xmlAttr *
1621
1.97k
xmlNewDocProp(xmlDoc *doc, const xmlChar *name, const xmlChar *value) {
1622
1.97k
    xmlAttrPtr cur;
1623
1624
1.97k
    if (name == NULL) {
1625
0
  return(NULL);
1626
0
    }
1627
1628
    /*
1629
     * Allocate a new property and fill the fields.
1630
     */
1631
1.97k
    cur = (xmlAttrPtr) xmlMalloc(sizeof(xmlAttr));
1632
1.97k
    if (cur == NULL)
1633
0
  return(NULL);
1634
1.97k
    memset(cur, 0, sizeof(xmlAttr));
1635
1.97k
    cur->type = XML_ATTRIBUTE_NODE;
1636
1637
1.97k
    if ((doc != NULL) && (doc->dict != NULL))
1638
0
  cur->name = xmlDictLookup(doc->dict, name, -1);
1639
1.97k
    else
1640
1.97k
  cur->name = xmlStrdup(name);
1641
1.97k
    if (cur->name == NULL)
1642
0
        goto error;
1643
1.97k
    cur->doc = doc;
1644
1.97k
    if (value != NULL) {
1645
        /*
1646
         * We shouldn't parse the attribute value here,
1647
         * but the API can't be changed.
1648
         */
1649
0
        if (xmlNodeParseAttValue(doc, cur, value, SIZE_MAX, NULL) < 0)
1650
0
            goto error;
1651
0
    }
1652
1653
1.97k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
1654
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)cur);
1655
1.97k
    return(cur);
1656
1657
0
error:
1658
0
    xmlFreeProp(cur);
1659
0
    return(NULL);
1660
1.97k
}
1661
1662
/**
1663
 * Free an attribute list including all children.
1664
 *
1665
 * @param cur  the first attribute in the list
1666
 */
1667
void
1668
31.8k
xmlFreePropList(xmlAttr *cur) {
1669
31.8k
    xmlAttrPtr next;
1670
31.8k
    if (cur == NULL) return;
1671
75.3k
    while (cur != NULL) {
1672
43.4k
        next = cur->next;
1673
43.4k
        xmlFreeProp(cur);
1674
43.4k
  cur = next;
1675
43.4k
    }
1676
31.8k
}
1677
1678
/**
1679
 * Free an attribute including all children.
1680
 *
1681
 * @param cur  an attribute
1682
 */
1683
void
1684
43.4k
xmlFreeProp(xmlAttr *cur) {
1685
43.4k
    xmlDictPtr dict = NULL;
1686
43.4k
    if (cur == NULL) return;
1687
1688
43.4k
    if (cur->doc != NULL) dict = cur->doc->dict;
1689
1690
43.4k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
1691
0
  xmlDeregisterNodeDefaultValue((xmlNodePtr)cur);
1692
1693
    /* Check for ID removal -> leading to invalid references ! */
1694
43.4k
    if (cur->doc != NULL && cur->id != NULL) {
1695
10
        xmlRemoveID(cur->doc, cur);
1696
10
    }
1697
43.4k
    if (cur->children != NULL) xmlFreeNodeList(cur->children);
1698
43.4k
    DICT_FREE(cur->name)
1699
43.4k
    xmlFree(cur);
1700
43.4k
}
1701
1702
/**
1703
 * Unlink and free an attribute including all children.
1704
 *
1705
 * Note this doesn't work for namespace declarations.
1706
 *
1707
 * The attribute must have a non-NULL parent pointer.
1708
 *
1709
 * @param cur  an attribute
1710
 * @returns 0 on success or -1 if the attribute was not found or
1711
 * arguments are invalid.
1712
 */
1713
int
1714
0
xmlRemoveProp(xmlAttr *cur) {
1715
0
    xmlAttrPtr tmp;
1716
0
    if (cur == NULL) {
1717
0
  return(-1);
1718
0
    }
1719
0
    if (cur->parent == NULL) {
1720
0
  return(-1);
1721
0
    }
1722
0
    tmp = cur->parent->properties;
1723
0
    if (tmp == cur) {
1724
0
        cur->parent->properties = cur->next;
1725
0
    if (cur->next != NULL)
1726
0
      cur->next->prev = NULL;
1727
0
  xmlFreeProp(cur);
1728
0
  return(0);
1729
0
    }
1730
0
    while (tmp != NULL) {
1731
0
  if (tmp->next == cur) {
1732
0
      tmp->next = cur->next;
1733
0
      if (tmp->next != NULL)
1734
0
    tmp->next->prev = tmp;
1735
0
      xmlFreeProp(cur);
1736
0
      return(0);
1737
0
  }
1738
0
        tmp = tmp->next;
1739
0
    }
1740
0
    return(-1);
1741
0
}
1742
1743
/**
1744
 * Create a processing instruction object.
1745
 *
1746
 * @param doc  the target document (optional)
1747
 * @param name  the processing instruction target
1748
 * @param content  the PI content (optional)
1749
 * @returns a pointer to the new node object or NULL if arguments are
1750
 * invalid or a memory allocation failed.
1751
 */
1752
xmlNode *
1753
3.14k
xmlNewDocPI(xmlDoc *doc, const xmlChar *name, const xmlChar *content) {
1754
3.14k
    xmlNodePtr cur;
1755
1756
3.14k
    if (name == NULL) {
1757
0
  return(NULL);
1758
0
    }
1759
1760
    /*
1761
     * Allocate a new node and fill the fields.
1762
     */
1763
3.14k
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
1764
3.14k
    if (cur == NULL)
1765
0
  return(NULL);
1766
3.14k
    memset(cur, 0, sizeof(xmlNode));
1767
3.14k
    cur->type = XML_PI_NODE;
1768
3.14k
    cur->doc = doc;
1769
1770
3.14k
    if ((doc != NULL) && (doc->dict != NULL))
1771
0
        cur->name = xmlDictLookup(doc->dict, name, -1);
1772
3.14k
    else
1773
3.14k
  cur->name = xmlStrdup(name);
1774
3.14k
    if (cur->name == NULL)
1775
0
        goto error;
1776
3.14k
    if (content != NULL) {
1777
1.43k
  cur->content = xmlStrdup(content);
1778
1.43k
        if (cur->content == NULL)
1779
0
            goto error;
1780
1.43k
    }
1781
1782
3.14k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
1783
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)cur);
1784
3.14k
    return(cur);
1785
1786
0
error:
1787
0
    xmlFreeNode(cur);
1788
0
    return(NULL);
1789
3.14k
}
1790
1791
/**
1792
 * Create a processing instruction node.
1793
 *
1794
 * Use of this function is DISCOURAGED in favor of #xmlNewDocPI.
1795
 *
1796
 * @param name  the processing instruction target
1797
 * @param content  the PI content (optional)
1798
 * @returns a pointer to the new node object or NULL if arguments are
1799
 * invalid or a memory allocation failed.
1800
 */
1801
xmlNode *
1802
0
xmlNewPI(const xmlChar *name, const xmlChar *content) {
1803
0
    return(xmlNewDocPI(NULL, name, content));
1804
0
}
1805
1806
/**
1807
 * Create an element node.
1808
 *
1809
 * Use of this function is DISCOURAGED in favor of #xmlNewDocNode.
1810
 *
1811
 * @param ns  namespace (optional)
1812
 * @param name  the node name
1813
 * @returns a pointer to the new node object or NULL if arguments are
1814
 * invalid or a memory allocation failed.
1815
 */
1816
xmlNode *
1817
0
xmlNewNode(xmlNs *ns, const xmlChar *name) {
1818
0
    return(xmlNewDocNode(NULL, ns, name, NULL));
1819
0
}
1820
1821
/**
1822
 * Create an element node.
1823
 *
1824
 * Use of this function is DISCOURAGED in favor of #xmlNewDocNodeEatName.
1825
 *
1826
 * Like #xmlNewNode, but the `name` string will be used directly
1827
 * without making a copy. Takes ownership of `name` which will also
1828
 * be freed on error.
1829
 *
1830
 * @param ns  namespace (optional)
1831
 * @param name  the node name
1832
 * @returns a pointer to the new node object or NULL if arguments are
1833
 * invalid or a memory allocation failed.
1834
 */
1835
xmlNode *
1836
0
xmlNewNodeEatName(xmlNs *ns, xmlChar *name) {
1837
0
    return(xmlNewDocNodeEatName(NULL, ns, name, NULL));
1838
0
}
1839
1840
static xmlNodePtr
1841
xmlNewElem(xmlDocPtr doc, xmlNsPtr ns, const xmlChar *name,
1842
67.6k
           const xmlChar *content) {
1843
67.6k
    xmlNodePtr cur;
1844
1845
67.6k
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
1846
67.6k
    if (cur == NULL)
1847
0
  return(NULL);
1848
67.6k
    memset(cur, 0, sizeof(xmlNode));
1849
1850
67.6k
    cur->type = XML_ELEMENT_NODE;
1851
67.6k
    cur->doc = doc;
1852
67.6k
    cur->name = name;
1853
67.6k
    cur->ns = ns;
1854
1855
67.6k
    if (content != NULL) {
1856
        /*
1857
         * We shouldn't parse the content as attribute value here,
1858
         * but the API can't be changed.
1859
         */
1860
0
        if (xmlNodeParseAttValue(doc, (xmlAttr *) cur, content, SIZE_MAX,
1861
0
                                 NULL) < 0) {
1862
            /* Don't free name on error */
1863
0
            xmlFree(cur);
1864
0
            return(NULL);
1865
0
        }
1866
0
    }
1867
1868
67.6k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
1869
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)cur);
1870
1871
67.6k
    return(cur);
1872
67.6k
}
1873
1874
/**
1875
 * Create an element node.
1876
 *
1877
 * If provided, `content` is expected to be a valid XML attribute value
1878
 * possibly containing character and entity references. Syntax errors
1879
 * and references to undeclared entities are ignored silently.
1880
 * Only references are handled, nested elements, comments or PIs are
1881
 * not. See #xmlNewDocRawNode for an alternative.
1882
 *
1883
 * General notes on object creation:
1884
 *
1885
 * Each node and all its children are associated with the same
1886
 * document. The document should be provided when creating nodes to
1887
 * avoid a performance penalty when adding the node to a document
1888
 * tree. Note that a document only owns nodes reachable from the root
1889
 * node. Unlinked subtrees must be freed manually.
1890
 *
1891
 * @param doc  the target document
1892
 * @param ns  namespace (optional)
1893
 * @param name  the node name
1894
 * @param content  text content with XML references (optional)
1895
 * @returns a pointer to the new node object or NULL if arguments are
1896
 * invalid or a memory allocation failed.
1897
 */
1898
xmlNode *
1899
xmlNewDocNode(xmlDoc *doc, xmlNs *ns,
1900
67.1k
              const xmlChar *name, const xmlChar *content) {
1901
67.1k
    xmlNodePtr cur;
1902
67.1k
    xmlChar *copy;
1903
1904
67.1k
    if (name == NULL)
1905
0
        return(NULL);
1906
1907
67.1k
    if ((doc != NULL) && (doc->dict != NULL)) {
1908
0
        const xmlChar *dictName = xmlDictLookup(doc->dict, name, -1);
1909
1910
0
        if (dictName == NULL)
1911
0
            return(NULL);
1912
0
        return(xmlNewElem(doc, ns, dictName, content));
1913
0
    }
1914
1915
67.1k
    copy = xmlStrdup(name);
1916
67.1k
    if (copy == NULL)
1917
0
        return(NULL);
1918
1919
67.1k
    cur = xmlNewElem(doc, ns, copy, content);
1920
67.1k
    if (cur == NULL) {
1921
0
        xmlFree(copy);
1922
0
        return(NULL);
1923
0
    }
1924
1925
67.1k
    return(cur);
1926
67.1k
}
1927
1928
/**
1929
 * Create an element node.
1930
 *
1931
 * Like #xmlNewDocNode, but the `name` string will be used directly
1932
 * without making a copy. Takes ownership of `name` which will also
1933
 * be freed on error.
1934
 *
1935
 * @param doc  the target document
1936
 * @param ns  namespace (optional)
1937
 * @param name  the node name
1938
 * @param content  text content with XML references (optional)
1939
 * @returns a pointer to the new node object or NULL if arguments are
1940
 * invalid or a memory allocation failed.
1941
 */
1942
xmlNode *
1943
xmlNewDocNodeEatName(xmlDoc *doc, xmlNs *ns,
1944
505
                     xmlChar *name, const xmlChar *content) {
1945
505
    xmlNodePtr cur;
1946
1947
505
    if (name == NULL)
1948
0
        return(NULL);
1949
1950
505
    cur = xmlNewElem(doc, ns, name, content);
1951
505
    if (cur == NULL) {
1952
        /* if name doesn't come from the doc dictionary free it here */
1953
0
        if ((doc == NULL) ||
1954
0
            (doc->dict == NULL) ||
1955
0
            (!xmlDictOwns(doc->dict, name)))
1956
0
            xmlFree(name);
1957
0
        return(NULL);
1958
0
    }
1959
1960
505
    return(cur);
1961
505
}
1962
1963
/**
1964
 * Create an element node.
1965
 *
1966
 * If provided, `value` should be a raw, unescaped string.
1967
 *
1968
 * @param doc  the target document
1969
 * @param ns  namespace (optional)
1970
 * @param name  the node name
1971
 * @param content  raw text content (optional)
1972
 * @returns a pointer to the new node object or NULL if arguments are
1973
 * invalid or a memory allocation failed.
1974
 */
1975
xmlNode *
1976
xmlNewDocRawNode(xmlDoc *doc, xmlNs *ns,
1977
0
                 const xmlChar *name, const xmlChar *content) {
1978
0
    xmlNodePtr cur;
1979
1980
0
    cur = xmlNewDocNode(doc, ns, name, NULL);
1981
0
    if (cur != NULL) {
1982
0
        cur->doc = doc;
1983
0
  if (content != NULL) {
1984
0
            xmlNodePtr text;
1985
1986
0
      text = xmlNewDocText(doc, content);
1987
0
            if (text == NULL) {
1988
0
                xmlFreeNode(cur);
1989
0
                return(NULL);
1990
0
            }
1991
1992
0
            cur->children = text;
1993
0
            cur->last = text;
1994
0
            text->parent = cur;
1995
0
  }
1996
0
    }
1997
0
    return(cur);
1998
0
}
1999
2000
/**
2001
 * Create a document fragment node.
2002
 *
2003
 * @param doc  the target document (optional)
2004
 * @returns a pointer to the new node object or NULL if a memory
2005
 * allocation failed.
2006
 */
2007
xmlNode *
2008
0
xmlNewDocFragment(xmlDoc *doc) {
2009
0
    xmlNodePtr cur;
2010
2011
    /*
2012
     * Allocate a new DocumentFragment node and fill the fields.
2013
     */
2014
0
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2015
0
    if (cur == NULL)
2016
0
  return(NULL);
2017
0
    memset(cur, 0, sizeof(xmlNode));
2018
0
    cur->type = XML_DOCUMENT_FRAG_NODE;
2019
2020
0
    cur->doc = doc;
2021
2022
0
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2023
0
  xmlRegisterNodeDefaultValue(cur);
2024
0
    return(cur);
2025
0
}
2026
2027
/**
2028
 * Create a text node.
2029
 *
2030
 * Use of this function is DISCOURAGED in favor of #xmlNewDocText.
2031
 *
2032
 * @param content  raw text content (optional)
2033
 * @returns a pointer to the new node object or NULL if a memory
2034
 * allocation failed.
2035
 */
2036
xmlNode *
2037
0
xmlNewText(const xmlChar *content) {
2038
0
    xmlNodePtr cur;
2039
2040
    /*
2041
     * Allocate a new node and fill the fields.
2042
     */
2043
0
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2044
0
    if (cur == NULL)
2045
0
  return(NULL);
2046
0
    memset(cur, 0, sizeof(xmlNode));
2047
0
    cur->type = XML_TEXT_NODE;
2048
2049
0
    cur->name = xmlStringText;
2050
0
    if (content != NULL) {
2051
0
  cur->content = xmlStrdup(content);
2052
0
        if (cur->content == NULL)
2053
0
            goto error;
2054
0
    }
2055
2056
0
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2057
0
  xmlRegisterNodeDefaultValue(cur);
2058
0
    return(cur);
2059
2060
0
error:
2061
0
    xmlFreeNode(cur);
2062
0
    return(NULL);
2063
0
}
2064
2065
/**
2066
 * Create a new child element and append it to a parent element.
2067
 *
2068
 * If `ns` is NULL, the newly created element inherits the namespace
2069
 * of the parent.
2070
 *
2071
 * If `content` is provided, a text node will be added to the child
2072
 * element, see #xmlNewDocRawNode.
2073
 *
2074
 * @param parent  the parent node
2075
 * @param ns  a namespace (optional)
2076
 * @param name  the name of the child
2077
 * @param content  raw text content of the child (optional)
2078
 * @returns a pointer to the new node object or NULL if arguments
2079
 * are invalid or a memory allocation failed.
2080
 */
2081
xmlNode *
2082
xmlNewTextChild(xmlNode *parent, xmlNs *ns,
2083
0
            const xmlChar *name, const xmlChar *content) {
2084
0
    xmlNodePtr cur, prev;
2085
2086
0
    if ((parent == NULL) || (name == NULL))
2087
0
  return(NULL);
2088
2089
0
    switch (parent->type) {
2090
0
        case XML_DOCUMENT_NODE:
2091
0
        case XML_HTML_DOCUMENT_NODE:
2092
0
        case XML_DOCUMENT_FRAG_NODE:
2093
0
            break;
2094
2095
0
        case XML_ELEMENT_NODE:
2096
0
            if (ns == NULL)
2097
0
                ns = parent->ns;
2098
0
            break;
2099
2100
0
        default:
2101
0
            return(NULL);
2102
0
    }
2103
2104
0
    cur = xmlNewDocRawNode(parent->doc, ns, name, content);
2105
0
    if (cur == NULL)
2106
0
        return(NULL);
2107
2108
    /*
2109
     * add the new element at the end of the children list.
2110
     */
2111
0
    cur->parent = parent;
2112
0
    if (parent->children == NULL) {
2113
0
        parent->children = cur;
2114
0
  parent->last = cur;
2115
0
    } else {
2116
0
        prev = parent->last;
2117
0
  prev->next = cur;
2118
0
  cur->prev = prev;
2119
0
  parent->last = cur;
2120
0
    }
2121
2122
0
    return(cur);
2123
0
}
2124
2125
/**
2126
 * Create an empty entity reference node. This function doesn't attempt
2127
 * to look up the entity in `doc`.
2128
 *
2129
 * `name` is consumed.
2130
 *
2131
 * @param doc  the target document (optional)
2132
 * @param name  the entity name
2133
 * @returns a pointer to the new node object or NULL if arguments are
2134
 * invalid or a memory allocation failed.
2135
 */
2136
static xmlNodePtr
2137
0
xmlNewEntityRef(xmlDocPtr doc, xmlChar *name) {
2138
0
    xmlNodePtr cur;
2139
2140
    /*
2141
     * Allocate a new node and fill the fields.
2142
     */
2143
0
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2144
0
    if (cur == NULL) {
2145
0
        xmlFree(name);
2146
0
  return(NULL);
2147
0
    }
2148
0
    memset(cur, 0, sizeof(xmlNode));
2149
0
    cur->type = XML_ENTITY_REF_NODE;
2150
0
    cur->doc = doc;
2151
0
    cur->name = name;
2152
2153
0
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2154
0
  xmlRegisterNodeDefaultValue(cur);
2155
2156
0
    return(cur);
2157
0
}
2158
2159
/**
2160
 * Create an empty entity reference node.
2161
 *
2162
 * This function is MISNAMED. It doesn't create a character reference
2163
 * but an entity reference.
2164
 *
2165
 * This function doesn't attempt to look up the entity in `doc`.
2166
 *
2167
 * Entity names like `&entity;` are handled as well.
2168
 *
2169
 * @param doc  the target document (optional)
2170
 * @param name  the entity name
2171
 * @returns a pointer to the new node object or NULL if arguments are
2172
 * invalid or a memory allocation failed.
2173
 */
2174
xmlNode *
2175
0
xmlNewCharRef(xmlDoc *doc, const xmlChar *name) {
2176
0
    xmlChar *copy;
2177
2178
0
    if (name == NULL)
2179
0
        return(NULL);
2180
2181
0
    if (name[0] == '&') {
2182
0
        int len;
2183
0
        name++;
2184
0
  len = xmlStrlen(name);
2185
0
  if (name[len - 1] == ';')
2186
0
      copy = xmlStrndup(name, len - 1);
2187
0
  else
2188
0
      copy = xmlStrndup(name, len);
2189
0
    } else
2190
0
  copy = xmlStrdup(name);
2191
0
    if (copy == NULL)
2192
0
        return(NULL);
2193
2194
0
    return(xmlNewEntityRef(doc, copy));
2195
0
}
2196
2197
/**
2198
 * Create a new entity reference node, linking the result with the
2199
 * entity in `doc` if found.
2200
 *
2201
 * Entity names like `&entity;` are handled as well.
2202
 *
2203
 * @param doc  the target document (optional)
2204
 * @param name  the entity name
2205
 * @returns a pointer to the new node object or NULL if arguments are
2206
 * invalid or a memory allocation failed.
2207
 */
2208
xmlNode *
2209
0
xmlNewReference(const xmlDoc *doc, const xmlChar *name) {
2210
0
    xmlNodePtr cur;
2211
0
    xmlEntityPtr ent;
2212
2213
0
    if (name == NULL)
2214
0
        return(NULL);
2215
2216
    /*
2217
     * Allocate a new node and fill the fields.
2218
     */
2219
0
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2220
0
    if (cur == NULL)
2221
0
  return(NULL);
2222
0
    memset(cur, 0, sizeof(xmlNode));
2223
0
    cur->type = XML_ENTITY_REF_NODE;
2224
2225
0
    cur->doc = (xmlDoc *)doc;
2226
0
    if (name[0] == '&') {
2227
0
        int len;
2228
0
        name++;
2229
0
  len = xmlStrlen(name);
2230
0
  if (name[len - 1] == ';')
2231
0
      cur->name = xmlStrndup(name, len - 1);
2232
0
  else
2233
0
      cur->name = xmlStrndup(name, len);
2234
0
    } else
2235
0
  cur->name = xmlStrdup(name);
2236
0
    if (cur->name == NULL)
2237
0
        goto error;
2238
2239
0
    ent = xmlGetDocEntity(doc, cur->name);
2240
0
    if (ent != NULL) {
2241
0
  cur->content = ent->content;
2242
  /*
2243
   * The parent pointer in entity is a DTD pointer and thus is NOT
2244
   * updated.  Not sure if this is 100% correct.
2245
   *  -George
2246
   */
2247
0
  cur->children = (xmlNodePtr) ent;
2248
0
  cur->last = (xmlNodePtr) ent;
2249
0
    }
2250
2251
0
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2252
0
  xmlRegisterNodeDefaultValue(cur);
2253
0
    return(cur);
2254
2255
0
error:
2256
0
    xmlFreeNode(cur);
2257
0
    return(NULL);
2258
0
}
2259
2260
/**
2261
 * Create a new text node.
2262
 *
2263
 * @param doc  the target document
2264
 * @param content  raw text content (optional)
2265
 * @returns a pointer to the new node object or NULL if a memory
2266
 * allocation failed.
2267
 */
2268
xmlNode *
2269
0
xmlNewDocText(const xmlDoc *doc, const xmlChar *content) {
2270
0
    xmlNodePtr cur;
2271
2272
0
    cur = xmlNewText(content);
2273
0
    if (cur != NULL) cur->doc = (xmlDoc *)doc;
2274
0
    return(cur);
2275
0
}
2276
2277
/**
2278
 * Create a new text node.
2279
 *
2280
 * Use of this function is DISCOURAGED in favor of #xmlNewDocTextLen.
2281
 *
2282
 * @param content  raw text content (optional)
2283
 * @param len  size of text content
2284
 * @returns a pointer to the new node object or NULL if a memory
2285
 * allocation failed.
2286
 */
2287
xmlNode *
2288
0
xmlNewTextLen(const xmlChar *content, int len) {
2289
0
    xmlNodePtr cur;
2290
2291
    /*
2292
     * Allocate a new node and fill the fields.
2293
     */
2294
0
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2295
0
    if (cur == NULL)
2296
0
  return(NULL);
2297
0
    memset(cur, 0, sizeof(xmlNode));
2298
0
    cur->type = XML_TEXT_NODE;
2299
2300
0
    cur->name = xmlStringText;
2301
0
    if (content != NULL) {
2302
0
  cur->content = xmlStrndup(content, len);
2303
0
        if (cur->content == NULL) {
2304
0
            xmlFreeNode(cur);
2305
0
            return(NULL);
2306
0
        }
2307
0
    }
2308
2309
0
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2310
0
  xmlRegisterNodeDefaultValue(cur);
2311
0
    return(cur);
2312
0
}
2313
2314
/**
2315
 * Create a new text node.
2316
 *
2317
 * @param doc  the target document
2318
 * @param content  raw text content (optional)
2319
 * @param len  size of text content
2320
 * @returns a pointer to the new node object or NULL if a memory
2321
 * allocation failed.
2322
 */
2323
xmlNode *
2324
0
xmlNewDocTextLen(xmlDoc *doc, const xmlChar *content, int len) {
2325
0
    xmlNodePtr cur;
2326
2327
0
    cur = xmlNewTextLen(content, len);
2328
0
    if (cur != NULL) cur->doc = doc;
2329
0
    return(cur);
2330
0
}
2331
2332
/**
2333
 * Use of this function is DISCOURAGED in favor of #xmlNewDocComment.
2334
 *
2335
 * Create a comment node.
2336
 *
2337
 * @param content  the comment content (optional)
2338
 * @returns a pointer to the new node object or NULL if a memory
2339
 * allocation failed.
2340
 */
2341
xmlNode *
2342
2.30k
xmlNewComment(const xmlChar *content) {
2343
2.30k
    xmlNodePtr cur;
2344
2345
    /*
2346
     * Allocate a new node and fill the fields.
2347
     */
2348
2.30k
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2349
2.30k
    if (cur == NULL)
2350
0
  return(NULL);
2351
2.30k
    memset(cur, 0, sizeof(xmlNode));
2352
2.30k
    cur->type = XML_COMMENT_NODE;
2353
2354
2.30k
    cur->name = xmlStringComment;
2355
2.30k
    if (content != NULL) {
2356
2.30k
  cur->content = xmlStrdup(content);
2357
2.30k
        if (cur->content == NULL)
2358
0
            goto error;
2359
2.30k
    }
2360
2361
2.30k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2362
0
  xmlRegisterNodeDefaultValue(cur);
2363
2.30k
    return(cur);
2364
2365
0
error:
2366
0
    xmlFreeNode(cur);
2367
0
    return(NULL);
2368
2.30k
}
2369
2370
/**
2371
 * Create a CDATA section node.
2372
 *
2373
 * @param doc  the target document (optional)
2374
 * @param content  raw text content (optional)
2375
 * @param len  size of text content
2376
 * @returns a pointer to the new node object or NULL if a memory
2377
 * allocation failed.
2378
 */
2379
xmlNode *
2380
2.46k
xmlNewCDataBlock(xmlDoc *doc, const xmlChar *content, int len) {
2381
2.46k
    xmlNodePtr cur;
2382
2383
    /*
2384
     * Allocate a new node and fill the fields.
2385
     */
2386
2.46k
    cur = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
2387
2.46k
    if (cur == NULL)
2388
0
  return(NULL);
2389
2.46k
    memset(cur, 0, sizeof(xmlNode));
2390
2.46k
    cur->type = XML_CDATA_SECTION_NODE;
2391
2.46k
    cur->doc = doc;
2392
2393
2.46k
    if (content != NULL) {
2394
2.46k
  cur->content = xmlStrndup(content, len);
2395
2.46k
        if (cur->content == NULL) {
2396
0
            xmlFree(cur);
2397
0
            return(NULL);
2398
0
        }
2399
2.46k
    }
2400
2401
2.46k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
2402
0
  xmlRegisterNodeDefaultValue(cur);
2403
2.46k
    return(cur);
2404
2.46k
}
2405
2406
/**
2407
 * Create a comment node.
2408
 *
2409
 * @param doc  the document
2410
 * @param content  the comment content
2411
 * @returns a pointer to the new node object or NULL if a memory
2412
 * allocation failed.
2413
 */
2414
xmlNode *
2415
2.30k
xmlNewDocComment(xmlDoc *doc, const xmlChar *content) {
2416
2.30k
    xmlNodePtr cur;
2417
2418
2.30k
    cur = xmlNewComment(content);
2419
2.30k
    if (cur != NULL) cur->doc = doc;
2420
2.30k
    return(cur);
2421
2.30k
}
2422
2423
static void
2424
0
xmlRemoveEntity(xmlEntityPtr ent) {
2425
0
    xmlDocPtr doc = ent->doc;
2426
0
    xmlDtdPtr intSubset, extSubset;
2427
2428
0
    if (doc == NULL)
2429
0
        return;
2430
0
    intSubset = doc->intSubset;
2431
0
    extSubset = doc->extSubset;
2432
2433
0
    if ((ent->etype == XML_INTERNAL_GENERAL_ENTITY) ||
2434
0
        (ent->etype == XML_EXTERNAL_GENERAL_PARSED_ENTITY) ||
2435
0
        (ent->etype == XML_EXTERNAL_GENERAL_UNPARSED_ENTITY)) {
2436
0
        if (intSubset != NULL) {
2437
0
            if (xmlHashLookup(intSubset->entities, ent->name) == ent)
2438
0
                xmlHashRemoveEntry(intSubset->entities, ent->name, NULL);
2439
0
        }
2440
0
        if (extSubset != NULL) {
2441
0
            if (xmlHashLookup(extSubset->entities, ent->name) == ent)
2442
0
                xmlHashRemoveEntry(extSubset->entities, ent->name, NULL);
2443
0
        }
2444
0
    } else if ((ent->etype == XML_INTERNAL_PARAMETER_ENTITY) ||
2445
0
               (ent->etype == XML_EXTERNAL_PARAMETER_ENTITY)) {
2446
0
        if (intSubset != NULL) {
2447
0
            if (xmlHashLookup(intSubset->pentities, ent->name) == ent)
2448
0
                xmlHashRemoveEntry(intSubset->entities, ent->name, NULL);
2449
0
        }
2450
0
        if (extSubset != NULL) {
2451
0
            if (xmlHashLookup(extSubset->pentities, ent->name) == ent)
2452
0
                xmlHashRemoveEntry(extSubset->entities, ent->name, NULL);
2453
0
        }
2454
0
    }
2455
0
}
2456
2457
static int
2458
0
xmlNodeSetDoc(xmlNodePtr node, xmlDocPtr doc) {
2459
0
    xmlDocPtr oldDoc;
2460
0
    xmlDictPtr oldDict, newDict;
2461
0
    int ret = 0;
2462
2463
    /*
2464
     * Remove name and content from old dictionary
2465
     */
2466
2467
0
    oldDoc = node->doc;
2468
0
    oldDict = oldDoc ? oldDoc->dict : NULL;
2469
0
    newDict = doc ? doc->dict : NULL;
2470
2471
0
    if ((oldDict != NULL) && (oldDict != newDict)) {
2472
0
        if ((node->name != NULL) &&
2473
0
            ((node->type == XML_ELEMENT_NODE) ||
2474
0
             (node->type == XML_ATTRIBUTE_NODE) ||
2475
0
             (node->type == XML_PI_NODE) ||
2476
0
             (node->type == XML_ENTITY_REF_NODE)) &&
2477
0
            (xmlDictOwns(oldDict, node->name))) {
2478
0
            if (newDict)
2479
0
                node->name = xmlDictLookup(newDict, node->name, -1);
2480
0
            else
2481
0
                node->name = xmlStrdup(node->name);
2482
0
            if (node->name == NULL)
2483
0
                ret = -1;
2484
0
        }
2485
2486
0
        if ((node->content != NULL) &&
2487
0
            ((node->type == XML_TEXT_NODE) ||
2488
0
             (node->type == XML_CDATA_SECTION_NODE)) &&
2489
0
            (xmlDictOwns(oldDict, node->content))) {
2490
0
            node->content = xmlStrdup(node->content);
2491
0
            if (node->content == NULL)
2492
0
                ret = -1;
2493
0
        }
2494
0
    }
2495
2496
0
    switch (node->type) {
2497
0
        case XML_ATTRIBUTE_NODE: {
2498
0
            xmlAttrPtr attr = (xmlAttrPtr) node;
2499
2500
            /*
2501
             * Handle IDs
2502
             *
2503
             * TODO: ID attributes should also be added to the new
2504
             * document, but it's not clear how to handle clashes.
2505
             */
2506
0
            if (attr->id != NULL)
2507
0
                xmlRemoveID(oldDoc, attr);
2508
2509
0
            break;
2510
0
        }
2511
2512
0
        case XML_ENTITY_REF_NODE:
2513
            /*
2514
             * Handle entity references
2515
             */
2516
0
            node->children = NULL;
2517
0
            node->last = NULL;
2518
0
            node->content = NULL;
2519
2520
0
            if ((doc != NULL) &&
2521
0
                ((doc->intSubset != NULL) || (doc->extSubset != NULL))) {
2522
0
                xmlEntityPtr ent;
2523
2524
                /*
2525
                * Assign new entity node if available
2526
                */
2527
0
                ent = xmlGetDocEntity(doc, node->name);
2528
0
                if (ent != NULL) {
2529
0
                    node->children = (xmlNodePtr) ent;
2530
0
                    node->last = (xmlNodePtr) ent;
2531
0
                    node->content = ent->content;
2532
0
                }
2533
0
            }
2534
2535
0
            break;
2536
2537
0
        case XML_DTD_NODE:
2538
0
            if (oldDoc != NULL) {
2539
0
                if (oldDoc->intSubset == (xmlDtdPtr) node)
2540
0
                    oldDoc->intSubset = NULL;
2541
0
                if (oldDoc->extSubset == (xmlDtdPtr) node)
2542
0
                    oldDoc->extSubset = NULL;
2543
0
            }
2544
2545
0
            break;
2546
2547
0
        case XML_ENTITY_DECL:
2548
0
            xmlRemoveEntity((xmlEntityPtr) node);
2549
0
            break;
2550
2551
        /*
2552
         * TODO:
2553
         * - Remove element decls from doc->elements
2554
         * - Remove attribtue decls form doc->attributes
2555
         */
2556
2557
0
        default:
2558
0
            break;
2559
0
    }
2560
2561
    /*
2562
     * Set new document
2563
     */
2564
0
    node->doc = doc;
2565
2566
0
    return(ret);
2567
0
}
2568
2569
/**
2570
 * Associate all nodes in a tree with a new document.
2571
 *
2572
 * This is an internal function which shouldn't be used. It is
2573
 * invoked by functions like #xmlAddChild, #xmlAddSibling or
2574
 * #xmlReplaceNode. `tree` must be the root node of an unlinked
2575
 * subtree.
2576
 *
2577
 * Also copy strings from the old document's dictionary and
2578
 * remove ID attributes from the old ID table.
2579
 *
2580
 * @param tree  root of a subtree
2581
 * @param doc  new document
2582
 * @returns 0 on success. If a memory allocation fails, returns -1.
2583
 * The whole tree will be updated on failure but some strings
2584
 * may be lost.
2585
 */
2586
int
2587
0
xmlSetTreeDoc(xmlNode *tree, xmlDoc *doc) {
2588
0
    int ret = 0;
2589
2590
0
    if ((tree == NULL) || (tree->type == XML_NAMESPACE_DECL))
2591
0
  return(0);
2592
0
    if (tree->doc == doc)
2593
0
        return(0);
2594
2595
0
    if (tree->type == XML_ELEMENT_NODE) {
2596
0
        xmlAttrPtr prop = tree->properties;
2597
2598
0
        while (prop != NULL) {
2599
0
            if (prop->children != NULL) {
2600
0
                if (xmlSetListDoc(prop->children, doc) < 0)
2601
0
                    ret = -1;
2602
0
            }
2603
2604
0
            if (xmlNodeSetDoc((xmlNodePtr) prop, doc) < 0)
2605
0
                ret = -1;
2606
2607
0
            prop = prop->next;
2608
0
        }
2609
0
    }
2610
2611
0
    if ((tree->children != NULL) &&
2612
0
        (tree->type != XML_ENTITY_REF_NODE)) {
2613
0
        if (xmlSetListDoc(tree->children, doc) < 0)
2614
0
            ret = -1;
2615
0
    }
2616
2617
0
    if (xmlNodeSetDoc(tree, doc) < 0)
2618
0
        ret = -1;
2619
2620
0
    return(ret);
2621
0
}
2622
2623
/**
2624
 * Associate all subtrees in `list` with a new document.
2625
 *
2626
 * Internal function, see #xmlSetTreeDoc.
2627
 *
2628
 * @param list  a node list
2629
 * @param doc  new document
2630
 * @returns 0 on success. If a memory allocation fails, returns -1.
2631
 * All subtrees will be updated on failure but some strings
2632
 * may be lost.
2633
 */
2634
int
2635
0
xmlSetListDoc(xmlNode *list, xmlDoc *doc) {
2636
0
    xmlNodePtr cur;
2637
0
    int ret = 0;
2638
2639
0
    if ((list == NULL) || (list->type == XML_NAMESPACE_DECL))
2640
0
  return(0);
2641
2642
0
    cur = list;
2643
0
    while (cur != NULL) {
2644
0
  if (cur->doc != doc) {
2645
0
      if (xmlSetTreeDoc(cur, doc) < 0)
2646
0
                ret = -1;
2647
0
        }
2648
0
  cur = cur->next;
2649
0
    }
2650
2651
0
    return(ret);
2652
0
}
2653
2654
/**
2655
 * Create a new child element and append it to a parent element.
2656
 *
2657
 * If `ns` is NULL, the newly created element inherits the namespace
2658
 * of the parent.
2659
 *
2660
 * If provided, `content` is expected to be a valid XML attribute
2661
 * value possibly containing character and entity references. Text
2662
 * and entity reference node will be added to the child element,
2663
 * see #xmlNewDocNode.
2664
 *
2665
 * @param parent  the parent node
2666
 * @param ns  a namespace (optional)
2667
 * @param name  the name of the child
2668
 * @param content  text content with XML references (optional)
2669
 * @returns a pointer to the new node object or NULL if arguments
2670
 * are invalid or a memory allocation failed.
2671
 */
2672
xmlNode *
2673
xmlNewChild(xmlNode *parent, xmlNs *ns,
2674
0
            const xmlChar *name, const xmlChar *content) {
2675
0
    xmlNodePtr cur, prev;
2676
2677
0
    if ((parent == NULL) || (name == NULL))
2678
0
  return(NULL);
2679
2680
0
    switch (parent->type) {
2681
0
        case XML_DOCUMENT_NODE:
2682
0
        case XML_HTML_DOCUMENT_NODE:
2683
0
        case XML_DOCUMENT_FRAG_NODE:
2684
0
            break;
2685
2686
0
        case XML_ELEMENT_NODE:
2687
0
            if (ns == NULL)
2688
0
                ns = parent->ns;
2689
0
            break;
2690
2691
0
        default:
2692
0
            return(NULL);
2693
0
    }
2694
2695
0
    cur = xmlNewDocNode(parent->doc, ns, name, content);
2696
0
    if (cur == NULL)
2697
0
        return(NULL);
2698
2699
    /*
2700
     * add the new element at the end of the children list.
2701
     */
2702
0
    cur->parent = parent;
2703
0
    if (parent->children == NULL) {
2704
0
        parent->children = cur;
2705
0
  parent->last = cur;
2706
0
    } else {
2707
0
        prev = parent->last;
2708
0
  prev->next = cur;
2709
0
  cur->prev = prev;
2710
0
  parent->last = cur;
2711
0
    }
2712
2713
0
    return(cur);
2714
0
}
2715
2716
static void
2717
0
xmlTextSetContent(xmlNodePtr text, xmlChar *content) {
2718
0
    if ((text->content != NULL) &&
2719
0
        (text->content != (xmlChar *) &text->properties)) {
2720
0
        xmlDocPtr doc = text->doc;
2721
2722
0
        if ((doc == NULL) ||
2723
0
            (doc->dict == NULL) ||
2724
0
            (!xmlDictOwns(doc->dict, text->content)))
2725
0
            xmlFree(text->content);
2726
0
    }
2727
2728
0
    text->content = content;
2729
0
    text->properties = NULL;
2730
0
}
2731
2732
static int
2733
0
xmlTextAddContent(xmlNodePtr text, const xmlChar *content, int len) {
2734
0
    xmlChar *merged;
2735
2736
0
    if (content == NULL)
2737
0
        return(0);
2738
2739
0
    merged = xmlStrncatNew(text->content, content, len);
2740
0
    if (merged == NULL)
2741
0
        return(-1);
2742
2743
0
    xmlTextSetContent(text, merged);
2744
0
    return(0);
2745
0
}
2746
2747
static xmlNodePtr
2748
xmlInsertProp(xmlDocPtr doc, xmlNodePtr cur, xmlNodePtr parent,
2749
0
              xmlNodePtr prev, xmlNodePtr next) {
2750
0
    xmlAttrPtr attr;
2751
2752
0
    if (((prev != NULL) && (prev->type != XML_ATTRIBUTE_NODE)) ||
2753
0
        ((next != NULL) && (next->type != XML_ATTRIBUTE_NODE)))
2754
0
        return(NULL);
2755
2756
    /* check if an attribute with the same name exists */
2757
0
    attr = xmlGetPropNodeInternal(parent, cur->name,
2758
0
                                  cur->ns ? cur->ns->href : NULL, 0);
2759
2760
0
    xmlUnlinkNodeInternal(cur);
2761
2762
0
    if (cur->doc != doc) {
2763
0
        if (xmlSetTreeDoc(cur, doc) < 0)
2764
0
            return(NULL);
2765
0
    }
2766
2767
0
    cur->parent = parent;
2768
0
    cur->prev = prev;
2769
0
    cur->next = next;
2770
2771
0
    if (prev == NULL) {
2772
0
        if (parent != NULL)
2773
0
            parent->properties = (xmlAttrPtr) cur;
2774
0
    } else {
2775
0
        prev->next = cur;
2776
0
    }
2777
0
    if (next != NULL) {
2778
0
        next->prev = cur;
2779
0
    }
2780
2781
0
    if ((attr != NULL) && (attr != (xmlAttrPtr) cur)) {
2782
        /* different instance, destroy it (attributes must be unique) */
2783
0
        xmlRemoveProp((xmlAttrPtr) attr);
2784
0
    }
2785
2786
0
    return cur;
2787
0
}
2788
2789
static xmlNodePtr
2790
xmlInsertNode(xmlDocPtr doc, xmlNodePtr cur, xmlNodePtr parent,
2791
0
              xmlNodePtr prev, xmlNodePtr next, int coalesce) {
2792
0
    xmlNodePtr oldParent;
2793
2794
0
    if (cur->type == XML_ATTRIBUTE_NODE)
2795
0
  return xmlInsertProp(doc, cur, parent, prev, next);
2796
2797
    /*
2798
     * Coalesce text nodes
2799
     */
2800
0
    if ((coalesce) && (cur->type == XML_TEXT_NODE)) {
2801
0
  if ((prev != NULL) && (prev->type == XML_TEXT_NODE) &&
2802
0
            (prev->name == cur->name)) {
2803
0
            if (xmlTextAddContent(prev, cur->content, -1) < 0)
2804
0
                return(NULL);
2805
0
            xmlUnlinkNodeInternal(cur);
2806
0
      xmlFreeNode(cur);
2807
0
      return(prev);
2808
0
  }
2809
2810
0
  if ((next != NULL) && (next->type == XML_TEXT_NODE) &&
2811
0
            (next->name == cur->name)) {
2812
0
            if (cur->content != NULL) {
2813
0
          xmlChar *merged;
2814
2815
0
                merged = xmlStrncatNew(cur->content, next->content, -1);
2816
0
                if (merged == NULL)
2817
0
                    return(NULL);
2818
0
                xmlTextSetContent(next, merged);
2819
0
            }
2820
2821
0
            xmlUnlinkNodeInternal(cur);
2822
0
      xmlFreeNode(cur);
2823
0
      return(next);
2824
0
  }
2825
0
    }
2826
2827
    /* Unlink */
2828
0
    oldParent = cur->parent;
2829
0
    if (oldParent != NULL) {
2830
0
        if (oldParent->children == cur)
2831
0
            oldParent->children = cur->next;
2832
0
        if (oldParent->last == cur)
2833
0
            oldParent->last = cur->prev;
2834
0
    }
2835
0
    if (cur->next != NULL)
2836
0
        cur->next->prev = cur->prev;
2837
0
    if (cur->prev != NULL)
2838
0
        cur->prev->next = cur->next;
2839
2840
0
    if (cur->doc != doc) {
2841
0
  if (xmlSetTreeDoc(cur, doc) < 0) {
2842
            /*
2843
             * We shouldn't make any modifications to the inserted
2844
             * tree if a memory allocation fails, but that's hard to
2845
             * implement. The tree has been moved to the target
2846
             * document now but some contents are corrupted.
2847
             * Unlinking is the best we can do.
2848
             */
2849
0
            cur->parent = NULL;
2850
0
            cur->prev = NULL;
2851
0
            cur->next = NULL;
2852
0
            return(NULL);
2853
0
        }
2854
0
    }
2855
2856
0
    cur->parent = parent;
2857
0
    cur->prev = prev;
2858
0
    cur->next = next;
2859
2860
0
    if (prev == NULL) {
2861
0
        if (parent != NULL)
2862
0
            parent->children = cur;
2863
0
    } else {
2864
0
        prev->next = cur;
2865
0
    }
2866
0
    if (next == NULL) {
2867
0
        if (parent != NULL)
2868
0
            parent->last = cur;
2869
0
    } else {
2870
0
        next->prev = cur;
2871
0
    }
2872
2873
0
    return(cur);
2874
0
}
2875
2876
/**
2877
 * Unlinks `cur` and inserts it as next sibling after `prev`.
2878
 *
2879
 * Unlike #xmlAddChild this function does not merge text nodes.
2880
 *
2881
 * If `cur` is an attribute node, it is inserted after attribute
2882
 * `prev`. If the attribute list contains an attribute with a name
2883
 * matching `cur`, the old attribute is destroyed.
2884
 *
2885
 * See the notes in #xmlAddChild.
2886
 *
2887
 * @param prev  the target node
2888
 * @param cur  the new node
2889
 * @returns `cur` or a sibling if `cur` was merged. Returns NULL
2890
 * if arguments are invalid or a memory allocation failed.
2891
 */
2892
xmlNode *
2893
0
xmlAddNextSibling(xmlNode *prev, xmlNode *cur) {
2894
0
    if ((prev == NULL) || (prev->type == XML_NAMESPACE_DECL) ||
2895
0
        (cur == NULL) || (cur->type == XML_NAMESPACE_DECL) ||
2896
0
        (cur == prev))
2897
0
  return(NULL);
2898
2899
0
    if (cur == prev->next)
2900
0
        return(cur);
2901
2902
0
    return(xmlInsertNode(prev->doc, cur, prev->parent, prev, prev->next, 0));
2903
0
}
2904
2905
/**
2906
 * Unlinks `cur` and inserts it as previous sibling before `next`.
2907
 *
2908
 * Unlike #xmlAddChild this function does not merge text nodes.
2909
 *
2910
 * If `cur` is an attribute node, it is inserted before attribute
2911
 * `next`. If the attribute list contains an attribute with a name
2912
 * matching `cur`, the old attribute is destroyed.
2913
 *
2914
 * See the notes in #xmlAddChild.
2915
 *
2916
 * @param next  the target node
2917
 * @param cur  the new node
2918
 * @returns `cur` or a sibling if `cur` was merged. Returns NULL
2919
 * if arguments are invalid or a memory allocation failed.
2920
 */
2921
xmlNode *
2922
0
xmlAddPrevSibling(xmlNode *next, xmlNode *cur) {
2923
0
    if ((next == NULL) || (next->type == XML_NAMESPACE_DECL) ||
2924
0
        (cur == NULL) || (cur->type == XML_NAMESPACE_DECL) ||
2925
0
        (cur == next))
2926
0
  return(NULL);
2927
2928
0
    if (cur == next->prev)
2929
0
        return(cur);
2930
2931
0
    return(xmlInsertNode(next->doc, cur, next->parent, next->prev, next, 0));
2932
0
}
2933
2934
/**
2935
 * Unlinks `cur` and inserts it as last sibling of `node`.
2936
 *
2937
 * If `cur` is a text node, it may be merged with an adjacent text
2938
 * node and freed. In this case the text node containing the merged
2939
 * content is returned.
2940
 *
2941
 * If `cur` is an attribute node, it is appended to the attribute
2942
 * list containing `node`. If the attribute list contains an attribute
2943
 * with a name matching `cur`, the old attribute is destroyed.
2944
 *
2945
 * See the notes in #xmlAddChild.
2946
 *
2947
 * @param node  the target node
2948
 * @param cur  the new node
2949
 * @returns `cur` or a sibling if `cur` was merged. Returns NULL
2950
 * if arguments are invalid or a memory allocation failed.
2951
 */
2952
xmlNode *
2953
0
xmlAddSibling(xmlNode *node, xmlNode *cur) {
2954
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL) ||
2955
0
        (cur == NULL) || (cur->type == XML_NAMESPACE_DECL) ||
2956
0
        (cur == node))
2957
0
  return(NULL);
2958
2959
    /*
2960
     * Constant time is we can rely on the ->parent->last to find
2961
     * the last sibling.
2962
     */
2963
0
    if ((node->type != XML_ATTRIBUTE_NODE) && (node->parent != NULL)) {
2964
0
        if (node->parent->last != NULL)
2965
0
      node = node->parent->last;
2966
0
    } else {
2967
0
  while (node->next != NULL)
2968
0
            node = node->next;
2969
0
    }
2970
2971
0
    if (cur == node)
2972
0
        return(cur);
2973
2974
0
    return(xmlInsertNode(node->doc, cur, node->parent, node, NULL, 1));
2975
0
}
2976
2977
/**
2978
 * Append a node list to another node.
2979
 *
2980
 * See #xmlAddChild.
2981
 *
2982
 * @param parent  the parent node
2983
 * @param cur  the first node in the list
2984
 * @returns the last child or NULL in case of error.
2985
 */
2986
xmlNode *
2987
0
xmlAddChildList(xmlNode *parent, xmlNode *cur) {
2988
0
    xmlNodePtr iter;
2989
0
    xmlNodePtr prev;
2990
0
    int oom;
2991
2992
0
    if ((parent == NULL) || (parent->type == XML_NAMESPACE_DECL)) {
2993
0
  return(NULL);
2994
0
    }
2995
2996
0
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL)) {
2997
0
  return(NULL);
2998
0
    }
2999
3000
0
    oom = 0;
3001
0
    for (iter = cur; iter != NULL; iter = iter->next) {
3002
0
  if (iter->doc != parent->doc) {
3003
0
      if (xmlSetTreeDoc(iter, parent->doc) < 0)
3004
0
                oom = 1;
3005
0
  }
3006
0
    }
3007
0
    if (oom)
3008
0
        return(NULL);
3009
3010
    /*
3011
     * add the first element at the end of the children list.
3012
     */
3013
3014
0
    if (parent->children == NULL) {
3015
0
        parent->children = cur;
3016
0
    } else {
3017
0
        prev = parent->last;
3018
3019
  /*
3020
   * If cur and parent->last both are TEXT nodes, then merge them.
3021
   */
3022
0
  if ((cur->type == XML_TEXT_NODE) &&
3023
0
      (prev->type == XML_TEXT_NODE) &&
3024
0
      (cur->name == prev->name)) {
3025
0
            xmlNodePtr next;
3026
3027
0
            if (xmlTextAddContent(prev, cur->content, -1) < 0)
3028
0
                return(NULL);
3029
0
            next = cur->next;
3030
0
      xmlFreeNode(cur);
3031
      /*
3032
       * if it's the only child, nothing more to be done.
3033
       */
3034
0
      if (next == NULL)
3035
0
    return(prev);
3036
0
      cur = next;
3037
0
  }
3038
3039
0
  prev->next = cur;
3040
0
  cur->prev = prev;
3041
0
    }
3042
0
    while (cur->next != NULL) {
3043
0
  cur->parent = parent;
3044
0
        cur = cur->next;
3045
0
    }
3046
0
    cur->parent = parent;
3047
0
    parent->last = cur;
3048
3049
0
    return(cur);
3050
0
}
3051
3052
/**
3053
 * Unlink `cur` and append it to the children of `parent`.
3054
 *
3055
 * If `cur` is a text node, it may be merged with an adjacent text
3056
 * node and freed. In this case the text node containing the merged
3057
 * content is returned.
3058
 *
3059
 * If `cur` is an attribute node, it is appended to the attributes of
3060
 * `parent`. If the attribute list contains an attribute with a name
3061
 * matching `cur`, the old attribute is destroyed.
3062
 *
3063
 * Before version 2.13, this function didn't unlink `cur` before
3064
 * moving it. Callers must unlink the node manually if it has
3065
 * siblings.
3066
 *
3067
 * General notes:
3068
 *
3069
 * Move operations like #xmlAddChild can cause element or attribute
3070
 * nodes to reference namespaces that aren't declared in one of
3071
 * their ancestors. This can lead to use-after-free errors if the
3072
 * elements containing the declarations are freed later, especially
3073
 * when moving nodes from one document to another. You should
3074
 * consider calling #xmlReconciliateNs after a move operation to
3075
 * normalize namespaces. Another option is to call
3076
 * #xmlDOMWrapAdoptNode with the target parent before moving a node.
3077
 *
3078
 * For the most part, move operations don't check whether the
3079
 * resulting tree structure is valid. Users must make sure that
3080
 * parent nodes only receive children of valid types. Inserted
3081
 * child nodes must never be an ancestor of the parent node to
3082
 * avoid cycles in the tree structure. In general, only
3083
 * document, document fragments, elements and attributes
3084
 * should be used as parent nodes.
3085
 *
3086
 * When moving a node between documents and a memory allocation
3087
 * fails, the node's content will be corrupted and it will be
3088
 * unlinked. In this case, the node must be freed manually.
3089
 *
3090
 * Moving DTDs between documents isn't supported.
3091
 *
3092
 * @param parent  the parent node
3093
 * @param cur  the child node
3094
 * @returns `cur` or a sibling if `cur` was merged. Returns NULL
3095
 * if arguments are invalid or a memory allocation failed.
3096
 */
3097
xmlNode *
3098
0
xmlAddChild(xmlNode *parent, xmlNode *cur) {
3099
0
    xmlNodePtr prev;
3100
3101
0
    if ((parent == NULL) || (parent->type == XML_NAMESPACE_DECL) ||
3102
0
        (cur == NULL) || (cur->type == XML_NAMESPACE_DECL) ||
3103
0
        (parent == cur))
3104
0
        return(NULL);
3105
3106
    /*
3107
     * If parent is a text node, call xmlTextAddContent. This
3108
     * undocumented quirk should probably be removed.
3109
     */
3110
0
    if (parent->type == XML_TEXT_NODE) {
3111
0
        if (xmlTextAddContent(parent, cur->content, -1) < 0)
3112
0
            return(NULL);
3113
0
        xmlFreeNode(cur);
3114
0
        return(parent);
3115
0
    }
3116
3117
0
    if (cur->type == XML_ATTRIBUTE_NODE) {
3118
0
        prev = (xmlNodePtr) parent->properties;
3119
0
        if (prev != NULL) {
3120
0
            while (prev->next != NULL)
3121
0
                prev = prev->next;
3122
0
        }
3123
0
    } else {
3124
0
        prev = parent->last;
3125
0
    }
3126
3127
0
    if (cur == prev)
3128
0
        return(cur);
3129
3130
0
    return(xmlInsertNode(parent->doc, cur, parent, prev, NULL, 1));
3131
0
}
3132
3133
/**
3134
 * Find the last child of a node.
3135
 *
3136
 * @param parent  the parent node
3137
 * @returns the last child or NULL if parent has no children.
3138
 */
3139
xmlNode *
3140
0
xmlGetLastChild(const xmlNode *parent) {
3141
0
    if ((parent == NULL) || (parent->type == XML_NAMESPACE_DECL)) {
3142
0
  return(NULL);
3143
0
    }
3144
0
    return(parent->last);
3145
0
}
3146
3147
/*
3148
 * 5 interfaces from DOM ElementTraversal
3149
 */
3150
3151
/**
3152
 * Count the number of child nodes which are elements.
3153
 *
3154
 * Note that entity references are not expanded.
3155
 *
3156
 * @param parent  the parent node
3157
 * @returns the number of element children or 0 if arguments are
3158
 * invalid.
3159
 */
3160
unsigned long
3161
0
xmlChildElementCount(xmlNode *parent) {
3162
0
    unsigned long ret = 0;
3163
0
    xmlNodePtr cur = NULL;
3164
3165
0
    if (parent == NULL)
3166
0
        return(0);
3167
0
    switch (parent->type) {
3168
0
        case XML_ELEMENT_NODE:
3169
0
        case XML_DOCUMENT_NODE:
3170
0
        case XML_DOCUMENT_FRAG_NODE:
3171
0
        case XML_HTML_DOCUMENT_NODE:
3172
0
        case XML_ENTITY_DECL:
3173
0
            cur = parent->children;
3174
0
            break;
3175
0
        default:
3176
0
            return(0);
3177
0
    }
3178
0
    while (cur != NULL) {
3179
0
        if (cur->type == XML_ELEMENT_NODE)
3180
0
            ret++;
3181
0
        cur = cur->next;
3182
0
    }
3183
0
    return(ret);
3184
0
}
3185
3186
/**
3187
 * Find the first child node which is an element.
3188
 *
3189
 * Note that entity references are not expanded.
3190
 *
3191
 * @param parent  the parent node
3192
 * @returns the first element or NULL if parent has no children.
3193
 */
3194
xmlNode *
3195
0
xmlFirstElementChild(xmlNode *parent) {
3196
0
    xmlNodePtr cur = NULL;
3197
3198
0
    if (parent == NULL)
3199
0
        return(NULL);
3200
0
    switch (parent->type) {
3201
0
        case XML_ELEMENT_NODE:
3202
0
        case XML_DOCUMENT_NODE:
3203
0
        case XML_DOCUMENT_FRAG_NODE:
3204
0
        case XML_HTML_DOCUMENT_NODE:
3205
0
        case XML_ENTITY_DECL:
3206
0
            cur = parent->children;
3207
0
            break;
3208
0
        default:
3209
0
            return(NULL);
3210
0
    }
3211
0
    while (cur != NULL) {
3212
0
        if (cur->type == XML_ELEMENT_NODE)
3213
0
            return(cur);
3214
0
        cur = cur->next;
3215
0
    }
3216
0
    return(NULL);
3217
0
}
3218
3219
/**
3220
 * Find the last child node which is an element.
3221
 *
3222
 * Note that entity references are not expanded.
3223
 *
3224
 * @param parent  the parent node
3225
 * @returns the last element or NULL if parent has no children.
3226
 */
3227
xmlNode *
3228
0
xmlLastElementChild(xmlNode *parent) {
3229
0
    xmlNodePtr cur = NULL;
3230
3231
0
    if (parent == NULL)
3232
0
        return(NULL);
3233
0
    switch (parent->type) {
3234
0
        case XML_ELEMENT_NODE:
3235
0
        case XML_DOCUMENT_NODE:
3236
0
        case XML_DOCUMENT_FRAG_NODE:
3237
0
        case XML_HTML_DOCUMENT_NODE:
3238
0
        case XML_ENTITY_DECL:
3239
0
            cur = parent->last;
3240
0
            break;
3241
0
        default:
3242
0
            return(NULL);
3243
0
    }
3244
0
    while (cur != NULL) {
3245
0
        if (cur->type == XML_ELEMENT_NODE)
3246
0
            return(cur);
3247
0
        cur = cur->prev;
3248
0
    }
3249
0
    return(NULL);
3250
0
}
3251
3252
/**
3253
 * Find the closest preceding sibling which is a element.
3254
 *
3255
 * Note that entity references are not expanded.
3256
 *
3257
 * @param node  the current node
3258
 * @returns the sibling or NULL if no sibling was found.
3259
 */
3260
xmlNode *
3261
0
xmlPreviousElementSibling(xmlNode *node) {
3262
0
    if (node == NULL)
3263
0
        return(NULL);
3264
0
    switch (node->type) {
3265
0
        case XML_ELEMENT_NODE:
3266
0
        case XML_TEXT_NODE:
3267
0
        case XML_CDATA_SECTION_NODE:
3268
0
        case XML_ENTITY_REF_NODE:
3269
0
        case XML_PI_NODE:
3270
0
        case XML_COMMENT_NODE:
3271
0
        case XML_XINCLUDE_START:
3272
0
        case XML_XINCLUDE_END:
3273
0
            node = node->prev;
3274
0
            break;
3275
0
        default:
3276
0
            return(NULL);
3277
0
    }
3278
0
    while (node != NULL) {
3279
0
        if (node->type == XML_ELEMENT_NODE)
3280
0
            return(node);
3281
0
        node = node->prev;
3282
0
    }
3283
0
    return(NULL);
3284
0
}
3285
3286
/**
3287
 * Find the closest following sibling which is a element.
3288
 *
3289
 * Note that entity references are not expanded.
3290
 *
3291
 * @param node  the current node
3292
 * @returns the sibling or NULL if no sibling was found.
3293
 */
3294
xmlNode *
3295
0
xmlNextElementSibling(xmlNode *node) {
3296
0
    if (node == NULL)
3297
0
        return(NULL);
3298
0
    switch (node->type) {
3299
0
        case XML_ELEMENT_NODE:
3300
0
        case XML_TEXT_NODE:
3301
0
        case XML_CDATA_SECTION_NODE:
3302
0
        case XML_ENTITY_REF_NODE:
3303
0
        case XML_PI_NODE:
3304
0
        case XML_COMMENT_NODE:
3305
0
        case XML_DTD_NODE:
3306
0
        case XML_XINCLUDE_START:
3307
0
        case XML_XINCLUDE_END:
3308
0
            node = node->next;
3309
0
            break;
3310
0
        default:
3311
0
            return(NULL);
3312
0
    }
3313
0
    while (node != NULL) {
3314
0
        if (node->type == XML_ELEMENT_NODE)
3315
0
            return(node);
3316
0
        node = node->next;
3317
0
    }
3318
0
    return(NULL);
3319
0
}
3320
3321
/**
3322
 * Free a node list including all children.
3323
 *
3324
 * @param cur  the first node in the list
3325
 */
3326
void
3327
50.1k
xmlFreeNodeList(xmlNode *cur) {
3328
50.1k
    xmlNodePtr next;
3329
50.1k
    xmlNodePtr parent;
3330
50.1k
    xmlDictPtr dict = NULL;
3331
50.1k
    size_t depth = 0;
3332
3333
50.1k
    if (cur == NULL) return;
3334
49.5k
    if (cur->type == XML_NAMESPACE_DECL) {
3335
0
  xmlFreeNsList((xmlNsPtr) cur);
3336
0
  return;
3337
0
    }
3338
49.5k
    if (cur->doc != NULL) dict = cur->doc->dict;
3339
134k
    while (1) {
3340
192k
        while ((cur->children != NULL) &&
3341
192k
               (cur->type != XML_DOCUMENT_NODE) &&
3342
192k
               (cur->type != XML_HTML_DOCUMENT_NODE) &&
3343
192k
               (cur->type != XML_DTD_NODE) &&
3344
192k
               (cur->type != XML_ENTITY_REF_NODE)) {
3345
58.4k
            cur = cur->children;
3346
58.4k
            depth += 1;
3347
58.4k
        }
3348
3349
134k
        next = cur->next;
3350
134k
        parent = cur->parent;
3351
134k
  if ((cur->type == XML_DOCUMENT_NODE) ||
3352
134k
            (cur->type == XML_HTML_DOCUMENT_NODE)) {
3353
0
            xmlFreeDoc((xmlDocPtr) cur);
3354
134k
        } else if (cur->type == XML_DTD_NODE) {
3355
            /*
3356
             * TODO: We should consider freeing the DTD if it isn't
3357
             * referenced from doc->intSubset or doc->extSubset.
3358
             */
3359
0
            cur->prev = NULL;
3360
0
            cur->next = NULL;
3361
134k
        } else {
3362
134k
      if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
3363
0
    xmlDeregisterNodeDefaultValue(cur);
3364
3365
134k
      if (((cur->type == XML_ELEMENT_NODE) ||
3366
134k
     (cur->type == XML_XINCLUDE_START) ||
3367
134k
     (cur->type == XML_XINCLUDE_END)) &&
3368
134k
    (cur->properties != NULL))
3369
31.8k
    xmlFreePropList(cur->properties);
3370
134k
      if ((cur->type != XML_ELEMENT_NODE) &&
3371
134k
    (cur->type != XML_XINCLUDE_START) &&
3372
134k
    (cur->type != XML_XINCLUDE_END) &&
3373
134k
    (cur->type != XML_ENTITY_REF_NODE) &&
3374
134k
    (cur->content != (xmlChar *) &(cur->properties))) {
3375
63.5k
    DICT_FREE(cur->content)
3376
63.5k
      }
3377
134k
      if (((cur->type == XML_ELEMENT_NODE) ||
3378
134k
           (cur->type == XML_XINCLUDE_START) ||
3379
134k
     (cur->type == XML_XINCLUDE_END)) &&
3380
134k
    (cur->nsDef != NULL))
3381
26.6k
    xmlFreeNsList(cur->nsDef);
3382
3383
      /*
3384
       * When a node is a text node or a comment, it uses a global static
3385
       * variable for the name of the node.
3386
       * Otherwise the node name might come from the document's
3387
       * dictionary
3388
       */
3389
134k
      if ((cur->name != NULL) &&
3390
134k
    (cur->type != XML_TEXT_NODE) &&
3391
134k
    (cur->type != XML_COMMENT_NODE))
3392
74.2k
    DICT_FREE(cur->name)
3393
134k
      xmlFree(cur);
3394
134k
  }
3395
3396
134k
        if (next != NULL) {
3397
26.4k
      cur = next;
3398
108k
        } else {
3399
108k
            if ((depth == 0) || (parent == NULL))
3400
49.5k
                break;
3401
58.4k
            depth -= 1;
3402
58.4k
            cur = parent;
3403
58.4k
            cur->children = NULL;
3404
58.4k
        }
3405
134k
    }
3406
49.5k
}
3407
3408
/**
3409
 * Free a node including all the children.
3410
 *
3411
 * This doesn't unlink the node from the tree. Call #xmlUnlinkNode first
3412
 * unless `cur` is a root node.
3413
 *
3414
 * @param cur  the node
3415
 */
3416
void
3417
2.52k
xmlFreeNode(xmlNode *cur) {
3418
2.52k
    xmlDictPtr dict = NULL;
3419
3420
2.52k
    if (cur == NULL) return;
3421
3422
    /* use xmlFreeDtd for DTD nodes */
3423
2.52k
    if (cur->type == XML_DTD_NODE) {
3424
0
  xmlFreeDtd((xmlDtdPtr) cur);
3425
0
  return;
3426
0
    }
3427
2.52k
    if (cur->type == XML_NAMESPACE_DECL) {
3428
0
  xmlFreeNs((xmlNsPtr) cur);
3429
0
        return;
3430
0
    }
3431
2.52k
    if (cur->type == XML_ATTRIBUTE_NODE) {
3432
0
  xmlFreeProp((xmlAttrPtr) cur);
3433
0
  return;
3434
0
    }
3435
2.52k
    if (cur->type == XML_ENTITY_DECL) {
3436
0
        xmlFreeEntity((xmlEntityPtr) cur);
3437
0
        return;
3438
0
    }
3439
3440
2.52k
    if ((xmlRegisterCallbacks) && (xmlDeregisterNodeDefaultValue))
3441
0
  xmlDeregisterNodeDefaultValue(cur);
3442
3443
2.52k
    if (cur->doc != NULL) dict = cur->doc->dict;
3444
3445
2.52k
    if ((cur->children != NULL) &&
3446
2.52k
  (cur->type != XML_ENTITY_REF_NODE))
3447
96
  xmlFreeNodeList(cur->children);
3448
3449
2.52k
    if ((cur->type == XML_ELEMENT_NODE) ||
3450
2.52k
        (cur->type == XML_XINCLUDE_START) ||
3451
2.52k
        (cur->type == XML_XINCLUDE_END)) {
3452
1.62k
        if (cur->properties != NULL)
3453
0
            xmlFreePropList(cur->properties);
3454
1.62k
        if (cur->nsDef != NULL)
3455
7
            xmlFreeNsList(cur->nsDef);
3456
1.62k
    } else if ((cur->content != NULL) &&
3457
898
               (cur->type != XML_ENTITY_REF_NODE) &&
3458
898
               (cur->content != (xmlChar *) &(cur->properties))) {
3459
552
        DICT_FREE(cur->content)
3460
552
    }
3461
3462
    /*
3463
     * When a node is a text node or a comment, it uses a global static
3464
     * variable for the name of the node.
3465
     * Otherwise the node name might come from the document's dictionary
3466
     */
3467
2.52k
    if ((cur->name != NULL) &&
3468
2.52k
        (cur->type != XML_TEXT_NODE) &&
3469
2.52k
        (cur->type != XML_COMMENT_NODE))
3470
2.07k
  DICT_FREE(cur->name)
3471
3472
2.52k
    xmlFree(cur);
3473
2.52k
}
3474
3475
/**
3476
 * Unlink a node from its tree.
3477
 *
3478
 * This function only unlinks the node from the tree. It doesn't
3479
 * clear references to DTD nodes.
3480
 *
3481
 * @param cur  the node
3482
 */
3483
static void
3484
20.3k
xmlUnlinkNodeInternal(xmlNodePtr cur) {
3485
20.3k
    if (cur->parent != NULL) {
3486
14.2k
  xmlNodePtr parent;
3487
14.2k
  parent = cur->parent;
3488
14.2k
  if (cur->type == XML_ATTRIBUTE_NODE) {
3489
0
      if (parent->properties == (xmlAttrPtr) cur)
3490
0
    parent->properties = ((xmlAttrPtr) cur)->next;
3491
14.2k
  } else {
3492
14.2k
      if (parent->children == cur)
3493
5.73k
    parent->children = cur->next;
3494
14.2k
      if (parent->last == cur)
3495
4.67k
    parent->last = cur->prev;
3496
14.2k
  }
3497
14.2k
  cur->parent = NULL;
3498
14.2k
    }
3499
3500
20.3k
    if (cur->next != NULL)
3501
9.61k
        cur->next->prev = cur->prev;
3502
20.3k
    if (cur->prev != NULL)
3503
8.55k
        cur->prev->next = cur->next;
3504
20.3k
    cur->next = NULL;
3505
20.3k
    cur->prev = NULL;
3506
20.3k
}
3507
3508
/**
3509
 * Unlink a node from its tree.
3510
 *
3511
 * The node is not freed. Unless it is reinserted, it must be managed
3512
 * manually and freed eventually by calling #xmlFreeNode.
3513
 *
3514
 * @param cur  the node
3515
 */
3516
void
3517
14.0k
xmlUnlinkNode(xmlNode *cur) {
3518
14.0k
    if (cur == NULL)
3519
0
  return;
3520
3521
14.0k
    if (cur->type == XML_NAMESPACE_DECL)
3522
0
        return;
3523
3524
14.0k
    if (cur->type == XML_DTD_NODE) {
3525
0
  xmlDocPtr doc = cur->doc;
3526
3527
0
  if (doc != NULL) {
3528
0
      if (doc->intSubset == (xmlDtdPtr) cur)
3529
0
    doc->intSubset = NULL;
3530
0
      if (doc->extSubset == (xmlDtdPtr) cur)
3531
0
    doc->extSubset = NULL;
3532
0
  }
3533
0
    }
3534
3535
14.0k
    if (cur->type == XML_ENTITY_DECL)
3536
0
        xmlRemoveEntity((xmlEntityPtr) cur);
3537
3538
14.0k
    xmlUnlinkNodeInternal(cur);
3539
14.0k
}
3540
3541
/**
3542
 * Unlink the old node. If `cur` is provided, it is unlinked and
3543
 * inserted in place of `old`.
3544
 *
3545
 * It is an error if `old` has no parent.
3546
 *
3547
 * Unlike #xmlAddChild, this function doesn't merge text nodes or
3548
 * delete duplicate attributes.
3549
 *
3550
 * See the notes in #xmlAddChild.
3551
 *
3552
 * @param old  the old node
3553
 * @param cur  the node (optional)
3554
 * @returns `old` or NULL if arguments are invalid or a memory
3555
 * allocation failed.
3556
 */
3557
xmlNode *
3558
0
xmlReplaceNode(xmlNode *old, xmlNode *cur) {
3559
0
    if (old == cur) return(NULL);
3560
0
    if ((old == NULL) || (old->type == XML_NAMESPACE_DECL) ||
3561
0
        (old->parent == NULL)) {
3562
0
  return(NULL);
3563
0
    }
3564
0
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL)) {
3565
        /* Don't call xmlUnlinkNodeInternal to handle DTDs. */
3566
0
  xmlUnlinkNode(old);
3567
0
  return(old);
3568
0
    }
3569
0
    if ((old->type==XML_ATTRIBUTE_NODE) && (cur->type!=XML_ATTRIBUTE_NODE)) {
3570
0
  return(old);
3571
0
    }
3572
0
    if ((cur->type==XML_ATTRIBUTE_NODE) && (old->type!=XML_ATTRIBUTE_NODE)) {
3573
0
  return(old);
3574
0
    }
3575
0
    xmlUnlinkNodeInternal(cur);
3576
0
    if (xmlSetTreeDoc(cur, old->doc) < 0)
3577
0
        return(NULL);
3578
0
    cur->parent = old->parent;
3579
0
    cur->next = old->next;
3580
0
    if (cur->next != NULL)
3581
0
  cur->next->prev = cur;
3582
0
    cur->prev = old->prev;
3583
0
    if (cur->prev != NULL)
3584
0
  cur->prev->next = cur;
3585
0
    if (cur->parent != NULL) {
3586
0
  if (cur->type == XML_ATTRIBUTE_NODE) {
3587
0
      if (cur->parent->properties == (xmlAttrPtr)old)
3588
0
    cur->parent->properties = ((xmlAttrPtr) cur);
3589
0
  } else {
3590
0
      if (cur->parent->children == old)
3591
0
    cur->parent->children = cur;
3592
0
      if (cur->parent->last == old)
3593
0
    cur->parent->last = cur;
3594
0
  }
3595
0
    }
3596
0
    old->next = old->prev = NULL;
3597
0
    old->parent = NULL;
3598
0
    return(old);
3599
0
}
3600
3601
/************************************************************************
3602
 *                  *
3603
 *    Copy operations           *
3604
 *                  *
3605
 ************************************************************************/
3606
3607
/**
3608
 * Copy a namespace.
3609
 *
3610
 * @param cur  the namespace
3611
 * @returns the copied namespace or NULL if a memory allocation
3612
 * failed.
3613
 */
3614
xmlNs *
3615
1.18k
xmlCopyNamespace(xmlNs *cur) {
3616
1.18k
    xmlNsPtr ret;
3617
3618
1.18k
    if (cur == NULL) return(NULL);
3619
1.18k
    switch (cur->type) {
3620
1.18k
  case XML_LOCAL_NAMESPACE:
3621
1.18k
      ret = xmlNewNs(NULL, cur->href, cur->prefix);
3622
1.18k
      break;
3623
0
  default:
3624
0
      return(NULL);
3625
1.18k
    }
3626
1.18k
    return(ret);
3627
1.18k
}
3628
3629
/**
3630
 * Copy a namespace list.
3631
 *
3632
 * @param cur  the first namespace
3633
 * @returns the head of the copied list or NULL if a memory
3634
 * allocation failed.
3635
 */
3636
xmlNs *
3637
837
xmlCopyNamespaceList(xmlNs *cur) {
3638
837
    xmlNsPtr ret = NULL;
3639
837
    xmlNsPtr p = NULL,q;
3640
3641
2.01k
    while (cur != NULL) {
3642
1.18k
        q = xmlCopyNamespace(cur);
3643
1.18k
        if (q == NULL) {
3644
0
            xmlFreeNsList(ret);
3645
0
            return(NULL);
3646
0
        }
3647
1.18k
  if (p == NULL) {
3648
837
      ret = p = q;
3649
837
  } else {
3650
343
      p->next = q;
3651
343
      p = q;
3652
343
  }
3653
1.18k
  cur = cur->next;
3654
1.18k
    }
3655
837
    return(ret);
3656
837
}
3657
3658
static xmlAttrPtr
3659
1.97k
xmlCopyPropInternal(xmlDocPtr doc, xmlNodePtr target, xmlAttrPtr cur) {
3660
1.97k
    xmlAttrPtr ret = NULL;
3661
3662
1.97k
    if (cur == NULL) return(NULL);
3663
1.97k
    if ((target != NULL) && (target->type != XML_ELEMENT_NODE))
3664
0
        return(NULL);
3665
1.97k
    if (target != NULL)
3666
1.97k
  ret = xmlNewDocProp(target->doc, cur->name, NULL);
3667
0
    else if (doc != NULL)
3668
0
  ret = xmlNewDocProp(doc, cur->name, NULL);
3669
0
    else if (cur->parent != NULL)
3670
0
  ret = xmlNewDocProp(cur->parent->doc, cur->name, NULL);
3671
0
    else if (cur->children != NULL)
3672
0
  ret = xmlNewDocProp(cur->children->doc, cur->name, NULL);
3673
0
    else
3674
0
  ret = xmlNewDocProp(NULL, cur->name, NULL);
3675
1.97k
    if (ret == NULL) return(NULL);
3676
1.97k
    ret->parent = target;
3677
3678
1.97k
    if ((cur->ns != NULL) && (target != NULL)) {
3679
706
      xmlNsPtr ns;
3680
706
      int res;
3681
3682
706
      res = xmlSearchNsSafe(target, cur->ns->prefix, &ns);
3683
706
      if (res < 0)
3684
0
          goto error;
3685
706
      if (ns == NULL) {
3686
        /*
3687
         * Humm, we are copying an element whose namespace is defined
3688
         * out of the new tree scope. Search it in the original tree
3689
         * and add it at the top of the new tree
3690
         */
3691
0
        res = xmlSearchNsSafe(cur->parent, cur->ns->prefix, &ns);
3692
0
        if (res < 0)
3693
0
          goto error;
3694
0
        if (ns != NULL) {
3695
0
          xmlNodePtr root = target;
3696
0
          xmlNodePtr pred = NULL;
3697
3698
0
          while (root->parent != NULL) {
3699
0
            pred = root;
3700
0
            root = root->parent;
3701
0
          }
3702
0
          if (root == (xmlNodePtr) target->doc) {
3703
            /* correct possibly cycling above the document elt */
3704
0
            root = pred;
3705
0
          }
3706
0
          ret->ns = xmlNewNs(root, ns->href, ns->prefix);
3707
0
          if (ret->ns == NULL)
3708
0
              goto error;
3709
0
        }
3710
706
      } else {
3711
        /*
3712
         * we have to find something appropriate here since
3713
         * we can't be sure, that the namespace we found is identified
3714
         * by the prefix
3715
         */
3716
706
        if (xmlStrEqual(ns->href, cur->ns->href)) {
3717
          /* this is the nice case */
3718
706
          ret->ns = ns;
3719
706
        } else {
3720
          /*
3721
           * we are in trouble: we need a new reconciled namespace.
3722
           * This is expensive
3723
           */
3724
0
          ret->ns = xmlNewReconciledNs(target, cur->ns);
3725
0
          if (ret->ns == NULL)
3726
0
              goto error;
3727
0
        }
3728
706
      }
3729
3730
706
    } else
3731
1.27k
        ret->ns = NULL;
3732
3733
1.97k
    if (cur->children != NULL) {
3734
1.97k
  xmlNodePtr tmp;
3735
3736
1.97k
  ret->children = xmlStaticCopyNodeList(cur->children, ret->doc, (xmlNodePtr) ret);
3737
1.97k
        if (ret->children == NULL)
3738
0
            goto error;
3739
1.97k
  ret->last = NULL;
3740
1.97k
  tmp = ret->children;
3741
3.95k
  while (tmp != NULL) {
3742
      /* tmp->parent = (xmlNodePtr)ret; */
3743
1.97k
      if (tmp->next == NULL)
3744
1.97k
          ret->last = tmp;
3745
1.97k
      tmp = tmp->next;
3746
1.97k
  }
3747
1.97k
    }
3748
    /*
3749
     * Try to handle IDs
3750
     */
3751
1.97k
    if ((target != NULL) && (cur != NULL) &&
3752
1.97k
  (target->doc != NULL) && (cur->doc != NULL) &&
3753
1.97k
        (cur->parent != NULL) &&
3754
1.97k
        (cur->children != NULL)) {
3755
1.97k
        int res = xmlIsID(cur->doc, cur->parent, cur);
3756
3757
1.97k
        if (res < 0)
3758
0
            goto error;
3759
1.97k
  if (res != 0) {
3760
227
      xmlChar *id;
3761
3762
227
      id = xmlNodeGetContent((xmlNodePtr) cur);
3763
227
      if (id == NULL)
3764
0
                goto error;
3765
227
            res = xmlAddIDSafe(ret, id);
3766
227
      xmlFree(id);
3767
227
            if (res < 0)
3768
0
                goto error;
3769
227
  }
3770
1.97k
    }
3771
1.97k
    return(ret);
3772
3773
0
error:
3774
0
    xmlFreeProp(ret);
3775
0
    return(NULL);
3776
1.97k
}
3777
3778
/**
3779
 * Create a copy of the attribute. This function sets the parent
3780
 * pointer of the copy to `target` but doesn't set the attribute on
3781
 * the target element. Users should consider to set the attribute
3782
 * by calling #xmlAddChild afterwards or reset the parent pointer to
3783
 * NULL.
3784
 *
3785
 * @param target  the element where the attribute will be grafted
3786
 * @param cur  the attribute
3787
 * @returns the copied attribute or NULL if a memory allocation
3788
 * failed.
3789
 */
3790
xmlAttr *
3791
1.97k
xmlCopyProp(xmlNode *target, xmlAttr *cur) {
3792
1.97k
  return xmlCopyPropInternal(NULL, target, cur);
3793
1.97k
}
3794
3795
/**
3796
 * Create a copy of an attribute list. This function sets the
3797
 * parent pointers of the copied attributes to `target` but doesn't
3798
 * set the attributes on the target element.
3799
 *
3800
 * @param target  the element where the attributes will be grafted
3801
 * @param cur  the first attribute
3802
 * @returns the head of the copied list or NULL if a memory
3803
 * allocation failed.
3804
 */
3805
xmlAttr *
3806
1.08k
xmlCopyPropList(xmlNode *target, xmlAttr *cur) {
3807
1.08k
    xmlAttrPtr ret = NULL;
3808
1.08k
    xmlAttrPtr p = NULL,q;
3809
3810
1.08k
    if ((target != NULL) && (target->type != XML_ELEMENT_NODE))
3811
0
        return(NULL);
3812
3.06k
    while (cur != NULL) {
3813
1.97k
        q = xmlCopyProp(target, cur);
3814
1.97k
  if (q == NULL) {
3815
0
            xmlFreePropList(ret);
3816
0
      return(NULL);
3817
0
        }
3818
1.97k
  if (p == NULL) {
3819
1.08k
      ret = p = q;
3820
1.08k
  } else {
3821
889
      p->next = q;
3822
889
      q->prev = p;
3823
889
      p = q;
3824
889
  }
3825
1.97k
  cur = cur->next;
3826
1.97k
    }
3827
1.08k
    return(ret);
3828
1.08k
}
3829
3830
/*
3831
 * NOTE about the CopyNode operations !
3832
 *
3833
 * They are split into external and internal parts for one
3834
 * tricky reason: namespaces. Doing a direct copy of a node
3835
 * say RPM:Copyright without changing the namespace pointer to
3836
 * something else can produce stale links. One way to do it is
3837
 * to keep a reference counter but this doesn't work as soon
3838
 * as one moves the element or the subtree out of the scope of
3839
 * the existing namespace. The actual solution seems to be to add
3840
 * a copy of the namespace at the top of the copied tree if
3841
 * not available in the subtree.
3842
 * Hence two functions, the public front-end call the inner ones
3843
 * The argument "recursive" normally indicates a recursive copy
3844
 * of the node with values 0 (no) and 1 (yes).  For XInclude,
3845
 * however, we allow a value of 2 to indicate copy properties and
3846
 * namespace info, but don't recurse on children.
3847
 */
3848
3849
/**
3850
 * Copy a node.
3851
 *
3852
 * @param node  source node
3853
 * @param doc  target document
3854
 * @param parent  target parent
3855
 * @param extended  flags
3856
 * @returns the copy or NULL if a memory allocation failed.
3857
 */
3858
xmlNode *
3859
xmlStaticCopyNode(xmlNode *node, xmlDoc *doc, xmlNode *parent,
3860
10.5k
                  int extended) {
3861
10.5k
    xmlNodePtr ret;
3862
3863
10.5k
    if (node == NULL) return(NULL);
3864
10.5k
    switch (node->type) {
3865
4.34k
        case XML_TEXT_NODE:
3866
4.79k
        case XML_CDATA_SECTION_NODE:
3867
9.72k
        case XML_ELEMENT_NODE:
3868
9.72k
        case XML_DOCUMENT_FRAG_NODE:
3869
9.72k
        case XML_ENTITY_REF_NODE:
3870
10.3k
        case XML_PI_NODE:
3871
10.5k
        case XML_COMMENT_NODE:
3872
10.5k
        case XML_XINCLUDE_START:
3873
10.5k
        case XML_XINCLUDE_END:
3874
10.5k
      break;
3875
0
        case XML_ATTRIBUTE_NODE:
3876
0
    return((xmlNodePtr) xmlCopyPropInternal(doc, parent, (xmlAttrPtr) node));
3877
0
        case XML_NAMESPACE_DECL:
3878
0
      return((xmlNodePtr) xmlCopyNamespaceList((xmlNsPtr) node));
3879
3880
0
        case XML_DOCUMENT_NODE:
3881
0
        case XML_HTML_DOCUMENT_NODE:
3882
0
      return((xmlNodePtr) xmlCopyDoc((xmlDocPtr) node, extended));
3883
0
        default:
3884
0
            return(NULL);
3885
10.5k
    }
3886
3887
    /*
3888
     * Allocate a new node and fill the fields.
3889
     */
3890
10.5k
    ret = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
3891
10.5k
    if (ret == NULL)
3892
0
  return(NULL);
3893
10.5k
    memset(ret, 0, sizeof(xmlNode));
3894
10.5k
    ret->type = node->type;
3895
3896
10.5k
    ret->doc = doc;
3897
10.5k
    ret->parent = parent;
3898
10.5k
    if (node->name == xmlStringText)
3899
4.34k
  ret->name = xmlStringText;
3900
6.22k
    else if (node->name == xmlStringTextNoenc)
3901
0
  ret->name = xmlStringTextNoenc;
3902
6.22k
    else if (node->name == xmlStringComment)
3903
245
  ret->name = xmlStringComment;
3904
5.97k
    else if (node->name != NULL) {
3905
5.52k
        if ((doc != NULL) && (doc->dict != NULL))
3906
0
      ret->name = xmlDictLookup(doc->dict, node->name, -1);
3907
5.52k
  else
3908
5.52k
      ret->name = xmlStrdup(node->name);
3909
5.52k
        if (ret->name == NULL)
3910
0
            goto error;
3911
5.52k
    }
3912
10.5k
    if ((node->type != XML_ELEMENT_NODE) &&
3913
10.5k
  (node->content != NULL) &&
3914
10.5k
  (node->type != XML_ENTITY_REF_NODE) &&
3915
10.5k
  (node->type != XML_XINCLUDE_END) &&
3916
10.5k
  (node->type != XML_XINCLUDE_START)) {
3917
5.03k
  ret->content = xmlStrdup(node->content);
3918
5.03k
        if (ret->content == NULL)
3919
0
            goto error;
3920
5.52k
    }else{
3921
5.52k
      if (node->type == XML_ELEMENT_NODE)
3922
4.93k
        ret->line = node->line;
3923
5.52k
    }
3924
3925
10.5k
    if (!extended)
3926
0
  goto out;
3927
10.5k
    if (((node->type == XML_ELEMENT_NODE) ||
3928
10.5k
         (node->type == XML_XINCLUDE_START)) && (node->nsDef != NULL)) {
3929
837
        ret->nsDef = xmlCopyNamespaceList(node->nsDef);
3930
837
        if (ret->nsDef == NULL)
3931
0
            goto error;
3932
837
    }
3933
3934
10.5k
    if ((node->type == XML_ELEMENT_NODE) && (node->ns != NULL)) {
3935
1.38k
        xmlNsPtr ns = NULL;
3936
1.38k
        int res;
3937
3938
1.38k
  res = xmlSearchNsSafe(ret, node->ns->prefix, &ns);
3939
1.38k
        if (res < 0)
3940
0
            goto error;
3941
1.38k
  if (ns == NULL) {
3942
      /*
3943
       * Humm, we are copying an element whose namespace is defined
3944
       * out of the new tree scope. Search it in the original tree
3945
       * and add it at the top of the new tree.
3946
             *
3947
             * TODO: Searching the original tree seems unnecessary. We
3948
             * already have a namespace URI.
3949
       */
3950
0
      res = xmlSearchNsSafe(node, node->ns->prefix, &ns);
3951
0
            if (res < 0)
3952
0
                goto error;
3953
0
      if (ns != NULL) {
3954
0
          xmlNodePtr root = ret;
3955
3956
0
    while (root->parent != NULL) root = root->parent;
3957
0
    ret->ns = xmlNewNs(root, ns->href, ns->prefix);
3958
0
            } else {
3959
0
                ret->ns = xmlNewReconciledNs(ret, node->ns);
3960
0
      }
3961
0
            if (ret->ns == NULL)
3962
0
                goto error;
3963
1.38k
  } else {
3964
      /*
3965
       * reference the existing namespace definition in our own tree.
3966
       */
3967
1.38k
      ret->ns = ns;
3968
1.38k
  }
3969
1.38k
    }
3970
10.5k
    if ((node->type == XML_ELEMENT_NODE) && (node->properties != NULL)) {
3971
1.08k
        ret->properties = xmlCopyPropList(ret, node->properties);
3972
1.08k
        if (ret->properties == NULL)
3973
0
            goto error;
3974
1.08k
    }
3975
10.5k
    if (node->type == XML_ENTITY_REF_NODE) {
3976
0
  if ((doc == NULL) || (node->doc != doc)) {
3977
      /*
3978
       * The copied node will go into a separate document, so
3979
       * to avoid dangling references to the ENTITY_DECL node
3980
       * we cannot keep the reference. Try to find it in the
3981
       * target document.
3982
       */
3983
0
      ret->children = (xmlNodePtr) xmlGetDocEntity(doc, ret->name);
3984
0
  } else {
3985
0
            ret->children = node->children;
3986
0
  }
3987
0
  ret->last = ret->children;
3988
10.5k
    } else if ((node->children != NULL) && (extended != 2)) {
3989
1.03k
        xmlNodePtr cur, insert;
3990
3991
1.03k
        cur = node->children;
3992
1.03k
        insert = ret;
3993
3.71k
        while (cur != NULL) {
3994
2.67k
            xmlNodePtr copy = xmlStaticCopyNode(cur, doc, insert, 2);
3995
2.67k
            if (copy == NULL)
3996
0
                goto error;
3997
3998
            /* Check for coalesced text nodes */
3999
2.67k
            if (insert->last != copy) {
4000
2.67k
                if (insert->last == NULL) {
4001
1.64k
                    insert->children = copy;
4002
1.64k
                } else {
4003
1.03k
                    copy->prev = insert->last;
4004
1.03k
                    insert->last->next = copy;
4005
1.03k
                }
4006
2.67k
                insert->last = copy;
4007
2.67k
            }
4008
4009
2.67k
            if ((cur->type != XML_ENTITY_REF_NODE) &&
4010
2.67k
                (cur->children != NULL)) {
4011
602
                cur = cur->children;
4012
602
                insert = copy;
4013
602
                continue;
4014
602
            }
4015
4016
2.67k
            while (1) {
4017
2.67k
                if (cur->next != NULL) {
4018
1.03k
                    cur = cur->next;
4019
1.03k
                    break;
4020
1.03k
                }
4021
4022
1.64k
                cur = cur->parent;
4023
1.64k
                insert = insert->parent;
4024
1.64k
                if (cur == node) {
4025
1.03k
                    cur = NULL;
4026
1.03k
                    break;
4027
1.03k
                }
4028
1.64k
            }
4029
2.07k
        }
4030
1.03k
    }
4031
4032
10.5k
out:
4033
10.5k
    if ((xmlRegisterCallbacks) && (xmlRegisterNodeDefaultValue))
4034
0
  xmlRegisterNodeDefaultValue((xmlNodePtr)ret);
4035
10.5k
    return(ret);
4036
4037
0
error:
4038
0
    xmlFreeNode(ret);
4039
0
    return(NULL);
4040
10.5k
}
4041
4042
/**
4043
 * Copy a node list. If `parent` is provided, sets the parent pointer
4044
 * of the copied nodes, but doesn't update the children and last
4045
 * pointer of `parent`.
4046
 *
4047
 * @param node  node to copy
4048
 * @param doc  target document
4049
 * @param parent  target node (optional)
4050
 * @returns a the copy or NULL in case of error.
4051
 */
4052
xmlNode *
4053
1.97k
xmlStaticCopyNodeList(xmlNode *node, xmlDoc *doc, xmlNode *parent) {
4054
1.97k
    xmlNodePtr ret = NULL;
4055
1.97k
    xmlNodePtr p = NULL,q;
4056
1.97k
    xmlDtdPtr newSubset = NULL;
4057
1.97k
    int linkedSubset = 0;
4058
4059
3.95k
    while (node != NULL) {
4060
1.97k
        xmlNodePtr next = node->next;
4061
4062
1.97k
  if (node->type == XML_DTD_NODE ) {
4063
0
      if (doc == NULL) {
4064
0
    node = next;
4065
0
    continue;
4066
0
      }
4067
0
      if ((doc->intSubset == NULL) && (newSubset == NULL)) {
4068
0
    q = (xmlNodePtr) xmlCopyDtd( (xmlDtdPtr) node );
4069
0
    if (q == NULL) goto error;
4070
                /* Can't fail on DTD */
4071
0
    xmlSetTreeDoc(q, doc);
4072
0
    q->parent = parent;
4073
0
    newSubset = (xmlDtdPtr) q;
4074
0
      } else {
4075
                /*
4076
                 * We don't allow multiple internal subsets in a document,
4077
                 * so we move the DTD instead of creating a copy.
4078
                 */
4079
0
                linkedSubset = 1;
4080
0
    q = (xmlNodePtr) doc->intSubset;
4081
                /* Unlink */
4082
0
                if (q->prev == NULL) {
4083
0
                    if (q->parent != NULL)
4084
0
                        q->parent->children = q->next;
4085
0
                } else {
4086
0
                    q->prev->next = q->next;
4087
0
                }
4088
0
                if (q->next == NULL) {
4089
0
                    if (q->parent != NULL)
4090
0
                        q->parent->last = q->prev;
4091
0
                } else {
4092
0
                    q->next->prev = q->prev;
4093
0
                }
4094
0
                q->parent = parent;
4095
0
                q->next = NULL;
4096
0
                q->prev = NULL;
4097
0
      }
4098
0
  } else
4099
1.97k
      q = xmlStaticCopyNode(node, doc, parent, 1);
4100
1.97k
  if (q == NULL) goto error;
4101
1.97k
  if (ret == NULL) {
4102
1.97k
      q->prev = NULL;
4103
1.97k
      ret = p = q;
4104
1.97k
  } else if (p != q) {
4105
  /* the test is required if xmlStaticCopyNode coalesced 2 text nodes */
4106
0
      p->next = q;
4107
0
      q->prev = p;
4108
0
      p = q;
4109
0
  }
4110
1.97k
  node = next;
4111
1.97k
    }
4112
1.97k
    if ((doc != NULL) && (newSubset != NULL))
4113
0
        doc->intSubset = newSubset;
4114
1.97k
    return(ret);
4115
0
error:
4116
0
    xmlFreeNodeList(ret);
4117
0
    if (newSubset != NULL)
4118
0
        xmlFreeDtd(newSubset);
4119
0
    if (linkedSubset != 0) {
4120
0
        doc->intSubset->next = NULL;
4121
0
        doc->intSubset->prev = NULL;
4122
0
    }
4123
0
    return(NULL);
4124
1.97k
}
4125
4126
/**
4127
 * Copy a node.
4128
 *
4129
 * If `extended` is 0, make a shallow copy.
4130
 *
4131
 * If `extended` is 1, make a deep copy (properties, namespaces
4132
 * and children when applicable).
4133
 *
4134
 * If `extended` is 2, make a shallow copy including properties and
4135
 * namespaces of elements.
4136
 *
4137
 * Use of this function is DISCOURAGED in favor of #xmlDocCopyNode.
4138
 *
4139
 * @param node  the node
4140
 * @param extended  mode of operation
4141
 * @returns the copied node or NULL if a memory allocation failed.
4142
 */
4143
xmlNode *
4144
0
xmlCopyNode(xmlNode *node, int extended) {
4145
0
    xmlNodePtr ret;
4146
4147
0
    ret = xmlStaticCopyNode(node, NULL, NULL, extended);
4148
0
    return(ret);
4149
0
}
4150
4151
/**
4152
 * Copy a node into another document.
4153
 *
4154
 * If `extended` is 0, make a shallow copy.
4155
 *
4156
 * If `extended` is 1, make a deep copy (properties, namespaces
4157
 * and children when applicable).
4158
 *
4159
 * If `extended` is 2, make a shallow copy including properties and
4160
 * namespaces of elements.
4161
 *
4162
 * @param node  the node
4163
 * @param doc  the document
4164
 * @param extended  mode of operation
4165
 * @returns the copied node or NULL if a memory allocation failed.
4166
 */
4167
xmlNode *
4168
5.90k
xmlDocCopyNode(xmlNode *node, xmlDoc *doc, int extended) {
4169
5.90k
    xmlNodePtr ret;
4170
4171
5.90k
    ret = xmlStaticCopyNode(node, doc, NULL, extended);
4172
5.90k
    return(ret);
4173
5.90k
}
4174
4175
/**
4176
 * Copy a node list and all children into a new document.
4177
 *
4178
 * @param doc  the target document
4179
 * @param node  the first node in the list.
4180
 * @returns the head of the copied list or NULL if a memory
4181
 * allocation failed.
4182
 */
4183
0
xmlNode *xmlDocCopyNodeList(xmlDoc *doc, xmlNode *node) {
4184
0
    xmlNodePtr ret = xmlStaticCopyNodeList(node, doc, NULL);
4185
0
    return(ret);
4186
0
}
4187
4188
/**
4189
 * Copy a node list and all children.
4190
 *
4191
 * Use of this function is DISCOURAGED in favor of #xmlDocCopyNodeList.
4192
 *
4193
 * @param node  the first node in the list.
4194
 * @returns the head of the copied list or NULL if a memory
4195
 * allocation failed.
4196
 */
4197
0
xmlNode *xmlCopyNodeList(xmlNode *node) {
4198
0
    xmlNodePtr ret = xmlStaticCopyNodeList(node, NULL, NULL);
4199
0
    return(ret);
4200
0
}
4201
4202
/**
4203
 * Copy a DTD.
4204
 *
4205
 * @param dtd  the DTD
4206
 * @returns the copied DTD or NULL if a memory allocation failed.
4207
 */
4208
xmlDtd *
4209
0
xmlCopyDtd(xmlDtd *dtd) {
4210
0
    xmlDtdPtr ret;
4211
0
    xmlNodePtr cur, p = NULL, q;
4212
4213
0
    if (dtd == NULL) return(NULL);
4214
0
    ret = xmlNewDtd(NULL, dtd->name, dtd->ExternalID, dtd->SystemID);
4215
0
    if (ret == NULL) return(NULL);
4216
0
    if (dtd->entities != NULL) {
4217
0
        ret->entities = (void *) xmlCopyEntitiesTable(
4218
0
                      (xmlEntitiesTablePtr) dtd->entities);
4219
0
        if (ret->entities == NULL)
4220
0
            goto error;
4221
0
    }
4222
0
    if (dtd->notations != NULL) {
4223
0
        ret->notations = (void *) xmlCopyNotationTable(
4224
0
                      (xmlNotationTablePtr) dtd->notations);
4225
0
        if (ret->notations == NULL)
4226
0
            goto error;
4227
0
    }
4228
0
    if (dtd->elements != NULL) {
4229
0
        ret->elements = (void *) xmlCopyElementTable(
4230
0
                      (xmlElementTablePtr) dtd->elements);
4231
0
        if (ret->elements == NULL)
4232
0
            goto error;
4233
0
    }
4234
0
    if (dtd->attributes != NULL) {
4235
0
        ret->attributes = (void *) xmlCopyAttributeTable(
4236
0
                      (xmlAttributeTablePtr) dtd->attributes);
4237
0
        if (ret->attributes == NULL)
4238
0
            goto error;
4239
0
    }
4240
0
    if (dtd->pentities != NULL) {
4241
0
  ret->pentities = (void *) xmlCopyEntitiesTable(
4242
0
          (xmlEntitiesTablePtr) dtd->pentities);
4243
0
        if (ret->pentities == NULL)
4244
0
            goto error;
4245
0
    }
4246
4247
0
    cur = dtd->children;
4248
0
    while (cur != NULL) {
4249
0
  q = NULL;
4250
4251
0
  if (cur->type == XML_ENTITY_DECL) {
4252
0
      xmlEntityPtr tmp = (xmlEntityPtr) cur;
4253
0
      switch (tmp->etype) {
4254
0
    case XML_INTERNAL_GENERAL_ENTITY:
4255
0
    case XML_EXTERNAL_GENERAL_PARSED_ENTITY:
4256
0
    case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY:
4257
0
        q = (xmlNodePtr) xmlGetEntityFromDtd(ret, tmp->name);
4258
0
        break;
4259
0
    case XML_INTERNAL_PARAMETER_ENTITY:
4260
0
    case XML_EXTERNAL_PARAMETER_ENTITY:
4261
0
        q = (xmlNodePtr)
4262
0
      xmlGetParameterEntityFromDtd(ret, tmp->name);
4263
0
        break;
4264
0
    case XML_INTERNAL_PREDEFINED_ENTITY:
4265
0
        break;
4266
0
      }
4267
0
  } else if (cur->type == XML_ELEMENT_DECL) {
4268
0
      xmlElementPtr tmp = (xmlElementPtr) cur;
4269
0
      q = (xmlNodePtr)
4270
0
    xmlGetDtdQElementDesc(ret, tmp->name, tmp->prefix);
4271
0
  } else if (cur->type == XML_ATTRIBUTE_DECL) {
4272
0
      xmlAttributePtr tmp = (xmlAttributePtr) cur;
4273
0
      q = (xmlNodePtr)
4274
0
    xmlGetDtdQAttrDesc(ret, tmp->elem, tmp->name, tmp->prefix);
4275
0
  } else if (cur->type == XML_COMMENT_NODE) {
4276
0
      q = xmlCopyNode(cur, 0);
4277
0
            if (q == NULL)
4278
0
                goto error;
4279
0
  }
4280
4281
0
  if (q == NULL) {
4282
0
      cur = cur->next;
4283
0
      continue;
4284
0
  }
4285
4286
0
  if (p == NULL)
4287
0
      ret->children = q;
4288
0
  else
4289
0
      p->next = q;
4290
4291
0
  q->prev = p;
4292
0
  q->parent = (xmlNodePtr) ret;
4293
0
  q->next = NULL;
4294
0
  ret->last = q;
4295
0
  p = q;
4296
0
  cur = cur->next;
4297
0
    }
4298
4299
0
    return(ret);
4300
4301
0
error:
4302
0
    xmlFreeDtd(ret);
4303
0
    return(NULL);
4304
0
}
4305
4306
/**
4307
 * Copy a document. If recursive, the content tree will
4308
 * be copied too as well as DTD, namespaces and entities.
4309
 *
4310
 * @param doc  the document
4311
 * @param recursive  if not zero do a recursive copy.
4312
 * @returns the copied document or NULL if a memory allocation
4313
 * failed.
4314
 */
4315
xmlDoc *
4316
0
xmlCopyDoc(xmlDoc *doc, int recursive) {
4317
0
    xmlDocPtr ret;
4318
4319
0
    if (doc == NULL) return(NULL);
4320
0
    ret = xmlNewDoc(doc->version);
4321
0
    if (ret == NULL) return(NULL);
4322
0
    ret->type = doc->type;
4323
0
    if (doc->name != NULL) {
4324
0
        ret->name = xmlMemStrdup(doc->name);
4325
0
        if (ret->name == NULL)
4326
0
            goto error;
4327
0
    }
4328
0
    if (doc->encoding != NULL) {
4329
0
        ret->encoding = xmlStrdup(doc->encoding);
4330
0
        if (ret->encoding == NULL)
4331
0
            goto error;
4332
0
    }
4333
0
    if (doc->URL != NULL) {
4334
0
        ret->URL = xmlStrdup(doc->URL);
4335
0
        if (ret->URL == NULL)
4336
0
            goto error;
4337
0
    }
4338
0
    ret->charset = doc->charset;
4339
0
    ret->compression = doc->compression;
4340
0
    ret->standalone = doc->standalone;
4341
0
    if (!recursive) return(ret);
4342
4343
0
    ret->last = NULL;
4344
0
    ret->children = NULL;
4345
0
    if (doc->intSubset != NULL) {
4346
0
        ret->intSubset = xmlCopyDtd(doc->intSubset);
4347
0
  if (ret->intSubset == NULL)
4348
0
            goto error;
4349
        /* Can't fail on DTD */
4350
0
  xmlSetTreeDoc((xmlNodePtr)ret->intSubset, ret);
4351
0
    }
4352
0
    if (doc->oldNs != NULL) {
4353
0
        ret->oldNs = xmlCopyNamespaceList(doc->oldNs);
4354
0
        if (ret->oldNs == NULL)
4355
0
            goto error;
4356
0
    }
4357
0
    if (doc->children != NULL) {
4358
0
  xmlNodePtr tmp;
4359
4360
0
  ret->children = xmlStaticCopyNodeList(doc->children, ret,
4361
0
                                   (xmlNodePtr)ret);
4362
0
        if (ret->children == NULL)
4363
0
            goto error;
4364
0
  ret->last = NULL;
4365
0
  tmp = ret->children;
4366
0
  while (tmp != NULL) {
4367
0
      if (tmp->next == NULL)
4368
0
          ret->last = tmp;
4369
0
      tmp = tmp->next;
4370
0
  }
4371
0
    }
4372
0
    return(ret);
4373
4374
0
error:
4375
0
    xmlFreeDoc(ret);
4376
0
    return(NULL);
4377
0
}
4378
4379
/************************************************************************
4380
 *                  *
4381
 *    Content access functions        *
4382
 *                  *
4383
 ************************************************************************/
4384
4385
/**
4386
 * Get line number of `node`. Try to work around the limitation of
4387
 * line numbers being stored as 16 bits ints. Requires xmlParserOption
4388
 * XML_PARSE_BIG_LINES to be set when parsing.
4389
 *
4390
 * @param node  valid node
4391
 * @param depth  used to limit any risk of recursion
4392
 * @returns the line number if successful, -1 otherwise
4393
 */
4394
static long
4395
xmlGetLineNoInternal(const xmlNode *node, int depth)
4396
5.71k
{
4397
5.71k
    long result = -1;
4398
4399
5.71k
    if (depth >= 5)
4400
561
        return(-1);
4401
4402
5.15k
    if (!node)
4403
0
        return result;
4404
5.15k
    if ((node->type == XML_ELEMENT_NODE) ||
4405
5.15k
        (node->type == XML_TEXT_NODE) ||
4406
5.15k
  (node->type == XML_COMMENT_NODE) ||
4407
5.15k
  (node->type == XML_PI_NODE)) {
4408
4.89k
  if (node->line == 65535) {
4409
2.76k
      if ((node->type == XML_TEXT_NODE) && (node->psvi != NULL))
4410
0
          result = XML_PTR_TO_INT(node->psvi);
4411
2.76k
      else if ((node->type == XML_ELEMENT_NODE) &&
4412
2.76k
               (node->children != NULL))
4413
91
          result = xmlGetLineNoInternal(node->children, depth + 1);
4414
2.67k
      else if (node->next != NULL)
4415
1.03k
          result = xmlGetLineNoInternal(node->next, depth + 1);
4416
1.64k
      else if (node->prev != NULL)
4417
1.54k
          result = xmlGetLineNoInternal(node->prev, depth + 1);
4418
2.76k
  }
4419
4.89k
  if ((result == -1) || (result == 65535))
4420
4.83k
      result = node->line;
4421
4.89k
    } else if ((node->prev != NULL) &&
4422
263
             ((node->prev->type == XML_ELEMENT_NODE) ||
4423
177
        (node->prev->type == XML_TEXT_NODE) ||
4424
177
        (node->prev->type == XML_COMMENT_NODE) ||
4425
177
        (node->prev->type == XML_PI_NODE)))
4426
147
        result = xmlGetLineNoInternal(node->prev, depth + 1);
4427
116
    else if ((node->parent != NULL) &&
4428
116
             (node->parent->type == XML_ELEMENT_NODE))
4429
100
        result = xmlGetLineNoInternal(node->parent, depth + 1);
4430
4431
5.15k
    return result;
4432
5.15k
}
4433
4434
/**
4435
 * Get line number of `node`. Try to work around the limitation of
4436
 * line numbers being stored as 16 bits ints. Requires xmlParserOption
4437
 * XML_PARSE_BIG_LINES to be set when parsing.
4438
 *
4439
 * @param node  valid node
4440
 * @returns the line number if successful, -1 otherwise
4441
 */
4442
long
4443
xmlGetLineNo(const xmlNode *node)
4444
2.80k
{
4445
2.80k
    return(xmlGetLineNoInternal(node, 0));
4446
2.80k
}
4447
4448
/**
4449
 * Build a structure based Path for the given node
4450
 *
4451
 * @param node  a node
4452
 * @returns the new path or NULL in case of error. The caller must free
4453
 *     the returned string
4454
 */
4455
xmlChar *
4456
xmlGetNodePath(const xmlNode *node)
4457
0
{
4458
0
    const xmlNode *cur, *tmp;
4459
0
    const xmlNode **nodes = NULL;
4460
0
    xmlChar *ret = NULL;
4461
0
    xmlBuf *buf;
4462
0
    size_t numNodes, i;
4463
4464
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL))
4465
0
        return(NULL);
4466
4467
0
    buf = xmlBufCreate(50);
4468
0
    if (buf == NULL)
4469
0
        return(NULL);
4470
4471
    /*
4472
     * Get list of ancestors
4473
     */
4474
0
    numNodes = 0;
4475
0
    for (cur = node; cur != NULL; cur = cur->parent)
4476
0
        numNodes += 1;
4477
0
    if (numNodes > SIZE_MAX / sizeof(nodes[0]))
4478
0
        goto error;
4479
0
    nodes = xmlMalloc(numNodes * sizeof(nodes[0]));
4480
0
    if (nodes == NULL)
4481
0
        goto error;
4482
0
    i = 0;
4483
0
    for (cur = node; cur != NULL && i < numNodes; cur = cur->parent)
4484
0
        nodes[i++] = cur;
4485
4486
    /*
4487
     * Iterate in reverse to start at root
4488
     */
4489
0
    while (i > 0) {
4490
0
        int occur = 0;
4491
4492
0
        i -= 1;
4493
0
        cur = nodes[i];
4494
4495
0
        if ((cur->type == XML_DOCUMENT_NODE) ||
4496
0
            (cur->type == XML_HTML_DOCUMENT_NODE)) {
4497
0
            if (i == 0)
4498
0
                xmlBufCat(buf, BAD_CAST "/");
4499
0
        } else if (cur->type == XML_ELEMENT_NODE) {
4500
0
            int generic = 0;
4501
4502
0
            xmlBufCat(buf, BAD_CAST "/");
4503
4504
0
            if (cur->ns) {
4505
0
    if (cur->ns->prefix != NULL) {
4506
0
                    xmlBufCat(buf, cur->ns->prefix);
4507
0
                    xmlBufCat(buf, BAD_CAST ":");
4508
0
                    xmlBufCat(buf, cur->name);
4509
0
    } else {
4510
        /*
4511
        * We cannot express named elements in the default
4512
        * namespace, so use "*".
4513
        */
4514
0
        generic = 1;
4515
0
                    xmlBufCat(buf, BAD_CAST "*");
4516
0
    }
4517
0
            } else {
4518
0
                xmlBufCat(buf, cur->name);
4519
0
            }
4520
4521
            /*
4522
             * Thumbler index computation
4523
       * TODO: the occurrence test seems bogus for namespaced names
4524
             */
4525
0
            tmp = cur->prev;
4526
0
            while (tmp != NULL) {
4527
0
                if ((tmp->type == XML_ELEMENT_NODE) &&
4528
0
        (generic ||
4529
0
         (xmlStrEqual(cur->name, tmp->name) &&
4530
0
         ((tmp->ns == cur->ns) ||
4531
0
          ((tmp->ns != NULL) && (cur->ns != NULL) &&
4532
0
           (xmlStrEqual(cur->ns->prefix, tmp->ns->prefix)))))))
4533
0
                    occur++;
4534
0
                tmp = tmp->prev;
4535
0
            }
4536
0
            if (occur == 0) {
4537
0
                tmp = cur->next;
4538
0
                while (tmp != NULL && occur == 0) {
4539
0
                    if ((tmp->type == XML_ELEMENT_NODE) &&
4540
0
      (generic ||
4541
0
       (xmlStrEqual(cur->name, tmp->name) &&
4542
0
       ((tmp->ns == cur->ns) ||
4543
0
        ((tmp->ns != NULL) && (cur->ns != NULL) &&
4544
0
         (xmlStrEqual(cur->ns->prefix, tmp->ns->prefix)))))))
4545
0
                        occur++;
4546
0
                    tmp = tmp->next;
4547
0
                }
4548
0
                if (occur != 0)
4549
0
                    occur = 1;
4550
0
            } else
4551
0
                occur++;
4552
0
        } else if (cur->type == XML_COMMENT_NODE) {
4553
0
            xmlBufCat(buf, BAD_CAST "/comment()");
4554
4555
            /*
4556
             * Thumbler index computation
4557
             */
4558
0
            tmp = cur->prev;
4559
0
            while (tmp != NULL) {
4560
0
                if (tmp->type == XML_COMMENT_NODE)
4561
0
        occur++;
4562
0
                tmp = tmp->prev;
4563
0
            }
4564
0
            if (occur == 0) {
4565
0
                tmp = cur->next;
4566
0
                while (tmp != NULL && occur == 0) {
4567
0
        if (tmp->type == XML_COMMENT_NODE)
4568
0
            occur++;
4569
0
                    tmp = tmp->next;
4570
0
                }
4571
0
                if (occur != 0)
4572
0
                    occur = 1;
4573
0
            } else
4574
0
                occur++;
4575
0
        } else if ((cur->type == XML_TEXT_NODE) ||
4576
0
                   (cur->type == XML_CDATA_SECTION_NODE)) {
4577
0
            xmlBufCat(buf, BAD_CAST "/text()");
4578
4579
            /*
4580
             * Thumbler index computation
4581
             */
4582
0
            tmp = cur->prev;
4583
0
            while (tmp != NULL) {
4584
0
                if ((tmp->type == XML_TEXT_NODE) ||
4585
0
        (tmp->type == XML_CDATA_SECTION_NODE))
4586
0
        occur++;
4587
0
                tmp = tmp->prev;
4588
0
            }
4589
      /*
4590
      * Evaluate if this is the only text- or CDATA-section-node;
4591
      * if yes, then we'll get "text()", otherwise "text()[1]".
4592
      */
4593
0
            if (occur == 0) {
4594
0
                tmp = cur->next;
4595
0
                while (tmp != NULL) {
4596
0
        if ((tmp->type == XML_TEXT_NODE) ||
4597
0
      (tmp->type == XML_CDATA_SECTION_NODE))
4598
0
        {
4599
0
      occur = 1;
4600
0
      break;
4601
0
        }
4602
0
        tmp = tmp->next;
4603
0
    }
4604
0
            } else
4605
0
                occur++;
4606
0
        } else if (cur->type == XML_PI_NODE) {
4607
0
            xmlBufCat(buf, BAD_CAST "/processing-instruction('");
4608
0
            xmlBufCat(buf, cur->name);
4609
0
            xmlBufCat(buf, BAD_CAST "')");
4610
4611
            /*
4612
             * Thumbler index computation
4613
             */
4614
0
            tmp = cur->prev;
4615
0
            while (tmp != NULL) {
4616
0
                if ((tmp->type == XML_PI_NODE) &&
4617
0
        (xmlStrEqual(cur->name, tmp->name)))
4618
0
                    occur++;
4619
0
                tmp = tmp->prev;
4620
0
            }
4621
0
            if (occur == 0) {
4622
0
                tmp = cur->next;
4623
0
                while (tmp != NULL && occur == 0) {
4624
0
                    if ((tmp->type == XML_PI_NODE) &&
4625
0
      (xmlStrEqual(cur->name, tmp->name)))
4626
0
                        occur++;
4627
0
                    tmp = tmp->next;
4628
0
                }
4629
0
                if (occur != 0)
4630
0
                    occur = 1;
4631
0
            } else
4632
0
                occur++;
4633
4634
0
        } else if (cur->type == XML_ATTRIBUTE_NODE) {
4635
0
            xmlBufCat(buf, BAD_CAST "/@");
4636
0
            if (cur->ns && cur->ns->prefix != NULL) {
4637
0
                xmlBufCat(buf, cur->ns->prefix);
4638
0
                xmlBufCat(buf, BAD_CAST ":");
4639
0
            }
4640
0
            xmlBufCat(buf, cur->name);
4641
0
        } else {
4642
0
            goto error;
4643
0
        }
4644
4645
0
        if (occur > 0) {
4646
0
            char tmpbuf[30];
4647
4648
0
            snprintf(tmpbuf, sizeof(tmpbuf), "[%d]", occur);
4649
0
            xmlBufCat(buf, BAD_CAST tmpbuf);
4650
0
        }
4651
0
    }
4652
4653
0
    ret = xmlBufDetach(buf);
4654
4655
0
error:
4656
0
    xmlBufFree(buf);
4657
0
    xmlFree(nodes);
4658
0
    return(ret);
4659
0
}
4660
4661
/**
4662
 * Get the root element of the document.
4663
 *
4664
 * Searches the document's children for the root element. The first
4665
 * child is not necessarily the root element, but could also be a
4666
 * DTD, comment or PI.
4667
 *
4668
 * @param doc  the document
4669
 * @returns the root element or NULL if no element was found.
4670
 */
4671
xmlNode *
4672
0
xmlDocGetRootElement(const xmlDoc *doc) {
4673
0
    xmlNodePtr ret;
4674
4675
0
    if (doc == NULL) return(NULL);
4676
0
    ret = doc->children;
4677
0
    while (ret != NULL) {
4678
0
  if (ret->type == XML_ELEMENT_NODE)
4679
0
      return(ret);
4680
0
        ret = ret->next;
4681
0
    }
4682
0
    return(ret);
4683
0
}
4684
4685
/**
4686
 * If `root` is NULL no action is taken. To remove a node from a
4687
 * document, use #xmlUnlinkNode instead.
4688
 *
4689
 * Set the root element of the document (`doc->children` is a list
4690
 * containing possibly comments, PIs, etc ...).
4691
 *
4692
 * `root` must be an element node. It is unlinked before insertion.
4693
 *
4694
 * @param doc  the document
4695
 * @param root  the new document root element
4696
 * @returns the unlinked old root element or NULL if the document
4697
 * didn't have a root element or a memory allocation failed.
4698
 */
4699
xmlNode *
4700
0
xmlDocSetRootElement(xmlDoc *doc, xmlNode *root) {
4701
0
    xmlNodePtr old = NULL;
4702
4703
0
    if (doc == NULL) return(NULL);
4704
0
    if ((root == NULL) || (root->type == XML_NAMESPACE_DECL))
4705
0
  return(NULL);
4706
0
    old = doc->children;
4707
0
    while (old != NULL) {
4708
0
  if (old->type == XML_ELEMENT_NODE)
4709
0
      break;
4710
0
        old = old->next;
4711
0
    }
4712
0
    if (old == root)
4713
0
        return(old);
4714
0
    xmlUnlinkNodeInternal(root);
4715
0
    if (xmlSetTreeDoc(root, doc) < 0)
4716
0
        return(NULL);
4717
0
    root->parent = (xmlNodePtr) doc;
4718
0
    if (old == NULL) {
4719
0
  if (doc->children == NULL) {
4720
0
      doc->children = root;
4721
0
      doc->last = root;
4722
0
  } else {
4723
0
      xmlAddSibling(doc->children, root);
4724
0
  }
4725
0
    } else {
4726
0
  xmlReplaceNode(old, root);
4727
0
    }
4728
0
    return(old);
4729
0
}
4730
4731
/**
4732
 * Set the `xml:lang` attribute of a node.
4733
 *
4734
 * @param cur  the node being changed
4735
 * @param lang  the language description
4736
 * @returns 0 on success, 1 if arguments are invalid, -1 if a
4737
 * memory allocation failed.
4738
 */
4739
int
4740
0
xmlNodeSetLang(xmlNode *cur, const xmlChar *lang) {
4741
0
    xmlNsPtr ns;
4742
0
    xmlAttrPtr attr;
4743
0
    int res;
4744
4745
0
    if ((cur == NULL) || (cur->type != XML_ELEMENT_NODE))
4746
0
        return(1);
4747
4748
0
    res = xmlSearchNsByHrefSafe(cur, XML_XML_NAMESPACE, &ns);
4749
0
    if (res != 0)
4750
0
        return(res);
4751
0
    attr = xmlSetNsProp(cur, ns, BAD_CAST "lang", lang);
4752
0
    if (attr == NULL)
4753
0
        return(-1);
4754
4755
0
    return(0);
4756
0
}
4757
4758
/**
4759
 * Find the `xml:lang` of a node.
4760
 *
4761
 * Look up the value of the `xml:lang` attribute or the one carried
4762
 * by the nearest ancestor.
4763
 *
4764
 * @param cur  the node being checked
4765
 * @returns a pointer to the lang value, or NULL if not found
4766
 *     It's up to the caller to free the memory with #xmlFree.
4767
 */
4768
xmlChar *
4769
0
xmlNodeGetLang(const xmlNode *cur) {
4770
0
    xmlChar *lang;
4771
0
    int res;
4772
4773
0
    if ((cur == NULL) || (cur->type == XML_NAMESPACE_DECL))
4774
0
        return(NULL);
4775
4776
0
    while (cur != NULL) {
4777
0
        res = xmlNodeGetAttrValue(cur, BAD_CAST "lang", XML_XML_NAMESPACE,
4778
0
                                  &lang);
4779
0
        if (res < 0)
4780
0
            return(NULL);
4781
0
  if (lang != NULL)
4782
0
      return(lang);
4783
4784
0
  cur = cur->parent;
4785
0
    }
4786
4787
0
    return(NULL);
4788
0
}
4789
4790
4791
/**
4792
 * Set the `xml:space` attribute of a node.
4793
 *
4794
 * @param cur  the node being changed
4795
 * @param val  the xml:space value ("0": default, 1: "preserve")
4796
 * @returns 0 on success, 1 if arguments are invalid, -1 if a
4797
 * memory allocation failed.
4798
 */
4799
int
4800
0
xmlNodeSetSpacePreserve(xmlNode *cur, int val) {
4801
0
    xmlNsPtr ns;
4802
0
    xmlAttrPtr attr;
4803
0
    const char *string;
4804
0
    int res;
4805
4806
0
    if ((cur == NULL) || (cur->type != XML_ELEMENT_NODE))
4807
0
        return(1);
4808
4809
0
    res = xmlSearchNsByHrefSafe(cur, XML_XML_NAMESPACE, &ns);
4810
0
    if (res != 0)
4811
0
  return(res);
4812
4813
0
    if (val == 0)
4814
0
        string = "default";
4815
0
    else
4816
0
        string = "preserve";
4817
4818
0
    attr = xmlSetNsProp(cur, ns, BAD_CAST "space", BAD_CAST string);
4819
0
    if (attr == NULL)
4820
0
        return(-1);
4821
4822
0
    return(0);
4823
0
}
4824
4825
/**
4826
 * Find the `xml:space` of a node.
4827
 *
4828
 * Look up the value of the `xml:space` attribute or the one carried
4829
 * by the nearest ancestor.
4830
 *
4831
 * @param cur  the node being checked
4832
 * @returns -1 if xml:space is not inherited, 0 if "default", 1 if "preserve"
4833
 */
4834
int
4835
0
xmlNodeGetSpacePreserve(const xmlNode *cur) {
4836
0
    xmlChar *space;
4837
0
        int res;
4838
4839
0
    if ((cur == NULL) || (cur->type != XML_ELEMENT_NODE))
4840
0
        return(-1);
4841
4842
0
    while (cur != NULL) {
4843
0
  res = xmlNodeGetAttrValue(cur, BAD_CAST "space", XML_XML_NAMESPACE,
4844
0
                                  &space);
4845
0
        if (res < 0)
4846
0
            return(-1);
4847
0
  if (space != NULL) {
4848
0
      if (xmlStrEqual(space, BAD_CAST "preserve")) {
4849
0
    xmlFree(space);
4850
0
    return(1);
4851
0
      }
4852
0
      if (xmlStrEqual(space, BAD_CAST "default")) {
4853
0
    xmlFree(space);
4854
0
    return(0);
4855
0
      }
4856
0
      xmlFree(space);
4857
0
  }
4858
4859
0
  cur = cur->parent;
4860
0
    }
4861
4862
0
    return(-1);
4863
0
}
4864
4865
/**
4866
 * Set (or reset) the name of a node.
4867
 *
4868
 * @param cur  the node being changed
4869
 * @param name  the new tag name
4870
 */
4871
void
4872
0
xmlNodeSetName(xmlNode *cur, const xmlChar *name) {
4873
0
    xmlDocPtr doc;
4874
0
    xmlDictPtr dict;
4875
0
    const xmlChar *copy;
4876
0
    const xmlChar *oldName;
4877
4878
0
    if (cur == NULL) return;
4879
0
    if (name == NULL) return;
4880
0
    switch(cur->type) {
4881
0
        case XML_ELEMENT_NODE:
4882
0
        case XML_ATTRIBUTE_NODE:
4883
0
        case XML_PI_NODE:
4884
0
        case XML_ENTITY_REF_NODE:
4885
0
      break;
4886
0
        default:
4887
0
            return;
4888
0
    }
4889
4890
0
    doc = cur->doc;
4891
0
    if (doc != NULL)
4892
0
  dict = doc->dict;
4893
0
    else
4894
0
        dict = NULL;
4895
4896
0
    if (dict != NULL)
4897
0
        copy = xmlDictLookup(dict, name, -1);
4898
0
    else
4899
0
        copy = xmlStrdup(name);
4900
0
    if (copy == NULL)
4901
0
        return;
4902
4903
0
    oldName = cur->name;
4904
0
    cur->name = copy;
4905
0
    if ((oldName != NULL) &&
4906
0
        ((dict == NULL) || (!xmlDictOwns(dict, oldName))))
4907
0
        xmlFree((xmlChar *) oldName);
4908
0
}
4909
4910
/**
4911
 * Set (or reset) the base URI of a node, i.e. the value of the
4912
 * `xml:base` attribute.
4913
 *
4914
 * @param cur  the node being changed
4915
 * @param uri  the new base URI
4916
 * @returns 0 on success, -1 on error.
4917
 */
4918
int
4919
0
xmlNodeSetBase(xmlNode *cur, const xmlChar* uri) {
4920
0
    xmlNsPtr ns;
4921
0
    xmlChar* fixed;
4922
4923
0
    if (cur == NULL)
4924
0
        return(-1);
4925
0
    switch(cur->type) {
4926
0
        case XML_ELEMENT_NODE:
4927
0
        case XML_ATTRIBUTE_NODE:
4928
0
      break;
4929
0
        case XML_DOCUMENT_NODE:
4930
0
        case XML_HTML_DOCUMENT_NODE: {
4931
0
      xmlDocPtr doc = (xmlDocPtr) cur;
4932
4933
0
      if (doc->URL != NULL)
4934
0
    xmlFree(doc->URL);
4935
0
      if (uri == NULL) {
4936
0
    doc->URL = NULL;
4937
0
            } else {
4938
0
    doc->URL = xmlPathToURI(uri);
4939
0
                if (doc->URL == NULL)
4940
0
                    return(-1);
4941
0
            }
4942
0
      return(0);
4943
0
  }
4944
0
        default:
4945
0
      return(-1);
4946
0
    }
4947
4948
0
    xmlSearchNsByHrefSafe(cur, XML_XML_NAMESPACE, &ns);
4949
0
    if (ns == NULL)
4950
0
  return(-1);
4951
0
    fixed = xmlPathToURI(uri);
4952
0
    if (fixed == NULL)
4953
0
        return(-1);
4954
0
    if (xmlSetNsProp(cur, ns, BAD_CAST "base", fixed) == NULL) {
4955
0
        xmlFree(fixed);
4956
0
        return(-1);
4957
0
    }
4958
0
    xmlFree(fixed);
4959
4960
0
    return(0);
4961
0
}
4962
4963
/**
4964
 * Searches for the base URI. The code should work on both XML
4965
 * and HTML document even if base mechanisms are completely different.
4966
 * It returns the base as defined in RFC 2396 sections "5.1.1. Base
4967
 * URI within Document Content" and "5.1.2. Base URI from the
4968
 * Encapsulating Entity". However it does not return the document base
4969
 * (5.1.3), use `doc->URL` in this case.
4970
 *
4971
 * @since 2.13.0
4972
 *
4973
 * @param doc  the document the node pertains to
4974
 * @param cur  the node being checked
4975
 * @param baseOut  pointer to base
4976
 * @returns 0 in case of success, 1 if a URI or argument is invalid, -1 if a
4977
 * memory allocation failed.
4978
 */
4979
int
4980
0
xmlNodeGetBaseSafe(const xmlDoc *doc, const xmlNode *cur, xmlChar **baseOut) {
4981
0
    xmlChar *ret = NULL;
4982
0
    xmlChar *base, *newbase;
4983
0
    int res;
4984
4985
0
    if (baseOut == NULL)
4986
0
        return(1);
4987
0
    *baseOut = NULL;
4988
0
    if ((cur == NULL) && (doc == NULL))
4989
0
        return(1);
4990
0
    if ((cur != NULL) && (cur->type == XML_NAMESPACE_DECL))
4991
0
        return(1);
4992
0
    if (doc == NULL)
4993
0
        doc = cur->doc;
4994
4995
0
    if ((doc != NULL) && (doc->type == XML_HTML_DOCUMENT_NODE)) {
4996
0
        cur = doc->children;
4997
0
  while ((cur != NULL) && (cur->name != NULL)) {
4998
0
      if (cur->type != XML_ELEMENT_NODE) {
4999
0
          cur = cur->next;
5000
0
    continue;
5001
0
      }
5002
0
      if (!xmlStrcasecmp(cur->name, BAD_CAST "html")) {
5003
0
          cur = cur->children;
5004
0
    continue;
5005
0
      }
5006
0
      if (!xmlStrcasecmp(cur->name, BAD_CAST "head")) {
5007
0
          cur = cur->children;
5008
0
    continue;
5009
0
      }
5010
0
      if (!xmlStrcasecmp(cur->name, BAD_CAST "base")) {
5011
0
                if (xmlNodeGetAttrValue(cur, BAD_CAST "href", NULL, &ret) < 0)
5012
0
                    return(-1);
5013
0
                if (ret == NULL)
5014
0
                    return(1);
5015
0
                goto found;
5016
0
      }
5017
0
      cur = cur->next;
5018
0
  }
5019
0
  return(0);
5020
0
    }
5021
5022
0
    while (cur != NULL) {
5023
0
  if (cur->type == XML_ENTITY_DECL) {
5024
0
      xmlEntityPtr ent = (xmlEntityPtr) cur;
5025
5026
0
            if (ent->URI == NULL)
5027
0
                break;
5028
0
            xmlFree(ret);
5029
0
      ret = xmlStrdup(ent->URI);
5030
0
            if (ret == NULL)
5031
0
                return(-1);
5032
0
            goto found;
5033
0
  }
5034
0
  if (cur->type == XML_ELEMENT_NODE) {
5035
0
      if (xmlNodeGetAttrValue(cur, BAD_CAST "base", XML_XML_NAMESPACE,
5036
0
                                    &base) < 0) {
5037
0
                xmlFree(ret);
5038
0
                return(-1);
5039
0
            }
5040
0
      if (base != NULL) {
5041
0
    if (ret != NULL) {
5042
0
        res = xmlBuildURISafe(ret, base, &newbase);
5043
0
                    xmlFree(ret);
5044
0
                    xmlFree(base);
5045
0
                    if (res != 0)
5046
0
                        return(res);
5047
0
        ret = newbase;
5048
0
    } else {
5049
0
        ret = base;
5050
0
    }
5051
0
    if ((!xmlStrncmp(ret, BAD_CAST "http://", 7)) ||
5052
0
        (!xmlStrncmp(ret, BAD_CAST "ftp://", 6)) ||
5053
0
        (!xmlStrncmp(ret, BAD_CAST "urn:", 4)))
5054
0
                    goto found;
5055
0
      }
5056
0
  }
5057
0
  cur = cur->parent;
5058
0
    }
5059
5060
0
    if ((doc != NULL) && (doc->URL != NULL)) {
5061
0
  if (ret == NULL) {
5062
0
      ret = xmlStrdup(doc->URL);
5063
0
            if (ret == NULL)
5064
0
                return(-1);
5065
0
        } else {
5066
0
            res = xmlBuildURISafe(ret, doc->URL, &newbase);
5067
0
            xmlFree(ret);
5068
0
            if (res != 0)
5069
0
                return(res);
5070
0
            ret = newbase;
5071
0
        }
5072
0
    }
5073
5074
0
found:
5075
0
    *baseOut = ret;
5076
0
    return(0);
5077
0
}
5078
5079
/**
5080
 * See #xmlNodeGetBaseSafe. This function doesn't allow to distinguish
5081
 * memory allocation failures from a non-existing base.
5082
 *
5083
 * @param doc  the document the node pertains to
5084
 * @param cur  the node being checked
5085
 * @returns a pointer to the base URL, or NULL if not found
5086
 *     It's up to the caller to free the memory with #xmlFree.
5087
 */
5088
xmlChar *
5089
0
xmlNodeGetBase(const xmlDoc *doc, const xmlNode *cur) {
5090
0
    xmlChar *base;
5091
5092
0
    xmlNodeGetBaseSafe(doc, cur, &base);
5093
0
    return(base);
5094
0
}
5095
5096
/**
5097
 * Append the string value of a node to `buffer`. For text nodes,
5098
 * the string value is the text content. Otherwise, the string value
5099
 * is the concatenation of the string values of the node's
5100
 * descendants.
5101
 *
5102
 * Entity references are substituted.
5103
 *
5104
 * @param buffer  a buffer
5105
 * @param cur  the node being read
5106
 * @returns 0 in case of success and -1 in case of error.
5107
 */
5108
int
5109
xmlNodeBufGetContent(xmlBuffer *buffer, const xmlNode *cur)
5110
0
{
5111
0
    xmlBufPtr buf;
5112
0
    int ret1, ret2;
5113
5114
0
    if ((cur == NULL) || (buffer == NULL)) return(-1);
5115
0
    buf = xmlBufFromBuffer(buffer);
5116
0
    ret1 = xmlBufGetNodeContent(buf, cur);
5117
0
    ret2 = xmlBufBackToBuffer(buf, buffer);
5118
0
    if ((ret1 < 0) || (ret2 < 0))
5119
0
        return(-1);
5120
0
    return(0);
5121
0
}
5122
5123
static void
5124
0
xmlBufGetEntityRefContent(xmlBufPtr buf, const xmlNode *ref) {
5125
0
    xmlEntityPtr ent;
5126
5127
0
    if (ref->children != NULL) {
5128
0
        ent = (xmlEntityPtr) ref->children;
5129
0
    } else {
5130
        /* lookup entity declaration */
5131
0
        ent = xmlGetDocEntity(ref->doc, ref->name);
5132
0
        if (ent == NULL)
5133
0
            return;
5134
0
    }
5135
5136
    /*
5137
     * The parser should always expand predefined entities but it's
5138
     * possible to create references to predefined entities using
5139
     * the tree API.
5140
     */
5141
0
    if (ent->etype == XML_INTERNAL_PREDEFINED_ENTITY) {
5142
0
        xmlBufCat(buf, ent->content);
5143
0
        return;
5144
0
    }
5145
5146
0
    if (ent->flags & XML_ENT_EXPANDING)
5147
0
        return;
5148
5149
0
    ent->flags |= XML_ENT_EXPANDING;
5150
0
    xmlBufGetChildContent(buf, (xmlNodePtr) ent);
5151
0
    ent->flags &= ~XML_ENT_EXPANDING;
5152
0
}
5153
5154
static void
5155
0
xmlBufGetChildContent(xmlBufPtr buf, const xmlNode *tree) {
5156
0
    const xmlNode *cur = tree->children;
5157
5158
0
    while (cur != NULL) {
5159
0
        switch (cur->type) {
5160
0
            case XML_TEXT_NODE:
5161
0
            case XML_CDATA_SECTION_NODE:
5162
0
                xmlBufCat(buf, cur->content);
5163
0
                break;
5164
5165
0
            case XML_ENTITY_REF_NODE:
5166
0
                xmlBufGetEntityRefContent(buf, cur);
5167
0
                break;
5168
5169
0
            default:
5170
0
                if (cur->children != NULL) {
5171
0
                    cur = cur->children;
5172
0
                    continue;
5173
0
                }
5174
0
                break;
5175
0
        }
5176
5177
0
        while (cur->next == NULL) {
5178
0
            cur = cur->parent;
5179
0
            if (cur == tree)
5180
0
                return;
5181
0
        }
5182
0
        cur = cur->next;
5183
0
    }
5184
0
}
5185
5186
/**
5187
 * Append the string value of a node to `buf`. For text nodes,
5188
 * the string value is the text content. Otherwise, the string value
5189
 * is the concatenation of the string values of the node's
5190
 * descendants.
5191
 *
5192
 * Entity references are substituted.
5193
 *
5194
 * @param buf  a buffer xmlBuf
5195
 * @param cur  the node being read
5196
 * @returns 0 in case of success and -1 in case of error.
5197
 */
5198
int
5199
xmlBufGetNodeContent(xmlBuf *buf, const xmlNode *cur)
5200
0
{
5201
0
    if ((cur == NULL) || (buf == NULL))
5202
0
        return(-1);
5203
5204
0
    switch (cur->type) {
5205
0
        case XML_DOCUMENT_NODE:
5206
0
        case XML_HTML_DOCUMENT_NODE:
5207
0
        case XML_DOCUMENT_FRAG_NODE:
5208
0
        case XML_ELEMENT_NODE:
5209
0
        case XML_ATTRIBUTE_NODE:
5210
0
        case XML_ENTITY_DECL:
5211
0
            xmlBufGetChildContent(buf, cur);
5212
0
            break;
5213
5214
0
        case XML_CDATA_SECTION_NODE:
5215
0
        case XML_TEXT_NODE:
5216
0
        case XML_COMMENT_NODE:
5217
0
        case XML_PI_NODE:
5218
0
      xmlBufCat(buf, cur->content);
5219
0
            break;
5220
5221
0
        case XML_ENTITY_REF_NODE:
5222
0
            xmlBufGetEntityRefContent(buf, cur);
5223
0
            break;
5224
5225
0
        case XML_NAMESPACE_DECL:
5226
0
      xmlBufCat(buf, ((xmlNsPtr) cur)->href);
5227
0
      break;
5228
5229
0
        default:
5230
0
            break;
5231
0
    }
5232
5233
0
    return(0);
5234
0
}
5235
5236
/**
5237
 * Returns the string value of a node. For text nodes, the string
5238
 * value is the text content. Otherwise, the string value is the
5239
 * concatenation of the string values of the node's descendants.
5240
 *
5241
 * Entity references are substituted.
5242
 *
5243
 * It's up to the caller to free the result with #xmlFree.
5244
 *
5245
 * @param cur  the node being read
5246
 * @returns a new string or NULL if arguments are invalid or a
5247
 * memory allocation failed.
5248
 */
5249
xmlChar *
5250
xmlNodeGetContent(const xmlNode *cur)
5251
227
{
5252
227
    xmlBufPtr buf;
5253
227
    xmlChar *ret;
5254
5255
227
    if (cur == NULL)
5256
0
        return (NULL);
5257
5258
227
    switch (cur->type) {
5259
0
        case XML_DOCUMENT_NODE:
5260
0
        case XML_HTML_DOCUMENT_NODE:
5261
0
        case XML_ENTITY_REF_NODE:
5262
0
            break;
5263
5264
0
        case XML_DOCUMENT_FRAG_NODE:
5265
0
        case XML_ELEMENT_NODE:
5266
227
        case XML_ATTRIBUTE_NODE:
5267
227
        case XML_ENTITY_DECL: {
5268
227
            xmlNodePtr children = cur->children;
5269
5270
227
            if (children == NULL)
5271
0
                return(xmlStrdup(BAD_CAST ""));
5272
5273
            /* Optimization for single text children */
5274
227
            if (((children->type == XML_TEXT_NODE) ||
5275
227
                 (children->type == XML_CDATA_SECTION_NODE)) &&
5276
227
                (children->next == NULL)) {
5277
227
                if (children->content == NULL)
5278
0
                    return(xmlStrdup(BAD_CAST ""));
5279
227
                return(xmlStrdup(children->content));
5280
227
            }
5281
5282
0
            break;
5283
227
        }
5284
5285
0
        case XML_CDATA_SECTION_NODE:
5286
0
        case XML_TEXT_NODE:
5287
0
        case XML_COMMENT_NODE:
5288
0
        case XML_PI_NODE:
5289
0
            if (cur->content != NULL)
5290
0
                return(xmlStrdup(cur->content));
5291
0
            else
5292
0
                return(xmlStrdup(BAD_CAST ""));
5293
5294
0
        case XML_NAMESPACE_DECL:
5295
0
      return(xmlStrdup(((xmlNsPtr) cur)->href));
5296
5297
0
        default:
5298
0
            return(NULL);
5299
227
    }
5300
5301
0
    buf = xmlBufCreate(50);
5302
0
    if (buf == NULL)
5303
0
        return (NULL);
5304
0
    xmlBufGetNodeContent(buf, cur);
5305
0
    ret = xmlBufDetach(buf);
5306
0
    xmlBufFree(buf);
5307
5308
0
    return(ret);
5309
0
}
5310
5311
static int
5312
0
xmlNodeSetContentInternal(xmlNodePtr cur, const xmlChar *content, int len) {
5313
0
    if (cur == NULL) {
5314
0
  return(1);
5315
0
    }
5316
0
    switch (cur->type) {
5317
0
        case XML_DOCUMENT_FRAG_NODE:
5318
0
        case XML_ELEMENT_NODE:
5319
0
        case XML_ATTRIBUTE_NODE: {
5320
0
            size_t maxSize = len < 0 ? SIZE_MAX : (size_t) len;
5321
5322
            /*
5323
             * We shouldn't parse the content as attribute value here,
5324
             * but the API can't be changed.
5325
             */
5326
0
            if (xmlNodeParseAttValue(cur->doc, (xmlAttr *) cur,
5327
0
                                     content, maxSize, NULL) < 0)
5328
0
                return(-1);
5329
0
      break;
5330
0
        }
5331
5332
0
        case XML_TEXT_NODE:
5333
0
        case XML_CDATA_SECTION_NODE:
5334
0
        case XML_PI_NODE:
5335
0
        case XML_COMMENT_NODE: {
5336
0
            xmlChar *copy = NULL;
5337
5338
0
      if (content != NULL) {
5339
0
                if (len < 0)
5340
0
                    copy = xmlStrdup(content);
5341
0
                else
5342
0
        copy = xmlStrndup(content, len);
5343
0
                if (copy == NULL)
5344
0
                    return(-1);
5345
0
      }
5346
5347
0
            xmlTextSetContent(cur, copy);
5348
0
      break;
5349
0
        }
5350
5351
0
        default:
5352
0
            break;
5353
0
    }
5354
5355
0
    return(0);
5356
0
}
5357
5358
/**
5359
 * Replace the text content of a node.
5360
 *
5361
 * Sets the raw text content of text, CDATA, comment or PI nodes.
5362
 *
5363
 * For element and attribute nodes, removes all children and
5364
 * replaces them by parsing `content` which is expected to be a
5365
 * valid XML attribute value possibly containing character and
5366
 * entity references. Syntax errors and references to undeclared
5367
 * entities are ignored silently. Unfortunately, there isn't an
5368
 * API to pass raw content directly. An inefficient work-around
5369
 * is to escape the content with #xmlEncodeSpecialChars before
5370
 * passing it. A better trick is clearing the old content
5371
 * with `xmlNodeSetContent(node, NULL)` first and then calling
5372
 * `xmlNodeAddContent(node, content)`. Unlike this function,
5373
 * #xmlNodeAddContent accepts raw text.
5374
 *
5375
 * @param cur  the node being modified
5376
 * @param content  the new value of the content
5377
 * @returns 0 on success, 1 on error, -1 if a memory allocation failed.
5378
 */
5379
int
5380
0
xmlNodeSetContent(xmlNode *cur, const xmlChar *content) {
5381
0
    return(xmlNodeSetContentInternal(cur, content, -1));
5382
0
}
5383
5384
/**
5385
 * See #xmlNodeSetContent.
5386
 *
5387
 * @param cur  the node being modified
5388
 * @param content  the new value of the content
5389
 * @param len  the size of `content`
5390
 * @returns 0 on success, 1 on error, -1 if a memory allocation failed.
5391
 */
5392
int
5393
0
xmlNodeSetContentLen(xmlNode *cur, const xmlChar *content, int len) {
5394
0
    return(xmlNodeSetContentInternal(cur, content, len));
5395
0
}
5396
5397
/**
5398
 * Append the extra substring to the node content.
5399
 *
5400
 * NOTE: In contrast to #xmlNodeSetContentLen, `content` is supposed
5401
 * to be raw text, so unescaped XML special chars are allowed, entity
5402
 * references are not supported.
5403
 *
5404
 * This doesn't work on attributes before version 2.15.
5405
 *
5406
 * @param cur  the node being modified
5407
 * @param content  extra content
5408
 * @param len  the size of `content`
5409
 * @returns 0 on success, 1 on error, -1 if a memory allocation failed.
5410
 */
5411
int
5412
0
xmlNodeAddContentLen(xmlNode *cur, const xmlChar *content, int len) {
5413
0
    if (cur == NULL)
5414
0
  return(1);
5415
0
    if ((content == NULL) || (len <= 0))
5416
0
        return(0);
5417
5418
0
    switch (cur->type) {
5419
0
        case XML_DOCUMENT_FRAG_NODE:
5420
0
        case XML_ELEMENT_NODE:
5421
0
        case XML_ATTRIBUTE_NODE: {
5422
0
      xmlNodePtr newNode, tmp;
5423
5424
0
      newNode = xmlNewDocTextLen(cur->doc, content, len);
5425
0
      if (newNode == NULL)
5426
0
                return(-1);
5427
0
            tmp = xmlAddChild(cur, newNode);
5428
0
            if (tmp == NULL) {
5429
0
                xmlFreeNode(newNode);
5430
0
                return(-1);
5431
0
            }
5432
0
      break;
5433
0
  }
5434
0
      break;
5435
0
        case XML_TEXT_NODE:
5436
0
        case XML_CDATA_SECTION_NODE:
5437
0
        case XML_PI_NODE:
5438
0
        case XML_COMMENT_NODE:
5439
0
            return(xmlTextAddContent(cur, content, len));
5440
0
        default:
5441
0
            break;
5442
0
    }
5443
5444
0
    return(0);
5445
0
}
5446
5447
/**
5448
 * Append the extra substring to the node content.
5449
 *
5450
 * NOTE: In contrast to #xmlNodeSetContent, `content` is supposed
5451
 * to be raw text, so unescaped XML special chars are allowed, entity
5452
 * references are not supported.
5453
 *
5454
 * This doesn't work on attributes before version 2.15.
5455
 *
5456
 * @param cur  the node being modified
5457
 * @param content  extra content
5458
 * @returns 0 on success, 1 on error, -1 if a memory allocation failed.
5459
 */
5460
int
5461
0
xmlNodeAddContent(xmlNode *cur, const xmlChar *content) {
5462
0
    return(xmlNodeAddContentLen(cur, content, xmlStrlen(content)));
5463
0
}
5464
5465
/**
5466
 * Merge the second text node into the first. If `first` is NULL,
5467
 * `second` is returned. Otherwise, the second node is unlinked and
5468
 * freed.
5469
 *
5470
 * @param first  the first text node
5471
 * @param second  the second text node being merged
5472
 * @returns the first text node augmented or NULL in case of error.
5473
 */
5474
xmlNode *
5475
0
xmlTextMerge(xmlNode *first, xmlNode *second) {
5476
0
    if (first == NULL)
5477
0
        return(second);
5478
0
    if (second == NULL)
5479
0
        return(first);
5480
5481
0
    if ((first->type != XML_TEXT_NODE) ||
5482
0
        (second->type != XML_TEXT_NODE) ||
5483
0
        (first == second) ||
5484
0
        (first->name != second->name))
5485
0
  return(NULL);
5486
5487
0
    if (xmlTextAddContent(first, second->content, -1) < 0)
5488
0
        return(NULL);
5489
5490
0
    xmlUnlinkNodeInternal(second);
5491
0
    xmlFreeNode(second);
5492
0
    return(first);
5493
0
}
5494
5495
/**
5496
 * Find all in-scope namespaces of a node. `out` returns a NULL
5497
 * terminated array of namespace pointers that must be freed by
5498
 * the caller.
5499
 *
5500
 * @since 2.13.0
5501
 *
5502
 * @param doc  the document
5503
 * @param node  the current node
5504
 * @param out  the returned namespace array
5505
 * @returns 0 on success, 1 if no namespaces were found, -1 if a
5506
 * memory allocation failed.
5507
 */
5508
int
5509
xmlGetNsListSafe(const xmlDoc *doc ATTRIBUTE_UNUSED, const xmlNode *node,
5510
                 xmlNs ***out)
5511
0
{
5512
0
    xmlNsPtr cur;
5513
0
    xmlNsPtr *namespaces = NULL;
5514
0
    int nbns = 0;
5515
0
    int maxns = 0;
5516
0
    int i;
5517
5518
0
    if (out == NULL)
5519
0
        return(1);
5520
0
    *out = NULL;
5521
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL))
5522
0
        return(1);
5523
5524
0
    while (node != NULL) {
5525
0
        if (node->type == XML_ELEMENT_NODE) {
5526
0
            cur = node->nsDef;
5527
0
            while (cur != NULL) {
5528
0
                for (i = 0; i < nbns; i++) {
5529
0
                    if ((cur->prefix == namespaces[i]->prefix) ||
5530
0
                        (xmlStrEqual(cur->prefix, namespaces[i]->prefix)))
5531
0
                        break;
5532
0
                }
5533
0
                if (i >= nbns) {
5534
0
                    if (nbns >= maxns) {
5535
0
                        xmlNsPtr *tmp;
5536
0
                        int newSize;
5537
5538
0
                        newSize = xmlGrowCapacity(maxns, sizeof(tmp[0]),
5539
0
                                                  10, XML_MAX_ITEMS);
5540
0
                        if (newSize < 0) {
5541
0
                            xmlFree(namespaces);
5542
0
                            return(-1);
5543
0
                        }
5544
0
#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
5545
0
                        if (newSize < 2)
5546
0
                            newSize = 2;
5547
0
#endif
5548
0
                        tmp = xmlRealloc(namespaces,
5549
0
                                         (newSize + 1) * sizeof(tmp[0]));
5550
0
                        if (tmp == NULL) {
5551
0
                            xmlFree(namespaces);
5552
0
                            return(-1);
5553
0
                        }
5554
0
                        namespaces = tmp;
5555
0
                        maxns = newSize;
5556
0
                    }
5557
0
                    namespaces[nbns++] = cur;
5558
0
                    namespaces[nbns] = NULL;
5559
0
                }
5560
5561
0
                cur = cur->next;
5562
0
            }
5563
0
        }
5564
0
        node = node->parent;
5565
0
    }
5566
5567
0
    *out = namespaces;
5568
0
    return((namespaces == NULL) ? 1 : 0);
5569
0
}
5570
5571
/**
5572
 * Find all in-scope namespaces of a node.
5573
 *
5574
 * Use #xmlGetNsListSafe for better error reporting.
5575
 *
5576
 * @param doc  the document
5577
 * @param node  the current node
5578
 * @returns a NULL terminated array of namespace pointers that must
5579
 * be freed by the caller or NULL if no namespaces were found or
5580
 * a memory allocation failed.
5581
 */
5582
xmlNs **
5583
xmlGetNsList(const xmlDoc *doc, const xmlNode *node)
5584
0
{
5585
0
    xmlNsPtr *ret;
5586
5587
0
    xmlGetNsListSafe(doc, node, &ret);
5588
0
    return(ret);
5589
0
}
5590
5591
static xmlNsPtr
5592
414
xmlNewXmlNs(void) {
5593
414
    xmlNsPtr ns;
5594
5595
414
    ns = (xmlNsPtr) xmlMalloc(sizeof(xmlNs));
5596
414
    if (ns == NULL)
5597
0
        return(NULL);
5598
414
    memset(ns, 0, sizeof(xmlNs));
5599
414
    ns->type = XML_LOCAL_NAMESPACE;
5600
414
    ns->href = xmlStrdup(XML_XML_NAMESPACE);
5601
414
    if (ns->href == NULL) {
5602
0
        xmlFreeNs(ns);
5603
0
        return(NULL);
5604
0
    }
5605
414
    ns->prefix = xmlStrdup(BAD_CAST "xml");
5606
414
    if (ns->prefix == NULL) {
5607
0
        xmlFreeNs(ns);
5608
0
        return(NULL);
5609
0
    }
5610
5611
414
    return(ns);
5612
414
}
5613
5614
/*
5615
* xmlTreeEnsureXMLDecl:
5616
* @doc: the doc
5617
*
5618
* Ensures that there is an XML namespace declaration on the doc.
5619
*
5620
* Returns the XML ns-struct or NULL if a memory allocation failed.
5621
*/
5622
static xmlNsPtr
5623
xmlTreeEnsureXMLDecl(xmlDocPtr doc)
5624
7.29k
{
5625
7.29k
    xmlNsPtr ns;
5626
5627
7.29k
    ns = doc->oldNs;
5628
7.29k
    if (ns != NULL)
5629
6.88k
  return (ns);
5630
5631
414
    ns = xmlNewXmlNs();
5632
414
    doc->oldNs = ns;
5633
5634
414
    return(ns);
5635
7.29k
}
5636
5637
/**
5638
 * Search for a namespace with `prefix` in scope of `node`.
5639
 *
5640
 * @param node  a node
5641
 * @param prefix  a namespace prefix
5642
 * @param out  pointer to resulting namespace
5643
 * @returns 0 on success, -1 if a memory allocation failed, 1 on
5644
 * other errors.
5645
 */
5646
int
5647
xmlSearchNsSafe(xmlNode *node, const xmlChar *prefix,
5648
9.01k
                xmlNs **out) {
5649
9.01k
    xmlNsPtr cur;
5650
9.01k
    xmlDocPtr doc;
5651
9.01k
    xmlNodePtr orig = node;
5652
9.01k
    xmlNodePtr parent;
5653
5654
9.01k
    if (out == NULL)
5655
0
        return(1);
5656
9.01k
    *out = NULL;
5657
9.01k
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL))
5658
0
        return(1);
5659
5660
9.01k
    doc = node->doc;
5661
5662
9.01k
    if ((doc != NULL) && (IS_STR_XML(prefix))) {
5663
7.29k
        cur = xmlTreeEnsureXMLDecl(doc);
5664
7.29k
        if (cur == NULL)
5665
0
            return(-1);
5666
7.29k
        *out = cur;
5667
7.29k
        return(0);
5668
7.29k
    }
5669
5670
1.71k
    while (node->type != XML_ELEMENT_NODE) {
5671
0
        node = node->parent;
5672
0
        if (node == NULL)
5673
0
            return(0);
5674
0
    }
5675
5676
1.71k
    parent = node;
5677
5678
2.99k
    while ((node != NULL) && (node->type == XML_ELEMENT_NODE)) {
5679
2.99k
        cur = node->nsDef;
5680
3.77k
        while (cur != NULL) {
5681
1.86k
            if ((xmlStrEqual(cur->prefix, prefix)) &&
5682
1.86k
                (cur->href != NULL)) {
5683
1.07k
                *out = cur;
5684
1.07k
                return(0);
5685
1.07k
            }
5686
786
            cur = cur->next;
5687
786
        }
5688
1.91k
        if (orig != node) {
5689
1.03k
            cur = node->ns;
5690
1.03k
            if ((cur != NULL) &&
5691
1.03k
                (xmlStrEqual(cur->prefix, prefix)) &&
5692
1.03k
                (cur->href != NULL)) {
5693
644
                *out = cur;
5694
644
                return(0);
5695
644
            }
5696
1.03k
        }
5697
5698
1.27k
  node = node->parent;
5699
1.27k
    }
5700
5701
    /*
5702
     * The XML-1.0 namespace is normally held on the document
5703
     * element. In this case exceptionally create it on the
5704
     * node element.
5705
     */
5706
0
    if ((doc == NULL) && (IS_STR_XML(prefix))) {
5707
0
        cur = xmlNewXmlNs();
5708
0
        if (cur == NULL)
5709
0
            return(-1);
5710
0
        cur->next = parent->nsDef;
5711
0
        parent->nsDef = cur;
5712
0
        *out = cur;
5713
0
    }
5714
5715
0
    return(0);
5716
0
}
5717
5718
/**
5719
 * Search for a namespace with `prefix` in scope of `node`.
5720
 *
5721
 * Recurse on the parents until it finds the defined namespace
5722
 * or return NULL otherwise.
5723
 *
5724
 * If `nameSpace` is NULL, the default namespace is looked up.
5725
 *
5726
 * Namespace search doesn't cross entity boundaries.
5727
 *
5728
 * @param doc  the document
5729
 * @param node  the current node
5730
 * @param nameSpace  the namespace prefix
5731
 * @returns the namespace pointer or NULL if no namespace was found or
5732
 * a memory allocation failed. Allocations can only fail if the `xml`
5733
 * namespace is queried.
5734
 */
5735
xmlNs *
5736
xmlSearchNs(xmlDoc *doc ATTRIBUTE_UNUSED, xmlNode *node,
5737
0
            const xmlChar *nameSpace) {
5738
0
    xmlNsPtr cur;
5739
5740
0
    xmlSearchNsSafe(node, nameSpace, &cur);
5741
0
    return(cur);
5742
0
}
5743
5744
/**
5745
 * Verify that the given namespace held on `ancestor` is still in scope
5746
 * on node.
5747
 *
5748
 * @param doc  the document
5749
 * @param node  the current node
5750
 * @param ancestor  the ancestor carrying the namespace
5751
 * @param prefix  the namespace prefix
5752
 * @returns 1 if true, 0 if false and -1 in case of error.
5753
 */
5754
static int
5755
xmlNsInScope(xmlDocPtr doc ATTRIBUTE_UNUSED, xmlNodePtr node,
5756
             xmlNodePtr ancestor, const xmlChar * prefix)
5757
0
{
5758
0
    xmlNsPtr tst;
5759
5760
0
    while ((node != NULL) && (node != ancestor)) {
5761
0
        if ((node->type == XML_ENTITY_REF_NODE) ||
5762
0
            (node->type == XML_ENTITY_DECL))
5763
0
            return (-1);
5764
0
        if (node->type == XML_ELEMENT_NODE) {
5765
0
            tst = node->nsDef;
5766
0
            while (tst != NULL) {
5767
0
                if ((tst->prefix == NULL)
5768
0
                    && (prefix == NULL))
5769
0
                    return (0);
5770
0
                if ((tst->prefix != NULL)
5771
0
                    && (prefix != NULL)
5772
0
                    && (xmlStrEqual(tst->prefix, prefix)))
5773
0
                    return (0);
5774
0
                tst = tst->next;
5775
0
            }
5776
0
        }
5777
0
        node = node->parent;
5778
0
    }
5779
0
    if (node != ancestor)
5780
0
        return (-1);
5781
0
    return (1);
5782
0
}
5783
5784
/**
5785
 * Search for a namespace matching `URI` in scope of `node`.
5786
 *
5787
 * @param node  a node
5788
 * @param href  a namespace URI
5789
 * @param out  pointer to resulting namespace
5790
 * @returns 0 on success, -1 if a memory allocation failed, 1 on
5791
 * other errors.
5792
 */
5793
int
5794
xmlSearchNsByHrefSafe(xmlNode *node, const xmlChar *href,
5795
0
                      xmlNs **out) {
5796
0
    xmlNsPtr cur;
5797
0
    xmlDocPtr doc;
5798
0
    xmlNodePtr orig = node;
5799
0
    xmlNodePtr parent;
5800
0
    int is_attr;
5801
5802
0
    if (out == NULL)
5803
0
        return(1);
5804
0
    *out = NULL;
5805
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL))
5806
0
        return(1);
5807
5808
0
    doc = node->doc;
5809
5810
0
    if ((doc != NULL) && (xmlStrEqual(href, XML_XML_NAMESPACE))) {
5811
0
        cur = xmlTreeEnsureXMLDecl(doc);
5812
0
        if (cur == NULL)
5813
0
            return(-1);
5814
0
        *out = cur;
5815
0
        return(0);
5816
0
    }
5817
5818
0
    is_attr = (node->type == XML_ATTRIBUTE_NODE);
5819
5820
0
    while (node->type != XML_ELEMENT_NODE) {
5821
0
        node = node->parent;
5822
0
        if (node == NULL)
5823
0
            return(0);
5824
0
    }
5825
5826
0
    parent = node;
5827
5828
0
    while ((node != NULL) && (node->type == XML_ELEMENT_NODE)) {
5829
0
        cur = node->nsDef;
5830
0
        while (cur != NULL) {
5831
0
            if (xmlStrEqual(cur->href, href)) {
5832
0
                if (((!is_attr) || (cur->prefix != NULL)) &&
5833
0
                    (xmlNsInScope(doc, orig, node, cur->prefix) == 1)) {
5834
0
                    *out = cur;
5835
0
                    return(0);
5836
0
                }
5837
0
            }
5838
0
            cur = cur->next;
5839
0
        }
5840
0
        if (orig != node) {
5841
0
            cur = node->ns;
5842
0
            if (cur != NULL) {
5843
0
                if (xmlStrEqual(cur->href, href)) {
5844
0
                    if (((!is_attr) || (cur->prefix != NULL)) &&
5845
0
                        (xmlNsInScope(doc, orig, node,
5846
0
                                      cur->prefix) == 1)) {
5847
0
                        *out = cur;
5848
0
                        return(0);
5849
0
                    }
5850
0
                }
5851
0
            }
5852
0
        }
5853
5854
0
        node = node->parent;
5855
0
    }
5856
5857
    /*
5858
     * The XML-1.0 namespace is normally held on the document
5859
     * element. In this case exceptionally create it on the
5860
     * node element.
5861
     */
5862
0
    if ((doc == NULL) && (xmlStrEqual(href, XML_XML_NAMESPACE))) {
5863
0
        cur = xmlNewXmlNs();
5864
0
        if (cur == NULL)
5865
0
            return(-1);
5866
0
        cur->next = parent->nsDef;
5867
0
        parent->nsDef = cur;
5868
0
        *out = cur;
5869
0
    }
5870
5871
0
    return(0);
5872
0
}
5873
5874
/**
5875
 * Search for a namespace matching `URI` in scope of `node`.
5876
 *
5877
 * @param doc  the document
5878
 * @param node  the current node
5879
 * @param href  the namespace value
5880
 * @returns the namespace pointer or NULL if no namespace was found or
5881
 * a memory allocation failed. Allocations can only fail if the `xml`
5882
 * namespace is queried.
5883
 */
5884
xmlNs *
5885
xmlSearchNsByHref(xmlDoc *doc ATTRIBUTE_UNUSED, xmlNode *node,
5886
0
                  const xmlChar * href) {
5887
0
    xmlNsPtr cur;
5888
5889
0
    xmlSearchNsByHrefSafe(node, href, &cur);
5890
0
    return(cur);
5891
0
}
5892
5893
/**
5894
 * Fix up namespace declarations.
5895
 *
5896
 * This function tries to locate a namespace definition in a tree
5897
 * ancestors, or create a new namespace definition node similar to
5898
 * `ns` trying to reuse the same prefix. However if the given prefix is
5899
 * null (default namespace) or reused within the subtree defined by
5900
 * `tree` or on one of its ancestors then a new prefix is generated.
5901
 *
5902
 * @param tree  a node expected to hold the new namespace
5903
 * @param ns  the original namespace
5904
 * @returns the (new) namespace definition or NULL in case of error
5905
 */
5906
static xmlNsPtr
5907
0
xmlNewReconciledNs(xmlNodePtr tree, xmlNsPtr ns) {
5908
0
    xmlNsPtr def;
5909
0
    xmlChar prefix[50];
5910
0
    int counter = 1;
5911
0
    int res;
5912
5913
0
    if ((tree == NULL) || (tree->type != XML_ELEMENT_NODE)) {
5914
0
  return(NULL);
5915
0
    }
5916
0
    if ((ns == NULL) || (ns->type != XML_NAMESPACE_DECL)) {
5917
0
  return(NULL);
5918
0
    }
5919
    /*
5920
     * Search an existing namespace definition inherited.
5921
     */
5922
0
    res = xmlSearchNsByHrefSafe(tree, ns->href, &def);
5923
0
    if (res < 0)
5924
0
        return(NULL);
5925
0
    if (def != NULL)
5926
0
        return(def);
5927
5928
    /*
5929
     * Find a close prefix which is not already in use.
5930
     * Let's strip namespace prefixes longer than 20 chars !
5931
     */
5932
0
    if (ns->prefix == NULL)
5933
0
  snprintf((char *) prefix, sizeof(prefix), "default");
5934
0
    else
5935
0
  snprintf((char *) prefix, sizeof(prefix), "%.20s", (char *)ns->prefix);
5936
5937
0
    res = xmlSearchNsSafe(tree, prefix, &def);
5938
0
    if (res < 0)
5939
0
        return(NULL);
5940
0
    while (def != NULL) {
5941
0
        if (counter > 1000) return(NULL);
5942
0
  if (ns->prefix == NULL)
5943
0
      snprintf((char *) prefix, sizeof(prefix), "default%d", counter++);
5944
0
  else
5945
0
      snprintf((char *) prefix, sizeof(prefix), "%.20s%d",
5946
0
    (char *)ns->prefix, counter++);
5947
0
  res = xmlSearchNsSafe(tree, prefix, &def);
5948
0
        if (res < 0)
5949
0
            return(NULL);
5950
0
    }
5951
5952
    /*
5953
     * OK, now we are ready to create a new one.
5954
     */
5955
0
    def = xmlNewNs(tree, ns->href, prefix);
5956
0
    return(def);
5957
0
}
5958
5959
typedef struct {
5960
    xmlNsPtr oldNs;
5961
    xmlNsPtr newNs;
5962
} xmlNsCache;
5963
5964
static int
5965
0
xmlGrowNsCache(xmlNsCache **cache, int *capacity) {
5966
0
    xmlNsCache *tmp;
5967
0
    int newSize;
5968
5969
0
    newSize = xmlGrowCapacity(*capacity, sizeof(tmp[0]),
5970
0
                              10, XML_MAX_ITEMS);
5971
0
    if (newSize < 0)
5972
0
        return(-1);
5973
0
    tmp = xmlRealloc(*cache, newSize * sizeof(tmp[0]));
5974
0
    if (tmp == NULL)
5975
0
        return(-1);
5976
0
    *cache = tmp;
5977
0
    *capacity = newSize;
5978
5979
0
    return(0);
5980
0
}
5981
5982
/**
5983
 * This function checks that all the namespaces declared within the given
5984
 * tree are properly declared. This is needed for example after copy or cut
5985
 * and then paste operations. The subtree may still hold pointers to
5986
 * namespace declarations outside the subtree or invalid/masked. As much
5987
 * as possible the function tries to reuse the existing namespaces found in
5988
 * the new environment. If not possible the new namespaces are redeclared
5989
 * on `tree` at the top of the given subtree.
5990
 *
5991
 * @param doc  the document
5992
 * @param tree  a node defining the subtree to reconciliate
5993
 * @returns 0 on success or -1 in case of error.
5994
 */
5995
int
5996
0
xmlReconciliateNs(xmlDoc *doc, xmlNode *tree) {
5997
0
    xmlNsCache *cache = NULL;
5998
0
    int sizeCache = 0;
5999
0
    int nbCache = 0;
6000
6001
0
    xmlNsPtr n;
6002
0
    xmlNodePtr node = tree;
6003
0
    xmlAttrPtr attr;
6004
0
    int ret = 0, i;
6005
6006
0
    if ((node == NULL) || (node->type != XML_ELEMENT_NODE)) return(-1);
6007
0
    if (node->doc != doc) return(-1);
6008
0
    while (node != NULL) {
6009
        /*
6010
   * Reconciliate the node namespace
6011
   */
6012
0
  if (node->ns != NULL) {
6013
0
      for (i = 0; i < nbCache; i++) {
6014
0
          if (cache[i].oldNs == node->ns) {
6015
0
        node->ns = cache[i].newNs;
6016
0
        break;
6017
0
    }
6018
0
      }
6019
0
      if (i == nbCache) {
6020
          /*
6021
     * OK we need to recreate a new namespace definition
6022
     */
6023
0
    n = xmlNewReconciledNs(tree, node->ns);
6024
0
    if (n == NULL) {
6025
0
                    ret = -1;
6026
0
                } else {
6027
        /*
6028
         * check if we need to grow the cache buffers.
6029
         */
6030
0
        if ((sizeCache <= nbCache) &&
6031
0
                        (xmlGrowNsCache(&cache, &sizeCache) < 0)) {
6032
0
                        ret = -1;
6033
0
        } else {
6034
0
                        cache[nbCache].newNs = n;
6035
0
                        cache[nbCache++].oldNs = node->ns;
6036
0
                    }
6037
0
                }
6038
0
    node->ns = n;
6039
0
      }
6040
0
  }
6041
  /*
6042
   * now check for namespace held by attributes on the node.
6043
   */
6044
0
  if (node->type == XML_ELEMENT_NODE) {
6045
0
      attr = node->properties;
6046
0
      while (attr != NULL) {
6047
0
    if (attr->ns != NULL) {
6048
0
        for (i = 0; i < nbCache; i++) {
6049
0
      if (cache[i].oldNs == attr->ns) {
6050
0
          attr->ns = cache[i].newNs;
6051
0
          break;
6052
0
      }
6053
0
        }
6054
0
        if (i == nbCache) {
6055
      /*
6056
       * OK we need to recreate a new namespace definition
6057
       */
6058
0
      n = xmlNewReconciledNs(tree, attr->ns);
6059
0
      if (n == NULL) {
6060
0
                            ret = -1;
6061
0
                        } else {
6062
          /*
6063
           * check if we need to grow the cache buffers.
6064
           */
6065
0
                            if ((sizeCache <= nbCache) &&
6066
0
                                (xmlGrowNsCache(&cache, &sizeCache) < 0)) {
6067
0
                                ret = -1;
6068
0
                            } else {
6069
0
                                cache[nbCache].newNs = n;
6070
0
                                cache[nbCache++].oldNs = attr->ns;
6071
0
          }
6072
0
      }
6073
0
      attr->ns = n;
6074
0
        }
6075
0
    }
6076
0
    attr = attr->next;
6077
0
      }
6078
0
  }
6079
6080
  /*
6081
   * Browse the full subtree, deep first
6082
   */
6083
0
        if ((node->children != NULL) && (node->type != XML_ENTITY_REF_NODE)) {
6084
      /* deep first */
6085
0
      node = node->children;
6086
0
  } else if ((node != tree) && (node->next != NULL)) {
6087
      /* then siblings */
6088
0
      node = node->next;
6089
0
  } else if (node != tree) {
6090
      /* go up to parents->next if needed */
6091
0
      while (node != tree) {
6092
0
          if (node->parent != NULL)
6093
0
        node = node->parent;
6094
0
    if ((node != tree) && (node->next != NULL)) {
6095
0
        node = node->next;
6096
0
        break;
6097
0
    }
6098
0
    if (node->parent == NULL) {
6099
0
        node = NULL;
6100
0
        break;
6101
0
    }
6102
0
      }
6103
      /* exit condition */
6104
0
      if (node == tree)
6105
0
          node = NULL;
6106
0
  } else
6107
0
      break;
6108
0
    }
6109
0
    if (cache != NULL)
6110
0
  xmlFree(cache);
6111
0
    return(ret);
6112
0
}
6113
6114
static xmlAttrPtr
6115
xmlGetPropNodeInternal(const xmlNode *node, const xmlChar *name,
6116
           const xmlChar *nsName, int useDTD)
6117
0
{
6118
0
    xmlAttrPtr prop;
6119
6120
    /* Avoid unused variable warning if features are disabled. */
6121
0
    (void) useDTD;
6122
6123
0
    if ((node == NULL) || (node->type != XML_ELEMENT_NODE) || (name == NULL))
6124
0
  return(NULL);
6125
6126
0
    if (node->properties != NULL) {
6127
0
  prop = node->properties;
6128
0
  if (nsName == NULL) {
6129
      /*
6130
      * We want the attr to be in no namespace.
6131
      */
6132
0
      do {
6133
0
    if ((prop->ns == NULL) && xmlStrEqual(prop->name, name)) {
6134
0
        return(prop);
6135
0
    }
6136
0
    prop = prop->next;
6137
0
      } while (prop != NULL);
6138
0
  } else {
6139
      /*
6140
      * We want the attr to be in the specified namespace.
6141
      */
6142
0
      do {
6143
0
    if ((prop->ns != NULL) && xmlStrEqual(prop->name, name) &&
6144
0
        ((prop->ns->href == nsName) ||
6145
0
         xmlStrEqual(prop->ns->href, nsName)))
6146
0
    {
6147
0
        return(prop);
6148
0
    }
6149
0
    prop = prop->next;
6150
0
      } while (prop != NULL);
6151
0
  }
6152
0
    }
6153
6154
0
    if (! useDTD)
6155
0
  return(NULL);
6156
    /*
6157
     * Check if there is a default/fixed attribute declaration in
6158
     * the internal or external subset.
6159
     */
6160
0
    if ((node->doc != NULL) && (node->doc->intSubset != NULL)) {
6161
0
  xmlDocPtr doc = node->doc;
6162
0
  xmlAttributePtr attrDecl = NULL;
6163
0
  xmlChar *elemQName, *tmpstr = NULL;
6164
6165
  /*
6166
  * We need the QName of the element for the DTD-lookup.
6167
  */
6168
0
  if ((node->ns != NULL) && (node->ns->prefix != NULL)) {
6169
0
      tmpstr = xmlStrdup(node->ns->prefix);
6170
0
      if (tmpstr == NULL)
6171
0
    return(NULL);
6172
0
      tmpstr = xmlStrcat(tmpstr, BAD_CAST ":");
6173
0
      if (tmpstr == NULL)
6174
0
    return(NULL);
6175
0
      tmpstr = xmlStrcat(tmpstr, node->name);
6176
0
      if (tmpstr == NULL)
6177
0
    return(NULL);
6178
0
      elemQName = tmpstr;
6179
0
  } else
6180
0
      elemQName = (xmlChar *) node->name;
6181
0
  if (nsName == NULL) {
6182
      /*
6183
      * The common and nice case: Attr in no namespace.
6184
      */
6185
0
      attrDecl = xmlGetDtdQAttrDesc(doc->intSubset,
6186
0
    elemQName, name, NULL);
6187
0
      if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
6188
0
    attrDecl = xmlGetDtdQAttrDesc(doc->extSubset,
6189
0
        elemQName, name, NULL);
6190
0
      }
6191
0
        } else if (xmlStrEqual(nsName, XML_XML_NAMESPACE)) {
6192
      /*
6193
      * The XML namespace must be bound to prefix 'xml'.
6194
      */
6195
0
      attrDecl = xmlGetDtdQAttrDesc(doc->intSubset,
6196
0
    elemQName, name, BAD_CAST "xml");
6197
0
      if ((attrDecl == NULL) && (doc->extSubset != NULL)) {
6198
0
    attrDecl = xmlGetDtdQAttrDesc(doc->extSubset,
6199
0
        elemQName, name, BAD_CAST "xml");
6200
0
      }
6201
0
  } else {
6202
0
      xmlNsPtr *nsList, *cur;
6203
6204
      /*
6205
      * The ugly case: Search using the prefixes of in-scope
6206
      * ns-decls corresponding to @nsName.
6207
      */
6208
0
      nsList = xmlGetNsList(node->doc, node);
6209
0
      if (nsList == NULL) {
6210
0
    if (tmpstr != NULL)
6211
0
        xmlFree(tmpstr);
6212
0
    return(NULL);
6213
0
      }
6214
0
      cur = nsList;
6215
0
      while (*cur != NULL) {
6216
0
    if (xmlStrEqual((*cur)->href, nsName)) {
6217
0
        attrDecl = xmlGetDtdQAttrDesc(doc->intSubset, elemQName,
6218
0
      name, (*cur)->prefix);
6219
0
        if (attrDecl)
6220
0
      break;
6221
0
        if (doc->extSubset != NULL) {
6222
0
      attrDecl = xmlGetDtdQAttrDesc(doc->extSubset, elemQName,
6223
0
          name, (*cur)->prefix);
6224
0
      if (attrDecl)
6225
0
          break;
6226
0
        }
6227
0
    }
6228
0
    cur++;
6229
0
      }
6230
0
      xmlFree(nsList);
6231
0
  }
6232
0
  if (tmpstr != NULL)
6233
0
      xmlFree(tmpstr);
6234
  /*
6235
  * Only default/fixed attrs are relevant.
6236
  */
6237
0
  if ((attrDecl != NULL) && (attrDecl->defaultValue != NULL))
6238
0
      return((xmlAttrPtr) attrDecl);
6239
0
    }
6240
6241
0
    return(NULL);
6242
0
}
6243
6244
static xmlChar*
6245
xmlGetPropNodeValueInternal(const xmlAttr *prop)
6246
0
{
6247
0
    if (prop == NULL)
6248
0
  return(NULL);
6249
0
    if (prop->type == XML_ATTRIBUTE_NODE) {
6250
0
  return(xmlNodeGetContent((xmlNodePtr) prop));
6251
0
    } else if (prop->type == XML_ATTRIBUTE_DECL) {
6252
0
  return(xmlStrdup(((xmlAttributePtr)prop)->defaultValue));
6253
0
    }
6254
0
    return(NULL);
6255
0
}
6256
6257
/**
6258
 * Search for an attribute of an element.
6259
 *
6260
 * This function also looks in DTD attribute declaration for \#FIXED or
6261
 * default declaration values.
6262
 *
6263
 * @param node  the element
6264
 * @param name  the attribute name
6265
 * @returns the attribute or the attribute declaration or NULL if
6266
 * neither was found. Also returns NULL if a memory allocation failed,
6267
 * making this function unreliable.
6268
 */
6269
xmlAttr *
6270
0
xmlHasProp(const xmlNode *node, const xmlChar *name) {
6271
0
    xmlAttrPtr prop;
6272
0
    xmlDocPtr doc;
6273
6274
0
    if ((node == NULL) || (node->type != XML_ELEMENT_NODE) || (name == NULL))
6275
0
        return(NULL);
6276
    /*
6277
     * Check on the properties attached to the node
6278
     */
6279
0
    prop = node->properties;
6280
0
    while (prop != NULL) {
6281
0
        if (xmlStrEqual(prop->name, name))  {
6282
0
      return(prop);
6283
0
        }
6284
0
  prop = prop->next;
6285
0
    }
6286
6287
    /*
6288
     * Check if there is a default declaration in the internal
6289
     * or external subsets
6290
     */
6291
0
    doc =  node->doc;
6292
0
    if (doc != NULL) {
6293
0
        xmlAttributePtr attrDecl;
6294
0
        if (doc->intSubset != NULL) {
6295
0
      attrDecl = xmlGetDtdAttrDesc(doc->intSubset, node->name, name);
6296
0
      if ((attrDecl == NULL) && (doc->extSubset != NULL))
6297
0
    attrDecl = xmlGetDtdAttrDesc(doc->extSubset, node->name, name);
6298
0
            if ((attrDecl != NULL) && (attrDecl->defaultValue != NULL))
6299
              /* return attribute declaration only if a default value is given
6300
                 (that includes #FIXED declarations) */
6301
0
    return((xmlAttrPtr) attrDecl);
6302
0
  }
6303
0
    }
6304
0
    return(NULL);
6305
0
}
6306
6307
/**
6308
 * Search for an attribute of an element.
6309
 *
6310
 * The attribute has to match the specified namespace. A namespace of
6311
 * NULL means that the attribute must have no namespace.
6312
 *
6313
 * This function also looks in DTD attribute declaration for \#FIXED or
6314
 * default declaration values.
6315
 *
6316
 * @param node  the element
6317
 * @param name  the attribute name
6318
 * @param nameSpace  the URI of the namespace
6319
 * @returns the attribute or the attribute declaration or NULL if
6320
 * neither was found. Also returns NULL if a memory allocation failed
6321
 * making this function unreliable.
6322
 */
6323
xmlAttr *
6324
0
xmlHasNsProp(const xmlNode *node, const xmlChar *name, const xmlChar *nameSpace) {
6325
6326
0
    return(xmlGetPropNodeInternal(node, name, nameSpace, 1));
6327
0
}
6328
6329
/**
6330
 * Look up the value of an element's attribute.
6331
 *
6332
 * The attribute has to match the specified namespace. A namespace of
6333
 * NULL means that the attribute must have no namespace.
6334
 *
6335
 * Entities are substituted. The returned value must be freed by the
6336
 * caller.
6337
 *
6338
 * @since 2.13.0
6339
 *
6340
 * @param node  the element
6341
 * @param name  the attribute name
6342
 * @param nsUri  the URI of the namespace
6343
 * @param out  the returned string
6344
 * @returns 0 on success, 1 if no attribute was found, -1 if a
6345
 * memory allocation failed.
6346
 */
6347
int
6348
xmlNodeGetAttrValue(const xmlNode *node, const xmlChar *name,
6349
0
                    const xmlChar *nsUri, xmlChar **out) {
6350
0
    xmlAttrPtr prop;
6351
6352
0
    if (out == NULL)
6353
0
        return(1);
6354
0
    *out = NULL;
6355
6356
0
    prop = xmlGetPropNodeInternal(node, name, nsUri, 0);
6357
0
    if (prop == NULL)
6358
0
  return(1);
6359
6360
0
    *out = xmlGetPropNodeValueInternal(prop);
6361
0
    if (*out == NULL)
6362
0
        return(-1);
6363
0
    return(0);
6364
0
}
6365
6366
/**
6367
 * Look up the value of an element's attribute.
6368
 *
6369
 * Entities are substituted. The returned value must be freed by the
6370
 * caller.
6371
 *
6372
 * This function looks in DTD attribute declarations for \#FIXED or
6373
 * default declaration values.
6374
 *
6375
 * NOTE: This function is ignores namespaces. Use #xmlGetNsProp or
6376
 * #xmlGetNoNsProp for namespace aware processing.
6377
 *
6378
 * NOTE: This function doesn't allow to distinguish malloc failures from
6379
 * missing attributes.
6380
 *
6381
 * @param node  the element
6382
 * @param name  the attribute name
6383
 * @returns the attribute value or NULL if not found or a memory allocation
6384
 * failed.
6385
 */
6386
xmlChar *
6387
0
xmlGetProp(const xmlNode *node, const xmlChar *name) {
6388
0
    xmlAttrPtr prop;
6389
6390
0
    prop = xmlHasProp(node, name);
6391
0
    if (prop == NULL)
6392
0
  return(NULL);
6393
0
    return(xmlGetPropNodeValueInternal(prop));
6394
0
}
6395
6396
/**
6397
 * Look up the value of an element's attribute.
6398
 *
6399
 * Entities are substituted. The returned value must be freed by the
6400
 * caller.
6401
 *
6402
 * This function looks in DTD attribute declarations for \#FIXED or
6403
 * default declaration values.
6404
 *
6405
 * This function is similar to #xmlGetProp except it will accept only
6406
 * an attribute in no namespace.
6407
 *
6408
 * NOTE: This function doesn't allow to distinguish malloc failures from
6409
 * missing attributes. It's more robust to use #xmlNodeGetAttrValue.
6410
 *
6411
 * @param node  the element
6412
 * @param name  the attribute name
6413
 * @returns the attribute value or NULL if not found or a memory allocation
6414
 * failed.
6415
 */
6416
xmlChar *
6417
0
xmlGetNoNsProp(const xmlNode *node, const xmlChar *name) {
6418
0
    xmlAttrPtr prop;
6419
6420
0
    prop = xmlGetPropNodeInternal(node, name, NULL, 1);
6421
0
    if (prop == NULL)
6422
0
  return(NULL);
6423
0
    return(xmlGetPropNodeValueInternal(prop));
6424
0
}
6425
6426
/**
6427
 * Look up the value of an element's attribute.
6428
 *
6429
 * The attribute has to match the specified namespace. A namespace of
6430
 * NULL means that the attribute must have no namespace.
6431
 *
6432
 * Entities are substituted. The returned value must be freed by the
6433
 * caller.
6434
 *
6435
 * This function looks in DTD attribute declaration for \#FIXED or
6436
 * default declaration values.
6437
 *
6438
 * NOTE: This function doesn't allow to distinguish malloc failures from
6439
 * missing attributes. It's more robust to use #xmlNodeGetAttrValue.
6440
 *
6441
 * @param node  the element
6442
 * @param name  the attribute name
6443
 * @param nameSpace  the URI of the namespace
6444
 * @returns the attribute value or NULL if not found or a memory allocation
6445
 * failed.
6446
 */
6447
xmlChar *
6448
0
xmlGetNsProp(const xmlNode *node, const xmlChar *name, const xmlChar *nameSpace) {
6449
0
    xmlAttrPtr prop;
6450
6451
0
    prop = xmlGetPropNodeInternal(node, name, nameSpace, 1);
6452
0
    if (prop == NULL)
6453
0
  return(NULL);
6454
0
    return(xmlGetPropNodeValueInternal(prop));
6455
0
}
6456
6457
/**
6458
 * Remove an attribute of an element.
6459
 * This handles only attributes in no namespace.
6460
 *
6461
 * @param node  the element
6462
 * @param name  the attribute name
6463
 * @returns 0 if successful, -1 if not found
6464
 */
6465
int
6466
0
xmlUnsetProp(xmlNode *node, const xmlChar *name) {
6467
0
    xmlAttrPtr prop;
6468
6469
0
    prop = xmlGetPropNodeInternal(node, name, NULL, 0);
6470
0
    if (prop == NULL)
6471
0
  return(-1);
6472
0
    xmlUnlinkNodeInternal((xmlNodePtr) prop);
6473
0
    xmlFreeProp(prop);
6474
0
    return(0);
6475
0
}
6476
6477
/**
6478
 * Remove an attribute of an element.
6479
 *
6480
 * @param node  the element
6481
 * @param ns  the namespace definition
6482
 * @param name  the attribute name
6483
 * @returns 0 if successful, -1 if not found
6484
 */
6485
int
6486
0
xmlUnsetNsProp(xmlNode *node, xmlNs *ns, const xmlChar *name) {
6487
0
    xmlAttrPtr prop;
6488
6489
0
    prop = xmlGetPropNodeInternal(node, name,
6490
0
                                  (ns != NULL) ? ns->href : NULL, 0);
6491
0
    if (prop == NULL)
6492
0
  return(-1);
6493
0
    xmlUnlinkNodeInternal((xmlNodePtr) prop);
6494
0
    xmlFreeProp(prop);
6495
0
    return(0);
6496
0
}
6497
6498
/**
6499
 * Set (or reset) an element's attribute. If `name` has a prefix,
6500
 * the corresponding namespace will be used. It is an error if
6501
 * there's no such binding for the prefix in scope.
6502
 *
6503
 * @param node  the node
6504
 * @param name  the attribute name (a QName)
6505
 * @param value  the attribute value
6506
 * @returns the attribute pointer.
6507
 */
6508
xmlAttr *
6509
0
xmlSetProp(xmlNode *node, const xmlChar *name, const xmlChar *value) {
6510
0
    xmlNsPtr ns = NULL;
6511
0
    const xmlChar *localname;
6512
0
    xmlChar *prefix;
6513
0
    int res;
6514
6515
0
    if ((node == NULL) || (name == NULL) || (node->type != XML_ELEMENT_NODE))
6516
0
  return(NULL);
6517
6518
    /*
6519
     * handle QNames
6520
     */
6521
0
    localname = xmlSplitQName4(name, &prefix);
6522
0
    if (localname == NULL)
6523
0
        return(NULL);
6524
6525
0
    if (prefix != NULL) {
6526
0
  res = xmlSearchNsSafe(node, prefix, &ns);
6527
0
  xmlFree(prefix);
6528
0
        if (res < 0)
6529
0
            return(NULL);
6530
0
        if (ns != NULL)
6531
0
            return(xmlSetNsProp(node, ns, localname, value));
6532
0
    }
6533
6534
0
    return(xmlSetNsProp(node, NULL, name, value));
6535
0
}
6536
6537
/**
6538
 * Set (or reset) an element's attribute.
6539
 *
6540
 * The namespace must be in scope.
6541
 *
6542
 * @param node  the node
6543
 * @param ns  the namespace definition
6544
 * @param name  the attribute name
6545
 * @param value  the attribute value
6546
 * @returns the attribute pointer.
6547
 */
6548
xmlAttr *
6549
xmlSetNsProp(xmlNode *node, xmlNs *ns, const xmlChar *name,
6550
       const xmlChar *value)
6551
0
{
6552
0
    xmlAttrPtr prop;
6553
6554
0
    if (ns && (ns->href == NULL))
6555
0
  return(NULL);
6556
0
    if (name == NULL)
6557
0
        return(NULL);
6558
0
    prop = xmlGetPropNodeInternal(node, name,
6559
0
                                  (ns != NULL) ? ns->href : NULL, 0);
6560
0
    if (prop != NULL) {
6561
0
        xmlNodePtr children = NULL;
6562
6563
  /*
6564
  * Modify the attribute's value.
6565
  */
6566
0
        if (value != NULL) {
6567
0
      children = xmlNewDocText(node->doc, value);
6568
0
            if (children == NULL)
6569
0
                return(NULL);
6570
0
        }
6571
6572
0
  if (prop->id != NULL) {
6573
0
      xmlRemoveID(node->doc, prop);
6574
0
      prop->atype = XML_ATTRIBUTE_ID;
6575
0
  }
6576
0
  if (prop->children != NULL)
6577
0
      xmlFreeNodeList(prop->children);
6578
0
  prop->children = NULL;
6579
0
  prop->last = NULL;
6580
0
  prop->ns = ns;
6581
0
  if (value != NULL) {
6582
0
      xmlNodePtr tmp;
6583
6584
0
      prop->children = children;
6585
0
      prop->last = NULL;
6586
0
      tmp = prop->children;
6587
0
      while (tmp != NULL) {
6588
0
    tmp->parent = (xmlNodePtr) prop;
6589
0
    if (tmp->next == NULL)
6590
0
        prop->last = tmp;
6591
0
    tmp = tmp->next;
6592
0
      }
6593
0
  }
6594
0
  if ((prop->atype == XML_ATTRIBUTE_ID) &&
6595
0
      (xmlAddIDSafe(prop, value) < 0)) {
6596
0
            return(NULL);
6597
0
        }
6598
0
  return(prop);
6599
0
    }
6600
    /*
6601
    * No equal attr found; create a new one.
6602
    */
6603
0
    return(xmlNewPropInternal(node, ns, name, value, 0));
6604
0
}
6605
6606
/**
6607
 * Check whether the node is a text node.
6608
 *
6609
 * @param node  the node
6610
 * @returns 1 if yes, 0 if no
6611
 */
6612
int
6613
0
xmlNodeIsText(const xmlNode *node) {
6614
0
    if (node == NULL) return(0);
6615
6616
0
    if (node->type == XML_TEXT_NODE) return(1);
6617
0
    return(0);
6618
0
}
6619
6620
/**
6621
 * Checks whether this node is an empty or whitespace-only
6622
 * text node.
6623
 *
6624
 * @param node  the node
6625
 * @returns 1 if yes, 0 if no
6626
 */
6627
int
6628
0
xmlIsBlankNode(const xmlNode *node) {
6629
0
    const xmlChar *cur;
6630
0
    if (node == NULL) return(0);
6631
6632
0
    if ((node->type != XML_TEXT_NODE) &&
6633
0
        (node->type != XML_CDATA_SECTION_NODE))
6634
0
  return(0);
6635
0
    if (node->content == NULL) return(1);
6636
0
    cur = node->content;
6637
0
    while (*cur != 0) {
6638
0
  if (!IS_BLANK_CH(*cur)) return(0);
6639
0
  cur++;
6640
0
    }
6641
6642
0
    return(1);
6643
0
}
6644
6645
/**
6646
 * Concat the given string at the end of the existing node content.
6647
 *
6648
 * If `len` is -1, the string length will be calculated.
6649
 *
6650
 * @param node  the node
6651
 * @param content  the content
6652
 * @param len  `content` length
6653
 * @returns -1 in case of error, 0 otherwise
6654
 */
6655
6656
int
6657
0
xmlTextConcat(xmlNode *node, const xmlChar *content, int len) {
6658
0
    if (node == NULL)
6659
0
        return(-1);
6660
6661
0
    if ((node->type != XML_TEXT_NODE) &&
6662
0
        (node->type != XML_CDATA_SECTION_NODE) &&
6663
0
  (node->type != XML_COMMENT_NODE) &&
6664
0
  (node->type != XML_PI_NODE))
6665
0
        return(-1);
6666
6667
0
    return(xmlTextAddContent(node, content, len));
6668
0
}
6669
6670
/**
6671
 * Get the compression level of a document, ZLIB based.
6672
 *
6673
 * @param doc  the document
6674
 * @returns 0 (uncompressed) to 9 (max compression)
6675
 */
6676
int
6677
0
xmlGetDocCompressMode (const xmlDoc *doc) {
6678
0
    if (doc == NULL) return(-1);
6679
0
    return(doc->compression);
6680
0
}
6681
6682
/**
6683
 * Set the compression level of a document, ZLIB based.
6684
 *
6685
 * Correct values: 0 (uncompressed) to 9 (max compression)
6686
 *
6687
 * @param doc  the document
6688
 * @param mode  the compression ratio
6689
 */
6690
void
6691
0
xmlSetDocCompressMode (xmlDoc *doc, int mode) {
6692
0
    if (doc == NULL) return;
6693
0
    if (mode < 0) doc->compression = 0;
6694
0
    else if (mode > 9) doc->compression = 9;
6695
0
    else doc->compression = mode;
6696
0
}
6697
6698
/**
6699
 * Get the global compression level, ZLIB based.
6700
 *
6701
 * @deprecated Use #xmlGetDocCompressMode
6702
 *
6703
 * @returns 0 (uncompressed) to 9 (max compression)
6704
 */
6705
int
6706
xmlGetCompressMode(void)
6707
0
{
6708
0
    return (xmlCompressMode);
6709
0
}
6710
6711
/**
6712
 * Set the global compression level, ZLIB based.
6713
 *
6714
 * @deprecated Use #xmlSetDocCompressMode
6715
 *
6716
 * Correct values: 0 (uncompressed) to 9 (max compression)
6717
 *
6718
 * @param mode  the compression ratio
6719
 */
6720
void
6721
0
xmlSetCompressMode(int mode) {
6722
0
    if (mode < 0) xmlCompressMode = 0;
6723
0
    else if (mode > 9) xmlCompressMode = 9;
6724
0
    else xmlCompressMode = mode;
6725
0
}
6726
6727
0
#define XML_TREE_NSMAP_PARENT -1
6728
#define XML_TREE_NSMAP_XML -2
6729
0
#define XML_TREE_NSMAP_DOC -3
6730
0
#define XML_TREE_NSMAP_CUSTOM -4
6731
6732
typedef struct xmlNsMapItem *xmlNsMapItemPtr;
6733
struct xmlNsMapItem {
6734
    xmlNsMapItemPtr next;
6735
    xmlNsMapItemPtr prev;
6736
    xmlNsPtr oldNs; /* old ns decl reference */
6737
    xmlNsPtr newNs; /* new ns decl reference */
6738
    int shadowDepth; /* Shadowed at this depth */
6739
    /*
6740
    * depth:
6741
    * >= 0 == @node's ns-decls
6742
    * -1   == @parent's ns-decls
6743
    * -2   == the doc->oldNs XML ns-decl
6744
    * -3   == the doc->oldNs storage ns-decls
6745
    * -4   == ns-decls provided via custom ns-handling
6746
    */
6747
    int depth;
6748
};
6749
6750
typedef struct xmlNsMap *xmlNsMapPtr;
6751
struct xmlNsMap {
6752
    xmlNsMapItemPtr first;
6753
    xmlNsMapItemPtr last;
6754
    xmlNsMapItemPtr pool;
6755
};
6756
6757
0
#define XML_NSMAP_NOTEMPTY(m) (((m) != NULL) && ((m)->first != NULL))
6758
0
#define XML_NSMAP_FOREACH(m, i) for (i = (m)->first; i != NULL; i = (i)->next)
6759
#define XML_NSMAP_POP(m, i) \
6760
0
    i = (m)->last; \
6761
0
    (m)->last = (i)->prev; \
6762
0
    if ((m)->last == NULL) \
6763
0
  (m)->first = NULL; \
6764
0
    else \
6765
0
  (m)->last->next = NULL; \
6766
0
    (i)->next = (m)->pool; \
6767
0
    (m)->pool = i;
6768
6769
/**
6770
 * Frees the ns-map
6771
 *
6772
 * @param nsmap  the ns-map
6773
 */
6774
static void
6775
xmlDOMWrapNsMapFree(xmlNsMapPtr nsmap)
6776
0
{
6777
0
    xmlNsMapItemPtr cur, tmp;
6778
6779
0
    if (nsmap == NULL)
6780
0
  return;
6781
0
    cur = nsmap->pool;
6782
0
    while (cur != NULL) {
6783
0
  tmp = cur;
6784
0
  cur = cur->next;
6785
0
  xmlFree(tmp);
6786
0
    }
6787
0
    cur = nsmap->first;
6788
0
    while (cur != NULL) {
6789
0
  tmp = cur;
6790
0
  cur = cur->next;
6791
0
  xmlFree(tmp);
6792
0
    }
6793
0
    xmlFree(nsmap);
6794
0
}
6795
6796
/**
6797
 * Adds an ns-mapping item.
6798
 *
6799
 * @param nsmap  the ns-map
6800
 * @param position  position
6801
 * @param oldNs  the old ns-struct
6802
 * @param newNs  the new ns-struct
6803
 * @param depth  depth and ns-kind information
6804
 * @returns the added item.
6805
 */
6806
static xmlNsMapItemPtr
6807
xmlDOMWrapNsMapAddItem(xmlNsMapPtr *nsmap, int position,
6808
           xmlNsPtr oldNs, xmlNsPtr newNs, int depth)
6809
0
{
6810
0
    xmlNsMapItemPtr ret;
6811
0
    xmlNsMapPtr map;
6812
6813
0
    if (nsmap == NULL)
6814
0
  return(NULL);
6815
0
    if ((position != -1) && (position != 0))
6816
0
  return(NULL);
6817
0
    map = *nsmap;
6818
6819
0
    if (map == NULL) {
6820
  /*
6821
  * Create the ns-map.
6822
  */
6823
0
  map = (xmlNsMapPtr) xmlMalloc(sizeof(struct xmlNsMap));
6824
0
  if (map == NULL)
6825
0
      return(NULL);
6826
0
  memset(map, 0, sizeof(struct xmlNsMap));
6827
0
  *nsmap = map;
6828
0
    }
6829
6830
0
    if (map->pool != NULL) {
6831
  /*
6832
  * Reuse an item from the pool.
6833
  */
6834
0
  ret = map->pool;
6835
0
  map->pool = ret->next;
6836
0
  memset(ret, 0, sizeof(struct xmlNsMapItem));
6837
0
    } else {
6838
  /*
6839
  * Create a new item.
6840
  */
6841
0
  ret = (xmlNsMapItemPtr) xmlMalloc(sizeof(struct xmlNsMapItem));
6842
0
  if (ret == NULL)
6843
0
      return(NULL);
6844
0
  memset(ret, 0, sizeof(struct xmlNsMapItem));
6845
0
    }
6846
6847
0
    if (map->first == NULL) {
6848
  /*
6849
  * First ever.
6850
  */
6851
0
  map->first = ret;
6852
0
  map->last = ret;
6853
0
    } else if (position == -1) {
6854
  /*
6855
  * Append.
6856
  */
6857
0
  ret->prev = map->last;
6858
0
  map->last->next = ret;
6859
0
  map->last = ret;
6860
0
    } else if (position == 0) {
6861
  /*
6862
  * Set on first position.
6863
  */
6864
0
  map->first->prev = ret;
6865
0
  ret->next = map->first;
6866
0
  map->first = ret;
6867
0
    }
6868
6869
0
    ret->oldNs = oldNs;
6870
0
    ret->newNs = newNs;
6871
0
    ret->shadowDepth = -1;
6872
0
    ret->depth = depth;
6873
0
    return (ret);
6874
0
}
6875
6876
/**
6877
 * Creates or reuses an xmlNs struct on doc->oldNs with
6878
 * the given prefix and namespace name.
6879
 *
6880
 * @param doc  the doc
6881
 * @param nsName  the namespace name
6882
 * @param prefix  the prefix
6883
 * @returns the acquired ns struct or NULL in case of an API
6884
 *          or internal error.
6885
 */
6886
static xmlNsPtr
6887
xmlDOMWrapStoreNs(xmlDocPtr doc,
6888
       const xmlChar *nsName,
6889
       const xmlChar *prefix)
6890
0
{
6891
0
    xmlNsPtr ns;
6892
6893
0
    if (doc == NULL)
6894
0
  return (NULL);
6895
0
    ns = xmlTreeEnsureXMLDecl(doc);
6896
0
    if (ns == NULL)
6897
0
  return (NULL);
6898
0
    if (ns->next != NULL) {
6899
  /* Reuse. */
6900
0
  ns = ns->next;
6901
0
  while (ns != NULL) {
6902
0
      if (((ns->prefix == prefix) ||
6903
0
    xmlStrEqual(ns->prefix, prefix)) &&
6904
0
    xmlStrEqual(ns->href, nsName)) {
6905
0
    return (ns);
6906
0
      }
6907
0
      if (ns->next == NULL)
6908
0
    break;
6909
0
      ns = ns->next;
6910
0
  }
6911
0
    }
6912
    /* Create. */
6913
0
    if (ns != NULL) {
6914
0
        ns->next = xmlNewNs(NULL, nsName, prefix);
6915
0
        return (ns->next);
6916
0
    }
6917
0
    return(NULL);
6918
0
}
6919
6920
/**
6921
 * Allocates and initializes a new DOM-wrapper context.
6922
 *
6923
 * @returns the xmlDOMWrapCtxt or NULL in case of an internal error.
6924
 */
6925
xmlDOMWrapCtxt *
6926
xmlDOMWrapNewCtxt(void)
6927
0
{
6928
0
    xmlDOMWrapCtxtPtr ret;
6929
6930
0
    ret = xmlMalloc(sizeof(xmlDOMWrapCtxt));
6931
0
    if (ret == NULL)
6932
0
  return (NULL);
6933
0
    memset(ret, 0, sizeof(xmlDOMWrapCtxt));
6934
0
    return (ret);
6935
0
}
6936
6937
/**
6938
 * Frees the DOM-wrapper context.
6939
 *
6940
 * @param ctxt  the DOM-wrapper context
6941
 */
6942
void
6943
xmlDOMWrapFreeCtxt(xmlDOMWrapCtxt *ctxt)
6944
0
{
6945
0
    if (ctxt == NULL)
6946
0
  return;
6947
0
    if (ctxt->namespaceMap != NULL)
6948
0
  xmlDOMWrapNsMapFree((xmlNsMapPtr) ctxt->namespaceMap);
6949
    /*
6950
    * TODO: Store the namespace map in the context.
6951
    */
6952
0
    xmlFree(ctxt);
6953
0
}
6954
6955
/**
6956
 * Searches for a ns-decl with the given prefix in `nsList`.
6957
 *
6958
 * @param nsList  a list of ns-structs
6959
 * @param prefix  the searched prefix
6960
 * @returns the ns-decl if found, NULL if not found and on
6961
 *          API errors.
6962
 */
6963
static xmlNsPtr
6964
xmlTreeNSListLookupByPrefix(xmlNsPtr nsList, const xmlChar *prefix)
6965
0
{
6966
0
    if (nsList == NULL)
6967
0
  return (NULL);
6968
0
    {
6969
0
  xmlNsPtr ns;
6970
0
  ns = nsList;
6971
0
  do {
6972
0
      if ((prefix == ns->prefix) ||
6973
0
    xmlStrEqual(prefix, ns->prefix)) {
6974
0
    return (ns);
6975
0
      }
6976
0
      ns = ns->next;
6977
0
  } while (ns != NULL);
6978
0
    }
6979
0
    return (NULL);
6980
0
}
6981
6982
/**
6983
 * Puts in-scope namespaces into the ns-map.
6984
 *
6985
 * @param map  the namespace map
6986
 * @param node  the node to start with
6987
 * @returns 0 on success, -1 on API or internal errors.
6988
 */
6989
static int
6990
xmlDOMWrapNSNormGatherInScopeNs(xmlNsMapPtr *map,
6991
        xmlNodePtr node)
6992
0
{
6993
0
    xmlNodePtr cur;
6994
0
    xmlNsPtr ns;
6995
0
    xmlNsMapItemPtr mi;
6996
0
    int shadowed;
6997
6998
0
    if ((map == NULL) || (*map != NULL))
6999
0
  return (-1);
7000
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL))
7001
0
        return (-1);
7002
    /*
7003
    * Get in-scope ns-decls of @parent.
7004
    */
7005
0
    cur = node;
7006
0
    while ((cur != NULL) && (cur != (xmlNodePtr) cur->doc)) {
7007
0
  if (cur->type == XML_ELEMENT_NODE) {
7008
0
      if (cur->nsDef != NULL) {
7009
0
    ns = cur->nsDef;
7010
0
    do {
7011
0
        shadowed = 0;
7012
0
        if (XML_NSMAP_NOTEMPTY(*map)) {
7013
      /*
7014
      * Skip shadowed prefixes.
7015
      */
7016
0
      XML_NSMAP_FOREACH(*map, mi) {
7017
0
          if ((ns->prefix == mi->newNs->prefix) ||
7018
0
        xmlStrEqual(ns->prefix, mi->newNs->prefix)) {
7019
0
        shadowed = 1;
7020
0
        break;
7021
0
          }
7022
0
      }
7023
0
        }
7024
        /*
7025
        * Insert mapping.
7026
        */
7027
0
        mi = xmlDOMWrapNsMapAddItem(map, 0, NULL,
7028
0
      ns, XML_TREE_NSMAP_PARENT);
7029
0
        if (mi == NULL)
7030
0
      return (-1);
7031
0
        if (shadowed)
7032
0
      mi->shadowDepth = 0;
7033
0
        ns = ns->next;
7034
0
    } while (ns != NULL);
7035
0
      }
7036
0
  }
7037
0
  cur = cur->parent;
7038
0
    }
7039
0
    return (0);
7040
0
}
7041
7042
/*
7043
 * For internal use. Adds a ns-decl mapping.
7044
 *
7045
 * Returns 0 on success, -1 on internal errors.
7046
 */
7047
static int
7048
xmlDOMWrapNSNormAddNsMapItem2(xmlNsPtr **list, int *size, int *number,
7049
      xmlNsPtr oldNs, xmlNsPtr newNs)
7050
0
{
7051
0
    if (*number >= *size) {
7052
0
        xmlNsPtr *tmp;
7053
0
        int newSize;
7054
7055
0
        newSize = xmlGrowCapacity(*size, 2 * sizeof(tmp[0]), 3, XML_MAX_ITEMS);
7056
0
        if (newSize < 0)
7057
0
            return(-1);
7058
0
        tmp = xmlRealloc(*list, newSize * 2 * sizeof(tmp[0]));
7059
0
        if (tmp == NULL)
7060
0
            return(-1);
7061
0
        *list = tmp;
7062
0
        *size = newSize;
7063
0
    }
7064
7065
0
    (*list)[2 * (*number)] = oldNs;
7066
0
    (*list)[2 * (*number) +1] = newNs;
7067
0
    (*number)++;
7068
0
    return (0);
7069
0
}
7070
7071
/**
7072
 * Unlinks the given node from its owner.
7073
 *
7074
 * This will substitute ns-references to node->nsDef for
7075
 * ns-references to doc->oldNs, thus ensuring the removed
7076
 * branch to be autark wrt ns-references.
7077
 *
7078
 * NOTE: This function was not intensively tested.
7079
 *
7080
 * @param ctxt  a DOM wrapper context
7081
 * @param doc  the doc
7082
 * @param node  the node to be removed.
7083
 * @param options  set of options, unused at the moment
7084
 * @returns 0 on success, 1 if the node is not supported,
7085
 *          -1 on API and internal errors.
7086
 */
7087
int
7088
xmlDOMWrapRemoveNode(xmlDOMWrapCtxt *ctxt, xmlDoc *doc,
7089
         xmlNode *node, int options ATTRIBUTE_UNUSED)
7090
0
{
7091
0
    xmlNsPtr *list = NULL;
7092
0
    int sizeList = 0, nbList = 0, ret = 0, i, j;
7093
0
    xmlNsPtr ns;
7094
7095
0
    if ((node == NULL) || (doc == NULL) || (node->doc != doc))
7096
0
  return (-1);
7097
7098
    /* TODO: 0 or -1 ? */
7099
0
    if (node->parent == NULL)
7100
0
  return (0);
7101
7102
0
    switch (node->type) {
7103
0
  case XML_TEXT_NODE:
7104
0
  case XML_CDATA_SECTION_NODE:
7105
0
  case XML_ENTITY_REF_NODE:
7106
0
  case XML_PI_NODE:
7107
0
  case XML_COMMENT_NODE:
7108
0
      xmlUnlinkNodeInternal(node);
7109
0
      return (0);
7110
0
  case XML_ELEMENT_NODE:
7111
0
  case XML_ATTRIBUTE_NODE:
7112
0
      break;
7113
0
  default:
7114
0
      return (1);
7115
0
    }
7116
0
    xmlUnlinkNodeInternal(node);
7117
    /*
7118
    * Save out-of-scope ns-references in doc->oldNs.
7119
    */
7120
0
    do {
7121
0
  switch (node->type) {
7122
0
      case XML_ELEMENT_NODE:
7123
0
    if ((ctxt == NULL) && (node->nsDef != NULL)) {
7124
0
        ns = node->nsDef;
7125
0
        do {
7126
0
      if (xmlDOMWrapNSNormAddNsMapItem2(&list, &sizeList,
7127
0
          &nbList, ns, ns) == -1)
7128
0
          ret = -1;
7129
0
      ns = ns->next;
7130
0
        } while (ns != NULL);
7131
0
    }
7132
                /* Falls through. */
7133
0
      case XML_ATTRIBUTE_NODE:
7134
0
    if (node->ns != NULL) {
7135
        /*
7136
        * Find a mapping.
7137
        */
7138
0
        if (list != NULL) {
7139
0
      for (i = 0, j = 0; i < nbList; i++, j += 2) {
7140
0
          if (node->ns == list[j]) {
7141
0
        node->ns = list[++j];
7142
0
        goto next_node;
7143
0
          }
7144
0
      }
7145
0
        }
7146
0
        ns = NULL;
7147
0
        if (ctxt != NULL) {
7148
      /*
7149
      * User defined.
7150
      */
7151
0
        } else {
7152
      /*
7153
      * Add to doc's oldNs.
7154
      */
7155
0
      ns = xmlDOMWrapStoreNs(doc, node->ns->href,
7156
0
          node->ns->prefix);
7157
0
      if (ns == NULL)
7158
0
          ret = -1;
7159
0
        }
7160
0
        if (ns != NULL) {
7161
      /*
7162
      * Add mapping.
7163
      */
7164
0
      if (xmlDOMWrapNSNormAddNsMapItem2(&list, &sizeList,
7165
0
          &nbList, node->ns, ns) == -1)
7166
0
          ret = -1;
7167
0
        }
7168
0
        node->ns = ns;
7169
0
    }
7170
0
    if ((node->type == XML_ELEMENT_NODE) &&
7171
0
        (node->properties != NULL)) {
7172
0
        node = (xmlNodePtr) node->properties;
7173
0
        continue;
7174
0
    }
7175
0
    break;
7176
0
      default:
7177
0
    goto next_sibling;
7178
0
  }
7179
0
next_node:
7180
0
  if ((node->type == XML_ELEMENT_NODE) &&
7181
0
      (node->children != NULL)) {
7182
0
      node = node->children;
7183
0
      continue;
7184
0
  }
7185
0
next_sibling:
7186
0
  if (node == NULL)
7187
0
      break;
7188
0
  if (node->next != NULL)
7189
0
      node = node->next;
7190
0
  else {
7191
0
            int type = node->type;
7192
7193
0
      node = node->parent;
7194
0
            if ((type == XML_ATTRIBUTE_NODE) &&
7195
0
                (node != NULL) &&
7196
0
                (node->children != NULL)) {
7197
0
                node = node->children;
7198
0
            } else {
7199
0
          goto next_sibling;
7200
0
            }
7201
0
  }
7202
0
    } while (node != NULL);
7203
7204
0
    if (list != NULL)
7205
0
  xmlFree(list);
7206
0
    return (ret);
7207
0
}
7208
7209
/**
7210
 * Dynamically searches for a ns-declaration which matches
7211
 * the given `nsName` in the ancestor-or-self axis of `node`.
7212
 *
7213
 * @param doc  the document
7214
 * @param node  the start node
7215
 * @param nsName  the searched namespace name
7216
 * @param retNs  the resulting ns-decl
7217
 * @param prefixed  if the found ns-decl must have a prefix
7218
 *                  (for attributes)
7219
 * @returns 1 if a ns-decl was found, 0 if not and -1 on API
7220
 *          and internal errors.
7221
 */
7222
static int
7223
xmlSearchNsByNamespaceStrict(xmlDocPtr doc, xmlNodePtr node,
7224
           const xmlChar* nsName,
7225
           xmlNsPtr *retNs, int prefixed)
7226
0
{
7227
0
    xmlNodePtr cur, prev = NULL, out = NULL;
7228
0
    xmlNsPtr ns, prevns;
7229
7230
0
    if ((doc == NULL) || (nsName == NULL) || (retNs == NULL))
7231
0
  return (-1);
7232
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL))
7233
0
        return(-1);
7234
7235
0
    *retNs = NULL;
7236
0
    if (xmlStrEqual(nsName, XML_XML_NAMESPACE)) {
7237
0
  *retNs = xmlTreeEnsureXMLDecl(doc);
7238
0
  if (*retNs == NULL)
7239
0
      return (-1);
7240
0
  return (1);
7241
0
    }
7242
0
    cur = node;
7243
0
    do {
7244
0
  if (cur->type == XML_ELEMENT_NODE) {
7245
0
      if (cur->nsDef != NULL) {
7246
0
    for (ns = cur->nsDef; ns != NULL; ns = ns->next) {
7247
0
        if (prefixed && (ns->prefix == NULL))
7248
0
      continue;
7249
0
        if (prev != NULL) {
7250
      /*
7251
      * Check the last level of ns-decls for a
7252
      * shadowing prefix.
7253
      */
7254
0
      prevns = prev->nsDef;
7255
0
      do {
7256
0
          if ((prevns->prefix == ns->prefix) ||
7257
0
        ((prevns->prefix != NULL) &&
7258
0
        (ns->prefix != NULL) &&
7259
0
        xmlStrEqual(prevns->prefix, ns->prefix))) {
7260
        /*
7261
        * Shadowed.
7262
        */
7263
0
        break;
7264
0
          }
7265
0
          prevns = prevns->next;
7266
0
      } while (prevns != NULL);
7267
0
      if (prevns != NULL)
7268
0
          continue;
7269
0
        }
7270
        /*
7271
        * Ns-name comparison.
7272
        */
7273
0
        if ((nsName == ns->href) ||
7274
0
      xmlStrEqual(nsName, ns->href)) {
7275
      /*
7276
      * At this point the prefix can only be shadowed,
7277
      * if we are the the (at least) 3rd level of
7278
      * ns-decls.
7279
      */
7280
0
      if (out) {
7281
0
          int ret;
7282
7283
0
          ret = xmlNsInScope(doc, node, prev, ns->prefix);
7284
0
          if (ret < 0)
7285
0
        return (-1);
7286
          /*
7287
          * TODO: Should we try to find a matching ns-name
7288
          * only once? This here keeps on searching.
7289
          * I think we should try further since, there might
7290
          * be an other matching ns-decl with an unshadowed
7291
          * prefix.
7292
          */
7293
0
          if (! ret)
7294
0
        continue;
7295
0
      }
7296
0
      *retNs = ns;
7297
0
      return (1);
7298
0
        }
7299
0
    }
7300
0
    out = prev;
7301
0
    prev = cur;
7302
0
      }
7303
0
  } else if (cur->type == XML_ENTITY_DECL)
7304
0
      return (0);
7305
0
  cur = cur->parent;
7306
0
    } while ((cur != NULL) && (cur->doc != (xmlDocPtr) cur));
7307
0
    return (0);
7308
0
}
7309
7310
/**
7311
 * Dynamically searches for a ns-declaration which matches
7312
 * the given `nsName` in the ancestor-or-self axis of `node`.
7313
 *
7314
 * @param doc  the document
7315
 * @param node  the start node
7316
 * @param prefix  the searched namespace prefix
7317
 * @param retNs  the resulting ns-decl
7318
 * @returns 1 if a ns-decl was found, 0 if not and -1 on API
7319
 *          and internal errors.
7320
 */
7321
static int
7322
xmlSearchNsByPrefixStrict(xmlDocPtr doc, xmlNodePtr node,
7323
        const xmlChar* prefix,
7324
        xmlNsPtr *retNs)
7325
0
{
7326
0
    xmlNodePtr cur;
7327
0
    xmlNsPtr ns;
7328
7329
0
    if ((doc == NULL) || (node == NULL) || (node->type == XML_NAMESPACE_DECL))
7330
0
        return(-1);
7331
7332
0
    if (retNs)
7333
0
  *retNs = NULL;
7334
0
    if (IS_STR_XML(prefix)) {
7335
0
  if (retNs) {
7336
0
      *retNs = xmlTreeEnsureXMLDecl(doc);
7337
0
      if (*retNs == NULL)
7338
0
    return (-1);
7339
0
  }
7340
0
  return (1);
7341
0
    }
7342
0
    cur = node;
7343
0
    do {
7344
0
  if (cur->type == XML_ELEMENT_NODE) {
7345
0
      if (cur->nsDef != NULL) {
7346
0
    ns = cur->nsDef;
7347
0
    do {
7348
0
        if ((prefix == ns->prefix) ||
7349
0
      xmlStrEqual(prefix, ns->prefix))
7350
0
        {
7351
      /*
7352
      * Disabled namespaces, e.g. xmlns:abc="".
7353
      */
7354
0
      if (ns->href == NULL)
7355
0
          return(0);
7356
0
      if (retNs)
7357
0
          *retNs = ns;
7358
0
      return (1);
7359
0
        }
7360
0
        ns = ns->next;
7361
0
    } while (ns != NULL);
7362
0
      }
7363
0
  } else if (cur->type == XML_ENTITY_DECL)
7364
0
      return (0);
7365
0
  cur = cur->parent;
7366
0
    } while ((cur != NULL) && (cur->doc != (xmlDocPtr) cur));
7367
0
    return (0);
7368
0
}
7369
7370
/**
7371
 * Declares a new namespace on `elem`. It tries to use the
7372
 * given `prefix`. If a ns-decl with the given prefix is already existent
7373
 * on `elem`, it will generate an other prefix.
7374
 *
7375
 * @param doc  the doc
7376
 * @param elem  the element-node to declare on
7377
 * @param nsName  the namespace-name of the ns-decl
7378
 * @param prefix  the preferred prefix of the ns-decl
7379
 * @param checkShadow  ensure that the new ns-decl doesn't shadow
7380
 *                     ancestor ns-decls
7381
 * @returns 1 if a ns-decl was found, 0 if not and -1 on API
7382
 *          and internal errors.
7383
 */
7384
static xmlNsPtr
7385
xmlDOMWrapNSNormDeclareNsForced(xmlDocPtr doc,
7386
        xmlNodePtr elem,
7387
        const xmlChar *nsName,
7388
        const xmlChar *prefix,
7389
        int checkShadow)
7390
0
{
7391
7392
0
    xmlNsPtr ret;
7393
0
    char buf[50];
7394
0
    const xmlChar *pref;
7395
0
    int counter = 0;
7396
7397
0
    if ((doc == NULL) || (elem == NULL) || (elem->type != XML_ELEMENT_NODE))
7398
0
        return(NULL);
7399
    /*
7400
    * Create a ns-decl on @anchor.
7401
    */
7402
0
    pref = prefix;
7403
0
    while (1) {
7404
  /*
7405
  * Lookup whether the prefix is unused in elem's ns-decls.
7406
  */
7407
0
  if ((elem->nsDef != NULL) &&
7408
0
      (xmlTreeNSListLookupByPrefix(elem->nsDef, pref) != NULL))
7409
0
      goto ns_next_prefix;
7410
0
  if (checkShadow && elem->parent &&
7411
0
      ((xmlNodePtr) elem->parent->doc != elem->parent)) {
7412
      /*
7413
      * Does it shadow ancestor ns-decls?
7414
      */
7415
0
      if (xmlSearchNsByPrefixStrict(doc, elem->parent, pref, NULL) == 1)
7416
0
    goto ns_next_prefix;
7417
0
  }
7418
0
  ret = xmlNewNs(NULL, nsName, pref);
7419
0
  if (ret == NULL)
7420
0
      return (NULL);
7421
0
  if (elem->nsDef == NULL)
7422
0
      elem->nsDef = ret;
7423
0
  else {
7424
0
      xmlNsPtr ns2 = elem->nsDef;
7425
0
      while (ns2->next != NULL)
7426
0
    ns2 = ns2->next;
7427
0
      ns2->next = ret;
7428
0
  }
7429
0
  return (ret);
7430
0
ns_next_prefix:
7431
0
  counter++;
7432
0
  if (counter > 1000)
7433
0
      return (NULL);
7434
0
  if (prefix == NULL) {
7435
0
      snprintf((char *) buf, sizeof(buf),
7436
0
    "ns_%d", counter);
7437
0
  } else
7438
0
      snprintf((char *) buf, sizeof(buf),
7439
0
      "%.30s_%d", (char *)prefix, counter);
7440
0
  pref = BAD_CAST buf;
7441
0
    }
7442
0
}
7443
7444
/**
7445
 * Searches for a matching ns-name in the ns-decls of `nsMap`, if not
7446
 * found it will either declare it on `elem`, or store it in `doc->oldNs`.
7447
 * If a new ns-decl needs to be declared on `elem`, it tries to use the
7448
 * `ns->prefix` for it, if this prefix is already in use on `elem`, it will
7449
 * change the prefix or the new ns-decl.
7450
 *
7451
 * @param doc  the doc
7452
 * @param elem  the element-node to declare namespaces on
7453
 * @param ns  the ns-struct to use for the search
7454
 * @param retNs  the found/created ns-struct
7455
 * @param nsMap  the ns-map
7456
 * @param depth  the current tree depth
7457
 * @param ancestorsOnly  search in ancestor ns-decls only
7458
 * @param prefixed  if the searched ns-decl must have a prefix
7459
 *                  (for attributes)
7460
 * @returns 0 if succeeded, -1 otherwise and on API/internal errors.
7461
 */
7462
static int
7463
xmlDOMWrapNSNormAcquireNormalizedNs(xmlDocPtr doc,
7464
           xmlNodePtr elem,
7465
           xmlNsPtr ns,
7466
           xmlNsPtr *retNs,
7467
           xmlNsMapPtr *nsMap,
7468
7469
           int depth,
7470
           int ancestorsOnly,
7471
           int prefixed)
7472
0
{
7473
0
    xmlNsMapItemPtr mi;
7474
7475
0
    if ((doc == NULL) || (ns == NULL) || (retNs == NULL) ||
7476
0
  (nsMap == NULL))
7477
0
  return (-1);
7478
7479
0
    *retNs = NULL;
7480
    /*
7481
    * Handle XML namespace.
7482
    */
7483
0
    if (IS_STR_XML(ns->prefix)) {
7484
  /*
7485
  * Insert XML namespace mapping.
7486
  */
7487
0
  *retNs = xmlTreeEnsureXMLDecl(doc);
7488
0
  if (*retNs == NULL)
7489
0
      return (-1);
7490
0
  return (0);
7491
0
    }
7492
    /*
7493
    * If the search should be done in ancestors only and no
7494
    * @elem (the first ancestor) was specified, then skip the search.
7495
    */
7496
0
    if ((XML_NSMAP_NOTEMPTY(*nsMap)) &&
7497
0
  (! (ancestorsOnly && (elem == NULL))))
7498
0
    {
7499
  /*
7500
  * Try to find an equal ns-name in in-scope ns-decls.
7501
  */
7502
0
  XML_NSMAP_FOREACH(*nsMap, mi) {
7503
0
      if ((mi->depth >= XML_TREE_NSMAP_PARENT) &&
7504
    /*
7505
    * ancestorsOnly: This should be turned on to gain speed,
7506
    * if one knows that the branch itself was already
7507
    * ns-wellformed and no stale references existed.
7508
    * I.e. it searches in the ancestor axis only.
7509
    */
7510
0
    ((! ancestorsOnly) || (mi->depth == XML_TREE_NSMAP_PARENT)) &&
7511
    /* Skip shadowed prefixes. */
7512
0
    (mi->shadowDepth == -1) &&
7513
    /* Skip xmlns="" or xmlns:foo="". */
7514
0
    ((mi->newNs->href != NULL) &&
7515
0
    (mi->newNs->href[0] != 0)) &&
7516
    /* Ensure a prefix if wanted. */
7517
0
    ((! prefixed) || (mi->newNs->prefix != NULL)) &&
7518
    /* Equal ns name */
7519
0
    ((mi->newNs->href == ns->href) ||
7520
0
    xmlStrEqual(mi->newNs->href, ns->href))) {
7521
    /* Set the mapping. */
7522
0
    mi->oldNs = ns;
7523
0
    *retNs = mi->newNs;
7524
0
    return (0);
7525
0
      }
7526
0
  }
7527
0
    }
7528
    /*
7529
    * No luck, the namespace is out of scope or shadowed.
7530
    */
7531
0
    if (elem == NULL) {
7532
0
  xmlNsPtr tmpns;
7533
7534
  /*
7535
  * Store ns-decls in "oldNs" of the document-node.
7536
  */
7537
0
  tmpns = xmlDOMWrapStoreNs(doc, ns->href, ns->prefix);
7538
0
  if (tmpns == NULL)
7539
0
      return (-1);
7540
  /*
7541
  * Insert mapping.
7542
  */
7543
0
  if (xmlDOMWrapNsMapAddItem(nsMap, -1, ns,
7544
0
    tmpns, XML_TREE_NSMAP_DOC) == NULL) {
7545
0
      return (-1);
7546
0
  }
7547
0
  *retNs = tmpns;
7548
0
    } else {
7549
0
  xmlNsPtr tmpns;
7550
7551
0
  tmpns = xmlDOMWrapNSNormDeclareNsForced(doc, elem, ns->href,
7552
0
      ns->prefix, 0);
7553
0
  if (tmpns == NULL)
7554
0
      return (-1);
7555
7556
0
  if (*nsMap != NULL) {
7557
      /*
7558
      * Does it shadow ancestor ns-decls?
7559
      */
7560
0
      XML_NSMAP_FOREACH(*nsMap, mi) {
7561
0
    if ((mi->depth < depth) &&
7562
0
        (mi->shadowDepth == -1) &&
7563
0
        ((ns->prefix == mi->newNs->prefix) ||
7564
0
        xmlStrEqual(ns->prefix, mi->newNs->prefix))) {
7565
        /*
7566
        * Shadows.
7567
        */
7568
0
        mi->shadowDepth = depth;
7569
0
        break;
7570
0
    }
7571
0
      }
7572
0
  }
7573
0
  if (xmlDOMWrapNsMapAddItem(nsMap, -1, ns, tmpns, depth) == NULL) {
7574
0
      return (-1);
7575
0
  }
7576
0
  *retNs = tmpns;
7577
0
    }
7578
0
    return (0);
7579
0
}
7580
7581
typedef enum {
7582
    XML_DOM_RECONNS_REMOVEREDUND = 1<<0
7583
} xmlDOMReconcileNSOptions;
7584
7585
/**
7586
 * Fix up namespaces.
7587
 *
7588
 * Ensures that ns-references point to ns-decls hold on element-nodes.
7589
 * Ensures that the tree is namespace wellformed by creating additional
7590
 * ns-decls where needed. Note that, since prefixes of already existent
7591
 * ns-decls can be shadowed by this process, it could break QNames in
7592
 * attribute values or element content.
7593
 *
7594
 * NOTE: This function was not intensively tested.
7595
 *
7596
 * @param ctxt  DOM wrapper context, unused at the moment
7597
 * @param elem  the element-node
7598
 * @param options  option flags
7599
 * @returns 0 if succeeded, -1 otherwise and on API/internal errors.
7600
 */
7601
int
7602
xmlDOMWrapReconcileNamespaces(xmlDOMWrapCtxt *ctxt ATTRIBUTE_UNUSED,
7603
            xmlNode *elem,
7604
            int options)
7605
0
{
7606
0
    int depth = -1, adoptns = 0, parnsdone = 0;
7607
0
    xmlNsPtr ns, prevns;
7608
0
    xmlDocPtr doc;
7609
0
    xmlNodePtr cur, curElem = NULL;
7610
0
    xmlNsMapPtr nsMap = NULL;
7611
0
    xmlNsMapItemPtr /* topmi = NULL, */ mi;
7612
    /* @ancestorsOnly should be set by an option flag. */
7613
0
    int ancestorsOnly = 0;
7614
0
    int optRemoveRedundantNS =
7615
0
  ((xmlDOMReconcileNSOptions) options & XML_DOM_RECONNS_REMOVEREDUND) ? 1 : 0;
7616
0
    xmlNsPtr *listRedund = NULL;
7617
0
    int sizeRedund = 0, nbRedund = 0, ret = 0, i, j;
7618
7619
0
    if ((elem == NULL) || (elem->doc == NULL) ||
7620
0
  (elem->type != XML_ELEMENT_NODE))
7621
0
  return (-1);
7622
7623
0
    doc = elem->doc;
7624
0
    cur = elem;
7625
0
    do {
7626
0
  switch (cur->type) {
7627
0
      case XML_ELEMENT_NODE:
7628
0
    adoptns = 1;
7629
0
    curElem = cur;
7630
0
    depth++;
7631
    /*
7632
    * Namespace declarations.
7633
    */
7634
0
    if (cur->nsDef != NULL) {
7635
0
        prevns = NULL;
7636
0
        ns = cur->nsDef;
7637
0
        while (ns != NULL) {
7638
0
      if (! parnsdone) {
7639
0
          if ((elem->parent) &&
7640
0
        ((xmlNodePtr) elem->parent->doc != elem->parent)) {
7641
        /*
7642
        * Gather ancestor in-scope ns-decls.
7643
        */
7644
0
        if (xmlDOMWrapNSNormGatherInScopeNs(&nsMap,
7645
0
            elem->parent) == -1)
7646
0
            ret = -1;
7647
0
          }
7648
0
          parnsdone = 1;
7649
0
      }
7650
7651
      /*
7652
      * Lookup the ns ancestor-axis for equal ns-decls in scope.
7653
      */
7654
0
      if (optRemoveRedundantNS && XML_NSMAP_NOTEMPTY(nsMap)) {
7655
0
          XML_NSMAP_FOREACH(nsMap, mi) {
7656
0
        if ((mi->depth >= XML_TREE_NSMAP_PARENT) &&
7657
0
            (mi->shadowDepth == -1) &&
7658
0
            ((ns->prefix == mi->newNs->prefix) ||
7659
0
              xmlStrEqual(ns->prefix, mi->newNs->prefix)) &&
7660
0
            ((ns->href == mi->newNs->href) ||
7661
0
              xmlStrEqual(ns->href, mi->newNs->href)))
7662
0
        {
7663
            /*
7664
            * A redundant ns-decl was found.
7665
            * Add it to the list of redundant ns-decls.
7666
            */
7667
0
            if (xmlDOMWrapNSNormAddNsMapItem2(&listRedund,
7668
0
          &sizeRedund, &nbRedund, ns, mi->newNs) == -1) {
7669
0
          ret = -1;
7670
0
                                    } else {
7671
                                        /*
7672
                                        * Remove the ns-decl from the element-node.
7673
                                        */
7674
0
                                        if (prevns)
7675
0
                                            prevns->next = ns->next;
7676
0
                                        else
7677
0
                                            cur->nsDef = ns->next;
7678
0
                                        goto next_ns_decl;
7679
0
                                    }
7680
0
        }
7681
0
          }
7682
0
      }
7683
7684
      /*
7685
      * Skip ns-references handling if the referenced
7686
      * ns-decl is declared on the same element.
7687
      */
7688
0
      if ((cur->ns != NULL) && adoptns && (cur->ns == ns))
7689
0
          adoptns = 0;
7690
      /*
7691
      * Does it shadow any ns-decl?
7692
      */
7693
0
      if (XML_NSMAP_NOTEMPTY(nsMap)) {
7694
0
          XML_NSMAP_FOREACH(nsMap, mi) {
7695
0
        if ((mi->depth >= XML_TREE_NSMAP_PARENT) &&
7696
0
            (mi->shadowDepth == -1) &&
7697
0
            ((ns->prefix == mi->newNs->prefix) ||
7698
0
            xmlStrEqual(ns->prefix, mi->newNs->prefix))) {
7699
7700
0
            mi->shadowDepth = depth;
7701
0
        }
7702
0
          }
7703
0
      }
7704
      /*
7705
      * Push mapping.
7706
      */
7707
0
      if (xmlDOMWrapNsMapAddItem(&nsMap, -1, ns, ns,
7708
0
          depth) == NULL)
7709
0
          ret = -1;
7710
7711
0
      prevns = ns;
7712
0
next_ns_decl:
7713
0
      ns = ns->next;
7714
0
        }
7715
0
    }
7716
0
    if (! adoptns)
7717
0
        goto ns_end;
7718
                /* Falls through. */
7719
0
      case XML_ATTRIBUTE_NODE:
7720
    /* No ns, no fun. */
7721
0
    if (cur->ns == NULL)
7722
0
        goto ns_end;
7723
7724
0
    if (! parnsdone) {
7725
0
        if ((elem->parent) &&
7726
0
      ((xmlNodePtr) elem->parent->doc != elem->parent)) {
7727
0
      if (xmlDOMWrapNSNormGatherInScopeNs(&nsMap,
7728
0
        elem->parent) == -1)
7729
0
          ret = -1;
7730
0
        }
7731
0
        parnsdone = 1;
7732
0
    }
7733
    /*
7734
    * Adjust the reference if this was a redundant ns-decl.
7735
    */
7736
0
    if (listRedund) {
7737
0
       for (i = 0, j = 0; i < nbRedund; i++, j += 2) {
7738
0
           if (cur->ns == listRedund[j]) {
7739
0
         cur->ns = listRedund[++j];
7740
0
         break;
7741
0
           }
7742
0
       }
7743
0
    }
7744
    /*
7745
    * Adopt ns-references.
7746
    */
7747
0
    if (XML_NSMAP_NOTEMPTY(nsMap)) {
7748
        /*
7749
        * Search for a mapping.
7750
        */
7751
0
        XML_NSMAP_FOREACH(nsMap, mi) {
7752
0
      if ((mi->shadowDepth == -1) &&
7753
0
          (cur->ns == mi->oldNs)) {
7754
7755
0
          cur->ns = mi->newNs;
7756
0
          goto ns_end;
7757
0
      }
7758
0
        }
7759
0
    }
7760
    /*
7761
    * Acquire a normalized ns-decl and add it to the map.
7762
    */
7763
0
    if (xmlDOMWrapNSNormAcquireNormalizedNs(doc, curElem,
7764
0
      cur->ns, &ns,
7765
0
      &nsMap, depth,
7766
0
      ancestorsOnly,
7767
0
      (cur->type == XML_ATTRIBUTE_NODE) ? 1 : 0) == -1)
7768
0
        ret = -1;
7769
0
    cur->ns = ns;
7770
7771
0
ns_end:
7772
0
    if ((cur->type == XML_ELEMENT_NODE) &&
7773
0
        (cur->properties != NULL)) {
7774
        /*
7775
        * Process attributes.
7776
        */
7777
0
        cur = (xmlNodePtr) cur->properties;
7778
0
        continue;
7779
0
    }
7780
0
    break;
7781
0
      default:
7782
0
    goto next_sibling;
7783
0
  }
7784
0
into_content:
7785
0
  if ((cur->type == XML_ELEMENT_NODE) &&
7786
0
      (cur->children != NULL)) {
7787
      /*
7788
      * Process content of element-nodes only.
7789
      */
7790
0
      cur = cur->children;
7791
0
      continue;
7792
0
  }
7793
0
next_sibling:
7794
0
  if (cur == elem)
7795
0
      break;
7796
0
  if (cur->type == XML_ELEMENT_NODE) {
7797
0
      if (XML_NSMAP_NOTEMPTY(nsMap)) {
7798
    /*
7799
    * Pop mappings.
7800
    */
7801
0
    while ((nsMap->last != NULL) &&
7802
0
        (nsMap->last->depth >= depth))
7803
0
    {
7804
0
        XML_NSMAP_POP(nsMap, mi)
7805
0
    }
7806
    /*
7807
    * Unshadow.
7808
    */
7809
0
    XML_NSMAP_FOREACH(nsMap, mi) {
7810
0
        if (mi->shadowDepth >= depth)
7811
0
      mi->shadowDepth = -1;
7812
0
    }
7813
0
      }
7814
0
      depth--;
7815
0
  }
7816
0
  if (cur->next != NULL)
7817
0
      cur = cur->next;
7818
0
  else {
7819
0
      if (cur->type == XML_ATTRIBUTE_NODE) {
7820
0
    cur = cur->parent;
7821
0
    goto into_content;
7822
0
      }
7823
0
      cur = cur->parent;
7824
0
      goto next_sibling;
7825
0
  }
7826
0
    } while (cur != NULL);
7827
7828
0
    if (listRedund) {
7829
0
  for (i = 0, j = 0; i < nbRedund; i++, j += 2) {
7830
0
      xmlFreeNs(listRedund[j]);
7831
0
  }
7832
0
  xmlFree(listRedund);
7833
0
    }
7834
0
    if (nsMap != NULL)
7835
0
  xmlDOMWrapNsMapFree(nsMap);
7836
0
    return (ret);
7837
0
}
7838
7839
/**
7840
 * Ensures that ns-references point to `destDoc`: either to
7841
 * `elements->nsDef` entries if `destParent` is given, or to
7842
 * `destDoc->oldNs` otherwise.
7843
 *
7844
 * If `destParent` is given, it ensures that the tree is namespace
7845
 * wellformed by creating additional ns-decls where needed.
7846
 * Note that, since prefixes of already existent ns-decls can be
7847
 * shadowed by this process, it could break QNames in attribute
7848
 * values or element content.
7849
 *
7850
 * NOTE: This function was not intensively tested.
7851
 *
7852
 * @param ctxt  the optional context for custom processing
7853
 * @param sourceDoc  the optional sourceDoc
7854
 * @param node  the element-node to start with
7855
 * @param destDoc  the destination doc for adoption
7856
 * @param destParent  the optional new parent of `node` in `destDoc`
7857
 * @param options  option flags
7858
 * @returns 0 if succeeded, -1 otherwise and on API/internal errors.
7859
 */
7860
static int
7861
xmlDOMWrapAdoptBranch(xmlDOMWrapCtxtPtr ctxt,
7862
          xmlDocPtr sourceDoc ATTRIBUTE_UNUSED,
7863
          xmlNodePtr node,
7864
          xmlDocPtr destDoc,
7865
          xmlNodePtr destParent,
7866
          int options ATTRIBUTE_UNUSED)
7867
0
{
7868
0
    int ret = 0;
7869
0
    xmlNodePtr cur, curElem = NULL;
7870
0
    xmlNsMapPtr nsMap = NULL;
7871
0
    xmlNsMapItemPtr mi;
7872
0
    xmlNsPtr ns = NULL;
7873
0
    int depth = -1;
7874
    /* gather @parent's ns-decls. */
7875
0
    int parnsdone;
7876
    /* @ancestorsOnly should be set per option. */
7877
0
    int ancestorsOnly = 0;
7878
7879
    /*
7880
    * Get the ns-map from the context if available.
7881
    */
7882
0
    if (ctxt)
7883
0
  nsMap = (xmlNsMapPtr) ctxt->namespaceMap;
7884
    /*
7885
    * Disable search for ns-decls in the parent-axis of the
7886
    * destination element, if:
7887
    * 1) there's no destination parent
7888
    * 2) custom ns-reference handling is used
7889
    */
7890
0
    if ((destParent == NULL) ||
7891
0
  (ctxt && ctxt->getNsForNodeFunc))
7892
0
    {
7893
0
  parnsdone = 1;
7894
0
    } else
7895
0
  parnsdone = 0;
7896
7897
0
    cur = node;
7898
7899
0
    while (cur != NULL) {
7900
0
        if (cur->doc != destDoc) {
7901
0
            if (xmlNodeSetDoc(cur, destDoc) < 0)
7902
0
                ret = -1;
7903
0
        }
7904
7905
0
  switch (cur->type) {
7906
0
      case XML_XINCLUDE_START:
7907
0
      case XML_XINCLUDE_END:
7908
    /*
7909
    * TODO
7910
    */
7911
0
    ret = -1;
7912
0
                goto leave_node;
7913
0
      case XML_ELEMENT_NODE:
7914
0
    curElem = cur;
7915
0
    depth++;
7916
    /*
7917
    * Namespace declarations.
7918
    * - ns->href and ns->prefix are never in the dict, so
7919
    *   we need not move the values over to the destination dict.
7920
    * - Note that for custom handling of ns-references,
7921
    *   the ns-decls need not be stored in the ns-map,
7922
    *   since they won't be referenced by node->ns.
7923
    */
7924
0
    if ((cur->nsDef) &&
7925
0
        ((ctxt == NULL) || (ctxt->getNsForNodeFunc == NULL)))
7926
0
    {
7927
0
        if (! parnsdone) {
7928
      /*
7929
      * Gather @parent's in-scope ns-decls.
7930
      */
7931
0
      if (xmlDOMWrapNSNormGatherInScopeNs(&nsMap,
7932
0
          destParent) == -1)
7933
0
          ret = -1;
7934
0
      parnsdone = 1;
7935
0
        }
7936
0
        for (ns = cur->nsDef; ns != NULL; ns = ns->next) {
7937
      /*
7938
      * NOTE: ns->prefix and ns->href are never in the dict.
7939
      */
7940
      /*
7941
      * Does it shadow any ns-decl?
7942
      */
7943
0
      if (XML_NSMAP_NOTEMPTY(nsMap)) {
7944
0
          XML_NSMAP_FOREACH(nsMap, mi) {
7945
0
        if ((mi->depth >= XML_TREE_NSMAP_PARENT) &&
7946
0
            (mi->shadowDepth == -1) &&
7947
0
            ((ns->prefix == mi->newNs->prefix) ||
7948
0
            xmlStrEqual(ns->prefix,
7949
0
            mi->newNs->prefix))) {
7950
7951
0
            mi->shadowDepth = depth;
7952
0
        }
7953
0
          }
7954
0
      }
7955
      /*
7956
      * Push mapping.
7957
      */
7958
0
      if (xmlDOMWrapNsMapAddItem(&nsMap, -1,
7959
0
          ns, ns, depth) == NULL)
7960
0
          ret = -1;
7961
0
        }
7962
0
    }
7963
                /* Falls through. */
7964
0
      case XML_ATTRIBUTE_NODE:
7965
    /* No namespace, no fun. */
7966
0
    if (cur->ns == NULL)
7967
0
        goto ns_end;
7968
7969
0
    if (! parnsdone) {
7970
0
        if (xmlDOMWrapNSNormGatherInScopeNs(&nsMap,
7971
0
      destParent) == -1)
7972
0
      ret = -1;
7973
0
        parnsdone = 1;
7974
0
    }
7975
    /*
7976
    * Adopt ns-references.
7977
    */
7978
0
    if (XML_NSMAP_NOTEMPTY(nsMap)) {
7979
        /*
7980
        * Search for a mapping.
7981
        */
7982
0
        XML_NSMAP_FOREACH(nsMap, mi) {
7983
0
      if ((mi->shadowDepth == -1) &&
7984
0
          (cur->ns == mi->oldNs)) {
7985
7986
0
          cur->ns = mi->newNs;
7987
0
          goto ns_end;
7988
0
      }
7989
0
        }
7990
0
    }
7991
    /*
7992
    * No matching namespace in scope. We need a new one.
7993
    */
7994
0
    if ((ctxt) && (ctxt->getNsForNodeFunc)) {
7995
        /*
7996
        * User-defined behaviour.
7997
        */
7998
0
        ns = ctxt->getNsForNodeFunc(ctxt, cur,
7999
0
      cur->ns->href, cur->ns->prefix);
8000
        /*
8001
        * Insert mapping if ns is available; it's the users fault
8002
        * if not.
8003
        */
8004
0
        if (xmlDOMWrapNsMapAddItem(&nsMap, -1,
8005
0
          cur->ns, ns, XML_TREE_NSMAP_CUSTOM) == NULL)
8006
0
      ret = -1;
8007
0
        cur->ns = ns;
8008
0
    } else {
8009
        /*
8010
        * Acquire a normalized ns-decl and add it to the map.
8011
        */
8012
0
        if (xmlDOMWrapNSNormAcquireNormalizedNs(destDoc,
8013
      /* ns-decls on curElem or on destDoc->oldNs */
8014
0
      destParent ? curElem : NULL,
8015
0
      cur->ns, &ns,
8016
0
      &nsMap, depth,
8017
0
      ancestorsOnly,
8018
      /* ns-decls must be prefixed for attributes. */
8019
0
      (cur->type == XML_ATTRIBUTE_NODE) ? 1 : 0) == -1)
8020
0
      ret = -1;
8021
0
        cur->ns = ns;
8022
0
    }
8023
8024
0
ns_end:
8025
0
    if (cur->type == XML_ELEMENT_NODE) {
8026
0
        cur->psvi = NULL;
8027
0
        cur->line = 0;
8028
0
        cur->extra = 0;
8029
        /*
8030
        * Walk attributes.
8031
        */
8032
0
        if (cur->properties != NULL) {
8033
      /*
8034
      * Process first attribute node.
8035
      */
8036
0
      cur = (xmlNodePtr) cur->properties;
8037
0
      continue;
8038
0
        }
8039
0
    }
8040
0
    break;
8041
0
      case XML_TEXT_NODE:
8042
0
      case XML_CDATA_SECTION_NODE:
8043
0
      case XML_PI_NODE:
8044
0
      case XML_COMMENT_NODE:
8045
0
      case XML_ENTITY_REF_NODE:
8046
0
    goto leave_node;
8047
0
      default:
8048
0
    ret = -1;
8049
0
  }
8050
  /*
8051
  * Walk the tree.
8052
  */
8053
0
  if (cur->children != NULL) {
8054
0
      cur = cur->children;
8055
0
      continue;
8056
0
  }
8057
8058
0
leave_node:
8059
0
  if (cur == node)
8060
0
      break;
8061
0
  if ((cur->type == XML_ELEMENT_NODE) ||
8062
0
      (cur->type == XML_XINCLUDE_START) ||
8063
0
      (cur->type == XML_XINCLUDE_END))
8064
0
  {
8065
      /*
8066
      * TODO: Do we expect nsDefs on XML_XINCLUDE_START?
8067
      */
8068
0
      if (XML_NSMAP_NOTEMPTY(nsMap)) {
8069
    /*
8070
    * Pop mappings.
8071
    */
8072
0
    while ((nsMap->last != NULL) &&
8073
0
        (nsMap->last->depth >= depth))
8074
0
    {
8075
0
        XML_NSMAP_POP(nsMap, mi)
8076
0
    }
8077
    /*
8078
    * Unshadow.
8079
    */
8080
0
    XML_NSMAP_FOREACH(nsMap, mi) {
8081
0
        if (mi->shadowDepth >= depth)
8082
0
      mi->shadowDepth = -1;
8083
0
    }
8084
0
      }
8085
0
      depth--;
8086
0
  }
8087
0
  if (cur->next != NULL)
8088
0
      cur = cur->next;
8089
0
  else if ((cur->type == XML_ATTRIBUTE_NODE) &&
8090
0
      (cur->parent->children != NULL))
8091
0
  {
8092
0
      cur = cur->parent->children;
8093
0
  } else {
8094
0
      cur = cur->parent;
8095
0
      goto leave_node;
8096
0
  }
8097
0
    }
8098
8099
    /*
8100
    * Cleanup.
8101
    */
8102
0
    if (nsMap != NULL) {
8103
0
  if ((ctxt) && (ctxt->namespaceMap == nsMap)) {
8104
      /*
8105
      * Just cleanup the map but don't free.
8106
      */
8107
0
      if (nsMap->first) {
8108
0
    if (nsMap->pool)
8109
0
        nsMap->last->next = nsMap->pool;
8110
0
    nsMap->pool = nsMap->first;
8111
0
    nsMap->first = NULL;
8112
0
      }
8113
0
  } else
8114
0
      xmlDOMWrapNsMapFree(nsMap);
8115
0
    }
8116
0
    return(ret);
8117
0
}
8118
8119
/**
8120
 * Clone a node and fix namespaces.
8121
 *
8122
 * References of out-of scope ns-decls are remapped to point to `destDoc`.
8123
 * If `destParent` is given, then nsDef entries on element-nodes are used.
8124
 * If *no* `destParent` is given, then `destDoc->oldNs` entries are used.
8125
 * This is the case when you don't know already where the cloned branch
8126
 * will be added to.
8127
 *
8128
 * If `destParent` is given, it ensures that the tree is namespace
8129
 * wellformed by creating additional ns-decls where needed.
8130
 * Note that, since prefixes of already existent ns-decls can be
8131
 * shadowed by this process, it could break QNames in attribute
8132
 * values or element content.
8133
 *
8134
 * @param ctxt  the optional context for custom processing
8135
 * @param sourceDoc  the optional sourceDoc
8136
 * @param node  the node to start with
8137
 * @param resNode  the clone of the given `node`
8138
 * @param destDoc  the destination doc
8139
 * @param destParent  the optional new parent of `node` in `destDoc`
8140
 * @param deep  descend into child if set
8141
 * @param options  option flags
8142
 * @returns 0 if the operation succeeded,
8143
 *          1 if a node of unsupported (or not yet supported) type was given,
8144
 *          -1 on API/internal errors.
8145
 */
8146
int
8147
xmlDOMWrapCloneNode(xmlDOMWrapCtxt *ctxt,
8148
          xmlDoc *sourceDoc,
8149
          xmlNode *node,
8150
          xmlNode **resNode,
8151
          xmlDoc *destDoc,
8152
          xmlNode *destParent,
8153
          int deep,
8154
          int options ATTRIBUTE_UNUSED)
8155
0
{
8156
0
    int ret = 0;
8157
0
    xmlNodePtr cur, cloneElem = NULL;
8158
0
    xmlNsMapPtr nsMap = NULL;
8159
0
    xmlNsMapItemPtr mi;
8160
0
    xmlNsPtr ns;
8161
0
    int depth = -1;
8162
    /* int adoptStr = 1; */
8163
    /* gather @parent's ns-decls. */
8164
0
    int parnsdone = 0;
8165
    /*
8166
    * @ancestorsOnly:
8167
    * TODO: @ancestorsOnly should be set per option.
8168
    *
8169
    */
8170
0
    int ancestorsOnly = 0;
8171
0
    xmlNodePtr resultClone = NULL, clone = NULL, parentClone = NULL, prevClone = NULL;
8172
0
    xmlNsPtr cloneNs = NULL, *cloneNsDefSlot = NULL;
8173
0
    xmlDictPtr dict; /* The destination dict */
8174
8175
0
    if ((node == NULL) || (resNode == NULL) || (destDoc == NULL) ||
8176
0
  ((destParent != NULL) && (destParent->doc != destDoc)))
8177
0
  return(-1);
8178
    /*
8179
    * TODO: Initially we support only element-nodes.
8180
    */
8181
0
    if (node->type != XML_ELEMENT_NODE)
8182
0
  return(1);
8183
    /*
8184
    * Check node->doc sanity.
8185
    */
8186
0
    if ((node->doc != NULL) && (sourceDoc != NULL) &&
8187
0
  (node->doc != sourceDoc)) {
8188
  /*
8189
  * Might be an XIncluded node.
8190
  */
8191
0
  return (-1);
8192
0
    }
8193
0
    if (sourceDoc == NULL)
8194
0
  sourceDoc = node->doc;
8195
0
    if (sourceDoc == NULL)
8196
0
        return (-1);
8197
8198
0
    dict = destDoc->dict;
8199
    /*
8200
    * Reuse the namespace map of the context.
8201
    */
8202
0
    if (ctxt)
8203
0
  nsMap = (xmlNsMapPtr) ctxt->namespaceMap;
8204
8205
0
    *resNode = NULL;
8206
8207
0
    cur = node;
8208
0
    while (cur != NULL) {
8209
0
  if (cur->doc != sourceDoc) {
8210
      /*
8211
      * We'll assume XIncluded nodes if the doc differs.
8212
      * TODO: Do we need to reconciliate XIncluded nodes?
8213
      * TODO: This here returns -1 in this case.
8214
      */
8215
0
      goto internal_error;
8216
0
  }
8217
  /*
8218
  * Create a new node.
8219
  */
8220
0
  switch (cur->type) {
8221
0
      case XML_XINCLUDE_START:
8222
0
      case XML_XINCLUDE_END:
8223
    /*
8224
    * TODO: What to do with XInclude?
8225
    */
8226
0
    goto internal_error;
8227
0
    break;
8228
0
      case XML_ELEMENT_NODE:
8229
0
      case XML_TEXT_NODE:
8230
0
      case XML_CDATA_SECTION_NODE:
8231
0
      case XML_COMMENT_NODE:
8232
0
      case XML_PI_NODE:
8233
0
      case XML_DOCUMENT_FRAG_NODE:
8234
0
      case XML_ENTITY_REF_NODE:
8235
    /*
8236
    * Nodes of xmlNode structure.
8237
    */
8238
0
    clone = (xmlNodePtr) xmlMalloc(sizeof(xmlNode));
8239
0
    if (clone == NULL)
8240
0
        goto internal_error;
8241
0
    memset(clone, 0, sizeof(xmlNode));
8242
    /*
8243
    * Set hierarchical links.
8244
    */
8245
0
    if (resultClone != NULL) {
8246
0
        clone->parent = parentClone;
8247
0
        if (prevClone) {
8248
0
      prevClone->next = clone;
8249
0
      clone->prev = prevClone;
8250
0
        } else
8251
0
      parentClone->children = clone;
8252
0
                    parentClone->last = clone;
8253
0
    } else
8254
0
        resultClone = clone;
8255
8256
0
    break;
8257
0
      case XML_ATTRIBUTE_NODE:
8258
    /*
8259
    * Attributes (xmlAttr).
8260
    */
8261
0
    clone = xmlMalloc(sizeof(xmlAttr));
8262
0
    if (clone == NULL)
8263
0
        goto internal_error;
8264
0
    memset(clone, 0, sizeof(xmlAttr));
8265
    /*
8266
    * Set hierarchical links.
8267
    * TODO: Change this to add to the end of attributes.
8268
    */
8269
0
    if (resultClone != NULL) {
8270
0
        clone->parent = parentClone;
8271
0
        if (prevClone) {
8272
0
      prevClone->next = clone;
8273
0
      clone->prev = prevClone;
8274
0
        } else
8275
0
      parentClone->properties = (xmlAttrPtr) clone;
8276
0
    } else
8277
0
        resultClone = clone;
8278
0
    break;
8279
0
      default:
8280
    /*
8281
    * TODO QUESTION: Any other nodes expected?
8282
    */
8283
0
    goto internal_error;
8284
0
  }
8285
8286
0
  clone->type = cur->type;
8287
0
  clone->doc = destDoc;
8288
8289
  /*
8290
  * Clone the name of the node if any.
8291
  */
8292
0
  if (cur->name == xmlStringText)
8293
0
      clone->name = xmlStringText;
8294
0
  else if (cur->name == xmlStringTextNoenc)
8295
      /*
8296
      * NOTE: Although xmlStringTextNoenc is never assigned to a node
8297
      *   in tree.c, it might be set in Libxslt via
8298
      *   "xsl:disable-output-escaping".
8299
      */
8300
0
      clone->name = xmlStringTextNoenc;
8301
0
  else if (cur->name == xmlStringComment)
8302
0
      clone->name = xmlStringComment;
8303
0
  else if (cur->name != NULL) {
8304
0
            if (dict != NULL)
8305
0
                clone->name = xmlDictLookup(dict, cur->name, -1);
8306
0
            else
8307
0
                clone->name = xmlStrdup(cur->name);
8308
0
            if (clone->name == NULL)
8309
0
                goto internal_error;
8310
0
  }
8311
8312
0
  switch (cur->type) {
8313
0
      case XML_XINCLUDE_START:
8314
0
      case XML_XINCLUDE_END:
8315
    /*
8316
    * TODO
8317
    */
8318
0
    return (-1);
8319
0
      case XML_ELEMENT_NODE:
8320
0
    cloneElem = clone;
8321
0
    depth++;
8322
    /*
8323
    * Namespace declarations.
8324
    */
8325
0
    if (cur->nsDef != NULL) {
8326
0
        if (! parnsdone) {
8327
0
      if (destParent && (ctxt == NULL)) {
8328
          /*
8329
          * Gather @parent's in-scope ns-decls.
8330
          */
8331
0
          if (xmlDOMWrapNSNormGatherInScopeNs(&nsMap,
8332
0
        destParent) == -1)
8333
0
        goto internal_error;
8334
0
      }
8335
0
      parnsdone = 1;
8336
0
        }
8337
        /*
8338
        * Clone namespace declarations.
8339
        */
8340
0
        cloneNsDefSlot = &(clone->nsDef);
8341
0
        for (ns = cur->nsDef; ns != NULL; ns = ns->next) {
8342
      /*
8343
      * Create a new xmlNs.
8344
      */
8345
0
      cloneNs = (xmlNsPtr) xmlMalloc(sizeof(xmlNs));
8346
0
      if (cloneNs == NULL)
8347
0
          goto internal_error;
8348
0
      memset(cloneNs, 0, sizeof(xmlNs));
8349
0
      cloneNs->type = XML_LOCAL_NAMESPACE;
8350
8351
0
      if (ns->href != NULL) {
8352
0
          cloneNs->href = xmlStrdup(ns->href);
8353
0
                            if (cloneNs->href == NULL) {
8354
0
                                xmlFreeNs(cloneNs);
8355
0
                                goto internal_error;
8356
0
                            }
8357
0
                        }
8358
0
      if (ns->prefix != NULL) {
8359
0
          cloneNs->prefix = xmlStrdup(ns->prefix);
8360
0
                            if (cloneNs->prefix == NULL) {
8361
0
                                xmlFreeNs(cloneNs);
8362
0
                                goto internal_error;
8363
0
                            }
8364
0
                        }
8365
8366
0
      *cloneNsDefSlot = cloneNs;
8367
0
      cloneNsDefSlot = &(cloneNs->next);
8368
8369
      /*
8370
      * Note that for custom handling of ns-references,
8371
      * the ns-decls need not be stored in the ns-map,
8372
      * since they won't be referenced by node->ns.
8373
      */
8374
0
      if ((ctxt == NULL) ||
8375
0
          (ctxt->getNsForNodeFunc == NULL))
8376
0
      {
8377
          /*
8378
          * Does it shadow any ns-decl?
8379
          */
8380
0
          if (XML_NSMAP_NOTEMPTY(nsMap)) {
8381
0
        XML_NSMAP_FOREACH(nsMap, mi) {
8382
0
            if ((mi->depth >= XML_TREE_NSMAP_PARENT) &&
8383
0
          (mi->shadowDepth == -1) &&
8384
0
          ((ns->prefix == mi->newNs->prefix) ||
8385
0
          xmlStrEqual(ns->prefix,
8386
0
          mi->newNs->prefix))) {
8387
          /*
8388
          * Mark as shadowed at the current
8389
          * depth.
8390
          */
8391
0
          mi->shadowDepth = depth;
8392
0
            }
8393
0
        }
8394
0
          }
8395
          /*
8396
          * Push mapping.
8397
          */
8398
0
          if (xmlDOMWrapNsMapAddItem(&nsMap, -1,
8399
0
        ns, cloneNs, depth) == NULL)
8400
0
        goto internal_error;
8401
0
      }
8402
0
        }
8403
0
    }
8404
    /* cur->ns will be processed further down. */
8405
0
    break;
8406
0
      case XML_ATTRIBUTE_NODE:
8407
    /* IDs will be processed further down. */
8408
    /* cur->ns will be processed further down. */
8409
0
    break;
8410
0
      case XML_PI_NODE:
8411
0
      case XML_COMMENT_NODE:
8412
0
      case XML_TEXT_NODE:
8413
0
      case XML_CDATA_SECTION_NODE:
8414
    /*
8415
    * Note that this will also cover the values of attributes.
8416
    */
8417
0
                if (cur->content != NULL) {
8418
0
                    clone->content = xmlStrdup(cur->content);
8419
0
                    if (clone->content == NULL)
8420
0
                        goto internal_error;
8421
0
                }
8422
0
    goto leave_node;
8423
0
      case XML_ENTITY_REF_NODE:
8424
0
    if (sourceDoc != destDoc) {
8425
0
        if ((destDoc->intSubset) || (destDoc->extSubset)) {
8426
0
      xmlEntityPtr ent;
8427
      /*
8428
      * Different doc: Assign new entity-node if available.
8429
      */
8430
0
      ent = xmlGetDocEntity(destDoc, cur->name);
8431
0
      if (ent != NULL) {
8432
0
          clone->content = ent->content;
8433
0
          clone->children = (xmlNodePtr) ent;
8434
0
          clone->last = (xmlNodePtr) ent;
8435
0
      }
8436
0
        }
8437
0
    } else {
8438
        /*
8439
        * Same doc: Use the current node's entity declaration
8440
        * and value.
8441
        */
8442
0
        clone->content = cur->content;
8443
0
        clone->children = cur->children;
8444
0
        clone->last = cur->last;
8445
0
    }
8446
0
    goto leave_node;
8447
0
      default:
8448
0
    goto internal_error;
8449
0
  }
8450
8451
0
  if (cur->ns == NULL)
8452
0
      goto end_ns_reference;
8453
8454
/* handle_ns_reference: */
8455
  /*
8456
  ** The following will take care of references to ns-decls ********
8457
  ** and is intended only for element- and attribute-nodes.
8458
  **
8459
  */
8460
0
  if (! parnsdone) {
8461
0
      if (destParent && (ctxt == NULL)) {
8462
0
    if (xmlDOMWrapNSNormGatherInScopeNs(&nsMap, destParent) == -1)
8463
0
        goto internal_error;
8464
0
      }
8465
0
      parnsdone = 1;
8466
0
  }
8467
  /*
8468
  * Adopt ns-references.
8469
  */
8470
0
  if (XML_NSMAP_NOTEMPTY(nsMap)) {
8471
      /*
8472
      * Search for a mapping.
8473
      */
8474
0
      XML_NSMAP_FOREACH(nsMap, mi) {
8475
0
    if ((mi->shadowDepth == -1) &&
8476
0
        (cur->ns == mi->oldNs)) {
8477
        /*
8478
        * This is the nice case: a mapping was found.
8479
        */
8480
0
        clone->ns = mi->newNs;
8481
0
        goto end_ns_reference;
8482
0
    }
8483
0
      }
8484
0
  }
8485
  /*
8486
  * No matching namespace in scope. We need a new one.
8487
  */
8488
0
  if ((ctxt != NULL) && (ctxt->getNsForNodeFunc != NULL)) {
8489
      /*
8490
      * User-defined behaviour.
8491
      */
8492
0
      ns = ctxt->getNsForNodeFunc(ctxt, cur,
8493
0
    cur->ns->href, cur->ns->prefix);
8494
      /*
8495
      * Add user's mapping.
8496
      */
8497
0
      if (xmlDOMWrapNsMapAddItem(&nsMap, -1,
8498
0
    cur->ns, ns, XML_TREE_NSMAP_CUSTOM) == NULL)
8499
0
    goto internal_error;
8500
0
      clone->ns = ns;
8501
0
  } else {
8502
      /*
8503
      * Acquire a normalized ns-decl and add it to the map.
8504
      */
8505
0
      if (xmlDOMWrapNSNormAcquireNormalizedNs(destDoc,
8506
    /* ns-decls on cloneElem or on destDoc->oldNs */
8507
0
    destParent ? cloneElem : NULL,
8508
0
    cur->ns, &ns,
8509
0
    &nsMap, depth,
8510
    /* if we need to search only in the ancestor-axis */
8511
0
    ancestorsOnly,
8512
    /* ns-decls must be prefixed for attributes. */
8513
0
    (cur->type == XML_ATTRIBUTE_NODE) ? 1 : 0) == -1)
8514
0
    goto internal_error;
8515
0
      clone->ns = ns;
8516
0
  }
8517
8518
0
end_ns_reference:
8519
8520
  /*
8521
  * Some post-processing.
8522
  *
8523
  * Handle ID attributes.
8524
  */
8525
0
  if ((clone->type == XML_ATTRIBUTE_NODE) &&
8526
0
      (clone->parent != NULL))
8527
0
  {
8528
0
            int res;
8529
8530
0
      res = xmlIsID(destDoc, clone->parent, (xmlAttrPtr) clone);
8531
0
            if (res < 0)
8532
0
                goto internal_error;
8533
0
            if (res == 1) {
8534
0
    xmlChar *idVal;
8535
8536
0
    idVal = xmlNodeGetContent(cur);
8537
0
                if (idVal == NULL)
8538
0
                    goto internal_error;
8539
0
                if (xmlAddIDSafe((xmlAttrPtr) cur, idVal) < 0) {
8540
0
                    xmlFree(idVal);
8541
0
                    goto internal_error;
8542
0
                }
8543
0
                xmlFree(idVal);
8544
0
      }
8545
0
  }
8546
  /*
8547
  **
8548
  ** The following will traverse the tree **************************
8549
  **
8550
  *
8551
  * Walk the element's attributes before descending into child-nodes.
8552
  */
8553
0
  if ((cur->type == XML_ELEMENT_NODE) && (cur->properties != NULL)) {
8554
0
      prevClone = NULL;
8555
0
      parentClone = clone;
8556
0
      cur = (xmlNodePtr) cur->properties;
8557
0
      continue;
8558
0
  }
8559
0
into_content:
8560
  /*
8561
  * Descend into child-nodes.
8562
  */
8563
0
  if (cur->children != NULL) {
8564
0
      if (deep || (cur->type == XML_ATTRIBUTE_NODE)) {
8565
0
    prevClone = NULL;
8566
0
    parentClone = clone;
8567
0
    cur = cur->children;
8568
0
    continue;
8569
0
      }
8570
0
  }
8571
8572
0
leave_node:
8573
  /*
8574
  * At this point we are done with the node, its content
8575
  * and an element-nodes's attribute-nodes.
8576
  */
8577
0
  if (cur == node)
8578
0
      break;
8579
0
  if ((cur->type == XML_ELEMENT_NODE) ||
8580
0
      (cur->type == XML_XINCLUDE_START) ||
8581
0
      (cur->type == XML_XINCLUDE_END)) {
8582
      /*
8583
      * TODO: Do we expect nsDefs on XML_XINCLUDE_START?
8584
      */
8585
0
      if (XML_NSMAP_NOTEMPTY(nsMap)) {
8586
    /*
8587
    * Pop mappings.
8588
    */
8589
0
    while ((nsMap->last != NULL) &&
8590
0
        (nsMap->last->depth >= depth))
8591
0
    {
8592
0
        XML_NSMAP_POP(nsMap, mi)
8593
0
    }
8594
    /*
8595
    * Unshadow.
8596
    */
8597
0
    XML_NSMAP_FOREACH(nsMap, mi) {
8598
0
        if (mi->shadowDepth >= depth)
8599
0
      mi->shadowDepth = -1;
8600
0
    }
8601
0
      }
8602
0
      depth--;
8603
0
  }
8604
0
  if (cur->next != NULL) {
8605
0
      prevClone = clone;
8606
0
      cur = cur->next;
8607
0
  } else if (cur->type != XML_ATTRIBUTE_NODE) {
8608
0
      clone = clone->parent;
8609
0
      if (clone != NULL)
8610
0
    parentClone = clone->parent;
8611
      /*
8612
      * Process parent --> next;
8613
      */
8614
0
      cur = cur->parent;
8615
0
      goto leave_node;
8616
0
  } else {
8617
      /* This is for attributes only. */
8618
0
      clone = clone->parent;
8619
0
      parentClone = clone->parent;
8620
      /*
8621
      * Process parent-element --> children.
8622
      */
8623
0
      cur = cur->parent;
8624
0
      goto into_content;
8625
0
  }
8626
0
    }
8627
0
    goto exit;
8628
8629
0
internal_error:
8630
0
    ret = -1;
8631
8632
0
exit:
8633
    /*
8634
    * Cleanup.
8635
    */
8636
0
    if (nsMap != NULL) {
8637
0
  if ((ctxt) && (ctxt->namespaceMap == nsMap)) {
8638
      /*
8639
      * Just cleanup the map but don't free.
8640
      */
8641
0
      if (nsMap->first) {
8642
0
    if (nsMap->pool)
8643
0
        nsMap->last->next = nsMap->pool;
8644
0
    nsMap->pool = nsMap->first;
8645
0
    nsMap->first = NULL;
8646
0
      }
8647
0
  } else
8648
0
      xmlDOMWrapNsMapFree(nsMap);
8649
0
    }
8650
    /*
8651
    * TODO: Should we try a cleanup of the cloned node in case of a
8652
    * fatal error?
8653
    */
8654
0
    *resNode = resultClone;
8655
0
    return (ret);
8656
0
}
8657
8658
/**
8659
 * `attr` is adopted by `destDoc`.
8660
 * Ensures that ns-references point to `destDoc`: either to
8661
 * `elements->nsDef` entries if `destParent` is given, or to
8662
 * `destDoc->oldNs` otherwise.
8663
 *
8664
 * @param ctxt  the optional context for custom processing
8665
 * @param sourceDoc  unused
8666
 * @param attr  the attribute-node to be adopted
8667
 * @param destDoc  the destination doc for adoption
8668
 * @param destParent  the optional new parent of `attr` in `destDoc`
8669
 * @param options  option flags (unused)
8670
 * @returns 0 if succeeded, -1 otherwise and on API/internal errors.
8671
 */
8672
static int
8673
xmlDOMWrapAdoptAttr(xmlDOMWrapCtxtPtr ctxt,
8674
        xmlDocPtr sourceDoc ATTRIBUTE_UNUSED,
8675
        xmlAttrPtr attr,
8676
        xmlDocPtr destDoc,
8677
        xmlNodePtr destParent,
8678
        int options ATTRIBUTE_UNUSED)
8679
0
{
8680
0
    int ret = 0;
8681
8682
0
    if ((attr == NULL) || (destDoc == NULL))
8683
0
  return (-1);
8684
8685
0
    if (attr->doc != destDoc) {
8686
0
        if (xmlSetTreeDoc((xmlNodePtr) attr, destDoc) < 0)
8687
0
            ret = -1;
8688
0
    }
8689
8690
0
    if (attr->ns != NULL) {
8691
0
  xmlNsPtr ns = NULL;
8692
8693
0
  if (ctxt != NULL) {
8694
      /* TODO: User defined. */
8695
0
  }
8696
  /* XML Namespace. */
8697
0
  if (IS_STR_XML(attr->ns->prefix)) {
8698
0
      ns = xmlTreeEnsureXMLDecl(destDoc);
8699
0
  } else if (destParent == NULL) {
8700
      /*
8701
      * Store in @destDoc->oldNs.
8702
      */
8703
0
      ns = xmlDOMWrapStoreNs(destDoc, attr->ns->href, attr->ns->prefix);
8704
0
  } else {
8705
      /*
8706
      * Declare on @destParent.
8707
      */
8708
0
      if (xmlSearchNsByNamespaceStrict(destDoc, destParent, attr->ns->href,
8709
0
    &ns, 1) == -1)
8710
0
    ret = -1;
8711
0
      if (ns == NULL) {
8712
0
    ns = xmlDOMWrapNSNormDeclareNsForced(destDoc, destParent,
8713
0
        attr->ns->href, attr->ns->prefix, 1);
8714
0
      }
8715
0
  }
8716
0
  if (ns == NULL)
8717
0
      ret = -1;
8718
0
  attr->ns = ns;
8719
0
    }
8720
8721
0
    return (ret);
8722
0
}
8723
8724
/**
8725
 * Fix up namespaces before moving a node.
8726
 *
8727
 * References of out-of scope ns-decls are remapped to point to `destDoc`:
8728
 * If `destParent` is given, then nsDef entries on element-nodes are used.
8729
 * If *no* `destParent` is given, then `destDoc->oldNs` entries are used
8730
 * This is the case when you have an unlinked node and just want to move it
8731
 * to the context of.
8732
 *
8733
 * If `destParent` is given, it ensures that the tree is namespace
8734
 * wellformed by creating additional ns-decls where needed.
8735
 * Note that, since prefixes of already existent ns-decls can be
8736
 * shadowed by this process, it could break QNames in attribute
8737
 * values or element content.
8738
 *
8739
 * NOTE: This function was not intensively tested.
8740
 *
8741
 * @param ctxt  the optional context for custom processing
8742
 * @param sourceDoc  the optional sourceDoc
8743
 * @param node  the node to start with
8744
 * @param destDoc  the destination doc
8745
 * @param destParent  the optional new parent of `node` in `destDoc`
8746
 * @param options  option flags
8747
 * @returns 0 if the operation succeeded,
8748
 *          1 if a node of unsupported type was given,
8749
 *          2 if a node of not yet supported type was given and
8750
 *          -1 on API/internal errors.
8751
 */
8752
int
8753
xmlDOMWrapAdoptNode(xmlDOMWrapCtxt *ctxt,
8754
        xmlDoc *sourceDoc,
8755
        xmlNode *node,
8756
        xmlDoc *destDoc,
8757
        xmlNode *destParent,
8758
        int options)
8759
0
{
8760
0
    int ret = 0;
8761
8762
0
    if ((node == NULL) || (node->type == XML_NAMESPACE_DECL) ||
8763
0
        (destDoc == NULL) ||
8764
0
  ((destParent != NULL) && (destParent->doc != destDoc)))
8765
0
  return(-1);
8766
    /*
8767
    * Check node->doc sanity.
8768
    */
8769
0
    if (sourceDoc == NULL) {
8770
0
        sourceDoc = node->doc;
8771
0
    } else if (node->doc != sourceDoc) {
8772
0
  return (-1);
8773
0
    }
8774
8775
    /*
8776
     * TODO: Shouldn't this be allowed?
8777
     */
8778
0
    if (sourceDoc == destDoc)
8779
0
  return (-1);
8780
8781
0
    switch (node->type) {
8782
0
  case XML_ELEMENT_NODE:
8783
0
  case XML_ATTRIBUTE_NODE:
8784
0
  case XML_TEXT_NODE:
8785
0
  case XML_CDATA_SECTION_NODE:
8786
0
  case XML_ENTITY_REF_NODE:
8787
0
  case XML_PI_NODE:
8788
0
  case XML_COMMENT_NODE:
8789
0
      break;
8790
0
  case XML_DOCUMENT_FRAG_NODE:
8791
      /* TODO: Support document-fragment-nodes. */
8792
0
      return (2);
8793
0
  default:
8794
0
      return (1);
8795
0
    }
8796
    /*
8797
    * Unlink only if @node was not already added to @destParent.
8798
    */
8799
0
    if ((node->parent != NULL) && (destParent != node->parent))
8800
0
  xmlUnlinkNodeInternal(node);
8801
8802
0
    if (node->type == XML_ELEMENT_NODE) {
8803
0
      return (xmlDOMWrapAdoptBranch(ctxt, sourceDoc, node,
8804
0
        destDoc, destParent, options));
8805
0
    } else if (node->type == XML_ATTRIBUTE_NODE) {
8806
0
      return (xmlDOMWrapAdoptAttr(ctxt, sourceDoc,
8807
0
    (xmlAttrPtr) node, destDoc, destParent, options));
8808
0
    } else {
8809
0
        if (node->doc != destDoc) {
8810
0
            if (xmlNodeSetDoc(node, destDoc) < 0)
8811
0
                ret = -1;
8812
0
        }
8813
0
    }
8814
0
    return (ret);
8815
0
}
8816
8817
/************************************************************************
8818
 *                  *
8819
 *      XHTML detection         *
8820
 *                  *
8821
 ************************************************************************/
8822
8823
0
#define XHTML_STRICT_PUBLIC_ID BAD_CAST \
8824
0
   "-//W3C//DTD XHTML 1.0 Strict//EN"
8825
0
#define XHTML_STRICT_SYSTEM_ID BAD_CAST \
8826
0
   "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"
8827
0
#define XHTML_FRAME_PUBLIC_ID BAD_CAST \
8828
0
   "-//W3C//DTD XHTML 1.0 Frameset//EN"
8829
0
#define XHTML_FRAME_SYSTEM_ID BAD_CAST \
8830
0
   "http://www.w3.org/TR/xhtml1/DTD/xhtml1-frameset.dtd"
8831
0
#define XHTML_TRANS_PUBLIC_ID BAD_CAST \
8832
0
   "-//W3C//DTD XHTML 1.0 Transitional//EN"
8833
0
#define XHTML_TRANS_SYSTEM_ID BAD_CAST \
8834
0
   "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"
8835
8836
/**
8837
 * Try to find if the document correspond to an XHTML DTD
8838
 *
8839
 * @param systemID  the system identifier
8840
 * @param publicID  the public identifier
8841
 * @returns 1 if true, 0 if not and -1 in case of error
8842
 */
8843
int
8844
0
xmlIsXHTML(const xmlChar *systemID, const xmlChar *publicID) {
8845
0
    if ((systemID == NULL) && (publicID == NULL))
8846
0
  return(-1);
8847
0
    if (publicID != NULL) {
8848
0
  if (xmlStrEqual(publicID, XHTML_STRICT_PUBLIC_ID)) return(1);
8849
0
  if (xmlStrEqual(publicID, XHTML_FRAME_PUBLIC_ID)) return(1);
8850
0
  if (xmlStrEqual(publicID, XHTML_TRANS_PUBLIC_ID)) return(1);
8851
0
    }
8852
0
    if (systemID != NULL) {
8853
0
  if (xmlStrEqual(systemID, XHTML_STRICT_SYSTEM_ID)) return(1);
8854
0
  if (xmlStrEqual(systemID, XHTML_FRAME_SYSTEM_ID)) return(1);
8855
0
  if (xmlStrEqual(systemID, XHTML_TRANS_SYSTEM_ID)) return(1);
8856
0
    }
8857
0
    return(0);
8858
0
}
8859
8860
/************************************************************************
8861
 *                  *
8862
 *      Node callbacks          *
8863
 *                  *
8864
 ************************************************************************/
8865
8866
/**
8867
 * Registers a callback for node creation
8868
 *
8869
 * @deprecated don't use
8870
 *
8871
 * @param func  function pointer to the new RegisterNodeFunc
8872
 * @returns the old value of the registration function
8873
 */
8874
xmlRegisterNodeFunc
8875
xmlRegisterNodeDefault(xmlRegisterNodeFunc func)
8876
0
{
8877
0
    xmlRegisterNodeFunc old = xmlRegisterNodeDefaultValue;
8878
8879
0
    xmlRegisterCallbacks = 1;
8880
0
    xmlRegisterNodeDefaultValue = func;
8881
0
    return(old);
8882
0
}
8883
8884
/**
8885
 * Registers a callback for node destruction
8886
 *
8887
 * @deprecated don't use
8888
 *
8889
 * @param func  function pointer to the new DeregisterNodeFunc
8890
 * @returns the previous value of the deregistration function
8891
 */
8892
xmlDeregisterNodeFunc
8893
xmlDeregisterNodeDefault(xmlDeregisterNodeFunc func)
8894
0
{
8895
0
    xmlDeregisterNodeFunc old = xmlDeregisterNodeDefaultValue;
8896
8897
0
    xmlRegisterCallbacks = 1;
8898
0
    xmlDeregisterNodeDefaultValue = func;
8899
0
    return(old);
8900
0
}
8901