2001-02-23 17:55:21 +00:00
/*
2001-12-17 15:30:10 +00:00
* tree . c : implementation of access function for an XML tree .
2001-02-23 17:55:21 +00:00
*
2002-11-21 14:10:52 +00:00
* References :
* XHTML 1.0 W3C REC : http : //www.w3.org/TR/2002/REC-xhtml1-20020801/
*
2001-02-23 17:55:21 +00:00
* See Copyright for the status of this software .
*
2001-06-24 12:13:24 +00:00
* daniel @ veillard . com
2001-02-23 17:55:21 +00:00
*
*/
2002-03-18 19:37:11 +00:00
# define IN_LIBXML
2001-04-21 16:57:29 +00:00
# include "libxml.h"
2001-02-23 17:55:21 +00:00
# include <string.h> /* for memset() only ! */
# ifdef HAVE_CTYPE_H
# include <ctype.h>
# endif
# ifdef HAVE_STDLIB_H
# include <stdlib.h>
# endif
# ifdef HAVE_ZLIB_H
# include <zlib.h>
# endif
# include <libxml/xmlmemory.h>
# include <libxml/tree.h>
# include <libxml/parser.h>
2001-07-09 16:01:19 +00:00
# include <libxml/uri.h>
2001-02-23 17:55:21 +00:00
# include <libxml/entities.h>
# include <libxml/valid.h>
# include <libxml/xmlerror.h>
2001-04-11 11:28:06 +00:00
# include <libxml/parserInternals.h>
2001-10-17 15:58:35 +00:00
# include <libxml/globals.h>
2002-11-21 14:10:52 +00:00
# ifdef LIBXML_HTML_ENABLED
# include <libxml/HTMLtree.h>
# endif
2001-02-23 17:55:21 +00:00
2001-03-24 17:00:36 +00:00
xmlNsPtr xmlNewReconciliedNs ( xmlDocPtr doc , xmlNodePtr tree , xmlNsPtr ns ) ;
/************************************************************************
* *
* A few static variables and macros *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-13 09:15:48 +00:00
/* #undef xmlStringText */
2001-07-16 00:06:07 +00:00
const xmlChar xmlStringText [ ] = { ' t ' , ' e ' , ' x ' , ' t ' , 0 } ;
2001-10-13 09:15:48 +00:00
/* #undef xmlStringTextNoenc */
2001-07-16 00:06:07 +00:00
const xmlChar xmlStringTextNoenc [ ] =
2001-02-23 17:55:21 +00:00
{ ' t ' , ' e ' , ' x ' , ' t ' , ' n ' , ' o ' , ' e ' , ' n ' , ' c ' , 0 } ;
2001-10-13 09:15:48 +00:00
/* #undef xmlStringComment */
2001-07-16 00:06:07 +00:00
const xmlChar xmlStringComment [ ] = { ' c ' , ' o ' , ' m ' , ' m ' , ' e ' , ' n ' , ' t ' , 0 } ;
2001-02-23 17:55:21 +00:00
static int xmlCompressMode = 0 ;
static int xmlCheckDTD = 1 ;
# define UPDATE_LAST_CHILD_AND_PARENT(n) if ((n) != NULL) { \
xmlNodePtr ulccur = ( n ) - > children ; \
if ( ulccur = = NULL ) { \
( n ) - > last = NULL ; \
} else { \
while ( ulccur - > next ! = NULL ) { \
ulccur - > parent = ( n ) ; \
ulccur = ulccur - > next ; \
} \
ulccur - > parent = ( n ) ; \
( n ) - > last = ulccur ; \
} }
/* #define DEBUG_BUFFER */
/* #define DEBUG_TREE */
2002-01-26 21:42:58 +00:00
/************************************************************************
* *
* Functions to move to entities . c once the *
* API freeze is smoothen and they can be made public . *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# include <libxml/hash.h>
/**
* xmlGetEntityFromDtd :
* @ dtd : A pointer to the DTD to search
* @ name : The entity name
*
* Do an entity lookup in the DTD entity hash table and
* return the corresponding entity , if found .
*
* Returns A pointer to the entity structure or NULL if not found .
*/
static xmlEntityPtr
xmlGetEntityFromDtd ( xmlDtdPtr dtd , const xmlChar * name ) {
xmlEntitiesTablePtr table ;
if ( ( dtd ! = NULL ) & & ( dtd - > entities ! = NULL ) ) {
table = ( xmlEntitiesTablePtr ) dtd - > entities ;
return ( ( xmlEntityPtr ) xmlHashLookup ( table , name ) ) ;
/* return(xmlGetEntityFromTable(table, name)); */
}
return ( NULL ) ;
}
/**
* xmlGetParameterEntityFromDtd :
* @ dtd : A pointer to the DTD to search
* @ name : The entity name
*
* Do an entity lookup in the DTD pararmeter entity hash table and
* return the corresponding entity , if found .
*
* Returns A pointer to the entity structure or NULL if not found .
*/
static xmlEntityPtr
xmlGetParameterEntityFromDtd ( xmlDtdPtr dtd , const xmlChar * name ) {
xmlEntitiesTablePtr table ;
if ( ( dtd ! = NULL ) & & ( dtd - > pentities ! = NULL ) ) {
table = ( xmlEntitiesTablePtr ) dtd - > pentities ;
return ( ( xmlEntityPtr ) xmlHashLookup ( table , name ) ) ;
/* return(xmlGetEntityFromTable(table, name)); */
}
return ( NULL ) ;
}
2003-02-14 16:54:11 +00:00
/************************************************************************
* *
* Check Name , NCName and QName strings *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define CUR_SCHAR(s, l) xmlStringCurrentChar(NULL, s, &l)
/**
* xmlValidateNCName :
* @ value : the value to check
* @ space : allow spaces in front and end of the string
*
* Check that a value conforms to the lexical space of NCName
*
* Returns 0 if this validates , a positive error code number otherwise
* and - 1 in case of internal or API error .
*/
int
xmlValidateNCName ( const xmlChar * value , int space ) {
const xmlChar * cur = value ;
int c , l ;
/*
* First quick algorithm for ASCII range
*/
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | | ( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( * cur = = ' _ ' ) )
cur + + ;
else
goto try_complex ;
while ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( ( * cur > = ' 0 ' ) & & ( * cur < = ' 9 ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' - ' ) | | ( * cur = = ' . ' ) )
cur + + ;
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( * cur = = 0 )
return ( 0 ) ;
try_complex :
/*
* Second check for chars outside the ASCII range
*/
cur = value ;
c = CUR_SCHAR ( cur , l ) ;
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( ( ! xmlIsLetter ( c ) ) & & ( c ! = ' _ ' ) )
return ( 1 ) ;
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
while ( xmlIsLetter ( c ) | | xmlIsDigit ( c ) | | ( c = = ' . ' ) | |
( c = = ' - ' ) | | ( c = = ' _ ' ) | | xmlIsCombining ( c ) | |
xmlIsExtender ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( c ! = 0 )
return ( 1 ) ;
return ( 0 ) ;
}
/**
* xmlValidateQName :
* @ value : the value to check
* @ space : allow spaces in front and end of the string
*
* Check that a value conforms to the lexical space of QName
*
* Returns 0 if this validates , a positive error code number otherwise
* and - 1 in case of internal or API error .
*/
int
xmlValidateQName ( const xmlChar * value , int space ) {
const xmlChar * cur = value ;
int c , l ;
/*
* First quick algorithm for ASCII range
*/
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | | ( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( * cur = = ' _ ' ) )
cur + + ;
else
goto try_complex ;
while ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( ( * cur > = ' 0 ' ) & & ( * cur < = ' 9 ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' - ' ) | | ( * cur = = ' . ' ) )
cur + + ;
if ( * cur = = ' : ' ) {
cur + + ;
if ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( * cur = = ' _ ' ) )
cur + + ;
else
goto try_complex ;
while ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( ( * cur > = ' 0 ' ) & & ( * cur < = ' 9 ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' - ' ) | | ( * cur = = ' . ' ) )
cur + + ;
}
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( * cur = = 0 )
return ( 0 ) ;
try_complex :
/*
* Second check for chars outside the ASCII range
*/
cur = value ;
c = CUR_SCHAR ( cur , l ) ;
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( ( ! xmlIsLetter ( c ) ) & & ( c ! = ' _ ' ) )
return ( 1 ) ;
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
while ( xmlIsLetter ( c ) | | xmlIsDigit ( c ) | | ( c = = ' . ' ) | |
( c = = ' - ' ) | | ( c = = ' _ ' ) | | xmlIsCombining ( c ) | |
xmlIsExtender ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
if ( c = = ' : ' ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
if ( ( ! xmlIsLetter ( c ) ) & & ( c ! = ' _ ' ) )
return ( 1 ) ;
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
while ( xmlIsLetter ( c ) | | xmlIsDigit ( c ) | | ( c = = ' . ' ) | |
( c = = ' - ' ) | | ( c = = ' _ ' ) | | xmlIsCombining ( c ) | |
xmlIsExtender ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( c ! = 0 )
return ( 1 ) ;
return ( 0 ) ;
}
/**
* xmlValidateName :
* @ value : the value to check
* @ space : allow spaces in front and end of the string
*
* Check that a value conforms to the lexical space of Name
*
* Returns 0 if this validates , a positive error code number otherwise
* and - 1 in case of internal or API error .
*/
int
xmlValidateName ( const xmlChar * value , int space ) {
const xmlChar * cur = value ;
int c , l ;
/*
* First quick algorithm for ASCII range
*/
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | | ( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' : ' ) )
cur + + ;
else
goto try_complex ;
while ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( ( * cur > = ' 0 ' ) & & ( * cur < = ' 9 ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' - ' ) | | ( * cur = = ' . ' ) | | ( * cur = = ' : ' ) )
cur + + ;
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( * cur = = 0 )
return ( 0 ) ;
try_complex :
/*
* Second check for chars outside the ASCII range
*/
cur = value ;
c = CUR_SCHAR ( cur , l ) ;
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( ( ! xmlIsLetter ( c ) ) & & ( c ! = ' _ ' ) & & ( c ! = ' : ' ) )
return ( 1 ) ;
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
2003-02-18 21:12:46 +00:00
while ( xmlIsLetter ( c ) | | xmlIsDigit ( c ) | | ( c = = ' . ' ) | | ( c = = ' : ' ) | |
( c = = ' - ' ) | | ( c = = ' _ ' ) | | xmlIsCombining ( c ) | | xmlIsExtender ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( c ! = 0 )
return ( 1 ) ;
return ( 0 ) ;
}
/**
* xmlValidateNMToken :
* @ value : the value to check
* @ space : allow spaces in front and end of the string
*
* Check that a value conforms to the lexical space of NMToken
*
* Returns 0 if this validates , a positive error code number otherwise
* and - 1 in case of internal or API error .
*/
int
xmlValidateNMToken ( const xmlChar * value , int space ) {
const xmlChar * cur = value ;
int c , l ;
/*
* First quick algorithm for ASCII range
*/
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( ( * cur > = ' 0 ' ) & & ( * cur < = ' 9 ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' - ' ) | | ( * cur = = ' . ' ) | | ( * cur = = ' : ' ) )
cur + + ;
else
goto try_complex ;
while ( ( ( * cur > = ' a ' ) & & ( * cur < = ' z ' ) ) | |
( ( * cur > = ' A ' ) & & ( * cur < = ' Z ' ) ) | |
( ( * cur > = ' 0 ' ) & & ( * cur < = ' 9 ' ) ) | |
( * cur = = ' _ ' ) | | ( * cur = = ' - ' ) | | ( * cur = = ' . ' ) | | ( * cur = = ' : ' ) )
cur + + ;
if ( space )
while ( IS_BLANK ( * cur ) ) cur + + ;
if ( * cur = = 0 )
return ( 0 ) ;
try_complex :
/*
* Second check for chars outside the ASCII range
*/
cur = value ;
c = CUR_SCHAR ( cur , l ) ;
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( ! ( xmlIsLetter ( c ) | | xmlIsDigit ( c ) | | ( c = = ' . ' ) | | ( c = = ' : ' ) | |
( c = = ' - ' ) | | ( c = = ' _ ' ) | | xmlIsCombining ( c ) | | xmlIsExtender ( c ) ) )
return ( 1 ) ;
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
2003-02-14 16:54:11 +00:00
while ( xmlIsLetter ( c ) | | xmlIsDigit ( c ) | | ( c = = ' . ' ) | | ( c = = ' : ' ) | |
( c = = ' - ' ) | | ( c = = ' _ ' ) | | xmlIsCombining ( c ) | | xmlIsExtender ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
if ( space ) {
while ( IS_BLANK ( c ) ) {
cur + = l ;
c = CUR_SCHAR ( cur , l ) ;
}
}
if ( c ! = 0 )
return ( 1 ) ;
return ( 0 ) ;
}
2001-02-23 17:55:21 +00:00
/************************************************************************
* *
* Allocation and deallocation of basic structures *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/**
* xmlSetBufferAllocationScheme :
* @ scheme : allocation method to use
*
* Set the buffer allocation method . Types are
* XML_BUFFER_ALLOC_EXACT - use exact sizes , keeps memory usage down
* XML_BUFFER_ALLOC_DOUBLEIT - double buffer when extra needed ,
* improves performance
*/
void
xmlSetBufferAllocationScheme ( xmlBufferAllocationScheme scheme ) {
xmlBufferAllocScheme = scheme ;
}
/**
* xmlGetBufferAllocationScheme :
*
* Types are
* XML_BUFFER_ALLOC_EXACT - use exact sizes , keeps memory usage down
* XML_BUFFER_ALLOC_DOUBLEIT - double buffer when extra needed ,
* improves performance
*
* Returns the current allocation scheme
*/
xmlBufferAllocationScheme
2001-03-24 17:00:36 +00:00
xmlGetBufferAllocationScheme ( void ) {
2001-04-16 14:08:07 +00:00
return ( xmlBufferAllocScheme ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNewNs :
* @ node : the element carrying the namespace
* @ href : the URI associated
* @ prefix : the prefix for the namespace
*
* Creation of a new Namespace . This function will refuse to create
* a namespace with a similar prefix than an existing one present on this
* node .
* We use href = = NULL in the case of an element creation where the namespace
* was not defined .
2001-12-17 15:30:10 +00:00
* Returns a new namespace pointer or NULL
2001-02-23 17:55:21 +00:00
*/
xmlNsPtr
xmlNewNs ( xmlNodePtr node , const xmlChar * href , const xmlChar * prefix ) {
xmlNsPtr cur ;
if ( ( node ! = NULL ) & & ( node - > type ! = XML_ELEMENT_NODE ) )
return ( NULL ) ;
2001-10-05 09:18:14 +00:00
if ( ( prefix ! = NULL ) & & ( xmlStrEqual ( prefix , BAD_CAST " xml " ) ) )
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
/*
* Allocate a new Namespace and fill the fields .
*/
cur = ( xmlNsPtr ) xmlMalloc ( sizeof ( xmlNs ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNs : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNs ) ) ;
cur - > type = XML_LOCAL_NAMESPACE ;
if ( href ! = NULL )
cur - > href = xmlStrdup ( href ) ;
if ( prefix ! = NULL )
cur - > prefix = xmlStrdup ( prefix ) ;
/*
* Add it at the end to preserve parsing order . . .
* and checks for existing use of the prefix
*/
if ( node ! = NULL ) {
if ( node - > nsDef = = NULL ) {
node - > nsDef = cur ;
} else {
xmlNsPtr prev = node - > nsDef ;
if ( ( ( prev - > prefix = = NULL ) & & ( cur - > prefix = = NULL ) ) | |
( xmlStrEqual ( prev - > prefix , cur - > prefix ) ) ) {
xmlFreeNs ( cur ) ;
return ( NULL ) ;
}
while ( prev - > next ! = NULL ) {
prev = prev - > next ;
if ( ( ( prev - > prefix = = NULL ) & & ( cur - > prefix = = NULL ) ) | |
( xmlStrEqual ( prev - > prefix , cur - > prefix ) ) ) {
xmlFreeNs ( cur ) ;
return ( NULL ) ;
}
}
prev - > next = cur ;
}
}
return ( cur ) ;
}
/**
* xmlSetNs :
* @ node : a node in the document
* @ ns : a namespace pointer
*
* Associate a namespace to a node , a posteriori .
*/
void
xmlSetNs ( xmlNodePtr node , xmlNsPtr ns ) {
if ( node = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlSetNs: node == NULL \n " ) ;
# endif
return ;
}
node - > ns = ns ;
}
/**
* xmlFreeNs :
* @ cur : the namespace pointer
*
* Free up the structures associated to a namespace
*/
void
xmlFreeNs ( xmlNsPtr cur ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeNs : ns == NULL \n " ) ;
# endif
return ;
}
if ( cur - > href ! = NULL ) xmlFree ( ( char * ) cur - > href ) ;
if ( cur - > prefix ! = NULL ) xmlFree ( ( char * ) cur - > prefix ) ;
xmlFree ( cur ) ;
}
/**
* xmlFreeNsList :
* @ cur : the first namespace pointer
*
* Free up all the structures associated to the chained namespaces .
*/
void
xmlFreeNsList ( xmlNsPtr cur ) {
xmlNsPtr next ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeNsList : ns == NULL \n " ) ;
# endif
return ;
}
while ( cur ! = NULL ) {
next = cur - > next ;
xmlFreeNs ( cur ) ;
cur = next ;
}
}
/**
* xmlNewDtd :
* @ doc : the document pointer
* @ name : the DTD name
* @ ExternalID : the external ID
* @ SystemID : the system ID
*
* Creation of a new DTD for the external subset . To create an
* internal subset , use xmlCreateIntSubset ( ) .
*
* Returns a pointer to the new DTD structure
*/
xmlDtdPtr
xmlNewDtd ( xmlDocPtr doc , const xmlChar * name ,
const xmlChar * ExternalID , const xmlChar * SystemID ) {
xmlDtdPtr cur ;
if ( ( doc ! = NULL ) & & ( doc - > extSubset ! = NULL ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewDtd(%s): document %s already have a DTD %s \n " ,
/* !!! */ ( char * ) name , doc - > name ,
/* !!! */ ( char * ) doc - > extSubset - > name ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new DTD and fill the fields .
*/
cur = ( xmlDtdPtr ) xmlMalloc ( sizeof ( xmlDtd ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewDtd : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlDtd ) ) ;
cur - > type = XML_DTD_NODE ;
if ( name ! = NULL )
cur - > name = xmlStrdup ( name ) ;
if ( ExternalID ! = NULL )
cur - > ExternalID = xmlStrdup ( ExternalID ) ;
if ( SystemID ! = NULL )
cur - > SystemID = xmlStrdup ( SystemID ) ;
if ( doc ! = NULL )
doc - > extSubset = cur ;
cur - > doc = doc ;
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlGetIntSubset :
* @ doc : the document pointer
*
* Get the internal subset of a document
* Returns a pointer to the DTD structure or NULL if not found
*/
xmlDtdPtr
xmlGetIntSubset ( xmlDocPtr doc ) {
xmlNodePtr cur ;
if ( doc = = NULL )
return ( NULL ) ;
cur = doc - > children ;
while ( cur ! = NULL ) {
if ( cur - > type = = XML_DTD_NODE )
return ( ( xmlDtdPtr ) cur ) ;
cur = cur - > next ;
}
return ( ( xmlDtdPtr ) doc - > intSubset ) ;
}
/**
* xmlCreateIntSubset :
* @ doc : the document pointer
* @ name : the DTD name
2001-03-10 12:32:04 +00:00
* @ ExternalID : the external ( PUBLIC ) ID
2001-02-23 17:55:21 +00:00
* @ SystemID : the system ID
*
* Create the internal subset of a document
* Returns a pointer to the new DTD structure
*/
xmlDtdPtr
xmlCreateIntSubset ( xmlDocPtr doc , const xmlChar * name ,
const xmlChar * ExternalID , const xmlChar * SystemID ) {
xmlDtdPtr cur ;
if ( ( doc ! = NULL ) & & ( xmlGetIntSubset ( doc ) ! = NULL ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlCreateIntSubset(): document %s already have an internal subset \n " ,
doc - > name ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new DTD and fill the fields .
*/
cur = ( xmlDtdPtr ) xmlMalloc ( sizeof ( xmlDtd ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlCreateIntSubset : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlDtd ) ) ;
cur - > type = XML_DTD_NODE ;
if ( name ! = NULL )
cur - > name = xmlStrdup ( name ) ;
if ( ExternalID ! = NULL )
cur - > ExternalID = xmlStrdup ( ExternalID ) ;
if ( SystemID ! = NULL )
cur - > SystemID = xmlStrdup ( SystemID ) ;
if ( doc ! = NULL ) {
doc - > intSubset = cur ;
cur - > parent = doc ;
cur - > doc = doc ;
if ( doc - > children = = NULL ) {
doc - > children = ( xmlNodePtr ) cur ;
doc - > last = ( xmlNodePtr ) cur ;
} else {
if ( doc - > type = = XML_HTML_DOCUMENT_NODE ) {
2001-03-10 12:32:04 +00:00
xmlNodePtr prev ;
2001-02-23 17:55:21 +00:00
prev = doc - > children ;
prev - > prev = ( xmlNodePtr ) cur ;
cur - > next = prev ;
doc - > children = ( xmlNodePtr ) cur ;
} else {
2001-03-10 12:32:04 +00:00
xmlNodePtr next ;
next = doc - > children ;
while ( ( next ! = NULL ) & & ( next - > type ! = XML_ELEMENT_NODE ) )
next = next - > next ;
if ( next = = NULL ) {
cur - > prev = doc - > last ;
cur - > prev - > next = ( xmlNodePtr ) cur ;
cur - > next = NULL ;
doc - > last = ( xmlNodePtr ) cur ;
} else {
cur - > next = next ;
cur - > prev = next - > prev ;
if ( cur - > prev = = NULL )
doc - > children = ( xmlNodePtr ) cur ;
else
cur - > prev - > next = ( xmlNodePtr ) cur ;
next - > prev = ( xmlNodePtr ) cur ;
}
2001-02-23 17:55:21 +00:00
}
}
}
2003-01-05 22:37:17 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlFreeDtd :
* @ cur : the DTD structure to free up
*
* Free a DTD structure .
*/
void
xmlFreeDtd ( xmlDtdPtr cur ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeDtd : DTD == NULL \n " ) ;
# endif
return ;
}
2003-01-01 20:59:38 +00:00
if ( xmlDeregisterNodeDefaultValue )
xmlDeregisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
if ( cur - > children ! = NULL ) {
xmlNodePtr next , c = cur - > children ;
/*
2001-12-31 16:16:02 +00:00
* Cleanup all the DTD comments they are not in the DTD
2001-02-23 17:55:21 +00:00
* indexes .
*/
while ( c ! = NULL ) {
next = c - > next ;
if ( c - > type = = XML_COMMENT_NODE ) {
xmlUnlinkNode ( c ) ;
xmlFreeNode ( c ) ;
}
c = next ;
}
}
if ( cur - > name ! = NULL ) xmlFree ( ( char * ) cur - > name ) ;
if ( cur - > SystemID ! = NULL ) xmlFree ( ( char * ) cur - > SystemID ) ;
if ( cur - > ExternalID ! = NULL ) xmlFree ( ( char * ) cur - > ExternalID ) ;
/* TODO !!! */
if ( cur - > notations ! = NULL )
xmlFreeNotationTable ( ( xmlNotationTablePtr ) cur - > notations ) ;
if ( cur - > elements ! = NULL )
xmlFreeElementTable ( ( xmlElementTablePtr ) cur - > elements ) ;
if ( cur - > attributes ! = NULL )
xmlFreeAttributeTable ( ( xmlAttributeTablePtr ) cur - > attributes ) ;
if ( cur - > entities ! = NULL )
xmlFreeEntitiesTable ( ( xmlEntitiesTablePtr ) cur - > entities ) ;
if ( cur - > pentities ! = NULL )
xmlFreeEntitiesTable ( ( xmlEntitiesTablePtr ) cur - > pentities ) ;
xmlFree ( cur ) ;
}
/**
* xmlNewDoc :
* @ version : xmlChar string giving the version of XML " 1.0 "
*
2001-07-18 19:30:27 +00:00
* Creates a new XML document
*
2001-02-23 17:55:21 +00:00
* Returns a new document
*/
xmlDocPtr
xmlNewDoc ( const xmlChar * version ) {
xmlDocPtr cur ;
if ( version = = NULL )
version = ( const xmlChar * ) " 1.0 " ;
/*
* Allocate a new document and fill the fields .
*/
cur = ( xmlDocPtr ) xmlMalloc ( sizeof ( xmlDoc ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewDoc : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlDoc ) ) ;
cur - > type = XML_DOCUMENT_NODE ;
cur - > version = xmlStrdup ( version ) ;
cur - > standalone = - 1 ;
cur - > compression = - 1 ; /* not initialized */
cur - > doc = cur ;
2001-04-11 07:50:02 +00:00
cur - > charset = XML_CHAR_ENCODING_UTF8 ;
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlFreeDoc :
* @ cur : pointer to the document
*
* Free up all the structures used by a document , tree included .
*/
void
xmlFreeDoc ( xmlDocPtr cur ) {
2001-06-19 11:07:54 +00:00
xmlDtdPtr extSubset , intSubset ;
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeDoc : document == NULL \n " ) ;
# endif
return ;
}
2003-01-01 20:59:38 +00:00
if ( xmlDeregisterNodeDefaultValue )
xmlDeregisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-05-16 21:05:17 +00:00
/*
* Do this before freeing the children list to avoid ID lookups
*/
if ( cur - > ids ! = NULL ) xmlFreeIDTable ( ( xmlIDTablePtr ) cur - > ids ) ;
cur - > ids = NULL ;
if ( cur - > refs ! = NULL ) xmlFreeRefTable ( ( xmlRefTablePtr ) cur - > refs ) ;
cur - > refs = NULL ;
2001-06-19 11:07:54 +00:00
extSubset = cur - > extSubset ;
intSubset = cur - > intSubset ;
2002-03-18 18:36:20 +00:00
if ( intSubset = = extSubset )
extSubset = NULL ;
2001-06-19 11:07:54 +00:00
if ( extSubset ! = NULL ) {
2001-05-16 21:05:17 +00:00
xmlUnlinkNode ( ( xmlNodePtr ) cur - > extSubset ) ;
cur - > extSubset = NULL ;
2001-06-19 11:07:54 +00:00
xmlFreeDtd ( extSubset ) ;
2001-05-16 21:05:17 +00:00
}
2001-06-19 11:07:54 +00:00
if ( intSubset ! = NULL ) {
2001-05-16 21:05:17 +00:00
xmlUnlinkNode ( ( xmlNodePtr ) cur - > intSubset ) ;
cur - > intSubset = NULL ;
2001-06-19 11:07:54 +00:00
xmlFreeDtd ( intSubset ) ;
2001-05-16 21:05:17 +00:00
}
if ( cur - > children ! = NULL ) xmlFreeNodeList ( cur - > children ) ;
2001-02-23 17:55:21 +00:00
if ( cur - > version ! = NULL ) xmlFree ( ( char * ) cur - > version ) ;
if ( cur - > name ! = NULL ) xmlFree ( ( char * ) cur - > name ) ;
if ( cur - > encoding ! = NULL ) xmlFree ( ( char * ) cur - > encoding ) ;
if ( cur - > oldNs ! = NULL ) xmlFreeNsList ( cur - > oldNs ) ;
if ( cur - > URL ! = NULL ) xmlFree ( ( char * ) cur - > URL ) ;
xmlFree ( cur ) ;
}
/**
* xmlStringLenGetNodeList :
* @ doc : the document
* @ value : the value of the text
* @ len : the length of the string value
*
* Parse the value string and build the node list associated . Should
* produce a flat tree with only TEXTs and ENTITY_REFs .
* Returns a pointer to the first child
*/
xmlNodePtr
xmlStringLenGetNodeList ( xmlDocPtr doc , const xmlChar * value , int len ) {
xmlNodePtr ret = NULL , last = NULL ;
xmlNodePtr node ;
xmlChar * val ;
const xmlChar * cur = value ;
const xmlChar * q ;
xmlEntityPtr ent ;
if ( value = = NULL ) return ( NULL ) ;
q = cur ;
while ( ( * cur ! = 0 ) & & ( cur - value < len ) ) {
if ( * cur = = ' & ' ) {
/*
* Save the current text .
*/
if ( cur ! = q ) {
if ( ( last ! = NULL ) & & ( last - > type = = XML_TEXT_NODE ) ) {
xmlNodeAddContentLen ( last , q , cur - q ) ;
} else {
node = xmlNewDocTextLen ( doc , q , cur - q ) ;
if ( node = = NULL ) return ( ret ) ;
if ( last = = NULL )
last = ret = node ;
else {
last - > next = node ;
node - > prev = last ;
last = node ;
}
}
}
/*
* Read the entity string
*/
cur + + ;
q = cur ;
while ( ( * cur ! = 0 ) & & ( cur - value < len ) & & ( * cur ! = ' ; ' ) ) cur + + ;
if ( ( * cur = = 0 ) | | ( cur - value > = len ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlStringLenGetNodeList: unterminated entity %30s \n " , q ) ;
# endif
return ( ret ) ;
}
if ( cur ! = q ) {
/*
* Predefined entities don ' t generate nodes
*/
val = xmlStrndup ( q , cur - q ) ;
ent = xmlGetDocEntity ( doc , val ) ;
if ( ( ent ! = NULL ) & &
( ent - > etype = = XML_INTERNAL_PREDEFINED_ENTITY ) ) {
if ( last = = NULL ) {
node = xmlNewDocText ( doc , ent - > content ) ;
last = ret = node ;
} else
xmlNodeAddContent ( last , ent - > content ) ;
} else {
/*
* Create a new REFERENCE_REF node
*/
node = xmlNewReference ( doc , val ) ;
if ( node = = NULL ) {
if ( val ! = NULL ) xmlFree ( val ) ;
return ( ret ) ;
}
2002-04-18 16:39:10 +00:00
else if ( ( ent ! = NULL ) & & ( ent - > children = = NULL ) ) {
xmlNodePtr tmp ;
ent - > children =
xmlStringGetNodeList ( doc , ( const xmlChar * ) node - > content ) ;
tmp = ent - > children ;
while ( tmp ) {
tmp - > parent = ( xmlNodePtr ) ent ;
tmp = tmp - > next ;
}
}
2001-02-23 17:55:21 +00:00
if ( last = = NULL )
last = ret = node ;
else {
last - > next = node ;
node - > prev = last ;
last = node ;
}
}
xmlFree ( val ) ;
}
cur + + ;
q = cur ;
} else
cur + + ;
}
if ( cur ! = q ) {
/*
* Handle the last piece of text .
*/
if ( ( last ! = NULL ) & & ( last - > type = = XML_TEXT_NODE ) ) {
xmlNodeAddContentLen ( last , q , cur - q ) ;
} else {
node = xmlNewDocTextLen ( doc , q , cur - q ) ;
if ( node = = NULL ) return ( ret ) ;
if ( last = = NULL )
last = ret = node ;
else {
last - > next = node ;
node - > prev = last ;
last = node ;
}
}
}
return ( ret ) ;
}
/**
* xmlStringGetNodeList :
* @ doc : the document
* @ value : the value of the attribute
*
* Parse the value string and build the node list associated . Should
* produce a flat tree with only TEXTs and ENTITY_REFs .
* Returns a pointer to the first child
*/
xmlNodePtr
xmlStringGetNodeList ( xmlDocPtr doc , const xmlChar * value ) {
xmlNodePtr ret = NULL , last = NULL ;
xmlNodePtr node ;
xmlChar * val ;
const xmlChar * cur = value ;
const xmlChar * q ;
xmlEntityPtr ent ;
if ( value = = NULL ) return ( NULL ) ;
q = cur ;
while ( * cur ! = 0 ) {
2001-04-11 11:28:06 +00:00
if ( cur [ 0 ] = = ' & ' ) {
int charval = 0 ;
xmlChar tmp ;
2001-02-23 17:55:21 +00:00
/*
* Save the current text .
*/
if ( cur ! = q ) {
if ( ( last ! = NULL ) & & ( last - > type = = XML_TEXT_NODE ) ) {
xmlNodeAddContentLen ( last , q , cur - q ) ;
} else {
node = xmlNewDocTextLen ( doc , q , cur - q ) ;
if ( node = = NULL ) return ( ret ) ;
if ( last = = NULL )
last = ret = node ;
else {
last - > next = node ;
node - > prev = last ;
last = node ;
}
}
}
q = cur ;
2001-04-11 11:28:06 +00:00
if ( ( cur [ 1 ] = = ' # ' ) & & ( cur [ 2 ] = = ' x ' ) ) {
cur + = 3 ;
tmp = * cur ;
while ( tmp ! = ' ; ' ) { /* Non input consuming loop */
if ( ( tmp > = ' 0 ' ) & & ( tmp < = ' 9 ' ) )
charval = charval * 16 + ( tmp - ' 0 ' ) ;
else if ( ( tmp > = ' a ' ) & & ( tmp < = ' f ' ) )
charval = charval * 16 + ( tmp - ' a ' ) + 10 ;
else if ( ( tmp > = ' A ' ) & & ( tmp < = ' F ' ) )
charval = charval * 16 + ( tmp - ' A ' ) + 10 ;
else {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlStringGetNodeList: invalid hexadecimal charvalue \n " ) ;
2001-04-11 11:28:06 +00:00
charval = 0 ;
break ;
}
cur + + ;
tmp = * cur ;
}
if ( tmp = = ' ; ' )
cur + + ;
q = cur ;
} else if ( cur [ 1 ] = = ' # ' ) {
cur + = 2 ;
tmp = * cur ;
while ( tmp ! = ' ; ' ) { /* Non input consuming loops */
if ( ( tmp > = ' 0 ' ) & & ( tmp < = ' 9 ' ) )
charval = charval * 10 + ( tmp - ' 0 ' ) ;
else {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlStringGetNodeList: invalid decimal charvalue \n " ) ;
2001-04-11 11:28:06 +00:00
charval = 0 ;
break ;
}
cur + + ;
tmp = * cur ;
}
if ( tmp = = ' ; ' )
cur + + ;
q = cur ;
} else {
2001-02-23 17:55:21 +00:00
/*
2001-04-11 11:28:06 +00:00
* Read the entity string
2001-02-23 17:55:21 +00:00
*/
2001-04-11 11:28:06 +00:00
cur + + ;
q = cur ;
while ( ( * cur ! = 0 ) & & ( * cur ! = ' ; ' ) ) cur + + ;
if ( * cur = = 0 ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlStringGetNodeList: unterminated entity %30s \n " , q ) ;
# endif
return ( ret ) ;
}
if ( cur ! = q ) {
2001-02-23 17:55:21 +00:00
/*
2001-04-11 11:28:06 +00:00
* Predefined entities don ' t generate nodes
2001-02-23 17:55:21 +00:00
*/
2001-04-11 11:28:06 +00:00
val = xmlStrndup ( q , cur - q ) ;
ent = xmlGetDocEntity ( doc , val ) ;
if ( ( ent ! = NULL ) & &
( ent - > etype = = XML_INTERNAL_PREDEFINED_ENTITY ) ) {
if ( last = = NULL ) {
node = xmlNewDocText ( doc , ent - > content ) ;
last = ret = node ;
2002-01-06 23:05:13 +00:00
} else if ( last - > type ! = XML_TEXT_NODE ) {
node = xmlNewDocText ( doc , ent - > content ) ;
last = xmlAddNextSibling ( last , node ) ;
2001-04-11 11:28:06 +00:00
} else
xmlNodeAddContent ( last , ent - > content ) ;
} else {
/*
* Create a new REFERENCE_REF node
*/
node = xmlNewReference ( doc , val ) ;
if ( node = = NULL ) {
if ( val ! = NULL ) xmlFree ( val ) ;
return ( ret ) ;
}
2002-04-18 16:39:10 +00:00
else if ( ( ent ! = NULL ) & & ( ent - > children = = NULL ) ) {
xmlNodePtr temp ;
ent - > children = xmlStringGetNodeList ( doc ,
( const xmlChar * ) node - > content ) ;
2002-12-30 00:01:08 +00:00
ent - > owner = 1 ;
2002-04-18 16:39:10 +00:00
temp = ent - > children ;
while ( temp ) {
temp - > parent = ( xmlNodePtr ) ent ;
temp = temp - > next ;
}
}
2001-04-11 11:28:06 +00:00
if ( last = = NULL ) {
last = ret = node ;
} else {
last = xmlAddNextSibling ( last , node ) ;
}
2001-02-23 17:55:21 +00:00
}
2001-04-11 11:28:06 +00:00
xmlFree ( val ) ;
}
cur + + ;
q = cur ;
}
if ( charval ! = 0 ) {
xmlChar buf [ 10 ] ;
int len ;
len = xmlCopyCharMultiByte ( buf , charval ) ;
buf [ len ] = 0 ;
node = xmlNewDocText ( doc , buf ) ;
if ( node ! = NULL ) {
if ( last = = NULL ) {
2001-02-23 17:55:21 +00:00
last = ret = node ;
2001-04-11 11:28:06 +00:00
} else {
last = xmlAddNextSibling ( last , node ) ;
2001-02-23 17:55:21 +00:00
}
}
2001-04-11 11:28:06 +00:00
charval = 0 ;
2001-02-23 17:55:21 +00:00
}
2001-04-11 11:28:06 +00:00
} else
2001-02-23 17:55:21 +00:00
cur + + ;
}
2001-05-11 17:41:21 +00:00
if ( ( cur ! = q ) | | ( ret = = NULL ) ) {
2001-02-23 17:55:21 +00:00
/*
* Handle the last piece of text .
*/
if ( ( last ! = NULL ) & & ( last - > type = = XML_TEXT_NODE ) ) {
xmlNodeAddContentLen ( last , q , cur - q ) ;
} else {
node = xmlNewDocTextLen ( doc , q , cur - q ) ;
if ( node = = NULL ) return ( ret ) ;
2001-04-11 11:28:06 +00:00
if ( last = = NULL ) {
2001-02-23 17:55:21 +00:00
last = ret = node ;
2001-04-11 11:28:06 +00:00
} else {
last = xmlAddNextSibling ( last , node ) ;
2001-02-23 17:55:21 +00:00
}
}
}
return ( ret ) ;
}
/**
* xmlNodeListGetString :
* @ doc : the document
* @ list : a Node list
* @ inLine : should we replace entity contents or show their external form
*
2002-12-11 14:23:49 +00:00
* Build the string equivalent to the text contained in the Node list
2001-02-23 17:55:21 +00:00
* made of TEXTs and ENTITY_REFs
2002-12-11 14:23:49 +00:00
*
2002-09-25 22:25:35 +00:00
* Returns a pointer to the string copy , the caller must free it with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
2002-04-20 06:41:40 +00:00
xmlNodeListGetString ( xmlDocPtr doc , xmlNodePtr list , int inLine )
{
2001-02-23 17:55:21 +00:00
xmlNodePtr node = list ;
xmlChar * ret = NULL ;
xmlEntityPtr ent ;
2002-04-20 06:41:40 +00:00
if ( list = = NULL )
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
while ( node ! = NULL ) {
if ( ( node - > type = = XML_TEXT_NODE ) | |
2002-04-20 06:41:40 +00:00
( node - > type = = XML_CDATA_SECTION_NODE ) ) {
if ( inLine ) {
ret = xmlStrcat ( ret , node - > content ) ;
} else {
xmlChar * buffer ;
2001-02-23 17:55:21 +00:00
2002-04-20 06:41:40 +00:00
buffer = xmlEncodeEntitiesReentrant ( doc , node - > content ) ;
if ( buffer ! = NULL ) {
ret = xmlStrcat ( ret , buffer ) ;
xmlFree ( buffer ) ;
}
2001-02-23 17:55:21 +00:00
}
2002-04-20 06:41:40 +00:00
} else if ( node - > type = = XML_ENTITY_REF_NODE ) {
if ( inLine ) {
ent = xmlGetDocEntity ( doc , node - > name ) ;
if ( ent ! = NULL ) {
xmlChar * buffer ;
/* an entity content can be any "well balanced chunk",
* i . e . the result of the content [ 43 ] production :
* http : //www.w3.org/TR/REC-xml#NT-content.
* So it can contain text , CDATA section or nested
* entity reference nodes ( among others ) .
* - > we recursive call xmlNodeListGetString ( )
* which handles these types */
buffer = xmlNodeListGetString ( doc , ent - > children , 1 ) ;
if ( buffer ! = NULL ) {
ret = xmlStrcat ( ret , buffer ) ;
xmlFree ( buffer ) ;
}
} else {
ret = xmlStrcat ( ret , node - > content ) ;
}
2001-02-23 17:55:21 +00:00
} else {
2002-04-20 06:41:40 +00:00
xmlChar buf [ 2 ] ;
buf [ 0 ] = ' & ' ;
buf [ 1 ] = 0 ;
ret = xmlStrncat ( ret , buf , 1 ) ;
ret = xmlStrcat ( ret , node - > name ) ;
buf [ 0 ] = ' ; ' ;
buf [ 1 ] = 0 ;
ret = xmlStrncat ( ret , buf , 1 ) ;
}
}
2001-02-23 17:55:21 +00:00
#if 0
2002-04-20 06:41:40 +00:00
else {
xmlGenericError ( xmlGenericErrorContext ,
" xmlGetNodeListString : invalid node type %d \n " ,
node - > type ) ;
}
2001-02-23 17:55:21 +00:00
# endif
2002-04-20 06:41:40 +00:00
node = node - > next ;
2001-02-23 17:55:21 +00:00
}
2002-04-20 06:41:40 +00:00
return ( ret ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNodeListGetRawString :
* @ doc : the document
* @ list : a Node list
* @ inLine : should we replace entity contents or show their external form
*
2002-12-11 14:23:49 +00:00
* Builds the string equivalent to the text contained in the Node list
2001-02-23 17:55:21 +00:00
* made of TEXTs and ENTITY_REFs , contrary to xmlNodeListGetString ( )
* this function doesn ' t do any character encoding handling .
*
2002-09-25 22:25:35 +00:00
* Returns a pointer to the string copy , the caller must free it with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
2002-04-20 06:41:40 +00:00
xmlNodeListGetRawString ( xmlDocPtr doc , xmlNodePtr list , int inLine )
{
2001-02-23 17:55:21 +00:00
xmlNodePtr node = list ;
xmlChar * ret = NULL ;
xmlEntityPtr ent ;
2002-04-20 06:41:40 +00:00
if ( list = = NULL )
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
while ( node ! = NULL ) {
2001-07-12 01:20:08 +00:00
if ( ( node - > type = = XML_TEXT_NODE ) | |
2002-04-20 06:41:40 +00:00
( node - > type = = XML_CDATA_SECTION_NODE ) ) {
if ( inLine ) {
ret = xmlStrcat ( ret , node - > content ) ;
} else {
xmlChar * buffer ;
2001-02-23 17:55:21 +00:00
2002-04-20 06:41:40 +00:00
buffer = xmlEncodeSpecialChars ( doc , node - > content ) ;
if ( buffer ! = NULL ) {
ret = xmlStrcat ( ret , buffer ) ;
xmlFree ( buffer ) ;
}
2001-02-23 17:55:21 +00:00
}
2002-04-20 06:41:40 +00:00
} else if ( node - > type = = XML_ENTITY_REF_NODE ) {
if ( inLine ) {
ent = xmlGetDocEntity ( doc , node - > name ) ;
if ( ent ! = NULL ) {
xmlChar * buffer ;
/* an entity content can be any "well balanced chunk",
* i . e . the result of the content [ 43 ] production :
* http : //www.w3.org/TR/REC-xml#NT-content.
* So it can contain text , CDATA section or nested
* entity reference nodes ( among others ) .
* - > we recursive call xmlNodeListGetRawString ( )
* which handles these types */
buffer =
xmlNodeListGetRawString ( doc , ent - > children , 1 ) ;
if ( buffer ! = NULL ) {
ret = xmlStrcat ( ret , buffer ) ;
xmlFree ( buffer ) ;
}
} else {
ret = xmlStrcat ( ret , node - > content ) ;
}
2001-02-23 17:55:21 +00:00
} else {
2002-04-20 06:41:40 +00:00
xmlChar buf [ 2 ] ;
buf [ 0 ] = ' & ' ;
buf [ 1 ] = 0 ;
ret = xmlStrncat ( ret , buf , 1 ) ;
ret = xmlStrcat ( ret , node - > name ) ;
buf [ 0 ] = ' ; ' ;
buf [ 1 ] = 0 ;
ret = xmlStrncat ( ret , buf , 1 ) ;
}
}
2001-02-23 17:55:21 +00:00
#if 0
2002-04-20 06:41:40 +00:00
else {
xmlGenericError ( xmlGenericErrorContext ,
" xmlGetNodeListString : invalid node type %d \n " ,
node - > type ) ;
}
2001-02-23 17:55:21 +00:00
# endif
2002-04-20 06:41:40 +00:00
node = node - > next ;
2001-02-23 17:55:21 +00:00
}
2002-04-20 06:41:40 +00:00
return ( ret ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNewProp :
* @ node : the holding node
* @ name : the name of the attribute
* @ value : the value of the attribute
*
* Create a new property carried by a node .
* Returns a pointer to the attribute
*/
xmlAttrPtr
xmlNewProp ( xmlNodePtr node , const xmlChar * name , const xmlChar * value ) {
xmlAttrPtr cur ;
xmlDocPtr doc = NULL ;
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewProp : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new property and fill the fields .
*/
cur = ( xmlAttrPtr ) xmlMalloc ( sizeof ( xmlAttr ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewProp : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlAttr ) ) ;
cur - > type = XML_ATTRIBUTE_NODE ;
cur - > parent = node ;
if ( node ! = NULL ) {
doc = node - > doc ;
cur - > doc = doc ;
}
cur - > name = xmlStrdup ( name ) ;
if ( value ! = NULL ) {
xmlChar * buffer ;
xmlNodePtr tmp ;
buffer = xmlEncodeEntitiesReentrant ( doc , value ) ;
cur - > children = xmlStringGetNodeList ( doc , buffer ) ;
cur - > last = NULL ;
tmp = cur - > children ;
while ( tmp ! = NULL ) {
tmp - > parent = ( xmlNodePtr ) cur ;
tmp - > doc = doc ;
if ( tmp - > next = = NULL )
cur - > last = tmp ;
tmp = tmp - > next ;
}
xmlFree ( buffer ) ;
}
/*
* Add it at the end to preserve parsing order . . .
*/
if ( node ! = NULL ) {
if ( node - > properties = = NULL ) {
node - > properties = cur ;
} else {
xmlAttrPtr prev = node - > properties ;
while ( prev - > next ! = NULL ) prev = prev - > next ;
prev - > next = cur ;
cur - > prev = prev ;
}
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewNsProp :
* @ node : the holding node
* @ ns : the namespace
* @ name : the name of the attribute
* @ value : the value of the attribute
*
* Create a new property tagged with a namespace and carried by a node .
* Returns a pointer to the attribute
*/
xmlAttrPtr
xmlNewNsProp ( xmlNodePtr node , xmlNsPtr ns , const xmlChar * name ,
const xmlChar * value ) {
xmlAttrPtr cur ;
2001-06-07 19:59:42 +00:00
xmlDocPtr doc = NULL ;
2001-02-23 17:55:21 +00:00
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewNsProp : name == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ( NULL ) ;
}
/*
* Allocate a new property and fill the fields .
*/
cur = ( xmlAttrPtr ) xmlMalloc ( sizeof ( xmlAttr ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewNsProp : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlAttr ) ) ;
cur - > type = XML_ATTRIBUTE_NODE ;
cur - > parent = node ;
2001-06-07 19:59:42 +00:00
if ( node ! = NULL ) {
doc = node - > doc ;
cur - > doc = doc ;
}
2001-02-23 17:55:21 +00:00
cur - > ns = ns ;
cur - > name = xmlStrdup ( name ) ;
if ( value ! = NULL ) {
xmlChar * buffer ;
xmlNodePtr tmp ;
2001-06-07 19:59:42 +00:00
buffer = xmlEncodeEntitiesReentrant ( doc , value ) ;
cur - > children = xmlStringGetNodeList ( doc , buffer ) ;
2001-02-23 17:55:21 +00:00
cur - > last = NULL ;
tmp = cur - > children ;
while ( tmp ! = NULL ) {
tmp - > parent = ( xmlNodePtr ) cur ;
if ( tmp - > next = = NULL )
cur - > last = tmp ;
tmp = tmp - > next ;
}
xmlFree ( buffer ) ;
}
/*
* Add it at the end to preserve parsing order . . .
*/
if ( node ! = NULL ) {
if ( node - > properties = = NULL ) {
node - > properties = cur ;
} else {
xmlAttrPtr prev = node - > properties ;
while ( prev - > next ! = NULL ) prev = prev - > next ;
prev - > next = cur ;
cur - > prev = prev ;
}
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
2002-05-29 08:21:33 +00:00
/**
* xmlNewNsPropEatName :
* @ node : the holding node
* @ ns : the namespace
* @ name : the name of the attribute
* @ value : the value of the attribute
*
* Create a new property tagged with a namespace and carried by a node .
* Returns a pointer to the attribute
*/
xmlAttrPtr
xmlNewNsPropEatName ( xmlNodePtr node , xmlNsPtr ns , xmlChar * name ,
const xmlChar * value ) {
xmlAttrPtr cur ;
xmlDocPtr doc = NULL ;
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNsPropEatName : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new property and fill the fields .
*/
cur = ( xmlAttrPtr ) xmlMalloc ( sizeof ( xmlAttr ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNsPropEatName : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlAttr ) ) ;
cur - > type = XML_ATTRIBUTE_NODE ;
cur - > parent = node ;
if ( node ! = NULL ) {
doc = node - > doc ;
cur - > doc = doc ;
}
cur - > ns = ns ;
cur - > name = name ;
if ( value ! = NULL ) {
xmlChar * buffer ;
xmlNodePtr tmp ;
buffer = xmlEncodeEntitiesReentrant ( doc , value ) ;
cur - > children = xmlStringGetNodeList ( doc , buffer ) ;
cur - > last = NULL ;
tmp = cur - > children ;
while ( tmp ! = NULL ) {
tmp - > parent = ( xmlNodePtr ) cur ;
if ( tmp - > next = = NULL )
cur - > last = tmp ;
tmp = tmp - > next ;
}
xmlFree ( buffer ) ;
}
/*
* Add it at the end to preserve parsing order . . .
*/
if ( node ! = NULL ) {
if ( node - > properties = = NULL ) {
node - > properties = cur ;
} else {
xmlAttrPtr prev = node - > properties ;
while ( prev - > next ! = NULL ) prev = prev - > next ;
prev - > next = cur ;
cur - > prev = prev ;
}
}
2003-01-05 22:37:17 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2002-05-29 08:21:33 +00:00
return ( cur ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlNewDocProp :
* @ doc : the document
* @ name : the name of the attribute
* @ value : the value of the attribute
*
* Create a new property carried by a document .
* Returns a pointer to the attribute
*/
xmlAttrPtr
xmlNewDocProp ( xmlDocPtr doc , const xmlChar * name , const xmlChar * value ) {
xmlAttrPtr cur ;
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewDocProp : name == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ( NULL ) ;
}
/*
* Allocate a new property and fill the fields .
*/
cur = ( xmlAttrPtr ) xmlMalloc ( sizeof ( xmlAttr ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewDocProp : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlAttr ) ) ;
cur - > type = XML_ATTRIBUTE_NODE ;
cur - > name = xmlStrdup ( name ) ;
cur - > doc = doc ;
if ( value ! = NULL ) {
xmlNodePtr tmp ;
cur - > children = xmlStringGetNodeList ( doc , value ) ;
cur - > last = NULL ;
tmp = cur - > children ;
while ( tmp ! = NULL ) {
tmp - > parent = ( xmlNodePtr ) cur ;
if ( tmp - > next = = NULL )
cur - > last = tmp ;
tmp = tmp - > next ;
}
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlFreePropList :
* @ cur : the first property in the list
*
* Free a property and all its siblings , all the children are freed too .
*/
void
xmlFreePropList ( xmlAttrPtr cur ) {
xmlAttrPtr next ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreePropList : property == NULL \n " ) ;
# endif
return ;
}
while ( cur ! = NULL ) {
next = cur - > next ;
xmlFreeProp ( cur ) ;
cur = next ;
}
}
/**
* xmlFreeProp :
* @ cur : an attribute
*
* Free one attribute , all the content is freed too
*/
void
xmlFreeProp ( xmlAttrPtr cur ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeProp : property == NULL \n " ) ;
# endif
return ;
}
2003-01-01 20:59:38 +00:00
if ( xmlDeregisterNodeDefaultValue )
xmlDeregisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
/* Check for ID removal -> leading to invalid references ! */
2001-05-16 21:05:17 +00:00
if ( ( cur - > parent ! = NULL ) & & ( cur - > parent - > doc ! = NULL ) & &
( ( cur - > parent - > doc - > intSubset ! = NULL ) | |
( cur - > parent - > doc - > extSubset ! = NULL ) ) ) {
if ( xmlIsID ( cur - > parent - > doc , cur - > parent , cur ) )
xmlRemoveID ( cur - > parent - > doc , cur ) ;
}
2001-02-23 17:55:21 +00:00
if ( cur - > name ! = NULL ) xmlFree ( ( char * ) cur - > name ) ;
if ( cur - > children ! = NULL ) xmlFreeNodeList ( cur - > children ) ;
xmlFree ( cur ) ;
}
/**
* xmlRemoveProp :
* @ cur : an attribute
*
* Unlink and free one attribute , all the content is freed too
* Note this doesn ' t work for namespace definition attributes
*
* Returns 0 if success and - 1 in case of error .
*/
int
xmlRemoveProp ( xmlAttrPtr cur ) {
xmlAttrPtr tmp ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlRemoveProp : cur == NULL \n " ) ;
# endif
return ( - 1 ) ;
}
if ( cur - > parent = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlRemoveProp : cur->parent == NULL \n " ) ;
# endif
return ( - 1 ) ;
}
tmp = cur - > parent - > properties ;
if ( tmp = = cur ) {
cur - > parent - > properties = cur - > next ;
xmlFreeProp ( cur ) ;
return ( 0 ) ;
}
while ( tmp ! = NULL ) {
if ( tmp - > next = = cur ) {
tmp - > next = cur - > next ;
if ( tmp - > next ! = NULL )
tmp - > next - > prev = tmp ;
xmlFreeProp ( cur ) ;
return ( 0 ) ;
}
tmp = tmp - > next ;
}
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlRemoveProp : attribute not owned by its node \n " ) ;
# endif
return ( - 1 ) ;
}
/**
* xmlNewPI :
* @ name : the processing instruction name
* @ content : the PI content
*
* Creation of a processing instruction element .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewPI ( const xmlChar * name , const xmlChar * content ) {
xmlNodePtr cur ;
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewPI : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewPI : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_PI_NODE ;
cur - > name = xmlStrdup ( name ) ;
if ( content ! = NULL ) {
cur - > content = xmlStrdup ( content ) ;
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewNode :
* @ ns : namespace if any
* @ name : the node name
*
2001-12-17 15:30:10 +00:00
* Creation of a new node element . @ ns is optional ( NULL ) .
2001-02-23 17:55:21 +00:00
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewNode ( xmlNsPtr ns , const xmlChar * name ) {
xmlNodePtr cur ;
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNode : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNode : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_ELEMENT_NODE ;
cur - > name = xmlStrdup ( name ) ;
cur - > ns = ns ;
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
2002-05-29 08:21:33 +00:00
/**
* xmlNewNodeEatName :
* @ ns : namespace if any
* @ name : the node name
*
* Creation of a new node element . @ ns is optional ( NULL ) .
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewNodeEatName ( xmlNsPtr ns , xmlChar * name ) {
xmlNodePtr cur ;
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNode : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewNode : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_ELEMENT_NODE ;
cur - > name = name ;
cur - > ns = ns ;
2003-01-05 22:37:17 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) cur ) ;
2002-05-29 08:21:33 +00:00
return ( cur ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlNewDocNode :
* @ doc : the document
* @ ns : namespace if any
* @ name : the node name
* @ content : the XML text content if any
*
* Creation of a new node element within a document . @ ns and @ content
2001-12-17 15:30:10 +00:00
* are optional ( NULL ) .
2001-02-23 17:55:21 +00:00
* NOTE : @ content is supposed to be a piece of XML CDATA , so it allow entities
* references , but XML special chars need to be escaped first by using
* xmlEncodeEntitiesReentrant ( ) . Use xmlNewDocRawNode ( ) if you don ' t
* need entities support .
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocNode ( xmlDocPtr doc , xmlNsPtr ns ,
const xmlChar * name , const xmlChar * content ) {
xmlNodePtr cur ;
cur = xmlNewNode ( ns , name ) ;
if ( cur ! = NULL ) {
cur - > doc = doc ;
if ( content ! = NULL ) {
cur - > children = xmlStringGetNodeList ( doc , content ) ;
UPDATE_LAST_CHILD_AND_PARENT ( cur )
}
}
2003-01-01 20:59:38 +00:00
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
2002-05-29 08:21:33 +00:00
/**
* xmlNewDocNodeEatName :
* @ doc : the document
* @ ns : namespace if any
* @ name : the node name
* @ content : the XML text content if any
*
* Creation of a new node element within a document . @ ns and @ content
* are optional ( NULL ) .
* NOTE : @ content is supposed to be a piece of XML CDATA , so it allow entities
* references , but XML special chars need to be escaped first by using
* xmlEncodeEntitiesReentrant ( ) . Use xmlNewDocRawNode ( ) if you don ' t
* need entities support .
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocNodeEatName ( xmlDocPtr doc , xmlNsPtr ns ,
xmlChar * name , const xmlChar * content ) {
xmlNodePtr cur ;
cur = xmlNewNodeEatName ( ns , name ) ;
if ( cur ! = NULL ) {
cur - > doc = doc ;
if ( content ! = NULL ) {
cur - > children = xmlStringGetNodeList ( doc , content ) ;
UPDATE_LAST_CHILD_AND_PARENT ( cur )
}
}
return ( cur ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlNewDocRawNode :
* @ doc : the document
* @ ns : namespace if any
* @ name : the node name
* @ content : the text content if any
*
* Creation of a new node element within a document . @ ns and @ content
2001-12-17 15:30:10 +00:00
* are optional ( NULL ) .
2001-02-23 17:55:21 +00:00
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocRawNode ( xmlDocPtr doc , xmlNsPtr ns ,
const xmlChar * name , const xmlChar * content ) {
xmlNodePtr cur ;
cur = xmlNewNode ( ns , name ) ;
if ( cur ! = NULL ) {
cur - > doc = doc ;
if ( content ! = NULL ) {
cur - > children = xmlNewDocText ( doc , content ) ;
UPDATE_LAST_CHILD_AND_PARENT ( cur )
}
}
return ( cur ) ;
}
/**
* xmlNewDocFragment :
* @ doc : the document owning the fragment
*
* Creation of a new Fragment node .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocFragment ( xmlDocPtr doc ) {
xmlNodePtr cur ;
/*
* Allocate a new DocumentFragment node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewDocFragment : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_DOCUMENT_FRAG_NODE ;
cur - > doc = doc ;
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewText :
* @ content : the text content
*
* Creation of a new text node .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewText ( const xmlChar * content ) {
xmlNodePtr cur ;
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewText : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_TEXT_NODE ;
cur - > name = xmlStringText ;
if ( content ! = NULL ) {
cur - > content = xmlStrdup ( content ) ;
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewTextChild :
* @ parent : the parent node
* @ ns : a namespace if any
* @ name : the name of the child
* @ content : the text content of the child if any .
*
* Creation of a new child element , added at the end of @ parent children list .
2001-12-17 15:30:10 +00:00
* @ ns and @ content parameters are optional ( NULL ) . If content is non NULL ,
2001-02-23 17:55:21 +00:00
* a child TEXT node will be created containing the string content .
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewTextChild ( xmlNodePtr parent , xmlNsPtr ns ,
const xmlChar * name , const xmlChar * content ) {
xmlNodePtr cur , prev ;
if ( parent = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewTextChild : parent == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewTextChild : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new node
*/
if ( ns = = NULL )
cur = xmlNewDocRawNode ( parent - > doc , parent - > ns , name , content ) ;
else
cur = xmlNewDocRawNode ( parent - > doc , ns , name , content ) ;
if ( cur = = NULL ) return ( NULL ) ;
/*
* add the new element at the end of the children list .
*/
cur - > type = XML_ELEMENT_NODE ;
cur - > parent = parent ;
cur - > doc = parent - > doc ;
if ( parent - > children = = NULL ) {
parent - > children = cur ;
parent - > last = cur ;
} else {
prev = parent - > last ;
prev - > next = cur ;
cur - > prev = prev ;
parent - > last = cur ;
}
return ( cur ) ;
}
/**
* xmlNewCharRef :
* @ doc : the document
* @ name : the char ref string , starting with # or " &# ... ; "
*
* Creation of a new character reference node .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewCharRef ( xmlDocPtr doc , const xmlChar * name ) {
xmlNodePtr cur ;
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewCharRef : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_ENTITY_REF_NODE ;
cur - > doc = doc ;
if ( name [ 0 ] = = ' & ' ) {
int len ;
name + + ;
len = xmlStrlen ( name ) ;
if ( name [ len - 1 ] = = ' ; ' )
cur - > name = xmlStrndup ( name , len - 1 ) ;
else
cur - > name = xmlStrndup ( name , len ) ;
} else
cur - > name = xmlStrdup ( name ) ;
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewReference :
* @ doc : the document
* @ name : the reference name , or the reference string with & and ;
*
* Creation of a new reference node .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewReference ( xmlDocPtr doc , const xmlChar * name ) {
xmlNodePtr cur ;
xmlEntityPtr ent ;
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewReference : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_ENTITY_REF_NODE ;
cur - > doc = doc ;
if ( name [ 0 ] = = ' & ' ) {
int len ;
name + + ;
len = xmlStrlen ( name ) ;
if ( name [ len - 1 ] = = ' ; ' )
cur - > name = xmlStrndup ( name , len - 1 ) ;
else
cur - > name = xmlStrndup ( name , len ) ;
} else
cur - > name = xmlStrdup ( name ) ;
ent = xmlGetDocEntity ( doc , cur - > name ) ;
if ( ent ! = NULL ) {
cur - > content = ent - > content ;
/*
2001-12-31 16:16:02 +00:00
* The parent pointer in entity is a DTD pointer and thus is NOT
2001-02-23 17:55:21 +00:00
* updated . Not sure if this is 100 % correct .
* - George
*/
cur - > children = ( xmlNodePtr ) ent ;
cur - > last = ( xmlNodePtr ) ent ;
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewDocText :
* @ doc : the document
* @ content : the text content
*
* Creation of a new text node within a document .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocText ( xmlDocPtr doc , const xmlChar * content ) {
xmlNodePtr cur ;
cur = xmlNewText ( content ) ;
if ( cur ! = NULL ) cur - > doc = doc ;
return ( cur ) ;
}
/**
* xmlNewTextLen :
* @ content : the text content
* @ len : the text len .
*
2001-10-10 09:45:09 +00:00
* Creation of a new text node with an extra parameter for the content ' s length
2001-02-23 17:55:21 +00:00
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewTextLen ( const xmlChar * content , int len ) {
xmlNodePtr cur ;
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNewTextLen : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_TEXT_NODE ;
cur - > name = xmlStringText ;
if ( content ! = NULL ) {
cur - > content = xmlStrndup ( content , len ) ;
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewDocTextLen :
* @ doc : the document
* @ content : the text content
* @ len : the text len .
*
2001-10-10 09:45:09 +00:00
* Creation of a new text node with an extra content length parameter . The
2001-02-23 17:55:21 +00:00
* text node pertain to a given document .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocTextLen ( xmlDocPtr doc , const xmlChar * content , int len ) {
xmlNodePtr cur ;
cur = xmlNewTextLen ( content , len ) ;
if ( cur ! = NULL ) cur - > doc = doc ;
return ( cur ) ;
}
/**
* xmlNewComment :
* @ content : the comment content
*
* Creation of a new node containing a comment .
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewComment ( const xmlChar * content ) {
xmlNodePtr cur ;
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewComment : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_COMMENT_NODE ;
cur - > name = xmlStringComment ;
if ( content ! = NULL ) {
cur - > content = xmlStrdup ( content ) ;
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewCDataBlock :
* @ doc : the document
2001-12-17 15:30:10 +00:00
* @ content : the CDATA block content content
2001-02-23 17:55:21 +00:00
* @ len : the length of the block
*
2001-12-17 15:30:10 +00:00
* Creation of a new node containing a CDATA block .
2001-02-23 17:55:21 +00:00
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewCDataBlock ( xmlDocPtr doc , const xmlChar * content , int len ) {
xmlNodePtr cur ;
/*
* Allocate a new node and fill the fields .
*/
cur = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewCDataBlock : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNode ) ) ;
cur - > type = XML_CDATA_SECTION_NODE ;
2001-03-24 17:00:36 +00:00
cur - > doc = doc ;
2001-02-23 17:55:21 +00:00
if ( content ! = NULL ) {
cur - > content = xmlStrndup ( content , len ) ;
}
2003-01-01 20:59:38 +00:00
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
return ( cur ) ;
}
/**
* xmlNewDocComment :
* @ doc : the document
* @ content : the comment content
*
2001-12-17 15:30:10 +00:00
* Creation of a new node containing a comment within a document .
2001-02-23 17:55:21 +00:00
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewDocComment ( xmlDocPtr doc , const xmlChar * content ) {
xmlNodePtr cur ;
cur = xmlNewComment ( content ) ;
if ( cur ! = NULL ) cur - > doc = doc ;
return ( cur ) ;
}
/**
* xmlSetTreeDoc :
* @ tree : the top element
* @ doc : the document
*
* update all nodes under the tree to point to the right document
*/
void
xmlSetTreeDoc ( xmlNodePtr tree , xmlDocPtr doc ) {
2001-07-09 10:32:59 +00:00
xmlAttrPtr prop ;
2001-02-23 17:55:21 +00:00
if ( tree = = NULL )
return ;
if ( tree - > doc ! = doc ) {
2002-01-24 15:02:46 +00:00
if ( tree - > type = = XML_ELEMENT_NODE ) {
prop = tree - > properties ;
while ( prop ! = NULL ) {
prop - > doc = doc ;
xmlSetListDoc ( prop - > children , doc ) ;
prop = prop - > next ;
}
2001-07-09 10:32:59 +00:00
}
2001-02-23 17:55:21 +00:00
if ( tree - > children ! = NULL )
xmlSetListDoc ( tree - > children , doc ) ;
tree - > doc = doc ;
}
}
/**
* xmlSetListDoc :
2001-12-17 15:30:10 +00:00
* @ list : the first element
2001-02-23 17:55:21 +00:00
* @ doc : the document
*
* update all nodes in the list to point to the right document
*/
void
xmlSetListDoc ( xmlNodePtr list , xmlDocPtr doc ) {
xmlNodePtr cur ;
if ( list = = NULL )
return ;
cur = list ;
while ( cur ! = NULL ) {
if ( cur - > doc ! = doc )
xmlSetTreeDoc ( cur , doc ) ;
cur = cur - > next ;
}
}
/**
* xmlNewChild :
* @ parent : the parent node
* @ ns : a namespace if any
* @ name : the name of the child
* @ content : the XML content of the child if any .
*
* Creation of a new child element , added at the end of @ parent children list .
2001-12-17 15:30:10 +00:00
* @ ns and @ content parameters are optional ( NULL ) . If content is non NULL ,
2001-02-23 17:55:21 +00:00
* a child list containing the TEXTs and ENTITY_REFs node will be created .
* NOTE : @ content is supposed to be a piece of XML CDATA , so it allow entities
* references , but XML special chars need to be escaped first by using
* xmlEncodeEntitiesReentrant ( ) . Use xmlNewTextChild ( ) if entities
* support is not needed .
*
* Returns a pointer to the new node object .
*/
xmlNodePtr
xmlNewChild ( xmlNodePtr parent , xmlNsPtr ns ,
const xmlChar * name , const xmlChar * content ) {
xmlNodePtr cur , prev ;
if ( parent = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewChild : parent == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( name = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewChild : name == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Allocate a new node
*/
2002-02-04 00:17:01 +00:00
if ( parent - > type = = XML_ELEMENT_NODE ) {
if ( ns = = NULL )
cur = xmlNewDocNode ( parent - > doc , parent - > ns , name , content ) ;
else
cur = xmlNewDocNode ( parent - > doc , ns , name , content ) ;
} else if ( ( parent - > type = = XML_DOCUMENT_NODE ) | |
( parent - > type = = XML_HTML_DOCUMENT_NODE ) ) {
if ( ns = = NULL )
cur = xmlNewDocNode ( ( xmlDocPtr ) parent , NULL , name , content ) ;
else
cur = xmlNewDocNode ( ( xmlDocPtr ) parent , ns , name , content ) ;
2002-10-28 18:52:57 +00:00
} else if ( parent - > type = = XML_DOCUMENT_FRAG_NODE ) {
cur = xmlNewDocNode ( parent - > doc , ns , name , content ) ;
2002-02-04 00:17:01 +00:00
} else {
return ( NULL ) ;
}
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) return ( NULL ) ;
/*
* add the new element at the end of the children list .
*/
cur - > type = XML_ELEMENT_NODE ;
cur - > parent = parent ;
cur - > doc = parent - > doc ;
if ( parent - > children = = NULL ) {
parent - > children = cur ;
parent - > last = cur ;
} else {
prev = parent - > last ;
prev - > next = cur ;
cur - > prev = prev ;
parent - > last = cur ;
}
return ( cur ) ;
}
/**
* xmlAddNextSibling :
* @ cur : the child node
* @ elem : the new node
*
2002-01-24 16:05:41 +00:00
* Add a new node @ elem as the next sibling of @ cur
* If the new node was already inserted in a document it is
2001-02-23 17:55:21 +00:00
* first unlinked from its existing context .
* As a result of text merging @ elem may be freed .
2002-01-24 16:05:41 +00:00
* If the new node is ATTRIBUTE , it is added into properties instead of children .
* If there is an attribute with equal name , it is first destroyed .
2001-02-23 17:55:21 +00:00
*
2002-01-24 16:05:41 +00:00
* Returns the new node or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlNodePtr
xmlAddNextSibling ( xmlNodePtr cur , xmlNodePtr elem ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddNextSibling : cur == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( elem = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddNextSibling : elem == NULL \n " ) ;
# endif
return ( NULL ) ;
}
xmlUnlinkNode ( elem ) ;
if ( elem - > type = = XML_TEXT_NODE ) {
if ( cur - > type = = XML_TEXT_NODE ) {
xmlNodeAddContent ( cur , elem - > content ) ;
xmlFreeNode ( elem ) ;
return ( cur ) ;
}
2001-08-31 20:03:19 +00:00
if ( ( cur - > next ! = NULL ) & & ( cur - > next - > type = = XML_TEXT_NODE ) & &
( cur - > name = = cur - > next - > name ) ) {
2001-02-23 17:55:21 +00:00
xmlChar * tmp ;
tmp = xmlStrdup ( elem - > content ) ;
tmp = xmlStrcat ( tmp , cur - > next - > content ) ;
xmlNodeSetContent ( cur - > next , tmp ) ;
xmlFree ( tmp ) ;
xmlFreeNode ( elem ) ;
return ( cur - > next ) ;
}
2002-01-24 16:05:41 +00:00
} else if ( elem - > type = = XML_ATTRIBUTE_NODE ) {
/* check if an attribute with the same name exists */
xmlAttrPtr attr ;
if ( elem - > ns = = NULL )
attr = xmlHasProp ( cur - > parent , elem - > name ) ;
else
attr = xmlHasNsProp ( cur - > parent , elem - > name , elem - > ns - > href ) ;
if ( ( attr ! = NULL ) & & ( attr ! = ( xmlAttrPtr ) elem ) ) {
/* different instance, destroy it (attributes must be unique) */
xmlFreeProp ( attr ) ;
}
2001-02-23 17:55:21 +00:00
}
if ( elem - > doc ! = cur - > doc ) {
xmlSetTreeDoc ( elem , cur - > doc ) ;
}
elem - > parent = cur - > parent ;
elem - > prev = cur ;
elem - > next = cur - > next ;
cur - > next = elem ;
if ( elem - > next ! = NULL )
elem - > next - > prev = elem ;
2002-01-24 16:05:41 +00:00
if ( ( elem - > parent ! = NULL ) & & ( elem - > parent - > last = = cur ) & & ( elem - > type ! = XML_ATTRIBUTE_NODE ) )
2001-02-23 17:55:21 +00:00
elem - > parent - > last = elem ;
return ( elem ) ;
}
/**
* xmlAddPrevSibling :
* @ cur : the child node
* @ elem : the new node
*
2002-01-24 16:05:41 +00:00
* Add a new node @ elem as the previous sibling of @ cur
2001-02-23 17:55:21 +00:00
* merging adjacent TEXT nodes ( @ elem may be freed )
2002-01-24 16:05:41 +00:00
* If the new node was already inserted in a document it is
2001-02-23 17:55:21 +00:00
* first unlinked from its existing context .
2002-01-24 16:05:41 +00:00
* If the new node is ATTRIBUTE , it is added into properties instead of children .
* If there is an attribute with equal name , it is first destroyed .
2001-02-23 17:55:21 +00:00
*
2002-01-24 16:05:41 +00:00
* Returns the new node or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlNodePtr
xmlAddPrevSibling ( xmlNodePtr cur , xmlNodePtr elem ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddPrevSibling : cur == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( elem = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddPrevSibling : elem == NULL \n " ) ;
# endif
return ( NULL ) ;
}
xmlUnlinkNode ( elem ) ;
if ( elem - > type = = XML_TEXT_NODE ) {
if ( cur - > type = = XML_TEXT_NODE ) {
xmlChar * tmp ;
tmp = xmlStrdup ( elem - > content ) ;
tmp = xmlStrcat ( tmp , cur - > content ) ;
xmlNodeSetContent ( cur , tmp ) ;
xmlFree ( tmp ) ;
xmlFreeNode ( elem ) ;
return ( cur ) ;
}
2001-08-31 20:03:19 +00:00
if ( ( cur - > prev ! = NULL ) & & ( cur - > prev - > type = = XML_TEXT_NODE ) & &
( cur - > name = = cur - > prev - > name ) ) {
2001-02-23 17:55:21 +00:00
xmlNodeAddContent ( cur - > prev , elem - > content ) ;
xmlFreeNode ( elem ) ;
return ( cur - > prev ) ;
}
2002-01-24 16:05:41 +00:00
} else if ( elem - > type = = XML_ATTRIBUTE_NODE ) {
/* check if an attribute with the same name exists */
xmlAttrPtr attr ;
if ( elem - > ns = = NULL )
attr = xmlHasProp ( cur - > parent , elem - > name ) ;
else
attr = xmlHasNsProp ( cur - > parent , elem - > name , elem - > ns - > href ) ;
if ( ( attr ! = NULL ) & & ( attr ! = ( xmlAttrPtr ) elem ) ) {
/* different instance, destroy it (attributes must be unique) */
xmlFreeProp ( attr ) ;
}
2001-02-23 17:55:21 +00:00
}
if ( elem - > doc ! = cur - > doc ) {
xmlSetTreeDoc ( elem , cur - > doc ) ;
}
elem - > parent = cur - > parent ;
elem - > next = cur ;
elem - > prev = cur - > prev ;
cur - > prev = elem ;
if ( elem - > prev ! = NULL )
elem - > prev - > next = elem ;
2002-01-24 16:05:41 +00:00
if ( elem - > parent ! = NULL ) {
if ( elem - > type = = XML_ATTRIBUTE_NODE ) {
if ( elem - > parent - > properties = = ( xmlAttrPtr ) cur ) {
elem - > parent - > properties = ( xmlAttrPtr ) elem ;
}
} else {
if ( elem - > parent - > children = = cur ) {
elem - > parent - > children = elem ;
}
}
}
2001-02-23 17:55:21 +00:00
return ( elem ) ;
}
/**
* xmlAddSibling :
* @ cur : the child node
* @ elem : the new node
*
* Add a new element @ elem to the list of siblings of @ cur
* merging adjacent TEXT nodes ( @ elem may be freed )
* If the new element was already inserted in a document it is
* first unlinked from its existing context .
*
* Returns the new element or NULL in case of error .
*/
xmlNodePtr
xmlAddSibling ( xmlNodePtr cur , xmlNodePtr elem ) {
xmlNodePtr parent ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddSibling : cur == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( elem = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddSibling : elem == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Constant time is we can rely on the - > parent - > last to find
* the last sibling .
*/
if ( ( cur - > parent ! = NULL ) & &
( cur - > parent - > children ! = NULL ) & &
( cur - > parent - > last ! = NULL ) & &
( cur - > parent - > last - > next = = NULL ) ) {
cur = cur - > parent - > last ;
} else {
while ( cur - > next ! = NULL ) cur = cur - > next ;
}
xmlUnlinkNode ( elem ) ;
if ( ( cur - > type = = XML_TEXT_NODE ) & & ( elem - > type = = XML_TEXT_NODE ) ) {
xmlNodeAddContent ( cur , elem - > content ) ;
xmlFreeNode ( elem ) ;
return ( cur ) ;
}
if ( elem - > doc ! = cur - > doc ) {
xmlSetTreeDoc ( elem , cur - > doc ) ;
}
parent = cur - > parent ;
elem - > prev = cur ;
elem - > next = NULL ;
elem - > parent = parent ;
cur - > next = elem ;
if ( parent ! = NULL )
parent - > last = elem ;
return ( elem ) ;
}
/**
* xmlAddChildList :
* @ parent : the parent node
* @ cur : the first node in the list
*
* Add a list of node at the end of the child list of the parent
* merging adjacent TEXT nodes ( @ cur may be freed )
*
* Returns the last child or NULL in case of error .
*/
xmlNodePtr
xmlAddChildList ( xmlNodePtr parent , xmlNodePtr cur ) {
xmlNodePtr prev ;
if ( parent = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlAddChildList : parent == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ( NULL ) ;
}
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlAddChildList : child == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ( NULL ) ;
}
if ( ( cur - > doc ! = NULL ) & & ( parent - > doc ! = NULL ) & &
( cur - > doc ! = parent - > doc ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" Elements moved to a different document \n " ) ;
# endif
}
/*
* add the first element at the end of the children list .
*/
2003-01-01 20:59:38 +00:00
2001-02-23 17:55:21 +00:00
if ( parent - > children = = NULL ) {
parent - > children = cur ;
} else {
/*
* If cur and parent - > last both are TEXT nodes , then merge them .
*/
if ( ( cur - > type = = XML_TEXT_NODE ) & &
( parent - > last - > type = = XML_TEXT_NODE ) & &
( cur - > name = = parent - > last - > name ) ) {
2003-01-01 20:59:38 +00:00
xmlNodeAddContent ( parent - > last , cur - > content ) ;
2001-02-23 17:55:21 +00:00
/*
* if it ' s the only child , nothing more to be done .
*/
if ( cur - > next = = NULL ) {
xmlFreeNode ( cur ) ;
return ( parent - > last ) ;
}
prev = cur ;
cur = cur - > next ;
xmlFreeNode ( prev ) ;
}
prev = parent - > last ;
prev - > next = cur ;
cur - > prev = prev ;
}
while ( cur - > next ! = NULL ) {
cur - > parent = parent ;
if ( cur - > doc ! = parent - > doc ) {
xmlSetTreeDoc ( cur , parent - > doc ) ;
}
cur = cur - > next ;
}
cur - > parent = parent ;
cur - > doc = parent - > doc ; /* the parent may not be linked to a doc ! */
parent - > last = cur ;
return ( cur ) ;
}
/**
* xmlAddChild :
* @ parent : the parent node
* @ cur : the child node
*
2002-01-24 16:05:41 +00:00
* Add a new node to @ parent , at the end of the child ( or property ) list
2001-02-23 17:55:21 +00:00
* merging adjacent TEXT nodes ( in which case @ cur is freed )
2002-01-24 16:05:41 +00:00
* If the new node is ATTRIBUTE , it is added into properties instead of children .
* If there is an attribute with equal name , it is first destroyed .
*
2001-02-23 17:55:21 +00:00
* Returns the child or NULL in case of error .
*/
xmlNodePtr
xmlAddChild ( xmlNodePtr parent , xmlNodePtr cur ) {
xmlNodePtr prev ;
if ( parent = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddChild : parent == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAddChild : child == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* If cur is a TEXT node , merge its content with adjacent TEXT nodes
* cur is then freed .
*/
if ( cur - > type = = XML_TEXT_NODE ) {
2001-07-12 01:20:08 +00:00
if ( ( parent - > type = = XML_TEXT_NODE ) & &
2003-01-01 20:59:38 +00:00
( parent - > content ! = NULL ) & &
( parent ! = cur ) ) {
2001-02-23 17:55:21 +00:00
xmlNodeAddContent ( parent , cur - > content ) ;
xmlFreeNode ( cur ) ;
return ( parent ) ;
}
if ( ( parent - > last ! = NULL ) & & ( parent - > last - > type = = XML_TEXT_NODE ) & &
2003-01-01 20:59:38 +00:00
( parent - > last - > name = = cur - > name ) & &
( parent - > last ! = cur ) ) {
2001-02-23 17:55:21 +00:00
xmlNodeAddContent ( parent - > last , cur - > content ) ;
xmlFreeNode ( cur ) ;
return ( parent - > last ) ;
}
}
/*
* add the new element at the end of the children list .
*/
2003-01-01 20:59:38 +00:00
prev = cur - > parent ;
2001-02-23 17:55:21 +00:00
cur - > parent = parent ;
if ( cur - > doc ! = parent - > doc ) {
xmlSetTreeDoc ( cur , parent - > doc ) ;
}
2003-01-01 20:59:38 +00:00
/* this check prevents a loop on tree-traversions if a developer
* tries to add a node to its parent multiple times
*/
if ( prev = = parent )
return ( cur ) ;
2001-02-23 17:55:21 +00:00
/*
2001-07-12 01:20:08 +00:00
* Coalescing
2001-02-23 17:55:21 +00:00
*/
2001-07-12 01:20:08 +00:00
if ( ( parent - > type = = XML_TEXT_NODE ) & &
2003-01-01 20:59:38 +00:00
( parent - > content ! = NULL ) & &
( parent ! = cur ) ) {
2001-07-12 01:20:08 +00:00
xmlNodeAddContent ( parent , cur - > content ) ;
xmlFreeNode ( cur ) ;
return ( parent ) ;
2001-02-23 17:55:21 +00:00
}
2002-01-24 16:05:41 +00:00
if ( cur - > type = = XML_ATTRIBUTE_NODE ) {
if ( parent - > properties = = NULL ) {
parent - > properties = ( xmlAttrPtr ) cur ;
} else {
/* check if an attribute with the same name exists */
xmlAttrPtr lastattr ;
if ( cur - > ns = = NULL )
lastattr = xmlHasProp ( parent , cur - > name ) ;
else
lastattr = xmlHasNsProp ( parent , cur - > name , cur - > ns - > href ) ;
if ( ( lastattr ! = NULL ) & & ( lastattr ! = ( xmlAttrPtr ) cur ) ) {
/* different instance, destroy it (attributes must be unique) */
xmlFreeProp ( lastattr ) ;
}
/* find the end */
lastattr = parent - > properties ;
while ( lastattr - > next ! = NULL ) {
lastattr = lastattr - > next ;
}
lastattr - > next = ( xmlAttrPtr ) cur ;
( ( xmlAttrPtr ) cur ) - > prev = lastattr ;
}
2001-02-23 17:55:21 +00:00
} else {
2002-01-24 16:05:41 +00:00
if ( parent - > children = = NULL ) {
parent - > children = cur ;
parent - > last = cur ;
} else {
prev = parent - > last ;
prev - > next = cur ;
cur - > prev = prev ;
parent - > last = cur ;
}
2001-02-23 17:55:21 +00:00
}
return ( cur ) ;
}
/**
* xmlGetLastChild :
* @ parent : the parent node
*
* Search the last child of a node .
* Returns the last child or NULL if none .
*/
xmlNodePtr
xmlGetLastChild ( xmlNodePtr parent ) {
if ( parent = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlGetLastChild : parent == NULL \n " ) ;
# endif
return ( NULL ) ;
}
return ( parent - > last ) ;
}
/**
* xmlFreeNodeList :
* @ cur : the first node in the list
*
* Free a node and all its siblings , this is a recursive behaviour , all
* the children are freed too .
*/
void
xmlFreeNodeList ( xmlNodePtr cur ) {
xmlNodePtr next ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeNodeList : node == NULL \n " ) ;
# endif
return ;
}
2002-01-14 17:11:53 +00:00
if ( cur - > type = = XML_NAMESPACE_DECL ) {
xmlFreeNsList ( ( xmlNsPtr ) cur ) ;
return ;
}
2001-02-23 17:55:21 +00:00
while ( cur ! = NULL ) {
next = cur - > next ;
2001-04-30 11:46:40 +00:00
/* unroll to speed up freeing the document */
if ( cur - > type ! = XML_DTD_NODE ) {
2003-01-01 20:59:38 +00:00
if ( xmlDeregisterNodeDefaultValue )
xmlDeregisterNodeDefaultValue ( cur ) ;
2001-04-30 11:46:40 +00:00
if ( ( cur - > children ! = NULL ) & &
( cur - > type ! = XML_ENTITY_REF_NODE ) )
xmlFreeNodeList ( cur - > children ) ;
2002-12-10 15:19:08 +00:00
if ( ( ( cur - > type = = XML_ELEMENT_NODE ) | |
( cur - > type = = XML_XINCLUDE_START ) | |
( cur - > type = = XML_XINCLUDE_END ) ) & &
2002-12-09 14:13:43 +00:00
( cur - > properties ! = NULL ) )
2001-04-30 11:46:40 +00:00
xmlFreePropList ( cur - > properties ) ;
2001-07-12 01:20:08 +00:00
if ( ( cur - > type ! = XML_ELEMENT_NODE ) & &
( cur - > type ! = XML_XINCLUDE_START ) & &
( cur - > type ! = XML_XINCLUDE_END ) & &
( cur - > type ! = XML_ENTITY_REF_NODE ) ) {
2001-04-30 11:46:40 +00:00
if ( cur - > content ! = NULL ) xmlFree ( cur - > content ) ;
2001-07-12 01:20:08 +00:00
}
if ( ( ( cur - > type = = XML_ELEMENT_NODE ) | |
( cur - > type = = XML_XINCLUDE_START ) | |
( cur - > type = = XML_XINCLUDE_END ) ) & &
( cur - > nsDef ! = NULL ) )
xmlFreeNsList ( cur - > nsDef ) ;
2001-06-11 08:09:20 +00:00
/*
* When a node is a text node or a comment , it uses a global static
* variable for the name of the node .
*
* The xmlStrEqual comparisons need to be done when ( happened with
* XML : : libXML and XML : : libXSLT ) the library is included twice
2001-12-31 16:16:02 +00:00
* statically in the binary and a tree allocated by one occurrence
2001-12-17 15:30:10 +00:00
* of the lib gets freed by the other occurrence , in this case
2001-06-11 08:09:20 +00:00
* the string addresses compare are not sufficient .
*/
2001-04-30 11:46:40 +00:00
if ( ( cur - > name ! = NULL ) & &
( cur - > name ! = xmlStringText ) & &
( cur - > name ! = xmlStringTextNoenc ) & &
2001-06-11 08:09:20 +00:00
( cur - > name ! = xmlStringComment ) ) {
if ( cur - > type = = XML_TEXT_NODE ) {
if ( ( ! xmlStrEqual ( cur - > name , xmlStringText ) ) & &
( ! xmlStrEqual ( cur - > name , xmlStringTextNoenc ) ) )
xmlFree ( ( char * ) cur - > name ) ;
} else if ( cur - > type = = XML_COMMENT_NODE ) {
if ( ! xmlStrEqual ( cur - > name , xmlStringComment ) )
xmlFree ( ( char * ) cur - > name ) ;
} else
xmlFree ( ( char * ) cur - > name ) ;
}
2001-04-30 11:46:40 +00:00
/* TODO : derecursivate this function */
xmlFree ( cur ) ;
}
2001-02-23 17:55:21 +00:00
cur = next ;
}
}
/**
* xmlFreeNode :
* @ cur : the node
*
* Free a node , this is a recursive behaviour , all the children are freed too .
* This doesn ' t unlink the child from the list , use xmlUnlinkNode ( ) first .
*/
void
xmlFreeNode ( xmlNodePtr cur ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlFreeNode : node == NULL \n " ) ;
# endif
return ;
}
2003-01-01 20:59:38 +00:00
2001-04-30 11:46:40 +00:00
/* use xmlFreeDtd for DTD nodes */
2002-01-14 17:11:53 +00:00
if ( cur - > type = = XML_DTD_NODE ) {
xmlFreeDtd ( ( xmlDtdPtr ) cur ) ;
2001-02-23 17:55:21 +00:00
return ;
2002-01-14 17:11:53 +00:00
}
if ( cur - > type = = XML_NAMESPACE_DECL ) {
xmlFreeNs ( ( xmlNsPtr ) cur ) ;
return ;
}
2002-11-07 14:18:03 +00:00
if ( cur - > type = = XML_ATTRIBUTE_NODE ) {
xmlFreeProp ( ( xmlAttrPtr ) cur ) ;
return ;
}
2003-01-01 20:59:38 +00:00
if ( xmlDeregisterNodeDefaultValue )
xmlDeregisterNodeDefaultValue ( cur ) ;
2001-02-23 17:55:21 +00:00
if ( ( cur - > children ! = NULL ) & &
( cur - > type ! = XML_ENTITY_REF_NODE ) )
xmlFreeNodeList ( cur - > children ) ;
2002-12-10 15:19:08 +00:00
if ( ( ( cur - > type = = XML_ELEMENT_NODE ) | |
( cur - > type = = XML_XINCLUDE_START ) | |
( cur - > type = = XML_XINCLUDE_END ) ) & &
( cur - > properties ! = NULL ) )
2001-04-30 11:46:40 +00:00
xmlFreePropList ( cur - > properties ) ;
2001-07-12 01:20:08 +00:00
if ( ( cur - > type ! = XML_ELEMENT_NODE ) & &
( cur - > content ! = NULL ) & &
( cur - > type ! = XML_ENTITY_REF_NODE ) & &
( cur - > type ! = XML_XINCLUDE_END ) & &
( cur - > type ! = XML_XINCLUDE_START ) ) {
xmlFree ( cur - > content ) ;
}
2001-06-09 17:17:51 +00:00
/*
* When a node is a text node or a comment , it uses a global static
* variable for the name of the node .
*
* The xmlStrEqual comparisons need to be done when ( happened with
* XML : : libXML and XML : : libXSLT ) the library is included twice statically
2001-12-31 16:16:02 +00:00
* in the binary and a tree allocated by one occurence of the lib gets
2001-12-17 15:30:10 +00:00
* freed by the other occurrence , in this case the string addresses compare
2001-06-09 17:17:51 +00:00
* are not sufficient .
*/
2001-02-23 17:55:21 +00:00
if ( ( cur - > name ! = NULL ) & &
( cur - > name ! = xmlStringText ) & &
( cur - > name ! = xmlStringTextNoenc ) & &
2001-06-09 17:17:51 +00:00
( cur - > name ! = xmlStringComment ) ) {
if ( cur - > type = = XML_TEXT_NODE ) {
if ( ( ! xmlStrEqual ( cur - > name , xmlStringText ) ) & &
( ! xmlStrEqual ( cur - > name , xmlStringTextNoenc ) ) )
xmlFree ( ( char * ) cur - > name ) ;
} else if ( cur - > type = = XML_COMMENT_NODE ) {
if ( ! xmlStrEqual ( cur - > name , xmlStringComment ) )
xmlFree ( ( char * ) cur - > name ) ;
} else
xmlFree ( ( char * ) cur - > name ) ;
}
2002-12-09 14:13:43 +00:00
if ( ( ( cur - > type = = XML_ELEMENT_NODE ) | |
( cur - > type = = XML_XINCLUDE_START ) | |
( cur - > type = = XML_XINCLUDE_END ) ) & &
( cur - > nsDef ! = NULL ) )
xmlFreeNsList ( cur - > nsDef ) ;
2001-02-23 17:55:21 +00:00
xmlFree ( cur ) ;
}
/**
* xmlUnlinkNode :
* @ cur : the node
*
* Unlink a node from it ' s current context , the node is not freed
*/
void
xmlUnlinkNode ( xmlNodePtr cur ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlUnlinkNode : node == NULL \n " ) ;
# endif
return ;
}
2001-12-13 22:21:58 +00:00
if ( cur - > type = = XML_DTD_NODE ) {
xmlDocPtr doc ;
doc = cur - > doc ;
if ( doc - > intSubset = = ( xmlDtdPtr ) cur )
doc - > intSubset = NULL ;
if ( doc - > extSubset = = ( xmlDtdPtr ) cur )
doc - > extSubset = NULL ;
}
2002-01-22 21:40:13 +00:00
if ( cur - > parent ! = NULL ) {
xmlNodePtr parent ;
parent = cur - > parent ;
if ( cur - > type = = XML_ATTRIBUTE_NODE ) {
if ( parent - > properties = = ( xmlAttrPtr ) cur )
parent - > properties = ( ( xmlAttrPtr ) cur ) - > next ;
} else {
if ( parent - > children = = cur )
parent - > children = cur - > next ;
if ( parent - > last = = cur )
parent - > last = cur - > prev ;
}
cur - > parent = NULL ;
}
2001-02-23 17:55:21 +00:00
if ( cur - > next ! = NULL )
cur - > next - > prev = cur - > prev ;
if ( cur - > prev ! = NULL )
cur - > prev - > next = cur - > next ;
cur - > next = cur - > prev = NULL ;
}
/**
* xmlReplaceNode :
* @ old : the old node
* @ cur : the node
*
* Unlink the old node from it ' s current context , prune the new one
2001-12-17 15:30:10 +00:00
* at the same place . If @ cur was already inserted in a document it is
2001-02-23 17:55:21 +00:00
* first unlinked from its existing context .
*
2001-12-17 15:30:10 +00:00
* Returns the @ old node
2001-02-23 17:55:21 +00:00
*/
xmlNodePtr
xmlReplaceNode ( xmlNodePtr old , xmlNodePtr cur ) {
if ( old = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlReplaceNode : old == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( cur = = NULL ) {
xmlUnlinkNode ( old ) ;
return ( old ) ;
}
if ( cur = = old ) {
return ( old ) ;
}
2002-01-22 21:40:13 +00:00
if ( ( old - > type = = XML_ATTRIBUTE_NODE ) & & ( cur - > type ! = XML_ATTRIBUTE_NODE ) ) {
2002-01-24 16:05:41 +00:00
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlReplaceNode : Trying to replace attribute node with other node type \n " ) ;
# endif
return ( old ) ;
}
if ( ( cur - > type = = XML_ATTRIBUTE_NODE ) & & ( old - > type ! = XML_ATTRIBUTE_NODE ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlReplaceNode : Trying to replace a non-attribute node with attribute node \n " ) ;
# endif
return ( old ) ;
}
if ( ( old - > type = = XML_ATTRIBUTE_NODE ) & & ( cur - > type ! = XML_ATTRIBUTE_NODE ) ) {
2002-01-22 21:40:13 +00:00
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlReplaceNode : Trying to replace attribute node with other node type \n " ) ;
# endif
return ( old ) ;
}
if ( ( cur - > type = = XML_ATTRIBUTE_NODE ) & & ( old - > type ! = XML_ATTRIBUTE_NODE ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlReplaceNode : Trying to replace a non-attribute node with attribute node \n " ) ;
# endif
return ( old ) ;
}
2001-02-23 17:55:21 +00:00
xmlUnlinkNode ( cur ) ;
cur - > doc = old - > doc ;
cur - > parent = old - > parent ;
cur - > next = old - > next ;
if ( cur - > next ! = NULL )
cur - > next - > prev = cur ;
cur - > prev = old - > prev ;
if ( cur - > prev ! = NULL )
cur - > prev - > next = cur ;
if ( cur - > parent ! = NULL ) {
2002-01-22 21:40:13 +00:00
if ( cur - > type = = XML_ATTRIBUTE_NODE ) {
if ( cur - > parent - > properties = = ( xmlAttrPtr ) old )
cur - > parent - > properties = ( ( xmlAttrPtr ) cur ) ;
} else {
if ( cur - > parent - > children = = old )
cur - > parent - > children = cur ;
if ( cur - > parent - > last = = old )
cur - > parent - > last = cur ;
}
2001-02-23 17:55:21 +00:00
}
old - > next = old - > prev = NULL ;
old - > parent = NULL ;
return ( old ) ;
}
/************************************************************************
* *
* Copy operations *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/**
* xmlCopyNamespace :
* @ cur : the namespace
*
* Do a copy of the namespace .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlNsPtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlNsPtr
xmlCopyNamespace ( xmlNsPtr cur ) {
xmlNsPtr ret ;
if ( cur = = NULL ) return ( NULL ) ;
switch ( cur - > type ) {
case XML_LOCAL_NAMESPACE :
ret = xmlNewNs ( NULL , cur - > href , cur - > prefix ) ;
break ;
default :
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlCopyNamespace: invalid type %d \n " , cur - > type ) ;
# endif
return ( NULL ) ;
}
return ( ret ) ;
}
/**
* xmlCopyNamespaceList :
* @ cur : the first namespace
*
* Do a copy of an namespace list .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlNsPtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlNsPtr
xmlCopyNamespaceList ( xmlNsPtr cur ) {
xmlNsPtr ret = NULL ;
xmlNsPtr p = NULL , q ;
while ( cur ! = NULL ) {
q = xmlCopyNamespace ( cur ) ;
if ( p = = NULL ) {
ret = p = q ;
} else {
p - > next = q ;
p = q ;
}
cur = cur - > next ;
}
return ( ret ) ;
}
static xmlNodePtr
xmlStaticCopyNodeList ( xmlNodePtr node , xmlDocPtr doc , xmlNodePtr parent ) ;
/**
* xmlCopyProp :
* @ target : the element where the attribute will be grafted
* @ cur : the attribute
*
* Do a copy of the attribute .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlAttrPtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlAttrPtr
xmlCopyProp ( xmlNodePtr target , xmlAttrPtr cur ) {
xmlAttrPtr ret ;
if ( cur = = NULL ) return ( NULL ) ;
if ( target ! = NULL )
ret = xmlNewDocProp ( target - > doc , cur - > name , NULL ) ;
else if ( cur - > parent ! = NULL )
ret = xmlNewDocProp ( cur - > parent - > doc , cur - > name , NULL ) ;
else if ( cur - > children ! = NULL )
ret = xmlNewDocProp ( cur - > children - > doc , cur - > name , NULL ) ;
else
ret = xmlNewDocProp ( NULL , cur - > name , NULL ) ;
if ( ret = = NULL ) return ( NULL ) ;
ret - > parent = target ;
2002-03-03 14:13:46 +00:00
2001-02-23 17:55:21 +00:00
if ( ( cur - > ns ! = NULL ) & & ( target ! = NULL ) ) {
2002-01-13 14:10:10 +00:00
xmlNsPtr ns ;
2002-03-03 14:13:46 +00:00
/*
* if ( target - > doc )
* ns = xmlSearchNs ( target - > doc , target , cur - > ns - > prefix ) ;
* else if ( cur - > doc ) / * target may not yet have a doc : KPI * /
* ns = xmlSearchNs ( cur - > doc , target , cur - > ns - > prefix ) ;
* else
* ns = NULL ;
* ret - > ns = ns ;
*/
ns = xmlSearchNs ( target - > doc , target , cur - > ns - > prefix ) ;
if ( ns = = NULL ) {
/*
* Humm , we are copying an element whose namespace is defined
* out of the new tree scope . Search it in the original tree
* and add it at the top of the new tree
*/
ns = xmlSearchNs ( cur - > doc , cur - > parent , cur - > ns - > prefix ) ;
if ( ns ! = NULL ) {
xmlNodePtr root = target ;
xmlNodePtr pred = NULL ;
while ( root - > parent ! = NULL ) {
pred = root ;
root = root - > parent ;
}
if ( root = = ( xmlNodePtr ) target - > doc ) {
/* correct possibly cycling above the document elt */
root = pred ;
}
ret - > ns = xmlNewNs ( root , ns - > href , ns - > prefix ) ;
}
} else {
/*
* we have to find something appropriate here since
* we cant be sure , that the namespce we found is identified
* by the prefix
*/
2002-03-04 17:09:44 +00:00
if ( xmlStrEqual ( ns - > href , cur - > ns - > href ) ) {
2002-03-03 14:13:46 +00:00
/* this is the nice case */
ret - > ns = ns ;
} else {
/*
* we are in trouble : we need a new reconcilied namespace .
* This is expensive
*/
ret - > ns = xmlNewReconciliedNs ( target - > doc , target , cur - > ns ) ;
}
}
2001-02-23 17:55:21 +00:00
} else
ret - > ns = NULL ;
if ( cur - > children ! = NULL ) {
xmlNodePtr tmp ;
ret - > children = xmlStaticCopyNodeList ( cur - > children , ret - > doc , ( xmlNodePtr ) ret ) ;
ret - > last = NULL ;
tmp = ret - > children ;
while ( tmp ! = NULL ) {
/* tmp->parent = (xmlNodePtr)ret; */
if ( tmp - > next = = NULL )
ret - > last = tmp ;
tmp = tmp - > next ;
}
}
2002-02-10 11:57:22 +00:00
/*
* Try to handle IDs
*/
2002-03-08 15:46:57 +00:00
if ( ( target ! = NULL ) & & ( cur ! = NULL ) & &
( target - > doc ! = NULL ) & & ( cur - > doc ! = NULL ) & &
2002-02-10 11:57:22 +00:00
( cur - > doc - > ids ! = NULL ) & & ( cur - > parent ! = NULL ) ) {
if ( xmlIsID ( cur - > doc , cur - > parent , cur ) ) {
xmlChar * id ;
id = xmlNodeListGetString ( cur - > doc , cur - > children , 1 ) ;
if ( id ! = NULL ) {
xmlAddID ( NULL , target - > doc , id , ret ) ;
xmlFree ( id ) ;
}
}
}
2001-02-23 17:55:21 +00:00
return ( ret ) ;
}
/**
* xmlCopyPropList :
* @ target : the element where the attributes will be grafted
* @ cur : the first attribute
*
* Do a copy of an attribute list .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlAttrPtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlAttrPtr
xmlCopyPropList ( xmlNodePtr target , xmlAttrPtr cur ) {
xmlAttrPtr ret = NULL ;
xmlAttrPtr p = NULL , q ;
while ( cur ! = NULL ) {
q = xmlCopyProp ( target , cur ) ;
if ( p = = NULL ) {
ret = p = q ;
} else {
p - > next = q ;
q - > prev = p ;
p = q ;
}
cur = cur - > next ;
}
return ( ret ) ;
}
/*
2001-12-17 15:30:10 +00:00
* NOTE about the CopyNode operations !
2001-02-23 17:55:21 +00:00
*
2001-12-17 15:30:10 +00:00
* They are split into external and internal parts for one
2001-02-23 17:55:21 +00:00
* tricky reason : namespaces . Doing a direct copy of a node
* say RPM : Copyright without changing the namespace pointer to
* something else can produce stale links . One way to do it is
* to keep a reference counter but this doesn ' t work as soon
* as one move the element or the subtree out of the scope of
* the existing namespace . The actual solution seems to add
* a copy of the namespace at the top of the copied tree if
* not available in the subtree .
* Hence two functions , the public front - end call the inner ones
*/
static xmlNodePtr
2001-08-28 20:39:49 +00:00
xmlStaticCopyNode ( const xmlNodePtr node , xmlDocPtr doc , xmlNodePtr parent ,
2001-02-23 17:55:21 +00:00
int recursive ) {
xmlNodePtr ret ;
if ( node = = NULL ) return ( NULL ) ;
2001-06-19 18:09:42 +00:00
switch ( node - > type ) {
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_ELEMENT_NODE :
2002-09-05 11:12:45 +00:00
case XML_DOCUMENT_FRAG_NODE :
2001-06-19 18:09:42 +00:00
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_PI_NODE :
case XML_COMMENT_NODE :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
break ;
case XML_ATTRIBUTE_NODE :
2001-07-26 11:49:41 +00:00
return ( ( xmlNodePtr ) xmlCopyProp ( parent , ( xmlAttrPtr ) node ) ) ;
2001-06-19 18:09:42 +00:00
case XML_NAMESPACE_DECL :
2001-07-26 11:49:41 +00:00
return ( ( xmlNodePtr ) xmlCopyNamespaceList ( ( xmlNsPtr ) node ) ) ;
2001-06-19 18:09:42 +00:00
2001-07-26 11:49:41 +00:00
case XML_DOCUMENT_NODE :
case XML_HTML_DOCUMENT_NODE :
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
# endif
return ( ( xmlNodePtr ) xmlCopyDoc ( ( xmlDocPtr ) node , recursive ) ) ;
2001-06-19 18:09:42 +00:00
case XML_DOCUMENT_TYPE_NODE :
case XML_NOTATION_NODE :
case XML_DTD_NODE :
case XML_ELEMENT_DECL :
case XML_ATTRIBUTE_DECL :
case XML_ENTITY_DECL :
return ( NULL ) ;
}
2001-04-25 12:59:04 +00:00
2001-02-23 17:55:21 +00:00
/*
* Allocate a new node and fill the fields .
*/
ret = ( xmlNodePtr ) xmlMalloc ( sizeof ( xmlNode ) ) ;
if ( ret = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlStaticCopyNode : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( ret , 0 , sizeof ( xmlNode ) ) ;
ret - > type = node - > type ;
ret - > doc = doc ;
ret - > parent = parent ;
if ( node - > name = = xmlStringText )
ret - > name = xmlStringText ;
else if ( node - > name = = xmlStringTextNoenc )
ret - > name = xmlStringTextNoenc ;
else if ( node - > name = = xmlStringComment )
ret - > name = xmlStringComment ;
else if ( node - > name ! = NULL )
ret - > name = xmlStrdup ( node - > name ) ;
2001-07-12 01:20:08 +00:00
if ( ( node - > type ! = XML_ELEMENT_NODE ) & &
( node - > content ! = NULL ) & &
( node - > type ! = XML_ENTITY_REF_NODE ) & &
( node - > type ! = XML_XINCLUDE_END ) & &
( node - > type ! = XML_XINCLUDE_START ) ) {
2001-02-23 17:55:21 +00:00
ret - > content = xmlStrdup ( node - > content ) ;
2002-01-13 14:10:10 +00:00
} else {
if ( node - > type = = XML_ELEMENT_NODE )
ret - > content = ( void * ) ( long ) node - > content ;
2001-02-23 17:55:21 +00:00
}
2002-01-13 16:15:43 +00:00
if ( parent ! = NULL ) {
xmlNodePtr tmp ;
2003-01-05 22:37:17 +00:00
/*
* this is a tricky part for the node register thing :
* in case ret does get coalesced in xmlAddChild
* the deregister - node callback is called ; so we register ret now already
*/
if ( xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) ret ) ;
2002-01-13 16:15:43 +00:00
tmp = xmlAddChild ( parent , ret ) ;
/* node could have coalesced */
if ( tmp ! = ret )
return ( tmp ) ;
}
2001-02-23 17:55:21 +00:00
2003-01-05 22:37:17 +00:00
if ( ! recursive )
goto out ;
2001-02-23 17:55:21 +00:00
if ( node - > nsDef ! = NULL )
ret - > nsDef = xmlCopyNamespaceList ( node - > nsDef ) ;
if ( node - > ns ! = NULL ) {
xmlNsPtr ns ;
ns = xmlSearchNs ( doc , ret , node - > ns - > prefix ) ;
if ( ns = = NULL ) {
/*
* Humm , we are copying an element whose namespace is defined
* out of the new tree scope . Search it in the original tree
* and add it at the top of the new tree
*/
ns = xmlSearchNs ( node - > doc , node , node - > ns - > prefix ) ;
if ( ns ! = NULL ) {
xmlNodePtr root = ret ;
while ( root - > parent ! = NULL ) root = root - > parent ;
2001-04-22 12:12:58 +00:00
ret - > ns = xmlNewNs ( root , ns - > href , ns - > prefix ) ;
2001-02-23 17:55:21 +00:00
}
} else {
/*
* reference the existing namespace definition in our own tree .
*/
ret - > ns = ns ;
}
}
if ( node - > properties ! = NULL )
ret - > properties = xmlCopyPropList ( ret , node - > properties ) ;
2001-04-25 12:59:04 +00:00
if ( node - > type = = XML_ENTITY_REF_NODE ) {
if ( ( doc = = NULL ) | | ( node - > doc ! = doc ) ) {
/*
* The copied node will go into a separate document , so
2001-12-17 15:30:10 +00:00
* to avoid dangling references to the ENTITY_DECL node
2001-04-25 12:59:04 +00:00
* we cannot keep the reference . Try to find it in the
* target document .
*/
ret - > children = ( xmlNodePtr ) xmlGetDocEntity ( doc , ret - > name ) ;
} else {
ret - > children = node - > children ;
}
2001-11-14 15:04:32 +00:00
ret - > last = ret - > children ;
} else if ( node - > children ! = NULL ) {
2001-02-23 17:55:21 +00:00
ret - > children = xmlStaticCopyNodeList ( node - > children , doc , ret ) ;
2001-11-14 15:04:32 +00:00
UPDATE_LAST_CHILD_AND_PARENT ( ret )
}
2003-01-05 22:37:17 +00:00
out :
/* if parent != NULL we already registered the node above */
if ( parent = = NULL & & xmlRegisterNodeDefaultValue )
xmlRegisterNodeDefaultValue ( ( xmlNodePtr ) ret ) ;
2001-02-23 17:55:21 +00:00
return ( ret ) ;
}
static xmlNodePtr
xmlStaticCopyNodeList ( xmlNodePtr node , xmlDocPtr doc , xmlNodePtr parent ) {
xmlNodePtr ret = NULL ;
xmlNodePtr p = NULL , q ;
while ( node ! = NULL ) {
2001-07-26 11:49:41 +00:00
if ( node - > type = = XML_DTD_NODE ) {
2001-06-09 14:19:02 +00:00
if ( doc = = NULL ) {
node = node - > next ;
continue ;
}
2001-04-25 12:59:04 +00:00
if ( doc - > intSubset = = NULL ) {
q = ( xmlNodePtr ) xmlCopyDtd ( ( xmlDtdPtr ) node ) ;
q - > doc = doc ;
q - > parent = parent ;
doc - > intSubset = ( xmlDtdPtr ) q ;
2002-01-26 21:42:58 +00:00
xmlAddChild ( parent , q ) ;
2001-04-25 12:59:04 +00:00
} else {
q = ( xmlNodePtr ) doc - > intSubset ;
2002-01-26 21:42:58 +00:00
xmlAddChild ( parent , q ) ;
2001-04-25 12:59:04 +00:00
}
} else
q = xmlStaticCopyNode ( node , doc , parent , 1 ) ;
2001-02-23 17:55:21 +00:00
if ( ret = = NULL ) {
q - > prev = NULL ;
ret = p = q ;
2002-01-13 16:15:43 +00:00
} else if ( p ! = q ) {
/* the test is required if xmlStaticCopyNode coalesced 2 text nodes */
2001-02-23 17:55:21 +00:00
p - > next = q ;
q - > prev = p ;
p = q ;
}
node = node - > next ;
}
return ( ret ) ;
}
/**
* xmlCopyNode :
* @ node : the node
* @ recursive : if 1 do a recursive copy .
*
* Do a copy of the node .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlNodePtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlNodePtr
2001-08-28 20:39:49 +00:00
xmlCopyNode ( const xmlNodePtr node , int recursive ) {
2001-02-23 17:55:21 +00:00
xmlNodePtr ret ;
ret = xmlStaticCopyNode ( node , NULL , NULL , recursive ) ;
return ( ret ) ;
}
2001-04-12 08:55:36 +00:00
/**
* xmlDocCopyNode :
* @ node : the node
2001-12-17 15:30:10 +00:00
* @ doc : the document
2001-04-12 08:55:36 +00:00
* @ recursive : if 1 do a recursive copy .
*
* Do a copy of the node to a given document .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlNodePtr , or NULL in case of error .
2001-04-12 08:55:36 +00:00
*/
xmlNodePtr
2001-08-28 20:39:49 +00:00
xmlDocCopyNode ( const xmlNodePtr node , xmlDocPtr doc , int recursive ) {
2001-04-12 08:55:36 +00:00
xmlNodePtr ret ;
ret = xmlStaticCopyNode ( node , doc , NULL , recursive ) ;
return ( ret ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlCopyNodeList :
* @ node : the first node in the list .
*
* Do a recursive copy of the node list .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlNodePtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
2001-08-28 20:39:49 +00:00
xmlNodePtr xmlCopyNodeList ( const xmlNodePtr node ) {
2001-02-23 17:55:21 +00:00
xmlNodePtr ret = xmlStaticCopyNodeList ( node , NULL , NULL ) ;
return ( ret ) ;
}
/**
* xmlCopyDtd :
* @ dtd : the dtd
*
* Do a copy of the dtd .
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlDtdPtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlDtdPtr
xmlCopyDtd ( xmlDtdPtr dtd ) {
xmlDtdPtr ret ;
2002-01-26 21:42:58 +00:00
xmlNodePtr cur , p = NULL , q ;
2001-02-23 17:55:21 +00:00
if ( dtd = = NULL ) return ( NULL ) ;
ret = xmlNewDtd ( NULL , dtd - > name , dtd - > ExternalID , dtd - > SystemID ) ;
if ( ret = = NULL ) return ( NULL ) ;
if ( dtd - > entities ! = NULL )
ret - > entities = ( void * ) xmlCopyEntitiesTable (
( xmlEntitiesTablePtr ) dtd - > entities ) ;
if ( dtd - > notations ! = NULL )
ret - > notations = ( void * ) xmlCopyNotationTable (
( xmlNotationTablePtr ) dtd - > notations ) ;
if ( dtd - > elements ! = NULL )
ret - > elements = ( void * ) xmlCopyElementTable (
( xmlElementTablePtr ) dtd - > elements ) ;
if ( dtd - > attributes ! = NULL )
ret - > attributes = ( void * ) xmlCopyAttributeTable (
( xmlAttributeTablePtr ) dtd - > attributes ) ;
2002-01-26 21:42:58 +00:00
if ( dtd - > pentities ! = NULL )
ret - > pentities = ( void * ) xmlCopyEntitiesTable (
( xmlEntitiesTablePtr ) dtd - > pentities ) ;
cur = dtd - > children ;
while ( cur ! = NULL ) {
q = NULL ;
if ( cur - > type = = XML_ENTITY_DECL ) {
xmlEntityPtr tmp = ( xmlEntityPtr ) cur ;
switch ( tmp - > etype ) {
case XML_INTERNAL_GENERAL_ENTITY :
case XML_EXTERNAL_GENERAL_PARSED_ENTITY :
case XML_EXTERNAL_GENERAL_UNPARSED_ENTITY :
q = ( xmlNodePtr ) xmlGetEntityFromDtd ( ret , tmp - > name ) ;
break ;
case XML_INTERNAL_PARAMETER_ENTITY :
case XML_EXTERNAL_PARAMETER_ENTITY :
q = ( xmlNodePtr )
xmlGetParameterEntityFromDtd ( ret , tmp - > name ) ;
break ;
case XML_INTERNAL_PREDEFINED_ENTITY :
break ;
}
} else if ( cur - > type = = XML_ELEMENT_DECL ) {
xmlElementPtr tmp = ( xmlElementPtr ) cur ;
q = ( xmlNodePtr )
xmlGetDtdQElementDesc ( ret , tmp - > name , tmp - > prefix ) ;
} else if ( cur - > type = = XML_ATTRIBUTE_DECL ) {
xmlAttributePtr tmp = ( xmlAttributePtr ) cur ;
q = ( xmlNodePtr )
xmlGetDtdQAttrDesc ( ret , tmp - > elem , tmp - > name , tmp - > prefix ) ;
} else if ( cur - > type = = XML_COMMENT_NODE ) {
q = xmlCopyNode ( cur , 0 ) ;
}
if ( q = = NULL ) {
cur = cur - > next ;
continue ;
}
if ( p = = NULL )
ret - > children = q ;
else
p - > next = q ;
q - > prev = p ;
q - > parent = ( xmlNodePtr ) ret ;
q - > next = NULL ;
ret - > last = q ;
p = q ;
cur = cur - > next ;
}
2001-02-23 17:55:21 +00:00
return ( ret ) ;
}
/**
* xmlCopyDoc :
* @ doc : the document
* @ recursive : if 1 do a recursive copy .
*
* Do a copy of the document info . If recursive , the content tree will
2001-12-31 16:16:02 +00:00
* be copied too as well as DTD , namespaces and entities .
2001-02-23 17:55:21 +00:00
*
2001-12-17 15:30:10 +00:00
* Returns : a new # xmlDocPtr , or NULL in case of error .
2001-02-23 17:55:21 +00:00
*/
xmlDocPtr
xmlCopyDoc ( xmlDocPtr doc , int recursive ) {
xmlDocPtr ret ;
if ( doc = = NULL ) return ( NULL ) ;
ret = xmlNewDoc ( doc - > version ) ;
if ( ret = = NULL ) return ( NULL ) ;
if ( doc - > name ! = NULL )
ret - > name = xmlMemStrdup ( doc - > name ) ;
if ( doc - > encoding ! = NULL )
ret - > encoding = xmlStrdup ( doc - > encoding ) ;
ret - > charset = doc - > charset ;
ret - > compression = doc - > compression ;
ret - > standalone = doc - > standalone ;
if ( ! recursive ) return ( ret ) ;
2001-04-25 12:59:04 +00:00
ret - > last = NULL ;
ret - > children = NULL ;
if ( doc - > intSubset ! = NULL ) {
2001-02-23 17:55:21 +00:00
ret - > intSubset = xmlCopyDtd ( doc - > intSubset ) ;
2002-01-26 21:42:58 +00:00
xmlSetTreeDoc ( ( xmlNodePtr ) ret - > intSubset , ret ) ;
2001-04-25 12:59:04 +00:00
ret - > intSubset - > parent = ret ;
}
2001-02-23 17:55:21 +00:00
if ( doc - > oldNs ! = NULL )
ret - > oldNs = xmlCopyNamespaceList ( doc - > oldNs ) ;
if ( doc - > children ! = NULL ) {
xmlNodePtr tmp ;
2001-04-25 12:59:04 +00:00
ret - > children = xmlStaticCopyNodeList ( doc - > children , ret ,
( xmlNodePtr ) ret ) ;
2001-02-23 17:55:21 +00:00
ret - > last = NULL ;
tmp = ret - > children ;
while ( tmp ! = NULL ) {
if ( tmp - > next = = NULL )
ret - > last = tmp ;
tmp = tmp - > next ;
}
}
return ( ret ) ;
}
/************************************************************************
* *
* Content access functions *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-11-26 15:58:08 +00:00
/**
* xmlGetLineNo :
2002-12-10 15:19:08 +00:00
* @ node : valid node
2001-11-26 15:58:08 +00:00
*
* Get line number of node . this requires activation of this option
2001-12-17 15:30:10 +00:00
* before invoking the parser by calling xmlLineNumbersDefault ( 1 )
2001-11-26 15:58:08 +00:00
*
2001-12-17 15:30:10 +00:00
* Returns the line number if successful , - 1 otherwise
2001-11-26 15:58:08 +00:00
*/
long
xmlGetLineNo ( xmlNodePtr node )
{
long result = - 1 ;
if ( ! node )
return result ;
if ( node - > type = = XML_ELEMENT_NODE )
result = ( long ) node - > content ;
else if ( ( node - > prev ! = NULL ) & &
( ( node - > prev - > type = = XML_ELEMENT_NODE ) | |
( node - > prev - > type = = XML_TEXT_NODE ) ) )
result = xmlGetLineNo ( node - > prev ) ;
else if ( ( node - > parent ! = NULL ) & &
( ( node - > parent - > type = = XML_ELEMENT_NODE ) | |
( node - > parent - > type = = XML_TEXT_NODE ) ) )
result = xmlGetLineNo ( node - > parent ) ;
return result ;
}
/**
* xmlGetNodePath :
* @ node : a node
*
* Build a structure based Path for the given node
*
* Returns the new path or NULL in case of error . The caller must free
* the returned string
*/
xmlChar *
xmlGetNodePath ( xmlNodePtr node )
{
xmlNodePtr cur , tmp , next ;
xmlChar * buffer = NULL , * temp ;
size_t buf_len ;
xmlChar * buf ;
2002-10-14 15:39:04 +00:00
const char * sep ;
2001-11-26 15:58:08 +00:00
const char * name ;
char nametemp [ 100 ] ;
int occur = 0 ;
if ( node = = NULL )
return ( NULL ) ;
buf_len = 500 ;
buffer = ( xmlChar * ) xmlMalloc ( buf_len * sizeof ( xmlChar ) ) ;
if ( buffer = = NULL )
return ( NULL ) ;
buf = ( xmlChar * ) xmlMalloc ( buf_len * sizeof ( xmlChar ) ) ;
if ( buf = = NULL ) {
xmlFree ( buffer ) ;
return ( NULL ) ;
}
buffer [ 0 ] = 0 ;
cur = node ;
do {
name = " " ;
2002-10-08 08:26:11 +00:00
sep = " ? " ;
2001-11-26 15:58:08 +00:00
occur = 0 ;
if ( ( cur - > type = = XML_DOCUMENT_NODE ) | |
( cur - > type = = XML_HTML_DOCUMENT_NODE ) ) {
if ( buffer [ 0 ] = = ' / ' )
break ;
2002-10-08 08:26:11 +00:00
sep = " / " ;
2001-11-26 15:58:08 +00:00
next = NULL ;
} else if ( cur - > type = = XML_ELEMENT_NODE ) {
2002-10-08 08:26:11 +00:00
sep = " / " ;
2001-11-26 15:58:08 +00:00
name = ( const char * ) cur - > name ;
if ( cur - > ns ) {
snprintf ( nametemp , sizeof ( nametemp ) - 1 ,
" %s:%s " , cur - > ns - > prefix , cur - > name ) ;
nametemp [ sizeof ( nametemp ) - 1 ] = 0 ;
name = nametemp ;
}
next = cur - > parent ;
/*
* Thumbler index computation
*/
tmp = cur - > prev ;
while ( tmp ! = NULL ) {
2002-09-17 23:04:40 +00:00
if ( ( tmp - > type = = XML_ELEMENT_NODE ) & &
( xmlStrEqual ( cur - > name , tmp - > name ) ) )
2001-11-26 15:58:08 +00:00
occur + + ;
tmp = tmp - > prev ;
}
if ( occur = = 0 ) {
tmp = cur - > next ;
2002-11-12 12:36:52 +00:00
while ( tmp ! = NULL & & occur = = 0 ) {
if ( ( tmp - > type = = XML_ELEMENT_NODE ) & &
( xmlStrEqual ( cur - > name , tmp - > name ) ) )
2001-11-26 15:58:08 +00:00
occur + + ;
tmp = tmp - > next ;
}
if ( occur ! = 0 )
occur = 1 ;
} else
occur + + ;
2002-11-12 12:36:52 +00:00
} else if ( cur - > type = = XML_COMMENT_NODE ) {
sep = " / " ;
name = " comment() " ;
next = cur - > parent ;
/*
* Thumbler index computation
*/
tmp = cur - > prev ;
while ( tmp ! = NULL ) {
if ( tmp - > type = = XML_COMMENT_NODE )
occur + + ;
tmp = tmp - > prev ;
}
if ( occur = = 0 ) {
tmp = cur - > next ;
while ( tmp ! = NULL & & occur = = 0 ) {
if ( tmp - > type = = XML_COMMENT_NODE )
occur + + ;
tmp = tmp - > next ;
}
if ( occur ! = 0 )
occur = 1 ;
} else
occur + + ;
} else if ( ( cur - > type = = XML_TEXT_NODE ) | |
( cur - > type = = XML_CDATA_SECTION_NODE ) ) {
sep = " / " ;
name = " text() " ;
next = cur - > parent ;
/*
* Thumbler index computation
*/
tmp = cur - > prev ;
while ( tmp ! = NULL ) {
if ( ( cur - > type = = XML_TEXT_NODE ) | |
( cur - > type = = XML_CDATA_SECTION_NODE ) )
occur + + ;
tmp = tmp - > prev ;
}
if ( occur = = 0 ) {
tmp = cur - > next ;
while ( tmp ! = NULL & & occur = = 0 ) {
if ( ( cur - > type = = XML_TEXT_NODE ) | |
( cur - > type = = XML_CDATA_SECTION_NODE ) )
occur + + ;
tmp = tmp - > next ;
}
if ( occur ! = 0 )
occur = 1 ;
} else
occur + + ;
} else if ( cur - > type = = XML_PI_NODE ) {
sep = " / " ;
snprintf ( nametemp , sizeof ( nametemp ) - 1 ,
" processing-instruction('%s') " , cur - > name ) ;
nametemp [ sizeof ( nametemp ) - 1 ] = 0 ;
name = nametemp ;
next = cur - > parent ;
/*
* Thumbler index computation
*/
tmp = cur - > prev ;
while ( tmp ! = NULL ) {
if ( ( tmp - > type = = XML_PI_NODE ) & &
( xmlStrEqual ( cur - > name , tmp - > name ) ) )
occur + + ;
tmp = tmp - > prev ;
}
if ( occur = = 0 ) {
tmp = cur - > next ;
while ( tmp ! = NULL & & occur = = 0 ) {
if ( ( tmp - > type = = XML_PI_NODE ) & &
( xmlStrEqual ( cur - > name , tmp - > name ) ) )
occur + + ;
tmp = tmp - > next ;
}
if ( occur ! = 0 )
occur = 1 ;
} else
occur + + ;
2001-11-26 15:58:08 +00:00
} else if ( cur - > type = = XML_ATTRIBUTE_NODE ) {
2002-10-08 08:26:11 +00:00
sep = " /@ " ;
2001-11-26 15:58:08 +00:00
name = ( const char * ) ( ( ( xmlAttrPtr ) cur ) - > name ) ;
next = ( ( xmlAttrPtr ) cur ) - > parent ;
} else {
next = cur - > parent ;
}
/*
* Make sure there is enough room
*/
if ( xmlStrlen ( buffer ) + sizeof ( nametemp ) + 20 > buf_len ) {
buf_len =
2 * buf_len + xmlStrlen ( buffer ) + sizeof ( nametemp ) + 20 ;
temp = ( xmlChar * ) xmlRealloc ( buffer , buf_len ) ;
if ( temp = = NULL ) {
xmlFree ( buf ) ;
xmlFree ( buffer ) ;
return ( NULL ) ;
}
buffer = temp ;
temp = ( xmlChar * ) xmlRealloc ( buf , buf_len ) ;
if ( temp = = NULL ) {
xmlFree ( buf ) ;
xmlFree ( buffer ) ;
return ( NULL ) ;
}
buf = temp ;
}
if ( occur = = 0 )
2002-10-08 08:26:11 +00:00
snprintf ( ( char * ) buf , buf_len , " %s%s%s " ,
2001-11-26 15:58:08 +00:00
sep , name , ( char * ) buffer ) ;
else
2002-10-08 08:26:11 +00:00
snprintf ( ( char * ) buf , buf_len , " %s%s[%d]%s " ,
2001-11-26 15:58:08 +00:00
sep , name , occur , ( char * ) buffer ) ;
snprintf ( ( char * ) buffer , buf_len , " %s " , buf ) ;
cur = next ;
} while ( cur ! = NULL ) ;
xmlFree ( buf ) ;
return ( buffer ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlDocGetRootElement :
* @ doc : the document
*
* Get the root element of the document ( doc - > children is a list
* containing possibly comments , PIs , etc . . . ) .
*
2001-12-17 15:30:10 +00:00
* Returns the # xmlNodePtr for the root or NULL
2001-02-23 17:55:21 +00:00
*/
xmlNodePtr
xmlDocGetRootElement ( xmlDocPtr doc ) {
xmlNodePtr ret ;
if ( doc = = NULL ) return ( NULL ) ;
ret = doc - > children ;
while ( ret ! = NULL ) {
if ( ret - > type = = XML_ELEMENT_NODE )
return ( ret ) ;
ret = ret - > next ;
}
return ( ret ) ;
}
/**
* xmlDocSetRootElement :
* @ doc : the document
* @ root : the new document root element
*
* Set the root element of the document ( doc - > children is a list
* containing possibly comments , PIs , etc . . . ) .
*
* Returns the old root element if any was found
*/
xmlNodePtr
xmlDocSetRootElement ( xmlDocPtr doc , xmlNodePtr root ) {
xmlNodePtr old = NULL ;
if ( doc = = NULL ) return ( NULL ) ;
2002-02-08 13:28:40 +00:00
if ( root = = NULL )
return ( NULL ) ;
xmlUnlinkNode ( root ) ;
root - > doc = doc ;
root - > parent = ( xmlNodePtr ) doc ;
2001-02-23 17:55:21 +00:00
old = doc - > children ;
while ( old ! = NULL ) {
if ( old - > type = = XML_ELEMENT_NODE )
break ;
old = old - > next ;
}
if ( old = = NULL ) {
if ( doc - > children = = NULL ) {
doc - > children = root ;
doc - > last = root ;
} else {
xmlAddSibling ( doc - > children , root ) ;
}
} else {
xmlReplaceNode ( old , root ) ;
}
return ( old ) ;
}
/**
* xmlNodeSetLang :
* @ cur : the node being changed
2001-12-17 15:30:10 +00:00
* @ lang : the language description
2001-02-23 17:55:21 +00:00
*
* Set the language of a node , i . e . the values of the xml : lang
* attribute .
*/
void
xmlNodeSetLang ( xmlNodePtr cur , const xmlChar * lang ) {
2002-01-17 08:46:58 +00:00
xmlNsPtr ns ;
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) return ;
switch ( cur - > type ) {
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_COMMENT_NODE :
case XML_DOCUMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_DOCUMENT_FRAG_NODE :
case XML_NOTATION_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_DTD_NODE :
case XML_ELEMENT_DECL :
case XML_ATTRIBUTE_DECL :
case XML_ENTITY_DECL :
case XML_PI_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_NAMESPACE_DECL :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
return ;
case XML_ELEMENT_NODE :
case XML_ATTRIBUTE_NODE :
break ;
}
2002-01-17 08:46:58 +00:00
ns = xmlSearchNsByHref ( cur - > doc , cur , XML_XML_NAMESPACE ) ;
if ( ns = = NULL )
return ;
xmlSetNsProp ( cur , ns , BAD_CAST " lang " , lang ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNodeGetLang :
* @ cur : the node being checked
*
* Searches the language of a node , i . e . the values of the xml : lang
* attribute or the one carried by the nearest ancestor .
*
* Returns a pointer to the lang value , or NULL if not found
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
xmlNodeGetLang ( xmlNodePtr cur ) {
xmlChar * lang ;
while ( cur ! = NULL ) {
2001-05-09 10:51:31 +00:00
lang = xmlGetNsProp ( cur , BAD_CAST " lang " , XML_XML_NAMESPACE ) ;
2001-02-23 17:55:21 +00:00
if ( lang ! = NULL )
return ( lang ) ;
cur = cur - > parent ;
}
return ( NULL ) ;
}
/**
* xmlNodeSetSpacePreserve :
* @ cur : the node being changed
* @ val : the xml : space value ( " 0 " : default , 1 : " preserve " )
*
* Set ( or reset ) the space preserving behaviour of a node , i . e . the
* value of the xml : space attribute .
*/
void
xmlNodeSetSpacePreserve ( xmlNodePtr cur , int val ) {
2002-01-17 08:46:58 +00:00
xmlNsPtr ns ;
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) return ;
switch ( cur - > type ) {
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_COMMENT_NODE :
case XML_DOCUMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_DOCUMENT_FRAG_NODE :
case XML_NOTATION_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_DTD_NODE :
case XML_ELEMENT_DECL :
case XML_ATTRIBUTE_DECL :
case XML_ENTITY_DECL :
case XML_PI_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_NAMESPACE_DECL :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
return ;
case XML_ELEMENT_NODE :
case XML_ATTRIBUTE_NODE :
break ;
}
2002-01-17 08:46:58 +00:00
ns = xmlSearchNsByHref ( cur - > doc , cur , XML_XML_NAMESPACE ) ;
if ( ns = = NULL )
return ;
2001-02-23 17:55:21 +00:00
switch ( val ) {
case 0 :
2002-01-17 08:46:58 +00:00
xmlSetNsProp ( cur , ns , BAD_CAST " space " , BAD_CAST " default " ) ;
2001-02-23 17:55:21 +00:00
break ;
case 1 :
2002-01-17 08:46:58 +00:00
xmlSetNsProp ( cur , ns , BAD_CAST " space " , BAD_CAST " preserve " ) ;
2001-02-23 17:55:21 +00:00
break ;
}
}
/**
* xmlNodeGetSpacePreserve :
* @ cur : the node being checked
*
* Searches the space preserving behaviour of a node , i . e . the values
* of the xml : space attribute or the one carried by the nearest
* ancestor .
*
2001-12-17 15:30:10 +00:00
* Returns - 1 if xml : space is not inherited , 0 if " default " , 1 if " preserve "
2001-02-23 17:55:21 +00:00
*/
int
xmlNodeGetSpacePreserve ( xmlNodePtr cur ) {
xmlChar * space ;
while ( cur ! = NULL ) {
2002-01-17 08:46:58 +00:00
space = xmlGetNsProp ( cur , BAD_CAST " space " , XML_XML_NAMESPACE ) ;
2001-02-23 17:55:21 +00:00
if ( space ! = NULL ) {
if ( xmlStrEqual ( space , BAD_CAST " preserve " ) ) {
xmlFree ( space ) ;
return ( 1 ) ;
}
if ( xmlStrEqual ( space , BAD_CAST " default " ) ) {
xmlFree ( space ) ;
return ( 0 ) ;
}
xmlFree ( space ) ;
}
cur = cur - > parent ;
}
return ( - 1 ) ;
}
/**
* xmlNodeSetName :
* @ cur : the node being changed
* @ name : the new tag name
*
* Set ( or reset ) the name of a node .
*/
void
xmlNodeSetName ( xmlNodePtr cur , const xmlChar * name ) {
if ( cur = = NULL ) return ;
if ( name = = NULL ) return ;
switch ( cur - > type ) {
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_COMMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_DOCUMENT_FRAG_NODE :
case XML_NOTATION_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_NAMESPACE_DECL :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
return ;
case XML_ELEMENT_NODE :
case XML_ATTRIBUTE_NODE :
case XML_PI_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_DTD_NODE :
case XML_DOCUMENT_NODE :
case XML_ELEMENT_DECL :
case XML_ATTRIBUTE_DECL :
case XML_ENTITY_DECL :
break ;
}
if ( cur - > name ! = NULL ) xmlFree ( ( xmlChar * ) cur - > name ) ;
cur - > name = xmlStrdup ( name ) ;
}
/**
* xmlNodeSetBase :
* @ cur : the node being changed
* @ uri : the new base URI
*
* Set ( or reset ) the base URI of a node , i . e . the value of the
* xml : base attribute .
*/
void
xmlNodeSetBase ( xmlNodePtr cur , xmlChar * uri ) {
2002-01-17 08:46:58 +00:00
xmlNsPtr ns ;
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) return ;
switch ( cur - > type ) {
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_COMMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_DOCUMENT_FRAG_NODE :
case XML_NOTATION_NODE :
case XML_DTD_NODE :
case XML_ELEMENT_DECL :
case XML_ATTRIBUTE_DECL :
case XML_ENTITY_DECL :
case XML_PI_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_NAMESPACE_DECL :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
return ;
case XML_ELEMENT_NODE :
case XML_ATTRIBUTE_NODE :
break ;
2002-05-05 06:57:27 +00:00
case XML_DOCUMENT_NODE :
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
# endif
case XML_HTML_DOCUMENT_NODE : {
xmlDocPtr doc = ( xmlDocPtr ) cur ;
if ( doc - > URL ! = NULL )
xmlFree ( ( xmlChar * ) doc - > URL ) ;
if ( uri = = NULL )
doc - > URL = NULL ;
else
doc - > URL = xmlStrdup ( uri ) ;
return ;
}
2001-02-23 17:55:21 +00:00
}
2002-01-17 08:46:58 +00:00
ns = xmlSearchNsByHref ( cur - > doc , cur , XML_XML_NAMESPACE ) ;
if ( ns = = NULL )
return ;
xmlSetNsProp ( cur , ns , BAD_CAST " base " , uri ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNodeGetBase :
* @ doc : the document the node pertains to
* @ cur : the node being checked
*
* Searches for the BASE URL . The code should work on both XML
* and HTML document even if base mechanisms are completely different .
* It returns the base as defined in RFC 2396 sections
* 5.1 .1 . Base URI within Document Content
* and
* 5.1 .2 . Base URI from the Encapsulating Entity
* However it does not return the document base ( 5.1 .3 ) , use
* xmlDocumentGetBase ( ) for this
*
* Returns a pointer to the base URL , or NULL if not found
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
xmlNodeGetBase ( xmlDocPtr doc , xmlNodePtr cur ) {
2001-07-09 16:01:19 +00:00
xmlChar * oldbase = NULL ;
xmlChar * base , * newbase ;
2001-02-23 17:55:21 +00:00
if ( ( cur = = NULL ) & & ( doc = = NULL ) )
return ( NULL ) ;
if ( doc = = NULL ) doc = cur - > doc ;
if ( ( doc ! = NULL ) & & ( doc - > type = = XML_HTML_DOCUMENT_NODE ) ) {
cur = doc - > children ;
while ( ( cur ! = NULL ) & & ( cur - > name ! = NULL ) ) {
if ( cur - > type ! = XML_ELEMENT_NODE ) {
cur = cur - > next ;
continue ;
}
if ( ! xmlStrcasecmp ( cur - > name , BAD_CAST " html " ) ) {
cur = cur - > children ;
continue ;
}
if ( ! xmlStrcasecmp ( cur - > name , BAD_CAST " head " ) ) {
cur = cur - > children ;
continue ;
}
if ( ! xmlStrcasecmp ( cur - > name , BAD_CAST " base " ) ) {
return ( xmlGetProp ( cur , BAD_CAST " href " ) ) ;
}
cur = cur - > next ;
}
return ( NULL ) ;
}
while ( cur ! = NULL ) {
if ( cur - > type = = XML_ENTITY_DECL ) {
xmlEntityPtr ent = ( xmlEntityPtr ) cur ;
return ( xmlStrdup ( ent - > URI ) ) ;
}
2001-05-22 16:57:14 +00:00
if ( cur - > type = = XML_ELEMENT_NODE ) {
2001-07-09 16:01:19 +00:00
base = xmlGetNsProp ( cur , BAD_CAST " base " , XML_XML_NAMESPACE ) ;
2001-05-22 16:57:14 +00:00
if ( base ! = NULL ) {
2001-07-09 16:01:19 +00:00
if ( oldbase ! = NULL ) {
newbase = xmlBuildURI ( oldbase , base ) ;
if ( newbase ! = NULL ) {
xmlFree ( oldbase ) ;
xmlFree ( base ) ;
oldbase = newbase ;
} else {
xmlFree ( oldbase ) ;
xmlFree ( base ) ;
return ( NULL ) ;
}
} else {
oldbase = base ;
}
if ( ( ! xmlStrncmp ( oldbase , BAD_CAST " http:// " , 7 ) ) | |
( ! xmlStrncmp ( oldbase , BAD_CAST " ftp:// " , 6 ) ) | |
( ! xmlStrncmp ( oldbase , BAD_CAST " urn: " , 4 ) ) )
return ( oldbase ) ;
2001-05-22 16:57:14 +00:00
}
}
2001-02-23 17:55:21 +00:00
cur = cur - > parent ;
}
2001-07-09 16:01:19 +00:00
if ( ( doc ! = NULL ) & & ( doc - > URL ! = NULL ) ) {
if ( oldbase = = NULL )
return ( xmlStrdup ( doc - > URL ) ) ;
newbase = xmlBuildURI ( oldbase , doc - > URL ) ;
xmlFree ( oldbase ) ;
return ( newbase ) ;
}
return ( oldbase ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNodeGetContent :
* @ cur : the node being read
*
* Read the value of a node , this can be either the text carried
* directly by this node if it ' s a TEXT node or the aggregate string
* of the values carried by this node child ' s ( TEXT and ENTITY_REF ) .
2001-12-17 15:30:10 +00:00
* Entity references are substituted .
* Returns a new # xmlChar * or NULL if no content is available .
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
2002-04-20 06:41:40 +00:00
xmlNodeGetContent ( xmlNodePtr cur )
{
if ( cur = = NULL )
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
switch ( cur - > type ) {
case XML_DOCUMENT_FRAG_NODE :
2002-04-20 06:41:40 +00:00
case XML_ELEMENT_NODE : {
xmlNodePtr tmp = cur ;
xmlBufferPtr buffer ;
xmlChar * ret ;
2003-01-23 18:24:20 +00:00
buffer = xmlBufferCreateSize ( 64 ) ;
2002-04-20 06:41:40 +00:00
if ( buffer = = NULL )
return ( NULL ) ;
while ( tmp ! = NULL ) {
switch ( tmp - > type ) {
case XML_CDATA_SECTION_NODE :
case XML_TEXT_NODE :
if ( tmp - > content ! = NULL )
xmlBufferCat ( buffer , tmp - > content ) ;
break ;
case XML_ENTITY_REF_NODE : {
/* recursive substitution of entity references */
xmlChar * cont = xmlNodeGetContent ( tmp ) ;
if ( cont ) {
xmlBufferCat ( buffer ,
( const xmlChar * ) cont ) ;
xmlFree ( cont ) ;
}
break ;
}
default :
break ;
}
/*
* Skip to next node
*/
if ( tmp - > children ! = NULL ) {
if ( tmp - > children - > type ! = XML_ENTITY_DECL ) {
tmp = tmp - > children ;
continue ;
}
}
if ( tmp = = cur )
break ;
if ( tmp - > next ! = NULL ) {
tmp = tmp - > next ;
continue ;
}
do {
tmp = tmp - > parent ;
if ( tmp = = NULL )
break ;
if ( tmp = = cur ) {
tmp = NULL ;
break ;
}
if ( tmp - > next ! = NULL ) {
tmp = tmp - > next ;
break ;
}
} while ( tmp ! = NULL ) ;
}
ret = buffer - > content ;
buffer - > content = NULL ;
xmlBufferFree ( buffer ) ;
return ( ret ) ;
}
case XML_ATTRIBUTE_NODE : {
xmlAttrPtr attr = ( xmlAttrPtr ) cur ;
if ( attr - > parent ! = NULL )
return ( xmlNodeListGetString
( attr - > parent - > doc , attr - > children , 1 ) ) ;
else
return ( xmlNodeListGetString ( NULL , attr - > children , 1 ) ) ;
break ;
}
2001-02-23 17:55:21 +00:00
case XML_COMMENT_NODE :
case XML_PI_NODE :
2002-04-20 06:41:40 +00:00
if ( cur - > content ! = NULL )
return ( xmlStrdup ( cur - > content ) ) ;
return ( NULL ) ;
case XML_ENTITY_REF_NODE : {
xmlEntityPtr ent ;
xmlNodePtr tmp ;
xmlBufferPtr buffer ;
xmlChar * ret ;
/* lookup entity declaration */
ent = xmlGetDocEntity ( cur - > doc , cur - > name ) ;
if ( ent = = NULL )
return ( NULL ) ;
buffer = xmlBufferCreate ( ) ;
if ( buffer = = NULL )
return ( NULL ) ;
/* an entity content can be any "well balanced chunk",
* i . e . the result of the content [ 43 ] production :
* http : //www.w3.org/TR/REC-xml#NT-content
* - > we iterate through child nodes and recursive call
* xmlNodeGetContent ( ) which handles all possible node types */
tmp = ent - > children ;
while ( tmp ) {
xmlChar * cont = xmlNodeGetContent ( tmp ) ;
if ( cont ) {
xmlBufferCat ( buffer , ( const xmlChar * ) cont ) ;
xmlFree ( cont ) ;
}
tmp = tmp - > next ;
}
ret = buffer - > content ;
buffer - > content = NULL ;
xmlBufferFree ( buffer ) ;
return ( ret ) ;
}
2001-02-23 17:55:21 +00:00
case XML_ENTITY_NODE :
case XML_DOCUMENT_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_NOTATION_NODE :
case XML_DTD_NODE :
2002-04-20 06:41:40 +00:00
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
2002-04-20 06:41:40 +00:00
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
2002-04-20 06:41:40 +00:00
return ( NULL ) ;
2002-10-14 15:39:04 +00:00
case XML_NAMESPACE_DECL : {
xmlChar * tmp ;
tmp = xmlStrdup ( ( ( xmlNsPtr ) cur ) - > href ) ;
return ( tmp ) ;
}
2001-02-23 17:55:21 +00:00
case XML_ELEMENT_DECL :
2002-04-20 06:41:40 +00:00
/* TODO !!! */
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
case XML_ATTRIBUTE_DECL :
2002-04-20 06:41:40 +00:00
/* TODO !!! */
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
case XML_ENTITY_DECL :
2002-04-20 06:41:40 +00:00
/* TODO !!! */
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
case XML_CDATA_SECTION_NODE :
case XML_TEXT_NODE :
2002-04-20 06:41:40 +00:00
if ( cur - > content ! = NULL )
return ( xmlStrdup ( cur - > content ) ) ;
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
}
2002-04-20 06:41:40 +00:00
return ( NULL ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlNodeSetContent :
* @ cur : the node being modified
* @ content : the new value of the content
*
* Replace the content of a node .
*/
void
xmlNodeSetContent ( xmlNodePtr cur , const xmlChar * content ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeSetContent : node == NULL \n " ) ;
# endif
return ;
}
switch ( cur - > type ) {
case XML_DOCUMENT_FRAG_NODE :
case XML_ELEMENT_NODE :
2002-01-16 15:37:50 +00:00
case XML_ATTRIBUTE_NODE :
2001-02-23 17:55:21 +00:00
if ( cur - > children ! = NULL ) xmlFreeNodeList ( cur - > children ) ;
cur - > children = xmlStringGetNodeList ( cur - > doc , content ) ;
UPDATE_LAST_CHILD_AND_PARENT ( cur )
break ;
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_PI_NODE :
case XML_COMMENT_NODE :
if ( cur - > content ! = NULL ) {
xmlFree ( cur - > content ) ;
}
if ( cur - > children ! = NULL ) xmlFreeNodeList ( cur - > children ) ;
cur - > last = cur - > children = NULL ;
if ( content ! = NULL ) {
cur - > content = xmlStrdup ( content ) ;
} else
cur - > content = NULL ;
break ;
case XML_DOCUMENT_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
break ;
case XML_NOTATION_NODE :
break ;
case XML_DTD_NODE :
break ;
case XML_NAMESPACE_DECL :
break ;
case XML_ELEMENT_DECL :
/* TODO !!! */
break ;
case XML_ATTRIBUTE_DECL :
/* TODO !!! */
break ;
case XML_ENTITY_DECL :
/* TODO !!! */
break ;
}
}
/**
* xmlNodeSetContentLen :
* @ cur : the node being modified
* @ content : the new value of the content
* @ len : the size of @ content
*
* Replace the content of a node .
*/
void
xmlNodeSetContentLen ( xmlNodePtr cur , const xmlChar * content , int len ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeSetContentLen : node == NULL \n " ) ;
# endif
return ;
}
switch ( cur - > type ) {
case XML_DOCUMENT_FRAG_NODE :
case XML_ELEMENT_NODE :
2002-01-16 15:37:50 +00:00
case XML_ATTRIBUTE_NODE :
2001-02-23 17:55:21 +00:00
if ( cur - > children ! = NULL ) xmlFreeNodeList ( cur - > children ) ;
cur - > children = xmlStringLenGetNodeList ( cur - > doc , content , len ) ;
UPDATE_LAST_CHILD_AND_PARENT ( cur )
break ;
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_PI_NODE :
case XML_COMMENT_NODE :
case XML_NOTATION_NODE :
if ( cur - > content ! = NULL ) {
xmlFree ( cur - > content ) ;
}
if ( cur - > children ! = NULL ) xmlFreeNodeList ( cur - > children ) ;
cur - > children = cur - > last = NULL ;
if ( content ! = NULL ) {
cur - > content = xmlStrndup ( content , len ) ;
} else
cur - > content = NULL ;
break ;
case XML_DOCUMENT_NODE :
case XML_DTD_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_NAMESPACE_DECL :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
break ;
case XML_ELEMENT_DECL :
/* TODO !!! */
break ;
case XML_ATTRIBUTE_DECL :
/* TODO !!! */
break ;
case XML_ENTITY_DECL :
/* TODO !!! */
break ;
}
}
/**
* xmlNodeAddContentLen :
* @ cur : the node being modified
* @ content : extra content
* @ len : the size of @ content
*
* Append the extra substring to the node content .
*/
void
xmlNodeAddContentLen ( xmlNodePtr cur , const xmlChar * content , int len ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeAddContentLen : node == NULL \n " ) ;
# endif
return ;
}
if ( len < = 0 ) return ;
switch ( cur - > type ) {
case XML_DOCUMENT_FRAG_NODE :
case XML_ELEMENT_NODE : {
2002-01-13 16:15:43 +00:00
xmlNodePtr last , newNode , tmp ;
2001-02-23 17:55:21 +00:00
2001-07-12 01:20:08 +00:00
last = cur - > last ;
2001-02-23 17:55:21 +00:00
newNode = xmlNewTextLen ( content , len ) ;
if ( newNode ! = NULL ) {
2002-01-13 16:15:43 +00:00
tmp = xmlAddChild ( cur , newNode ) ;
if ( tmp ! = newNode )
return ;
2001-02-23 17:55:21 +00:00
if ( ( last ! = NULL ) & & ( last - > next = = newNode ) ) {
xmlTextMerge ( last , newNode ) ;
}
}
break ;
}
case XML_ATTRIBUTE_NODE :
break ;
case XML_TEXT_NODE :
case XML_CDATA_SECTION_NODE :
case XML_ENTITY_REF_NODE :
case XML_ENTITY_NODE :
case XML_PI_NODE :
case XML_COMMENT_NODE :
case XML_NOTATION_NODE :
if ( content ! = NULL ) {
cur - > content = xmlStrncat ( cur - > content , content , len ) ;
}
case XML_DOCUMENT_NODE :
case XML_DTD_NODE :
case XML_HTML_DOCUMENT_NODE :
case XML_DOCUMENT_TYPE_NODE :
case XML_NAMESPACE_DECL :
case XML_XINCLUDE_START :
case XML_XINCLUDE_END :
2001-04-23 13:41:34 +00:00
# ifdef LIBXML_DOCB_ENABLED
case XML_DOCB_DOCUMENT_NODE :
2001-02-23 17:55:21 +00:00
# endif
break ;
case XML_ELEMENT_DECL :
case XML_ATTRIBUTE_DECL :
case XML_ENTITY_DECL :
break ;
}
}
/**
* xmlNodeAddContent :
* @ cur : the node being modified
* @ content : extra content
*
* Append the extra substring to the node content .
*/
void
xmlNodeAddContent ( xmlNodePtr cur , const xmlChar * content ) {
int len ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeAddContent : node == NULL \n " ) ;
# endif
return ;
}
if ( content = = NULL ) return ;
len = xmlStrlen ( content ) ;
xmlNodeAddContentLen ( cur , content , len ) ;
}
/**
* xmlTextMerge :
* @ first : the first text node
* @ second : the second text node being merged
*
* Merge two text nodes into one
* Returns the first text node augmented
*/
xmlNodePtr
xmlTextMerge ( xmlNodePtr first , xmlNodePtr second ) {
if ( first = = NULL ) return ( second ) ;
if ( second = = NULL ) return ( first ) ;
if ( first - > type ! = XML_TEXT_NODE ) return ( first ) ;
if ( second - > type ! = XML_TEXT_NODE ) return ( first ) ;
if ( second - > name ! = first - > name )
return ( first ) ;
xmlNodeAddContent ( first , second - > content ) ;
xmlUnlinkNode ( second ) ;
xmlFreeNode ( second ) ;
return ( first ) ;
}
/**
* xmlGetNsList :
* @ doc : the document
* @ node : the current node
*
* Search all the namespace applying to a given element .
2001-12-17 15:30:10 +00:00
* Returns an NULL terminated array of all the # xmlNsPtr found
2001-02-23 17:55:21 +00:00
* that need to be freed by the caller or NULL if no
* namespace if defined
*/
xmlNsPtr *
2001-06-29 21:31:07 +00:00
xmlGetNsList ( xmlDocPtr doc ATTRIBUTE_UNUSED , xmlNodePtr node )
{
2001-02-23 17:55:21 +00:00
xmlNsPtr cur ;
xmlNsPtr * ret = NULL ;
int nbns = 0 ;
int maxns = 10 ;
int i ;
while ( node ! = NULL ) {
2001-06-29 21:31:07 +00:00
if ( node - > type = = XML_ELEMENT_NODE ) {
cur = node - > nsDef ;
while ( cur ! = NULL ) {
if ( ret = = NULL ) {
ret =
( xmlNsPtr * ) xmlMalloc ( ( maxns + 1 ) *
sizeof ( xmlNsPtr ) ) ;
if ( ret = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlGetNsList : out of memory! \n " ) ;
return ( NULL ) ;
}
ret [ nbns ] = NULL ;
}
for ( i = 0 ; i < nbns ; i + + ) {
if ( ( cur - > prefix = = ret [ i ] - > prefix ) | |
( xmlStrEqual ( cur - > prefix , ret [ i ] - > prefix ) ) )
break ;
}
if ( i > = nbns ) {
if ( nbns > = maxns ) {
maxns * = 2 ;
ret = ( xmlNsPtr * ) xmlRealloc ( ret ,
( maxns +
1 ) *
sizeof ( xmlNsPtr ) ) ;
if ( ret = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlGetNsList : realloc failed! \n " ) ;
return ( NULL ) ;
}
}
ret [ nbns + + ] = cur ;
ret [ nbns ] = NULL ;
}
cur = cur - > next ;
}
}
node = node - > parent ;
2001-02-23 17:55:21 +00:00
}
2001-06-29 21:31:07 +00:00
return ( ret ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlSearchNs :
* @ doc : the document
* @ node : the current node
2001-02-27 21:54:07 +00:00
* @ nameSpace : the namespace prefix
2001-02-23 17:55:21 +00:00
*
* Search a Ns registered under a given name space for a document .
* recurse on the parents until it finds the defined namespace
* or return NULL otherwise .
* @ nameSpace can be NULL , this is a search for the default namespace .
* We don ' t allow to cross entities boundaries . If you don ' t declare
* the namespace within those you will be in troubles ! ! ! A warning
* is generated to cover this case .
*
* Returns the namespace pointer or NULL .
*/
xmlNsPtr
xmlSearchNs ( xmlDocPtr doc , xmlNodePtr node , const xmlChar * nameSpace ) {
xmlNsPtr cur ;
if ( node = = NULL ) return ( NULL ) ;
if ( ( nameSpace ! = NULL ) & &
( xmlStrEqual ( nameSpace , ( const xmlChar * ) " xml " ) ) ) {
2002-08-01 12:22:24 +00:00
if ( ( doc = = NULL ) & & ( node - > type = = XML_ELEMENT_NODE ) ) {
/*
* The XML - 1.0 namespace is normally held on the root
* element . In this case exceptionally create it on the
* node element .
*/
cur = ( xmlNsPtr ) xmlMalloc ( sizeof ( xmlNs ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlSearchNs : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNs ) ) ;
cur - > type = XML_LOCAL_NAMESPACE ;
cur - > href = xmlStrdup ( XML_XML_NAMESPACE ) ;
cur - > prefix = xmlStrdup ( ( const xmlChar * ) " xml " ) ;
cur - > next = node - > nsDef ;
node - > nsDef = cur ;
return ( cur ) ;
}
2001-02-23 17:55:21 +00:00
if ( doc - > oldNs = = NULL ) {
/*
* Allocate a new Namespace and fill the fields .
*/
doc - > oldNs = ( xmlNsPtr ) xmlMalloc ( sizeof ( xmlNs ) ) ;
if ( doc - > oldNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlSearchNs : malloc failed \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
memset ( doc - > oldNs , 0 , sizeof ( xmlNs ) ) ;
doc - > oldNs - > type = XML_LOCAL_NAMESPACE ;
doc - > oldNs - > href = xmlStrdup ( XML_XML_NAMESPACE ) ;
doc - > oldNs - > prefix = xmlStrdup ( ( const xmlChar * ) " xml " ) ;
}
return ( doc - > oldNs ) ;
}
while ( node ! = NULL ) {
if ( ( node - > type = = XML_ENTITY_REF_NODE ) | |
( node - > type = = XML_ENTITY_NODE ) | |
( node - > type = = XML_ENTITY_DECL ) )
return ( NULL ) ;
if ( node - > type = = XML_ELEMENT_NODE ) {
cur = node - > nsDef ;
while ( cur ! = NULL ) {
if ( ( cur - > prefix = = NULL ) & & ( nameSpace = = NULL ) & &
( cur - > href ! = NULL ) )
return ( cur ) ;
if ( ( cur - > prefix ! = NULL ) & & ( nameSpace ! = NULL ) & &
( cur - > href ! = NULL ) & &
( xmlStrEqual ( cur - > prefix , nameSpace ) ) )
return ( cur ) ;
cur = cur - > next ;
}
}
node = node - > parent ;
}
return ( NULL ) ;
}
/**
* xmlSearchNsByHref :
* @ doc : the document
* @ node : the current node
* @ href : the namespace value
*
* Search a Ns aliasing a given URI . Recurse on the parents until it finds
* the defined namespace or return NULL otherwise .
* Returns the namespace pointer or NULL .
*/
xmlNsPtr
xmlSearchNsByHref ( xmlDocPtr doc , xmlNodePtr node , const xmlChar * href ) {
xmlNsPtr cur ;
xmlNodePtr orig = node ;
if ( ( node = = NULL ) | | ( href = = NULL ) ) return ( NULL ) ;
if ( xmlStrEqual ( href , XML_XML_NAMESPACE ) ) {
2002-01-17 08:46:58 +00:00
/*
* Only the document can hold the XML spec namespace .
*/
2002-08-14 08:32:18 +00:00
if ( ( doc = = NULL ) & & ( node - > type = = XML_ELEMENT_NODE ) ) {
/*
* The XML - 1.0 namespace is normally held on the root
* element . In this case exceptionally create it on the
* node element .
*/
cur = ( xmlNsPtr ) xmlMalloc ( sizeof ( xmlNs ) ) ;
if ( cur = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlSearchNs : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( cur , 0 , sizeof ( xmlNs ) ) ;
cur - > type = XML_LOCAL_NAMESPACE ;
cur - > href = xmlStrdup ( XML_XML_NAMESPACE ) ;
cur - > prefix = xmlStrdup ( ( const xmlChar * ) " xml " ) ;
cur - > next = node - > nsDef ;
node - > nsDef = cur ;
return ( cur ) ;
}
2001-02-23 17:55:21 +00:00
if ( doc - > oldNs = = NULL ) {
/*
* Allocate a new Namespace and fill the fields .
*/
doc - > oldNs = ( xmlNsPtr ) xmlMalloc ( sizeof ( xmlNs ) ) ;
if ( doc - > oldNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlSearchNsByHref : malloc failed \n " ) ;
return ( NULL ) ;
}
memset ( doc - > oldNs , 0 , sizeof ( xmlNs ) ) ;
doc - > oldNs - > type = XML_LOCAL_NAMESPACE ;
doc - > oldNs - > href = xmlStrdup ( XML_XML_NAMESPACE ) ;
doc - > oldNs - > prefix = xmlStrdup ( ( const xmlChar * ) " xml " ) ;
}
return ( doc - > oldNs ) ;
}
while ( node ! = NULL ) {
cur = node - > nsDef ;
while ( cur ! = NULL ) {
if ( ( cur - > href ! = NULL ) & & ( href ! = NULL ) & &
( xmlStrEqual ( cur - > href , href ) ) ) {
/*
* Check that the prefix is not shadowed between orig and node
*/
xmlNodePtr check = orig ;
xmlNsPtr tst ;
while ( check ! = node ) {
tst = check - > nsDef ;
while ( tst ! = NULL ) {
if ( ( tst - > prefix = = NULL ) & & ( cur - > prefix = = NULL ) )
goto shadowed ;
if ( ( tst - > prefix ! = NULL ) & & ( cur - > prefix ! = NULL ) & &
( xmlStrEqual ( tst - > prefix , cur - > prefix ) ) )
goto shadowed ;
tst = tst - > next ;
}
check = check - > parent ;
}
return ( cur ) ;
}
shadowed :
cur = cur - > next ;
}
node = node - > parent ;
}
return ( NULL ) ;
}
/**
2002-12-10 15:19:08 +00:00
* xmlNewReconciliedNs :
2001-02-23 17:55:21 +00:00
* @ doc : the document
* @ tree : a node expected to hold the new namespace
* @ ns : the original namespace
*
* This function tries to locate a namespace definition in a tree
* ancestors , or create a new namespace definition node similar to
* @ ns trying to reuse the same prefix . However if the given prefix is
* null ( default namespace ) or reused within the subtree defined by
* @ tree or on one of its ancestors then a new prefix is generated .
* Returns the ( new ) namespace definition or NULL in case of error
*/
xmlNsPtr
xmlNewReconciliedNs ( xmlDocPtr doc , xmlNodePtr tree , xmlNsPtr ns ) {
xmlNsPtr def ;
xmlChar prefix [ 50 ] ;
int counter = 1 ;
if ( tree = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewReconciliedNs : tree == NULL \n " ) ;
# endif
return ( NULL ) ;
}
if ( ns = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNewReconciliedNs : ns == NULL \n " ) ;
# endif
return ( NULL ) ;
}
/*
* Search an existing namespace definition inherited .
*/
def = xmlSearchNsByHref ( doc , tree , ns - > href ) ;
if ( def ! = NULL )
return ( def ) ;
/*
* Find a close prefix which is not already in use .
* Let ' s strip namespace prefixes longer than 20 chars !
*/
2002-03-07 00:05:35 +00:00
if ( ns - > prefix = = NULL )
2002-06-14 17:07:10 +00:00
snprintf ( ( char * ) prefix , sizeof ( prefix ) , " default " ) ;
2002-03-07 00:05:35 +00:00
else
2002-06-14 17:07:10 +00:00
snprintf ( ( char * ) prefix , sizeof ( prefix ) , " %.20s " , ns - > prefix ) ;
2002-03-07 00:05:35 +00:00
2001-02-23 17:55:21 +00:00
def = xmlSearchNs ( doc , tree , prefix ) ;
while ( def ! = NULL ) {
if ( counter > 1000 ) return ( NULL ) ;
2002-03-07 00:05:35 +00:00
if ( ns - > prefix = = NULL )
2002-06-14 17:07:10 +00:00
snprintf ( ( char * ) prefix , sizeof ( prefix ) , " default%d " , counter + + ) ;
2002-03-07 00:05:35 +00:00
else
2002-06-14 17:07:10 +00:00
snprintf ( ( char * ) prefix , sizeof ( prefix ) , " %.20s%d " , ns - > prefix , counter + + ) ;
2001-02-23 17:55:21 +00:00
def = xmlSearchNs ( doc , tree , prefix ) ;
}
/*
2001-12-17 15:30:10 +00:00
* OK , now we are ready to create a new one .
2001-02-23 17:55:21 +00:00
*/
def = xmlNewNs ( tree , ns - > href , prefix ) ;
return ( def ) ;
}
/**
2002-12-10 15:19:08 +00:00
* xmlReconciliateNs :
2001-02-23 17:55:21 +00:00
* @ doc : the document
* @ tree : a node defining the subtree to reconciliate
*
* This function checks that all the namespaces declared within the given
* tree are properly declared . This is needed for example after Copy or Cut
* and then paste operations . The subtree may still hold pointers to
* namespace declarations outside the subtree or invalid / masked . As much
2001-12-17 15:30:10 +00:00
* as possible the function try to reuse the existing namespaces found in
2001-02-23 17:55:21 +00:00
* the new environment . If not possible the new namespaces are redeclared
* on @ tree at the top of the given subtree .
* Returns the number of namespace declarations created or - 1 in case of error .
*/
int
xmlReconciliateNs ( xmlDocPtr doc , xmlNodePtr tree ) {
xmlNsPtr * oldNs = NULL ;
xmlNsPtr * newNs = NULL ;
int sizeCache = 0 ;
int nbCache = 0 ;
xmlNsPtr n ;
xmlNodePtr node = tree ;
xmlAttrPtr attr ;
int ret = 0 , i ;
while ( node ! = NULL ) {
/*
* Reconciliate the node namespace
*/
if ( node - > ns ! = NULL ) {
/*
* initialize the cache if needed
*/
if ( sizeCache = = 0 ) {
sizeCache = 10 ;
oldNs = ( xmlNsPtr * ) xmlMalloc ( sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( oldNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
return ( - 1 ) ;
}
newNs = ( xmlNsPtr * ) xmlMalloc ( sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( newNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
xmlFree ( oldNs ) ;
return ( - 1 ) ;
}
}
for ( i = 0 ; i < nbCache ; i + + ) {
if ( oldNs [ i ] = = node - > ns ) {
node - > ns = newNs [ i ] ;
break ;
}
}
if ( i = = nbCache ) {
/*
2001-12-17 15:30:10 +00:00
* OK we need to recreate a new namespace definition
2001-02-23 17:55:21 +00:00
*/
n = xmlNewReconciliedNs ( doc , tree , node - > ns ) ;
if ( n ! = NULL ) { /* :-( what if else ??? */
/*
* check if we need to grow the cache buffers .
*/
if ( sizeCache < = nbCache ) {
sizeCache * = 2 ;
oldNs = ( xmlNsPtr * ) xmlRealloc ( oldNs , sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( oldNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
xmlFree ( newNs ) ;
return ( - 1 ) ;
}
newNs = ( xmlNsPtr * ) xmlRealloc ( newNs , sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( newNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
xmlFree ( oldNs ) ;
return ( - 1 ) ;
}
}
newNs [ nbCache ] = n ;
oldNs [ nbCache + + ] = node - > ns ;
node - > ns = n ;
}
}
}
/*
* now check for namespace hold by attributes on the node .
*/
attr = node - > properties ;
while ( attr ! = NULL ) {
if ( attr - > ns ! = NULL ) {
/*
* initialize the cache if needed
*/
if ( sizeCache = = 0 ) {
sizeCache = 10 ;
oldNs = ( xmlNsPtr * ) xmlMalloc ( sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( oldNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
return ( - 1 ) ;
}
newNs = ( xmlNsPtr * ) xmlMalloc ( sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( newNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
xmlFree ( oldNs ) ;
return ( - 1 ) ;
}
}
for ( i = 0 ; i < nbCache ; i + + ) {
if ( oldNs [ i ] = = attr - > ns ) {
2002-10-28 19:01:59 +00:00
attr - > ns = newNs [ i ] ;
2001-02-23 17:55:21 +00:00
break ;
}
}
if ( i = = nbCache ) {
/*
2001-12-17 15:30:10 +00:00
* OK we need to recreate a new namespace definition
2001-02-23 17:55:21 +00:00
*/
n = xmlNewReconciliedNs ( doc , tree , attr - > ns ) ;
if ( n ! = NULL ) { /* :-( what if else ??? */
/*
* check if we need to grow the cache buffers .
*/
if ( sizeCache < = nbCache ) {
sizeCache * = 2 ;
oldNs = ( xmlNsPtr * ) xmlRealloc ( oldNs , sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( oldNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
xmlFree ( newNs ) ;
return ( - 1 ) ;
}
newNs = ( xmlNsPtr * ) xmlRealloc ( newNs , sizeCache *
sizeof ( xmlNsPtr ) ) ;
if ( newNs = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlReconciliateNs : memory pbm \n " ) ;
xmlFree ( oldNs ) ;
return ( - 1 ) ;
}
}
newNs [ nbCache ] = n ;
oldNs [ nbCache + + ] = attr - > ns ;
attr - > ns = n ;
}
}
}
attr = attr - > next ;
}
/*
* Browse the full subtree , deep first
*/
if ( node - > children ! = NULL ) {
/* deep first */
node = node - > children ;
} else if ( ( node ! = tree ) & & ( node - > next ! = NULL ) ) {
/* then siblings */
node = node - > next ;
} else if ( node ! = tree ) {
/* go up to parents->next if needed */
while ( node ! = tree ) {
if ( node - > parent ! = NULL )
node = node - > parent ;
if ( ( node ! = tree ) & & ( node - > next ! = NULL ) ) {
node = node - > next ;
break ;
}
if ( node - > parent = = NULL ) {
node = NULL ;
break ;
}
}
/* exit condition */
if ( node = = tree )
node = NULL ;
2002-03-06 17:35:40 +00:00
} else
break ;
2001-02-23 17:55:21 +00:00
}
2002-03-07 00:05:35 +00:00
if ( oldNs ! = NULL )
xmlFree ( oldNs ) ;
if ( newNs ! = NULL )
xmlFree ( newNs ) ;
2001-02-23 17:55:21 +00:00
return ( ret ) ;
}
/**
* xmlHasProp :
* @ node : the node
* @ name : the attribute name
*
* Search an attribute associated to a node
* This function also looks in DTD attribute declaration for # FIXED or
* default declaration values unless DTD use has been turned off .
*
* Returns the attribute or the attribute declaration or NULL if
* neither was found .
*/
xmlAttrPtr
xmlHasProp ( xmlNodePtr node , const xmlChar * name ) {
xmlAttrPtr prop ;
xmlDocPtr doc ;
if ( ( node = = NULL ) | | ( name = = NULL ) ) return ( NULL ) ;
/*
* Check on the properties attached to the node
*/
prop = node - > properties ;
while ( prop ! = NULL ) {
if ( xmlStrEqual ( prop - > name , name ) ) {
return ( prop ) ;
}
prop = prop - > next ;
}
if ( ! xmlCheckDTD ) return ( NULL ) ;
/*
* Check if there is a default declaration in the internal
* or external subsets
*/
doc = node - > doc ;
if ( doc ! = NULL ) {
xmlAttributePtr attrDecl ;
if ( doc - > intSubset ! = NULL ) {
attrDecl = xmlGetDtdAttrDesc ( doc - > intSubset , node - > name , name ) ;
if ( ( attrDecl = = NULL ) & & ( doc - > extSubset ! = NULL ) )
attrDecl = xmlGetDtdAttrDesc ( doc - > extSubset , node - > name , name ) ;
if ( attrDecl ! = NULL )
return ( ( xmlAttrPtr ) attrDecl ) ;
}
}
return ( NULL ) ;
}
2001-06-06 10:46:28 +00:00
/**
* xmlHasNsProp :
* @ node : the node
* @ name : the attribute name
2001-06-11 12:09:01 +00:00
* @ nameSpace : the URI of the namespace
2001-06-06 10:46:28 +00:00
*
* Search for an attribute associated to a node
* This attribute has to be anchored in the namespace specified .
* This does the entity substitution .
* This function looks in DTD attribute declaration for # FIXED or
* default declaration values unless DTD use has been turned off .
*
* Returns the attribute or the attribute declaration or NULL
* if neither was found .
*/
xmlAttrPtr
2001-06-11 12:09:01 +00:00
xmlHasNsProp ( xmlNodePtr node , const xmlChar * name , const xmlChar * nameSpace ) {
2001-06-06 10:46:28 +00:00
xmlAttrPtr prop ;
xmlDocPtr doc ;
if ( node = = NULL )
return ( NULL ) ;
prop = node - > properties ;
2001-06-11 12:09:01 +00:00
if ( nameSpace = = NULL )
2001-06-06 10:46:28 +00:00
return ( xmlHasProp ( node , name ) ) ;
while ( prop ! = NULL ) {
/*
* One need to have
* - same attribute names
* - and the attribute carrying that namespace
*/
if ( ( xmlStrEqual ( prop - > name , name ) ) & &
2001-06-17 14:50:34 +00:00
( ( prop - > ns ! = NULL ) & & ( xmlStrEqual ( prop - > ns - > href , nameSpace ) ) ) ) {
return ( prop ) ;
2001-06-06 10:46:28 +00:00
}
prop = prop - > next ;
}
if ( ! xmlCheckDTD ) return ( NULL ) ;
/*
* Check if there is a default declaration in the internal
* or external subsets
*/
doc = node - > doc ;
if ( doc ! = NULL ) {
if ( doc - > intSubset ! = NULL ) {
2002-03-07 22:21:56 +00:00
xmlAttributePtr attrDecl = NULL ;
xmlNsPtr * nsList , * cur ;
xmlChar * ename ;
2001-06-06 10:46:28 +00:00
2002-03-07 22:21:56 +00:00
nsList = xmlGetNsList ( node - > doc , node ) ;
if ( nsList = = NULL )
return ( NULL ) ;
if ( ( node - > ns ! = NULL ) & & ( node - > ns - > prefix ! = NULL ) ) {
ename = xmlStrdup ( node - > ns - > prefix ) ;
ename = xmlStrcat ( ename , BAD_CAST " : " ) ;
ename = xmlStrcat ( ename , node - > name ) ;
} else {
ename = xmlStrdup ( node - > name ) ;
}
if ( ename = = NULL ) {
xmlFree ( nsList ) ;
return ( NULL ) ;
}
cur = nsList ;
while ( * cur ! = NULL ) {
if ( xmlStrEqual ( ( * cur ) - > href , nameSpace ) ) {
attrDecl = xmlGetDtdQAttrDesc ( doc - > intSubset , ename ,
name , ( * cur ) - > prefix ) ;
if ( ( attrDecl = = NULL ) & & ( doc - > extSubset ! = NULL ) )
attrDecl = xmlGetDtdQAttrDesc ( doc - > extSubset , ename ,
name , ( * cur ) - > prefix ) ;
}
cur + + ;
2001-06-06 10:46:28 +00:00
}
2002-03-07 22:21:56 +00:00
xmlFree ( nsList ) ;
xmlFree ( ename ) ;
return ( ( xmlAttrPtr ) attrDecl ) ;
2001-06-06 10:46:28 +00:00
}
}
return ( NULL ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlGetProp :
* @ node : the node
* @ name : the attribute name
*
* Search and get the value of an attribute associated to a node
* This does the entity substitution .
* This function looks in DTD attribute declaration for # FIXED or
* default declaration values unless DTD use has been turned off .
2003-02-16 15:50:27 +00:00
* NOTE : this function acts independently of namespaces associated
2003-02-05 13:19:53 +00:00
* to the attribute . Use xmlGetNsProp ( ) or xmlGetNoNsProp ( )
* for namespace aware processing .
2001-02-23 17:55:21 +00:00
*
* Returns the attribute value or NULL if not found .
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
xmlGetProp ( xmlNodePtr node , const xmlChar * name ) {
xmlAttrPtr prop ;
xmlDocPtr doc ;
if ( ( node = = NULL ) | | ( name = = NULL ) ) return ( NULL ) ;
/*
* Check on the properties attached to the node
*/
prop = node - > properties ;
while ( prop ! = NULL ) {
if ( xmlStrEqual ( prop - > name , name ) ) {
xmlChar * ret ;
ret = xmlNodeListGetString ( node - > doc , prop - > children , 1 ) ;
if ( ret = = NULL ) return ( xmlStrdup ( ( xmlChar * ) " " ) ) ;
return ( ret ) ;
}
prop = prop - > next ;
}
if ( ! xmlCheckDTD ) return ( NULL ) ;
/*
* Check if there is a default declaration in the internal
* or external subsets
*/
doc = node - > doc ;
if ( doc ! = NULL ) {
xmlAttributePtr attrDecl ;
if ( doc - > intSubset ! = NULL ) {
attrDecl = xmlGetDtdAttrDesc ( doc - > intSubset , node - > name , name ) ;
if ( ( attrDecl = = NULL ) & & ( doc - > extSubset ! = NULL ) )
attrDecl = xmlGetDtdAttrDesc ( doc - > extSubset , node - > name , name ) ;
if ( attrDecl ! = NULL )
return ( xmlStrdup ( attrDecl - > defaultValue ) ) ;
}
}
return ( NULL ) ;
}
2003-02-05 13:19:53 +00:00
/**
* xmlGetNoNsProp :
* @ node : the node
* @ name : the attribute name
*
* Search and get the value of an attribute associated to a node
* This does the entity substitution .
* This function looks in DTD attribute declaration for # FIXED or
* default declaration values unless DTD use has been turned off .
* This function is similar to xmlGetProp except it will accept only
* an attribute in no namespace .
*
* Returns the attribute value or NULL if not found .
* It ' s up to the caller to free the memory with xmlFree ( ) .
*/
xmlChar *
xmlGetNoNsProp ( xmlNodePtr node , const xmlChar * name ) {
xmlAttrPtr prop ;
xmlDocPtr doc ;
if ( ( node = = NULL ) | | ( name = = NULL ) ) return ( NULL ) ;
/*
* Check on the properties attached to the node
*/
prop = node - > properties ;
while ( prop ! = NULL ) {
if ( ( prop - > ns = = NULL ) & & ( xmlStrEqual ( prop - > name , name ) ) ) {
xmlChar * ret ;
ret = xmlNodeListGetString ( node - > doc , prop - > children , 1 ) ;
if ( ret = = NULL ) return ( xmlStrdup ( ( xmlChar * ) " " ) ) ;
return ( ret ) ;
}
prop = prop - > next ;
}
if ( ! xmlCheckDTD ) return ( NULL ) ;
/*
* Check if there is a default declaration in the internal
* or external subsets
*/
doc = node - > doc ;
if ( doc ! = NULL ) {
xmlAttributePtr attrDecl ;
if ( doc - > intSubset ! = NULL ) {
attrDecl = xmlGetDtdAttrDesc ( doc - > intSubset , node - > name , name ) ;
if ( ( attrDecl = = NULL ) & & ( doc - > extSubset ! = NULL ) )
attrDecl = xmlGetDtdAttrDesc ( doc - > extSubset , node - > name , name ) ;
if ( attrDecl ! = NULL )
return ( xmlStrdup ( attrDecl - > defaultValue ) ) ;
}
}
return ( NULL ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlGetNsProp :
* @ node : the node
* @ name : the attribute name
2001-06-11 12:09:01 +00:00
* @ nameSpace : the URI of the namespace
2001-02-23 17:55:21 +00:00
*
* Search and get the value of an attribute associated to a node
* This attribute has to be anchored in the namespace specified .
* This does the entity substitution .
* This function looks in DTD attribute declaration for # FIXED or
* default declaration values unless DTD use has been turned off .
*
* Returns the attribute value or NULL if not found .
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
xmlChar *
2001-06-11 12:09:01 +00:00
xmlGetNsProp ( xmlNodePtr node , const xmlChar * name , const xmlChar * nameSpace ) {
2001-02-23 17:55:21 +00:00
xmlAttrPtr prop ;
xmlDocPtr doc ;
xmlNsPtr ns ;
if ( node = = NULL )
return ( NULL ) ;
prop = node - > properties ;
2001-06-11 12:09:01 +00:00
if ( nameSpace = = NULL )
2003-02-05 13:19:53 +00:00
return ( xmlGetNoNsProp ( node , name ) ) ;
2001-02-23 17:55:21 +00:00
while ( prop ! = NULL ) {
/*
* One need to have
* - same attribute names
* - and the attribute carrying that namespace
*/
if ( ( xmlStrEqual ( prop - > name , name ) ) & &
2001-06-07 19:35:47 +00:00
( ( prop - > ns ! = NULL ) & &
2001-06-11 12:09:01 +00:00
( xmlStrEqual ( prop - > ns - > href , nameSpace ) ) ) ) {
2001-02-23 17:55:21 +00:00
xmlChar * ret ;
ret = xmlNodeListGetString ( node - > doc , prop - > children , 1 ) ;
if ( ret = = NULL ) return ( xmlStrdup ( ( xmlChar * ) " " ) ) ;
return ( ret ) ;
}
prop = prop - > next ;
}
if ( ! xmlCheckDTD ) return ( NULL ) ;
/*
* Check if there is a default declaration in the internal
* or external subsets
*/
doc = node - > doc ;
if ( doc ! = NULL ) {
if ( doc - > intSubset ! = NULL ) {
2001-04-30 17:44:45 +00:00
xmlAttributePtr attrDecl ;
2001-02-23 17:55:21 +00:00
attrDecl = xmlGetDtdAttrDesc ( doc - > intSubset , node - > name , name ) ;
if ( ( attrDecl = = NULL ) & & ( doc - > extSubset ! = NULL ) )
attrDecl = xmlGetDtdAttrDesc ( doc - > extSubset , node - > name , name ) ;
if ( ( attrDecl ! = NULL ) & & ( attrDecl - > prefix ! = NULL ) ) {
/*
* The DTD declaration only allows a prefix search
*/
ns = xmlSearchNs ( doc , node , attrDecl - > prefix ) ;
2001-06-11 12:09:01 +00:00
if ( ( ns ! = NULL ) & & ( xmlStrEqual ( ns - > href , nameSpace ) ) )
2001-02-23 17:55:21 +00:00
return ( xmlStrdup ( attrDecl - > defaultValue ) ) ;
}
}
}
return ( NULL ) ;
}
/**
* xmlSetProp :
* @ node : the node
* @ name : the attribute name
* @ value : the attribute value
*
* Set ( or reset ) an attribute carried by a node .
* Returns the attribute pointer .
*/
xmlAttrPtr
xmlSetProp ( xmlNodePtr node , const xmlChar * name , const xmlChar * value ) {
2001-11-25 10:35:25 +00:00
xmlAttrPtr prop ;
xmlDocPtr doc ;
2001-02-23 17:55:21 +00:00
if ( ( node = = NULL ) | | ( name = = NULL ) )
return ( NULL ) ;
doc = node - > doc ;
2001-11-25 10:35:25 +00:00
prop = node - > properties ;
2001-02-23 17:55:21 +00:00
while ( prop ! = NULL ) {
2001-05-11 17:41:21 +00:00
if ( ( xmlStrEqual ( prop - > name , name ) ) & &
( prop - > ns = = NULL ) ) {
2001-11-25 10:35:25 +00:00
xmlNodePtr oldprop = prop - > children ;
2001-02-23 17:55:21 +00:00
prop - > children = NULL ;
prop - > last = NULL ;
if ( value ! = NULL ) {
xmlChar * buffer ;
xmlNodePtr tmp ;
buffer = xmlEncodeEntitiesReentrant ( node - > doc , value ) ;
prop - > children = xmlStringGetNodeList ( node - > doc , buffer ) ;
prop - > last = NULL ;
prop - > doc = doc ;
tmp = prop - > children ;
while ( tmp ! = NULL ) {
tmp - > parent = ( xmlNodePtr ) prop ;
tmp - > doc = doc ;
if ( tmp - > next = = NULL )
prop - > last = tmp ;
tmp = tmp - > next ;
}
xmlFree ( buffer ) ;
2001-05-11 17:41:21 +00:00
}
2001-11-25 10:35:25 +00:00
if ( oldprop ! = NULL )
xmlFreeNodeList ( oldprop ) ;
2001-02-23 17:55:21 +00:00
return ( prop ) ;
}
prop = prop - > next ;
}
prop = xmlNewProp ( node , name , value ) ;
return ( prop ) ;
}
2001-05-11 17:41:21 +00:00
/**
* xmlUnsetProp :
* @ node : the node
* @ name : the attribute name
*
* Remove an attribute carried by a node .
* Returns 0 if successful , - 1 if not found
*/
int
xmlUnsetProp ( xmlNodePtr node , const xmlChar * name ) {
2003-01-23 18:24:20 +00:00
xmlAttrPtr prop , prev = NULL ; ;
2001-05-11 17:41:21 +00:00
if ( ( node = = NULL ) | | ( name = = NULL ) )
return ( - 1 ) ;
2003-01-23 18:24:20 +00:00
prop = node - > properties ;
2001-05-11 17:41:21 +00:00
while ( prop ! = NULL ) {
if ( ( xmlStrEqual ( prop - > name , name ) ) & &
( prop - > ns = = NULL ) ) {
if ( prev = = NULL )
node - > properties = prop - > next ;
else
prev - > next = prop - > next ;
xmlFreeProp ( prop ) ;
return ( 0 ) ;
}
prev = prop ;
prop = prop - > next ;
}
return ( - 1 ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlSetNsProp :
* @ node : the node
* @ ns : the namespace definition
* @ name : the attribute name
* @ value : the attribute value
*
* Set ( or reset ) an attribute carried by a node .
* The ns structure must be in scope , this is not checked .
*
* Returns the attribute pointer .
*/
xmlAttrPtr
xmlSetNsProp ( xmlNodePtr node , xmlNsPtr ns , const xmlChar * name ,
const xmlChar * value ) {
xmlAttrPtr prop ;
if ( ( node = = NULL ) | | ( name = = NULL ) )
return ( NULL ) ;
if ( ns = = NULL )
return ( xmlSetProp ( node , name , value ) ) ;
if ( ns - > href = = NULL )
return ( NULL ) ;
prop = node - > properties ;
while ( prop ! = NULL ) {
/*
* One need to have
* - same attribute names
* - and the attribute carrying that namespace
*/
if ( ( xmlStrEqual ( prop - > name , name ) ) & &
2002-08-01 12:52:24 +00:00
( prop - > ns ! = NULL ) & & ( xmlStrEqual ( prop - > ns - > href , ns - > href ) ) ) {
2001-02-23 17:55:21 +00:00
if ( prop - > children ! = NULL )
xmlFreeNodeList ( prop - > children ) ;
prop - > children = NULL ;
prop - > last = NULL ;
prop - > ns = ns ;
if ( value ! = NULL ) {
xmlChar * buffer ;
xmlNodePtr tmp ;
buffer = xmlEncodeEntitiesReentrant ( node - > doc , value ) ;
prop - > children = xmlStringGetNodeList ( node - > doc , buffer ) ;
prop - > last = NULL ;
tmp = prop - > children ;
while ( tmp ! = NULL ) {
tmp - > parent = ( xmlNodePtr ) prop ;
if ( tmp - > next = = NULL )
prop - > last = tmp ;
tmp = tmp - > next ;
}
xmlFree ( buffer ) ;
}
return ( prop ) ;
}
prop = prop - > next ;
}
prop = xmlNewNsProp ( node , ns , name , value ) ;
return ( prop ) ;
}
2001-05-11 17:41:21 +00:00
/**
* xmlUnsetNsProp :
* @ node : the node
* @ ns : the namespace definition
* @ name : the attribute name
*
* Remove an attribute carried by a node .
* Returns 0 if successful , - 1 if not found
*/
int
xmlUnsetNsProp ( xmlNodePtr node , xmlNsPtr ns , const xmlChar * name ) {
xmlAttrPtr prop = node - > properties , prev = NULL ; ;
if ( ( node = = NULL ) | | ( name = = NULL ) )
return ( - 1 ) ;
if ( ns = = NULL )
return ( xmlUnsetProp ( node , name ) ) ;
if ( ns - > href = = NULL )
return ( - 1 ) ;
while ( prop ! = NULL ) {
if ( ( xmlStrEqual ( prop - > name , name ) ) & &
2002-08-01 12:54:11 +00:00
( prop - > ns ! = NULL ) & & ( xmlStrEqual ( prop - > ns - > href , ns - > href ) ) ) {
2001-05-11 17:41:21 +00:00
if ( prev = = NULL )
node - > properties = prop - > next ;
else
prev - > next = prop - > next ;
xmlFreeProp ( prop ) ;
return ( 0 ) ;
}
prev = prop ;
prop = prop - > next ;
}
return ( - 1 ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlNodeIsText :
* @ node : the node
*
* Is this node a Text node ?
* Returns 1 yes , 0 no
*/
int
xmlNodeIsText ( xmlNodePtr node ) {
if ( node = = NULL ) return ( 0 ) ;
if ( node - > type = = XML_TEXT_NODE ) return ( 1 ) ;
return ( 0 ) ;
}
/**
* xmlIsBlankNode :
* @ node : the node
*
* Checks whether this node is an empty or whitespace only
* ( and possibly ignorable ) text - node .
*
* Returns 1 yes , 0 no
*/
int
xmlIsBlankNode ( xmlNodePtr node ) {
const xmlChar * cur ;
if ( node = = NULL ) return ( 0 ) ;
2001-07-12 01:20:08 +00:00
if ( ( node - > type ! = XML_TEXT_NODE ) & &
( node - > type ! = XML_CDATA_SECTION_NODE ) )
return ( 0 ) ;
2001-02-23 17:55:21 +00:00
if ( node - > content = = NULL ) return ( 1 ) ;
cur = node - > content ;
while ( * cur ! = 0 ) {
if ( ! IS_BLANK ( * cur ) ) return ( 0 ) ;
cur + + ;
}
return ( 1 ) ;
}
/**
* xmlTextConcat :
* @ node : the node
* @ content : the content
2001-10-10 09:45:09 +00:00
* @ len : @ content length
2001-02-23 17:55:21 +00:00
*
* Concat the given string at the end of the existing node content
*/
void
xmlTextConcat ( xmlNodePtr node , const xmlChar * content , int len ) {
if ( node = = NULL ) return ;
if ( ( node - > type ! = XML_TEXT_NODE ) & &
( node - > type ! = XML_CDATA_SECTION_NODE ) ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlTextConcat: node is not text nor CDATA \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
node - > content = xmlStrncat ( node - > content , content , len ) ;
}
/************************************************************************
* *
* Output : to a FILE or in memory *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/**
* xmlBufferCreate :
*
* routine to create an XML buffer .
* returns the new structure .
*/
xmlBufferPtr
xmlBufferCreate ( void ) {
xmlBufferPtr ret ;
ret = ( xmlBufferPtr ) xmlMalloc ( sizeof ( xmlBuffer ) ) ;
if ( ret = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferCreate : out of memory! \n " ) ;
return ( NULL ) ;
}
ret - > use = 0 ;
2001-03-10 12:32:04 +00:00
ret - > size = xmlDefaultBufferSize ;
2001-02-23 17:55:21 +00:00
ret - > alloc = xmlBufferAllocScheme ;
ret - > content = ( xmlChar * ) xmlMalloc ( ret - > size * sizeof ( xmlChar ) ) ;
if ( ret - > content = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferCreate : out of memory! \n " ) ;
xmlFree ( ret ) ;
return ( NULL ) ;
}
ret - > content [ 0 ] = 0 ;
return ( ret ) ;
}
/**
* xmlBufferCreateSize :
* @ size : initial size of buffer
*
* routine to create an XML buffer .
* returns the new structure .
*/
xmlBufferPtr
xmlBufferCreateSize ( size_t size ) {
xmlBufferPtr ret ;
ret = ( xmlBufferPtr ) xmlMalloc ( sizeof ( xmlBuffer ) ) ;
if ( ret = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferCreate : out of memory! \n " ) ;
return ( NULL ) ;
}
ret - > use = 0 ;
ret - > alloc = xmlBufferAllocScheme ;
ret - > size = ( size ? size + 2 : 0 ) ; /* +1 for ending null */
if ( ret - > size ) {
ret - > content = ( xmlChar * ) xmlMalloc ( ret - > size * sizeof ( xmlChar ) ) ;
if ( ret - > content = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferCreate : out of memory! \n " ) ;
xmlFree ( ret ) ;
return ( NULL ) ;
}
ret - > content [ 0 ] = 0 ;
} else
ret - > content = NULL ;
return ( ret ) ;
}
/**
* xmlBufferSetAllocationScheme :
2002-09-25 22:25:35 +00:00
* @ buf : the buffer to tune
2001-02-23 17:55:21 +00:00
* @ scheme : allocation scheme to use
*
* Sets the allocation scheme for this buffer
*/
void
xmlBufferSetAllocationScheme ( xmlBufferPtr buf ,
xmlBufferAllocationScheme scheme ) {
if ( buf = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferSetAllocationScheme: buf == NULL \n " ) ;
# endif
return ;
}
buf - > alloc = scheme ;
}
/**
* xmlBufferFree :
* @ buf : the buffer to free
*
2002-01-22 18:15:52 +00:00
* Frees an XML buffer . It frees both the content and the structure which
* encapsulate it .
2001-02-23 17:55:21 +00:00
*/
void
xmlBufferFree ( xmlBufferPtr buf ) {
if ( buf = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferFree: buf == NULL \n " ) ;
# endif
return ;
}
2002-03-20 21:55:57 +00:00
if ( buf - > content ! = NULL ) {
2001-02-23 17:55:21 +00:00
xmlFree ( buf - > content ) ;
}
xmlFree ( buf ) ;
}
/**
* xmlBufferEmpty :
* @ buf : the buffer
*
* empty a buffer .
*/
void
xmlBufferEmpty ( xmlBufferPtr buf ) {
if ( buf - > content = = NULL ) return ;
buf - > use = 0 ;
2001-03-27 12:47:33 +00:00
memset ( buf - > content , 0 , buf - > size ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlBufferShrink :
* @ buf : the buffer to dump
* @ len : the number of xmlChar to remove
*
* Remove the beginning of an XML buffer .
*
2001-12-17 15:30:10 +00:00
* Returns the number of # xmlChar removed , or - 1 in case of failure .
2001-02-23 17:55:21 +00:00
*/
int
xmlBufferShrink ( xmlBufferPtr buf , unsigned int len ) {
if ( len = = 0 ) return ( 0 ) ;
if ( len > buf - > use ) return ( - 1 ) ;
buf - > use - = len ;
memmove ( buf - > content , & buf - > content [ len ] , buf - > use * sizeof ( xmlChar ) ) ;
buf - > content [ buf - > use ] = 0 ;
return ( len ) ;
}
/**
* xmlBufferGrow :
* @ buf : the buffer
* @ len : the minimum free size to allocate
*
* Grow the available space of an XML buffer .
*
* Returns the new available space or - 1 in case of error
*/
int
xmlBufferGrow ( xmlBufferPtr buf , unsigned int len ) {
int size ;
xmlChar * newbuf ;
if ( len + buf - > use < buf - > size ) return ( 0 ) ;
size = buf - > use + len + 100 ;
newbuf = ( xmlChar * ) xmlRealloc ( buf - > content , size ) ;
if ( newbuf = = NULL ) return ( - 1 ) ;
buf - > content = newbuf ;
buf - > size = size ;
return ( buf - > size - buf - > use ) ;
}
/**
* xmlBufferDump :
* @ file : the file output
* @ buf : the buffer to dump
*
* Dumps an XML buffer to a FILE * .
2001-12-17 15:30:10 +00:00
* Returns the number of # xmlChar written
2001-02-23 17:55:21 +00:00
*/
int
xmlBufferDump ( FILE * file , xmlBufferPtr buf ) {
int ret ;
if ( buf = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferDump: buf == NULL \n " ) ;
# endif
return ( 0 ) ;
}
if ( buf - > content = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferDump: buf->content == NULL \n " ) ;
# endif
return ( 0 ) ;
}
2001-11-22 18:20:37 +00:00
if ( file = = NULL )
file = stdout ;
2001-02-23 17:55:21 +00:00
ret = fwrite ( buf - > content , sizeof ( xmlChar ) , buf - > use , file ) ;
return ( ret ) ;
}
/**
* xmlBufferContent :
* @ buf : the buffer
*
2001-07-18 19:30:27 +00:00
* Function to extract the content of a buffer
*
2001-02-23 17:55:21 +00:00
* Returns the internal content
*/
2001-07-18 19:30:27 +00:00
const xmlChar *
2001-02-23 17:55:21 +00:00
xmlBufferContent ( const xmlBufferPtr buf )
{
if ( ! buf )
return NULL ;
return buf - > content ;
}
/**
* xmlBufferLength :
* @ buf : the buffer
*
2001-07-18 19:30:27 +00:00
* Function to get the length of a buffer
*
2001-02-23 17:55:21 +00:00
* Returns the length of data in the internal content
*/
int
xmlBufferLength ( const xmlBufferPtr buf )
{
if ( ! buf )
return 0 ;
return buf - > use ;
}
/**
* xmlBufferResize :
* @ buf : the buffer to resize
* @ size : the desired size
*
2001-12-17 15:30:10 +00:00
* Resize a buffer to accommodate minimum size of @ size .
2001-02-23 17:55:21 +00:00
*
* Returns 0 in case of problems , 1 otherwise
*/
int
xmlBufferResize ( xmlBufferPtr buf , unsigned int size )
{
unsigned int newSize ;
xmlChar * rebuf = NULL ;
/*take care of empty case*/
newSize = ( buf - > size ? buf - > size * 2 : size ) ;
/* Don't resize if we don't have to */
if ( size < buf - > size )
return 1 ;
/* figure out new size */
switch ( buf - > alloc ) {
case XML_BUFFER_ALLOC_DOUBLEIT :
while ( size > newSize ) newSize * = 2 ;
break ;
case XML_BUFFER_ALLOC_EXACT :
newSize = size + 10 ;
break ;
default :
newSize = size + 10 ;
break ;
}
if ( buf - > content = = NULL )
rebuf = ( xmlChar * ) xmlMalloc ( newSize * sizeof ( xmlChar ) ) ;
else
rebuf = ( xmlChar * ) xmlRealloc ( buf - > content ,
newSize * sizeof ( xmlChar ) ) ;
if ( rebuf = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlBufferResize : out of memory! \n " ) ;
2001-02-23 17:55:21 +00:00
return 0 ;
}
buf - > content = rebuf ;
buf - > size = newSize ;
return 1 ;
}
/**
* xmlBufferAdd :
* @ buf : the buffer to dump
2001-12-17 15:30:10 +00:00
* @ str : the # xmlChar string
* @ len : the number of # xmlChar to add
2001-02-23 17:55:21 +00:00
*
2001-10-10 09:45:09 +00:00
* Add a string range to an XML buffer . if len = = - 1 , the length of
2001-02-23 17:55:21 +00:00
* str is recomputed .
*/
void
xmlBufferAdd ( xmlBufferPtr buf , const xmlChar * str , int len ) {
unsigned int needSize ;
if ( str = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferAdd: str == NULL \n " ) ;
# endif
return ;
}
if ( len < - 1 ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferAdd: len < 0 \n " ) ;
# endif
return ;
}
if ( len = = 0 ) return ;
if ( len < 0 )
len = xmlStrlen ( str ) ;
if ( len < = 0 ) return ;
needSize = buf - > use + len + 2 ;
if ( needSize > buf - > size ) {
if ( ! xmlBufferResize ( buf , needSize ) ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferAdd : out of memory! \n " ) ;
return ;
}
}
memmove ( & buf - > content [ buf - > use ] , str , len * sizeof ( xmlChar ) ) ;
buf - > use + = len ;
buf - > content [ buf - > use ] = 0 ;
}
/**
* xmlBufferAddHead :
* @ buf : the buffer
2001-12-17 15:30:10 +00:00
* @ str : the # xmlChar string
* @ len : the number of # xmlChar to add
2001-02-23 17:55:21 +00:00
*
* Add a string range to the beginning of an XML buffer .
2001-10-10 09:45:09 +00:00
* if len = = - 1 , the length of @ str is recomputed .
2001-02-23 17:55:21 +00:00
*/
void
xmlBufferAddHead ( xmlBufferPtr buf , const xmlChar * str , int len ) {
unsigned int needSize ;
if ( str = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlBufferAddHead: str == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
if ( len < - 1 ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlBufferAddHead: len < 0 \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
if ( len = = 0 ) return ;
if ( len < 0 )
len = xmlStrlen ( str ) ;
if ( len < = 0 ) return ;
needSize = buf - > use + len + 2 ;
if ( needSize > buf - > size ) {
if ( ! xmlBufferResize ( buf , needSize ) ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferAddHead : out of memory! \n " ) ;
return ;
}
}
memmove ( & buf - > content [ len ] , & buf - > content [ 0 ] , buf - > use * sizeof ( xmlChar ) ) ;
memmove ( & buf - > content [ 0 ] , str , len * sizeof ( xmlChar ) ) ;
buf - > use + = len ;
buf - > content [ buf - > use ] = 0 ;
}
/**
* xmlBufferCat :
* @ buf : the buffer to dump
2001-12-17 15:30:10 +00:00
* @ str : the # xmlChar string
2001-02-23 17:55:21 +00:00
*
* Append a zero terminated string to an XML buffer .
*/
void
xmlBufferCat ( xmlBufferPtr buf , const xmlChar * str ) {
if ( str ! = NULL )
xmlBufferAdd ( buf , str , - 1 ) ;
}
/**
* xmlBufferCCat :
* @ buf : the buffer to dump
* @ str : the C char string
*
* Append a zero terminated C string to an XML buffer .
*/
void
xmlBufferCCat ( xmlBufferPtr buf , const char * str ) {
const char * cur ;
if ( str = = NULL ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlBufferCCat: str == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
for ( cur = str ; * cur ! = 0 ; cur + + ) {
if ( buf - > use + 10 > = buf - > size ) {
if ( ! xmlBufferResize ( buf , buf - > use + 10 ) ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferCCat : out of memory! \n " ) ;
return ;
}
}
buf - > content [ buf - > use + + ] = * cur ;
}
buf - > content [ buf - > use ] = 0 ;
}
/**
* xmlBufferWriteCHAR :
* @ buf : the XML buffer
* @ string : the string to add
*
* routine which manages and grows an output buffer . This one adds
* xmlChars at the end of the buffer .
*/
void
xmlBufferWriteCHAR
( xmlBufferPtr buf , const xmlChar * string ) {
xmlBufferCat ( buf , string ) ;
}
/**
* xmlBufferWriteChar :
* @ buf : the XML buffer output
* @ string : the string to add
*
* routine which manage and grows an output buffer . This one add
* C chars at the end of the array .
*/
void
xmlBufferWriteChar ( xmlBufferPtr buf , const char * string ) {
xmlBufferCCat ( buf , string ) ;
}
/**
* xmlBufferWriteQuotedString :
* @ buf : the XML buffer output
* @ string : the string to add
*
* routine which manage and grows an output buffer . This one writes
2001-12-17 15:30:10 +00:00
* a quoted or double quoted # xmlChar string , checking first if it holds
2001-02-23 17:55:21 +00:00
* quote or double - quotes internally
*/
void
xmlBufferWriteQuotedString ( xmlBufferPtr buf , const xmlChar * string ) {
if ( xmlStrchr ( string , ' " ' ) ) {
if ( xmlStrchr ( string , ' \' ' ) ) {
# ifdef DEBUG_BUFFER
xmlGenericError ( xmlGenericErrorContext ,
" xmlBufferWriteQuotedString: string contains quote and double-quotes ! \n " ) ;
# endif
}
xmlBufferCCat ( buf , " ' " ) ;
xmlBufferCat ( buf , string ) ;
xmlBufferCCat ( buf , " ' " ) ;
} else {
xmlBufferCCat ( buf , " \" " ) ;
xmlBufferCat ( buf , string ) ;
xmlBufferCCat ( buf , " \" " ) ;
}
}
/************************************************************************
* *
* Dumping XML tree content to a simple buffer *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-02-13 13:07:41 +00:00
/**
* xmlAttrSerializeContent :
* @ buf : the XML buffer output
* @ doc : the document
* @ attr : the attribute pointer
*
* Serialize the attribute in the buffer
*/
static void
2002-11-27 11:43:05 +00:00
xmlAttrSerializeContent ( xmlBufferPtr buf , xmlDocPtr doc , xmlAttrPtr attr )
{
2002-02-13 13:07:41 +00:00
const xmlChar * cur , * base ;
xmlNodePtr children ;
children = attr - > children ;
while ( children ! = NULL ) {
2002-11-27 11:43:05 +00:00
switch ( children - > type ) {
case XML_TEXT_NODE :
base = cur = children - > content ;
while ( * cur ! = 0 ) {
if ( * cur = = ' \n ' ) {
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
xmlBufferAdd ( buf , BAD_CAST " " , 5 ) ;
cur + + ;
base = cur ;
2002-02-13 13:07:41 +00:00
#if 0
2002-11-27 11:43:05 +00:00
} else if ( * cur = = ' \' ' ) {
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
xmlBufferAdd ( buf , BAD_CAST " ' " , 6 ) ;
cur + + ;
base = cur ;
2002-02-13 13:07:41 +00:00
# endif
2002-11-27 11:43:05 +00:00
} else if ( * cur = = ' " ' ) {
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
xmlBufferAdd ( buf , BAD_CAST " " " , 6 ) ;
cur + + ;
base = cur ;
} else if ( * cur = = ' < ' ) {
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
xmlBufferAdd ( buf , BAD_CAST " < " , 4 ) ;
cur + + ;
base = cur ;
} else if ( * cur = = ' > ' ) {
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
xmlBufferAdd ( buf , BAD_CAST " > " , 4 ) ;
cur + + ;
base = cur ;
} else if ( * cur = = ' & ' ) {
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
xmlBufferAdd ( buf , BAD_CAST " & " , 5 ) ;
cur + + ;
base = cur ;
} else if ( ( * cur > = 0x80 ) & & ( ( doc = = NULL ) | |
( doc - > encoding = =
NULL ) ) ) {
/*
* We assume we have UTF - 8 content .
*/
char tmp [ 10 ] ;
int val = 0 , l = 1 ;
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
if ( * cur < 0xC0 ) {
xmlGenericError ( xmlGenericErrorContext ,
2002-02-13 13:07:41 +00:00
" xmlAttrSerializeContent : input not UTF-8 \n " ) ;
2002-11-27 11:43:05 +00:00
if ( doc ! = NULL )
doc - > encoding =
xmlStrdup ( BAD_CAST " ISO-8859-1 " ) ;
snprintf ( tmp , sizeof ( tmp ) , " &#%d; " , * cur ) ;
tmp [ sizeof ( tmp ) - 1 ] = 0 ;
xmlBufferAdd ( buf , ( xmlChar * ) tmp , - 1 ) ;
cur + + ;
base = cur ;
continue ;
} else if ( * cur < 0xE0 ) {
val = ( cur [ 0 ] ) & 0x1F ;
val < < = 6 ;
val | = ( cur [ 1 ] ) & 0x3F ;
l = 2 ;
} else if ( * cur < 0xF0 ) {
val = ( cur [ 0 ] ) & 0x0F ;
val < < = 6 ;
val | = ( cur [ 1 ] ) & 0x3F ;
val < < = 6 ;
val | = ( cur [ 2 ] ) & 0x3F ;
l = 3 ;
} else if ( * cur < 0xF8 ) {
val = ( cur [ 0 ] ) & 0x07 ;
val < < = 6 ;
val | = ( cur [ 1 ] ) & 0x3F ;
val < < = 6 ;
val | = ( cur [ 2 ] ) & 0x3F ;
val < < = 6 ;
val | = ( cur [ 3 ] ) & 0x3F ;
l = 4 ;
}
if ( ( l = = 1 ) | | ( ! IS_CHAR ( val ) ) ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlAttrSerializeContent : char out of range \n " ) ;
if ( doc ! = NULL )
doc - > encoding =
xmlStrdup ( BAD_CAST " ISO-8859-1 " ) ;
snprintf ( tmp , sizeof ( tmp ) , " &#%d; " , * cur ) ;
tmp [ sizeof ( tmp ) - 1 ] = 0 ;
xmlBufferAdd ( buf , ( xmlChar * ) tmp , - 1 ) ;
cur + + ;
base = cur ;
continue ;
}
/*
* We could do multiple things here . Just save
* as a char ref
*/
snprintf ( tmp , sizeof ( tmp ) , " &#x%X; " , val ) ;
tmp [ sizeof ( tmp ) - 1 ] = 0 ;
xmlBufferAdd ( buf , ( xmlChar * ) tmp , - 1 ) ;
cur + = l ;
base = cur ;
} else {
cur + + ;
}
}
if ( base ! = cur )
xmlBufferAdd ( buf , base , cur - base ) ;
break ;
case XML_ENTITY_REF_NODE :
xmlBufferAdd ( buf , BAD_CAST " & " , 1 ) ;
xmlBufferAdd ( buf , children - > name ,
xmlStrlen ( children - > name ) ) ;
xmlBufferAdd ( buf , BAD_CAST " ; " , 1 ) ;
break ;
default :
/* should not happen unless we have a badly built tree */
break ;
}
children = children - > next ;
2001-02-23 17:55:21 +00:00
}
}
/**
* xmlNodeDump :
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the current node
* @ level : the imbrication level for indenting
* @ format : is formatting allowed
*
* Dump an XML node , recursive behaviour , children are printed too .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-03-19 14:36:46 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
2002-11-27 11:43:05 +00:00
*
* Returns the number of bytes written to the buffer or - 1 in case of error
2001-02-23 17:55:21 +00:00
*/
2002-11-27 11:43:05 +00:00
int
2001-02-23 17:55:21 +00:00
xmlNodeDump ( xmlBufferPtr buf , xmlDocPtr doc , xmlNodePtr cur , int level ,
2002-11-27 11:43:05 +00:00
int format )
{
unsigned int use ;
int ret ;
xmlOutputBufferPtr outbuf ;
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2002-11-27 11:43:05 +00:00
" xmlNodeDump : node == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
2002-11-27 11:43:05 +00:00
return ( - 1 ) ;
2001-02-23 17:55:21 +00:00
}
2002-11-27 11:43:05 +00:00
if ( buf = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeDump : buf == NULL \n " ) ;
# endif
return ( - 1 ) ;
2001-02-23 17:55:21 +00:00
}
2002-11-27 11:43:05 +00:00
outbuf = ( xmlOutputBufferPtr ) xmlMalloc ( sizeof ( xmlOutputBuffer ) ) ;
if ( outbuf = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeDump: out of memory! \n " ) ;
return ( - 1 ) ;
}
memset ( outbuf , 0 , ( size_t ) sizeof ( xmlOutputBuffer ) ) ;
outbuf - > buffer = buf ;
outbuf - > encoder = NULL ;
outbuf - > writecallback = NULL ;
outbuf - > closecallback = NULL ;
outbuf - > context = NULL ;
outbuf - > written = 0 ;
use = buf - > use ;
xmlNodeDumpOutput ( outbuf , doc , cur , level , format , NULL ) ;
xmlFree ( outbuf ) ;
ret = buf - > use - use ;
return ( ret ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlElemDump :
* @ f : the FILE * for the output
* @ doc : the document
* @ cur : the current node
*
2001-12-17 15:30:10 +00:00
* Dump an XML / HTML node , recursive behaviour , children are printed too .
2001-02-23 17:55:21 +00:00
*/
void
2002-11-27 11:43:05 +00:00
xmlElemDump ( FILE * f , xmlDocPtr doc , xmlNodePtr cur )
{
xmlOutputBufferPtr outbuf ;
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2002-11-27 11:43:05 +00:00
" xmlElemDump : cur == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
2002-11-27 11:43:05 +00:00
return ;
2001-02-23 17:55:21 +00:00
}
# ifdef DEBUG_TREE
2001-06-21 22:07:42 +00:00
if ( doc = = NULL ) {
2001-02-23 17:55:21 +00:00
xmlGenericError ( xmlGenericErrorContext ,
2002-11-27 11:43:05 +00:00
" xmlElemDump : doc == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
}
2001-06-21 22:07:42 +00:00
# endif
2002-11-27 11:43:05 +00:00
outbuf = xmlOutputBufferCreateFile ( f , NULL ) ;
if ( outbuf = = NULL )
return ;
if ( ( doc ! = NULL ) & & ( doc - > type = = XML_HTML_DOCUMENT_NODE ) ) {
2001-02-23 17:55:21 +00:00
# ifdef LIBXML_HTML_ENABLED
2002-11-27 11:43:05 +00:00
htmlNodeDumpOutput ( outbuf , doc , cur , NULL ) ;
# else
xmlGenericError ( xmlGenericErrorContext ,
" HTML support not compiled in \n " ) ;
2001-02-23 17:55:21 +00:00
# endif /* LIBXML_HTML_ENABLED */
} else
2002-11-27 11:43:05 +00:00
xmlNodeDumpOutput ( outbuf , doc , cur , 0 , 1 , NULL ) ;
xmlOutputBufferClose ( outbuf ) ;
2001-02-23 17:55:21 +00:00
}
/************************************************************************
* *
* Dumping XML tree content to an I / O output buffer *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void
2002-11-21 14:10:52 +00:00
xhtmlNodeDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc , xmlNodePtr cur ,
int level , int format , const char * encoding ) ;
static void
2001-02-23 17:55:21 +00:00
xmlNodeListDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc , xmlNodePtr cur ,
int level , int format , const char * encoding ) ;
2002-11-21 14:10:52 +00:00
static void
xmlNodeDumpOutputInternal ( xmlOutputBufferPtr buf , xmlDocPtr doc ,
xmlNodePtr cur , int level , int format , const char * encoding ) ;
2003-01-09 13:19:33 +00:00
void xmlNsListDumpOutput ( xmlOutputBufferPtr buf , xmlNsPtr cur ) ;
2001-02-23 17:55:21 +00:00
/**
* xmlNsDumpOutput :
* @ buf : the XML buffer output
* @ cur : a namespace
*
* Dump a local Namespace definition .
* Should be called in the context of attributes dumps .
*/
static void
xmlNsDumpOutput ( xmlOutputBufferPtr buf , xmlNsPtr cur ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNsDumpOutput : Ns == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
if ( ( cur - > type = = XML_LOCAL_NAMESPACE ) & & ( cur - > href ! = NULL ) ) {
2002-08-01 12:22:24 +00:00
if ( xmlStrEqual ( cur - > prefix , BAD_CAST " xml " ) )
return ;
2001-02-23 17:55:21 +00:00
/* Within the context of an element attributes */
if ( cur - > prefix ! = NULL ) {
xmlOutputBufferWriteString ( buf , " xmlns: " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > prefix ) ;
} else
xmlOutputBufferWriteString ( buf , " xmlns " ) ;
xmlOutputBufferWriteString ( buf , " = " ) ;
xmlBufferWriteQuotedString ( buf - > buffer , cur - > href ) ;
}
}
/**
* xmlNsListDumpOutput :
* @ buf : the XML buffer output
* @ cur : the first namespace
*
* Dump a list of local Namespace definitions .
* Should be called in the context of attributes dumps .
*/
2003-01-09 13:19:33 +00:00
void
2001-02-23 17:55:21 +00:00
xmlNsListDumpOutput ( xmlOutputBufferPtr buf , xmlNsPtr cur ) {
while ( cur ! = NULL ) {
xmlNsDumpOutput ( buf , cur ) ;
cur = cur - > next ;
}
}
/**
* xmlDtdDumpOutput :
* @ buf : the XML buffer output
* @ doc : the document
* @ encoding : an optional encoding string
*
* Dump the XML document DTD , if any .
*/
static void
xmlDtdDumpOutput ( xmlOutputBufferPtr buf , xmlDtdPtr dtd , const char * encoding ) {
if ( dtd = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlDtdDumpOutput : no internal subset \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
xmlOutputBufferWriteString ( buf , " <!DOCTYPE " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) dtd - > name ) ;
if ( dtd - > ExternalID ! = NULL ) {
xmlOutputBufferWriteString ( buf , " PUBLIC " ) ;
xmlBufferWriteQuotedString ( buf - > buffer , dtd - > ExternalID ) ;
xmlOutputBufferWriteString ( buf , " " ) ;
xmlBufferWriteQuotedString ( buf - > buffer , dtd - > SystemID ) ;
} else if ( dtd - > SystemID ! = NULL ) {
xmlOutputBufferWriteString ( buf , " SYSTEM " ) ;
xmlBufferWriteQuotedString ( buf - > buffer , dtd - > SystemID ) ;
}
if ( ( dtd - > entities = = NULL ) & & ( dtd - > elements = = NULL ) & &
( dtd - > attributes = = NULL ) & & ( dtd - > notations = = NULL ) ) {
xmlOutputBufferWriteString ( buf , " > " ) ;
return ;
}
xmlOutputBufferWriteString ( buf , " [ \n " ) ;
xmlNodeListDumpOutput ( buf , dtd - > doc , dtd - > children , - 1 , 0 , encoding ) ;
xmlOutputBufferWriteString ( buf , " ]> " ) ;
}
/**
* xmlAttrDumpOutput :
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the attribute pointer
* @ encoding : an optional encoding string
*
* Dump an XML attribute
*/
static void
xmlAttrDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc , xmlAttrPtr cur ,
2001-03-26 16:28:29 +00:00
const char * encoding ATTRIBUTE_UNUSED ) {
2001-02-23 17:55:21 +00:00
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlAttrDumpOutput : property == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
xmlOutputBufferWriteString ( buf , " " ) ;
if ( ( cur - > ns ! = NULL ) & & ( cur - > ns - > prefix ! = NULL ) ) {
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > ns - > prefix ) ;
xmlOutputBufferWriteString ( buf , " : " ) ;
}
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
2002-02-13 13:07:41 +00:00
xmlOutputBufferWriteString ( buf , " = \" " ) ;
xmlAttrSerializeContent ( buf - > buffer , doc , cur ) ;
xmlOutputBufferWriteString ( buf , " \" " ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlAttrListDumpOutput :
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the first attribute pointer
* @ encoding : an optional encoding string
*
* Dump a list of XML attributes
*/
static void
xmlAttrListDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc ,
xmlAttrPtr cur , const char * encoding ) {
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlAttrListDumpOutput : property == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
while ( cur ! = NULL ) {
xmlAttrDumpOutput ( buf , doc , cur , encoding ) ;
cur = cur - > next ;
}
}
/**
* xmlNodeListDumpOutput :
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the first node
* @ level : the imbrication level for indenting
* @ format : is formatting allowed
* @ encoding : an optional encoding string
*
2001-12-17 15:30:10 +00:00
* Dump an XML node list , recursive behaviour , children are printed too .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-03-19 14:36:46 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
static void
xmlNodeListDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc ,
xmlNodePtr cur , int level , int format , const char * encoding ) {
int i ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNodeListDumpOutput : node == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
while ( cur ! = NULL ) {
if ( ( format ) & & ( xmlIndentTreeOutput ) & &
( cur - > type = = XML_ELEMENT_NODE ) )
for ( i = 0 ; i < level ; i + + )
2002-05-24 07:18:40 +00:00
xmlOutputBufferWriteString ( buf , xmlTreeIndentString ) ;
2002-11-21 14:10:52 +00:00
xmlNodeDumpOutputInternal ( buf , doc , cur , level , format , encoding ) ;
2001-02-23 17:55:21 +00:00
if ( format ) {
xmlOutputBufferWriteString ( buf , " \n " ) ;
}
cur = cur - > next ;
}
}
/**
2002-11-21 14:10:52 +00:00
* xmlNodeDumpOutputInternal :
2001-02-23 17:55:21 +00:00
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the current node
* @ level : the imbrication level for indenting
* @ format : is formatting allowed
* @ encoding : an optional encoding string
*
2001-12-17 15:30:10 +00:00
* Dump an XML node , recursive behaviour , children are printed too .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-03-19 14:36:46 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
2002-11-21 14:10:52 +00:00
static void
xmlNodeDumpOutputInternal ( xmlOutputBufferPtr buf , xmlDocPtr doc ,
xmlNodePtr cur , int level , int format , const char * encoding ) {
2001-02-23 17:55:21 +00:00
int i ;
xmlNodePtr tmp ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlNodeDumpOutput : node == NULL \n " ) ;
2001-02-23 17:55:21 +00:00
# endif
return ;
}
if ( cur - > type = = XML_XINCLUDE_START )
return ;
if ( cur - > type = = XML_XINCLUDE_END )
return ;
if ( cur - > type = = XML_DTD_NODE ) {
xmlDtdDumpOutput ( buf , ( xmlDtdPtr ) cur , encoding ) ;
return ;
}
if ( cur - > type = = XML_ELEMENT_DECL ) {
xmlDumpElementDecl ( buf - > buffer , ( xmlElementPtr ) cur ) ;
return ;
}
if ( cur - > type = = XML_ATTRIBUTE_DECL ) {
xmlDumpAttributeDecl ( buf - > buffer , ( xmlAttributePtr ) cur ) ;
return ;
}
if ( cur - > type = = XML_ENTITY_DECL ) {
xmlDumpEntityDecl ( buf - > buffer , ( xmlEntityPtr ) cur ) ;
return ;
}
if ( cur - > type = = XML_TEXT_NODE ) {
if ( cur - > content ! = NULL ) {
if ( ( cur - > name = = xmlStringText ) | |
( cur - > name ! = xmlStringTextNoenc ) ) {
xmlChar * buffer ;
if ( encoding = = NULL )
buffer = xmlEncodeEntitiesReentrant ( doc , cur - > content ) ;
else
buffer = xmlEncodeSpecialChars ( doc , cur - > content ) ;
if ( buffer ! = NULL ) {
xmlOutputBufferWriteString ( buf , ( const char * ) buffer ) ;
xmlFree ( buffer ) ;
}
} else {
/*
* Disable escaping , needed for XSLT
*/
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
}
}
return ;
}
if ( cur - > type = = XML_PI_NODE ) {
if ( cur - > content ! = NULL ) {
xmlOutputBufferWriteString ( buf , " <? " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
if ( cur - > content ! = NULL ) {
xmlOutputBufferWriteString ( buf , " " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
}
xmlOutputBufferWriteString ( buf , " ?> " ) ;
} else {
xmlOutputBufferWriteString ( buf , " <? " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " ?> " ) ;
}
return ;
}
if ( cur - > type = = XML_COMMENT_NODE ) {
if ( cur - > content ! = NULL ) {
xmlOutputBufferWriteString ( buf , " <!-- " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
xmlOutputBufferWriteString ( buf , " --> " ) ;
}
return ;
}
if ( cur - > type = = XML_ENTITY_REF_NODE ) {
xmlOutputBufferWriteString ( buf , " & " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " ; " ) ;
return ;
}
if ( cur - > type = = XML_CDATA_SECTION_NODE ) {
xmlOutputBufferWriteString ( buf , " <![CDATA[ " ) ;
if ( cur - > content ! = NULL )
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
xmlOutputBufferWriteString ( buf , " ]]> " ) ;
return ;
}
2003-02-10 00:01:56 +00:00
if ( cur - > type = = XML_NAMESPACE_DECL ) {
2003-02-10 16:12:39 +00:00
xmlNsDumpOutput ( buf , ( xmlNsPtr ) cur ) ;
2003-02-10 00:01:56 +00:00
return ;
}
2001-02-23 17:55:21 +00:00
if ( format = = 1 ) {
tmp = cur - > children ;
while ( tmp ! = NULL ) {
if ( ( tmp - > type = = XML_TEXT_NODE ) | |
( tmp - > type = = XML_ENTITY_REF_NODE ) ) {
format = 0 ;
break ;
}
tmp = tmp - > next ;
}
}
xmlOutputBufferWriteString ( buf , " < " ) ;
if ( ( cur - > ns ! = NULL ) & & ( cur - > ns - > prefix ! = NULL ) ) {
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > ns - > prefix ) ;
xmlOutputBufferWriteString ( buf , " : " ) ;
}
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
if ( cur - > nsDef )
xmlNsListDumpOutput ( buf , cur - > nsDef ) ;
if ( cur - > properties ! = NULL )
xmlAttrListDumpOutput ( buf , doc , cur - > properties , encoding ) ;
2001-07-12 01:20:08 +00:00
if ( ( ( cur - > type = = XML_ELEMENT_NODE ) | | ( cur - > content = = NULL ) ) & &
( cur - > children = = NULL ) & & ( ! xmlSaveNoEmptyTags ) ) {
2001-02-23 17:55:21 +00:00
xmlOutputBufferWriteString ( buf , " /> " ) ;
return ;
}
xmlOutputBufferWriteString ( buf , " > " ) ;
2001-07-12 01:20:08 +00:00
if ( ( cur - > type ! = XML_ELEMENT_NODE ) & & ( cur - > content ! = NULL ) ) {
2001-02-23 17:55:21 +00:00
xmlChar * buffer ;
if ( encoding = = NULL )
buffer = xmlEncodeEntitiesReentrant ( doc , cur - > content ) ;
else
buffer = xmlEncodeSpecialChars ( doc , cur - > content ) ;
if ( buffer ! = NULL ) {
xmlOutputBufferWriteString ( buf , ( const char * ) buffer ) ;
xmlFree ( buffer ) ;
}
}
if ( cur - > children ! = NULL ) {
if ( format ) xmlOutputBufferWriteString ( buf , " \n " ) ;
xmlNodeListDumpOutput ( buf , doc , cur - > children ,
( level > = 0 ? level + 1 : - 1 ) , format , encoding ) ;
if ( ( xmlIndentTreeOutput ) & & ( format ) )
for ( i = 0 ; i < level ; i + + )
2002-05-24 07:18:40 +00:00
xmlOutputBufferWriteString ( buf , xmlTreeIndentString ) ;
2001-02-23 17:55:21 +00:00
}
xmlOutputBufferWriteString ( buf , " </ " ) ;
if ( ( cur - > ns ! = NULL ) & & ( cur - > ns - > prefix ! = NULL ) ) {
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > ns - > prefix ) ;
xmlOutputBufferWriteString ( buf , " : " ) ;
}
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " > " ) ;
}
2002-11-21 14:10:52 +00:00
/**
* xmlNodeDumpOutput :
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the current node
* @ level : the imbrication level for indenting
* @ format : is formatting allowed
* @ encoding : an optional encoding string
*
* Dump an XML node , recursive behaviour , children are printed too .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-11-21 14:10:52 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
*/
void
xmlNodeDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc , xmlNodePtr cur ,
2002-11-27 11:43:05 +00:00
int level , int format , const char * encoding )
{
2002-11-21 14:10:52 +00:00
# ifdef LIBXML_HTML_ENABLED
xmlDtdPtr dtd ;
int is_xhtml = 0 ;
dtd = xmlGetIntSubset ( doc ) ;
if ( dtd ! = NULL ) {
2002-11-27 11:43:05 +00:00
is_xhtml = xmlIsXHTML ( dtd - > SystemID , dtd - > ExternalID ) ;
if ( is_xhtml < 0 )
is_xhtml = 0 ;
if ( ( is_xhtml ) & & ( cur - > parent = = ( xmlNodePtr ) doc ) & &
( cur - > type = = XML_ELEMENT_NODE ) & &
( xmlStrEqual ( cur - > name , BAD_CAST " html " ) ) ) {
if ( encoding ! = NULL )
2002-12-18 14:53:54 +00:00
htmlSetMetaEncoding ( ( htmlDocPtr ) doc ,
2002-11-27 11:43:05 +00:00
( const xmlChar * ) encoding ) ;
else
2002-12-18 14:53:54 +00:00
htmlSetMetaEncoding ( ( htmlDocPtr ) doc , BAD_CAST " UTF-8 " ) ;
2002-11-27 11:43:05 +00:00
}
2002-11-21 14:10:52 +00:00
}
if ( is_xhtml )
2002-11-27 11:43:05 +00:00
xhtmlNodeDumpOutput ( buf , doc , cur , level , format , encoding ) ;
2002-11-21 14:10:52 +00:00
else
# endif
2002-11-27 11:43:05 +00:00
xmlNodeDumpOutputInternal ( buf , doc , cur , level , format , encoding ) ;
2002-11-21 14:10:52 +00:00
}
2001-02-23 17:55:21 +00:00
/**
* xmlDocContentDumpOutput :
* @ buf : the XML buffer output
* @ cur : the document
* @ encoding : an optional encoding string
* @ format : should formatting spaces been added
*
* Dump an XML document .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-03-19 14:36:46 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
static void
xmlDocContentDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr cur ,
const char * encoding , int format ) {
2002-11-21 14:10:52 +00:00
# ifdef LIBXML_HTML_ENABLED
xmlDtdPtr dtd ;
int is_xhtml = 0 ;
# endif
2001-02-23 17:55:21 +00:00
xmlOutputBufferWriteString ( buf , " <?xml version= " ) ;
if ( cur - > version ! = NULL )
xmlBufferWriteQuotedString ( buf - > buffer , cur - > version ) ;
else
xmlOutputBufferWriteString ( buf , " \" 1.0 \" " ) ;
if ( encoding = = NULL ) {
if ( cur - > encoding ! = NULL )
encoding = ( const char * ) cur - > encoding ;
else if ( cur - > charset ! = XML_CHAR_ENCODING_UTF8 )
encoding = xmlGetCharEncodingName ( ( xmlCharEncoding ) cur - > charset ) ;
}
if ( encoding ! = NULL ) {
xmlOutputBufferWriteString ( buf , " encoding= " ) ;
xmlBufferWriteQuotedString ( buf - > buffer , ( xmlChar * ) encoding ) ;
}
switch ( cur - > standalone ) {
case 0 :
xmlOutputBufferWriteString ( buf , " standalone= \" no \" " ) ;
break ;
case 1 :
xmlOutputBufferWriteString ( buf , " standalone= \" yes \" " ) ;
break ;
}
xmlOutputBufferWriteString ( buf , " ?> \n " ) ;
2002-11-21 14:10:52 +00:00
# ifdef LIBXML_HTML_ENABLED
dtd = xmlGetIntSubset ( cur ) ;
if ( dtd ! = NULL ) {
is_xhtml = xmlIsXHTML ( dtd - > SystemID , dtd - > ExternalID ) ;
if ( is_xhtml < 0 ) is_xhtml = 0 ;
}
if ( is_xhtml ) {
if ( encoding ! = NULL )
htmlSetMetaEncoding ( cur , ( const xmlChar * ) encoding ) ;
else
htmlSetMetaEncoding ( cur , BAD_CAST " UTF-8 " ) ;
}
# endif
2001-02-23 17:55:21 +00:00
if ( cur - > children ! = NULL ) {
xmlNodePtr child = cur - > children ;
while ( child ! = NULL ) {
2002-11-21 14:10:52 +00:00
# ifdef LIBXML_HTML_ENABLED
if ( is_xhtml )
xhtmlNodeDumpOutput ( buf , cur , child , 0 , format , encoding ) ;
else
# endif
xmlNodeDumpOutputInternal ( buf , cur , child , 0 , format , encoding ) ;
2001-02-23 17:55:21 +00:00
xmlOutputBufferWriteString ( buf , " \n " ) ;
child = child - > next ;
}
}
}
2002-11-21 14:10:52 +00:00
# ifdef LIBXML_HTML_ENABLED
/************************************************************************
* *
* Functions specific to XHTML serialization *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define XHTML_STRICT_PUBLIC_ID BAD_CAST \
" -//W3C//DTD XHTML 1.0 Strict//EN "
# define XHTML_STRICT_SYSTEM_ID BAD_CAST \
" http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd "
# define XHTML_FRAME_PUBLIC_ID BAD_CAST \
" -//W3C//DTD XHTML 1.0 Frameset//EN "
# define XHTML_FRAME_SYSTEM_ID BAD_CAST \
" http://www.w3.org/TR/xhtml1/DTD/xhtml1-frameset.dtd "
# define XHTML_TRANS_PUBLIC_ID BAD_CAST \
" -//W3C//DTD XHTML 1.0 Transitional//EN "
# define XHTML_TRANS_SYSTEM_ID BAD_CAST \
" http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd "
# define XHTML_NS_NAME BAD_CAST "http: //www.w3.org/1999/xhtml"
/**
* xmlIsXHTML :
* @ systemID : the system identifier
* @ publicID : the public identifier
*
* Try to find if the document correspond to an XHTML DTD
*
* Returns 1 if true , 0 if not and - 1 in case of error
*/
int
xmlIsXHTML ( const xmlChar * systemID , const xmlChar * publicID ) {
if ( ( systemID = = NULL ) & & ( publicID = = NULL ) )
return ( - 1 ) ;
if ( publicID ! = NULL ) {
if ( xmlStrEqual ( publicID , XHTML_STRICT_PUBLIC_ID ) ) return ( 1 ) ;
if ( xmlStrEqual ( publicID , XHTML_FRAME_PUBLIC_ID ) ) return ( 1 ) ;
if ( xmlStrEqual ( publicID , XHTML_TRANS_PUBLIC_ID ) ) return ( 1 ) ;
}
if ( systemID ! = NULL ) {
if ( xmlStrEqual ( systemID , XHTML_STRICT_SYSTEM_ID ) ) return ( 1 ) ;
if ( xmlStrEqual ( systemID , XHTML_FRAME_SYSTEM_ID ) ) return ( 1 ) ;
if ( xmlStrEqual ( systemID , XHTML_TRANS_SYSTEM_ID ) ) return ( 1 ) ;
}
return ( 0 ) ;
}
/**
* xhtmlIsEmpty :
* @ node : the node
*
* Check if a node is an empty xhtml node
*
* Returns 1 if the node is an empty node , 0 if not and - 1 in case of error
*/
static int
xhtmlIsEmpty ( xmlNodePtr node ) {
if ( node = = NULL )
return ( - 1 ) ;
if ( node - > type ! = XML_ELEMENT_NODE )
return ( 0 ) ;
if ( ( node - > ns ! = NULL ) & & ( ! xmlStrEqual ( node - > ns - > href , XHTML_NS_NAME ) ) )
return ( 0 ) ;
if ( node - > children ! = NULL )
return ( 0 ) ;
switch ( node - > name [ 0 ] ) {
case ' a ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " area " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' b ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " br " ) )
return ( 1 ) ;
if ( xmlStrEqual ( node - > name , BAD_CAST " base " ) )
return ( 1 ) ;
if ( xmlStrEqual ( node - > name , BAD_CAST " basefont " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' c ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " col " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' f ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " frame " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' h ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " hr " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' i ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " img " ) )
return ( 1 ) ;
if ( xmlStrEqual ( node - > name , BAD_CAST " input " ) )
return ( 1 ) ;
if ( xmlStrEqual ( node - > name , BAD_CAST " isindex " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' l ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " link " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' m ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " meta " ) )
return ( 1 ) ;
return ( 0 ) ;
case ' p ' :
if ( xmlStrEqual ( node - > name , BAD_CAST " param " ) )
return ( 1 ) ;
return ( 0 ) ;
}
return ( 0 ) ;
}
/**
* xhtmlAttrListDumpOutput :
* @ buf : the XML buffer output
* @ doc : the document
* @ cur : the first attribute pointer
* @ encoding : an optional encoding string
*
* Dump a list of XML attributes
*/
static void
xhtmlAttrListDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc ,
xmlAttrPtr cur , const char * encoding ) {
xmlAttrPtr xml_lang = NULL ;
xmlAttrPtr lang = NULL ;
xmlAttrPtr name = NULL ;
xmlAttrPtr id = NULL ;
2003-02-10 16:12:39 +00:00
xmlNodePtr parent ;
2002-11-21 14:10:52 +00:00
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlAttrListDumpOutput : property == NULL \n " ) ;
# endif
return ;
}
2003-02-10 16:12:39 +00:00
parent = cur - > parent ;
2002-11-21 14:10:52 +00:00
while ( cur ! = NULL ) {
if ( ( cur - > ns = = NULL ) & & ( xmlStrEqual ( cur - > name , BAD_CAST " id " ) ) )
id = cur ;
else
if ( ( cur - > ns = = NULL ) & & ( xmlStrEqual ( cur - > name , BAD_CAST " name " ) ) )
name = cur ;
else
if ( ( cur - > ns = = NULL ) & & ( xmlStrEqual ( cur - > name , BAD_CAST " lang " ) ) )
lang = cur ;
else
if ( ( cur - > ns ! = NULL ) & & ( xmlStrEqual ( cur - > name , BAD_CAST " lang " ) ) & &
( xmlStrEqual ( cur - > ns - > prefix , BAD_CAST " xml " ) ) )
xml_lang = cur ;
else if ( ( cur - > ns = = NULL ) & &
( ( cur - > children = = NULL ) | |
( cur - > children - > content = = NULL ) | |
( cur - > children - > content [ 0 ] = = 0 ) ) & &
( htmlIsBooleanAttr ( cur - > name ) ) ) {
if ( cur - > children ! = NULL )
xmlFreeNode ( cur - > children ) ;
cur - > children = xmlNewText ( cur - > name ) ;
if ( cur - > children ! = NULL )
cur - > children - > parent = ( xmlNodePtr ) cur ;
}
xmlAttrDumpOutput ( buf , doc , cur , encoding ) ;
cur = cur - > next ;
}
/*
* C .8
*/
if ( ( name ! = NULL ) & & ( id = = NULL ) ) {
2003-02-10 16:12:39 +00:00
if ( ( parent ! = NULL ) & & ( parent - > name ! = NULL ) & &
( ( xmlStrEqual ( parent - > name , BAD_CAST " a " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " p " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " div " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " img " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " map " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " applet " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " form " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " frame " ) ) | |
( xmlStrEqual ( parent - > name , BAD_CAST " iframe " ) ) ) ) {
xmlOutputBufferWriteString ( buf , " id= \" " ) ;
xmlAttrSerializeContent ( buf - > buffer , doc , name ) ;
xmlOutputBufferWriteString ( buf , " \" " ) ;
}
2002-11-21 14:10:52 +00:00
}
/*
* C .7 .
*/
if ( ( lang ! = NULL ) & & ( xml_lang = = NULL ) ) {
xmlOutputBufferWriteString ( buf , " xml:lang= \" " ) ;
xmlAttrSerializeContent ( buf - > buffer , doc , lang ) ;
xmlOutputBufferWriteString ( buf , " \" " ) ;
} else
if ( ( xml_lang ! = NULL ) & & ( lang = = NULL ) ) {
xmlOutputBufferWriteString ( buf , " lang= \" " ) ;
xmlAttrSerializeContent ( buf - > buffer , doc , xml_lang ) ;
xmlOutputBufferWriteString ( buf , " \" " ) ;
}
}
/**
* xhtmlNodeListDumpOutput :
* @ buf : the XML buffer output
* @ doc : the XHTML document
* @ cur : the first node
* @ level : the imbrication level for indenting
* @ format : is formatting allowed
* @ encoding : an optional encoding string
*
* Dump an XML node list , recursive behaviour , children are printed too .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-11-21 14:10:52 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
*/
static void
xhtmlNodeListDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc ,
xmlNodePtr cur , int level , int format , const char * encoding ) {
int i ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xhtmlNodeListDumpOutput : node == NULL \n " ) ;
# endif
return ;
}
while ( cur ! = NULL ) {
if ( ( format ) & & ( xmlIndentTreeOutput ) & &
( cur - > type = = XML_ELEMENT_NODE ) )
for ( i = 0 ; i < level ; i + + )
xmlOutputBufferWriteString ( buf , xmlTreeIndentString ) ;
xhtmlNodeDumpOutput ( buf , doc , cur , level , format , encoding ) ;
if ( format ) {
xmlOutputBufferWriteString ( buf , " \n " ) ;
}
cur = cur - > next ;
}
}
/**
* xhtmlNodeDumpOutput :
* @ buf : the XML buffer output
* @ doc : the XHTML document
* @ cur : the current node
* @ level : the imbrication level for indenting
* @ format : is formatting allowed
* @ encoding : an optional encoding string
*
* Dump an XHTML node , recursive behaviour , children are printed too .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-11-21 14:10:52 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
*/
static void
xhtmlNodeDumpOutput ( xmlOutputBufferPtr buf , xmlDocPtr doc , xmlNodePtr cur ,
int level , int format , const char * encoding ) {
int i ;
xmlNodePtr tmp ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlNodeDumpOutput : node == NULL \n " ) ;
# endif
return ;
}
if ( cur - > type = = XML_XINCLUDE_START )
return ;
if ( cur - > type = = XML_XINCLUDE_END )
return ;
if ( cur - > type = = XML_DTD_NODE ) {
xmlDtdDumpOutput ( buf , ( xmlDtdPtr ) cur , encoding ) ;
return ;
}
if ( cur - > type = = XML_ELEMENT_DECL ) {
xmlDumpElementDecl ( buf - > buffer , ( xmlElementPtr ) cur ) ;
return ;
}
if ( cur - > type = = XML_ATTRIBUTE_DECL ) {
xmlDumpAttributeDecl ( buf - > buffer , ( xmlAttributePtr ) cur ) ;
return ;
}
if ( cur - > type = = XML_ENTITY_DECL ) {
xmlDumpEntityDecl ( buf - > buffer , ( xmlEntityPtr ) cur ) ;
return ;
}
if ( cur - > type = = XML_TEXT_NODE ) {
if ( cur - > content ! = NULL ) {
if ( ( cur - > name = = xmlStringText ) | |
( cur - > name ! = xmlStringTextNoenc ) ) {
xmlChar * buffer ;
if ( encoding = = NULL )
buffer = xmlEncodeEntitiesReentrant ( doc , cur - > content ) ;
else
buffer = xmlEncodeSpecialChars ( doc , cur - > content ) ;
if ( buffer ! = NULL ) {
xmlOutputBufferWriteString ( buf , ( const char * ) buffer ) ;
xmlFree ( buffer ) ;
}
} else {
/*
* Disable escaping , needed for XSLT
*/
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
}
}
return ;
}
if ( cur - > type = = XML_PI_NODE ) {
if ( cur - > content ! = NULL ) {
xmlOutputBufferWriteString ( buf , " <? " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
if ( cur - > content ! = NULL ) {
xmlOutputBufferWriteString ( buf , " " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
}
xmlOutputBufferWriteString ( buf , " ?> " ) ;
} else {
xmlOutputBufferWriteString ( buf , " <? " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " ?> " ) ;
}
return ;
}
if ( cur - > type = = XML_COMMENT_NODE ) {
if ( cur - > content ! = NULL ) {
xmlOutputBufferWriteString ( buf , " <!-- " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
xmlOutputBufferWriteString ( buf , " --> " ) ;
}
return ;
}
if ( cur - > type = = XML_ENTITY_REF_NODE ) {
xmlOutputBufferWriteString ( buf , " & " ) ;
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " ; " ) ;
return ;
}
if ( cur - > type = = XML_CDATA_SECTION_NODE ) {
xmlOutputBufferWriteString ( buf , " <![CDATA[ " ) ;
if ( cur - > content ! = NULL )
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > content ) ;
xmlOutputBufferWriteString ( buf , " ]]> " ) ;
return ;
}
if ( format = = 1 ) {
tmp = cur - > children ;
while ( tmp ! = NULL ) {
if ( ( tmp - > type = = XML_TEXT_NODE ) | |
( tmp - > type = = XML_ENTITY_REF_NODE ) ) {
format = 0 ;
break ;
}
tmp = tmp - > next ;
}
}
xmlOutputBufferWriteString ( buf , " < " ) ;
if ( ( cur - > ns ! = NULL ) & & ( cur - > ns - > prefix ! = NULL ) ) {
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > ns - > prefix ) ;
xmlOutputBufferWriteString ( buf , " : " ) ;
}
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
if ( cur - > nsDef )
xmlNsListDumpOutput ( buf , cur - > nsDef ) ;
if ( ( xmlStrEqual ( cur - > name , BAD_CAST " html " ) & &
( cur - > ns = = NULL ) & & ( cur - > nsDef = = NULL ) ) ) {
/*
* 3.1 .1 . Strictly Conforming Documents A .3 .1 .1 3 /
*/
xmlOutputBufferWriteString ( buf ,
" xmlns= \" http://www.w3.org/1999/xhtml \" " ) ;
}
if ( cur - > properties ! = NULL )
xhtmlAttrListDumpOutput ( buf , doc , cur - > properties , encoding ) ;
if ( ( cur - > type = = XML_ELEMENT_NODE ) & & ( cur - > children = = NULL ) ) {
if ( ( ( cur - > ns = = NULL ) | | ( cur - > ns - > prefix = = NULL ) ) & &
( xhtmlIsEmpty ( cur ) = = 1 ) ) {
/*
* C .2 . Empty Elements
*/
xmlOutputBufferWriteString ( buf , " /> " ) ;
} else {
/*
* C .3 . Element Minimization and Empty Element Content
*/
xmlOutputBufferWriteString ( buf , " ></ " ) ;
if ( ( cur - > ns ! = NULL ) & & ( cur - > ns - > prefix ! = NULL ) ) {
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > ns - > prefix ) ;
xmlOutputBufferWriteString ( buf , " : " ) ;
}
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " > " ) ;
}
return ;
}
xmlOutputBufferWriteString ( buf , " > " ) ;
if ( ( cur - > type ! = XML_ELEMENT_NODE ) & & ( cur - > content ! = NULL ) ) {
xmlChar * buffer ;
if ( encoding = = NULL )
buffer = xmlEncodeEntitiesReentrant ( doc , cur - > content ) ;
else
buffer = xmlEncodeSpecialChars ( doc , cur - > content ) ;
if ( buffer ! = NULL ) {
xmlOutputBufferWriteString ( buf , ( const char * ) buffer ) ;
xmlFree ( buffer ) ;
}
}
/*
* 4.8 . Script and Style elements
*/
if ( ( cur - > type = = XML_ELEMENT_NODE ) & &
( ( xmlStrEqual ( cur - > name , BAD_CAST " script " ) ) | |
( xmlStrEqual ( cur - > name , BAD_CAST " style " ) ) ) & &
( ( cur - > ns = = NULL ) | |
( xmlStrEqual ( cur - > ns - > href , XHTML_NS_NAME ) ) ) ) {
xmlNodePtr child = cur - > children ;
while ( child ! = NULL ) {
if ( ( child - > type = = XML_TEXT_NODE ) | |
( child - > type = = XML_CDATA_SECTION_NODE ) ) {
2002-12-04 15:10:40 +00:00
/*
* Apparently CDATA escaping for style just break on IE ,
* mozilla and galeon , so . . .
*/
if ( xmlStrEqual ( cur - > name , BAD_CAST " style " ) & &
( xmlStrchr ( child - > content , ' < ' ) = = NULL ) & &
( xmlStrchr ( child - > content , ' > ' ) = = NULL ) & &
( xmlStrchr ( child - > content , ' & ' ) = = NULL ) ) {
xhtmlNodeDumpOutput ( buf , doc , child , 0 , 0 , encoding ) ;
} else {
xmlOutputBufferWriteString ( buf , " <![CDATA[ " ) ;
if ( child - > content ! = NULL )
xmlOutputBufferWriteString ( buf ,
( const char * ) child - > content ) ;
xmlOutputBufferWriteString ( buf , " ]]> " ) ;
}
2002-11-21 14:10:52 +00:00
} else {
xhtmlNodeDumpOutput ( buf , doc , child , 0 , 0 , encoding ) ;
}
child = child - > next ;
}
} else if ( cur - > children ! = NULL ) {
if ( format ) xmlOutputBufferWriteString ( buf , " \n " ) ;
xhtmlNodeListDumpOutput ( buf , doc , cur - > children ,
( level > = 0 ? level + 1 : - 1 ) , format , encoding ) ;
if ( ( xmlIndentTreeOutput ) & & ( format ) )
for ( i = 0 ; i < level ; i + + )
xmlOutputBufferWriteString ( buf , xmlTreeIndentString ) ;
}
xmlOutputBufferWriteString ( buf , " </ " ) ;
if ( ( cur - > ns ! = NULL ) & & ( cur - > ns - > prefix ! = NULL ) ) {
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > ns - > prefix ) ;
xmlOutputBufferWriteString ( buf , " : " ) ;
}
xmlOutputBufferWriteString ( buf , ( const char * ) cur - > name ) ;
xmlOutputBufferWriteString ( buf , " > " ) ;
}
# endif
2001-02-23 17:55:21 +00:00
/************************************************************************
* *
* Saving functions front - ends *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/**
2001-07-18 19:30:27 +00:00
* xmlDocDumpFormatMemoryEnc :
2001-02-23 17:55:21 +00:00
* @ out_doc : Document to generate XML text from
* @ doc_txt_ptr : Memory pointer for allocated XML text
* @ doc_txt_len : Length of the generated XML text
* @ txt_encoding : Character encoding to use when generating XML text
* @ format : should formatting spaces been added
*
* Dump the current DOM tree into memory using the character encoding specified
* by the caller . Note it is up to the caller of this function to free the
2002-09-25 22:25:35 +00:00
* allocated memory with xmlFree ( ) .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-03-19 14:36:46 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
void
xmlDocDumpFormatMemoryEnc ( xmlDocPtr out_doc , xmlChar * * doc_txt_ptr ,
2001-03-24 17:00:36 +00:00
int * doc_txt_len , const char * txt_encoding ,
2001-04-10 16:38:06 +00:00
int format ) {
2001-02-23 17:55:21 +00:00
int dummy = 0 ;
xmlCharEncoding doc_charset ;
xmlOutputBufferPtr out_buff = NULL ;
xmlCharEncodingHandlerPtr conv_hdlr = NULL ;
if ( doc_txt_len = = NULL ) {
doc_txt_len = & dummy ; /* Continue, caller just won't get length */
}
if ( doc_txt_ptr = = NULL ) {
* doc_txt_len = 0 ;
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDumpFormatMemoryEnc: Null return buffer pointer. " ) ;
return ;
}
* doc_txt_ptr = NULL ;
* doc_txt_len = 0 ;
if ( out_doc = = NULL ) {
/* No document, no output */
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDumpFormatMemoryEnc: Null DOM tree document pointer. \n " ) ;
return ;
}
/*
* Validate the encoding value , if provided .
* This logic is copied from xmlSaveFileEnc .
*/
if ( txt_encoding = = NULL )
txt_encoding = ( const char * ) out_doc - > encoding ;
if ( txt_encoding ! = NULL ) {
doc_charset = xmlParseCharEncoding ( txt_encoding ) ;
if ( out_doc - > charset ! = XML_CHAR_ENCODING_UTF8 ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDumpFormatMemoryEnc: Source document not in UTF8 \n " ) ;
return ;
} else if ( doc_charset ! = XML_CHAR_ENCODING_UTF8 ) {
conv_hdlr = xmlFindCharEncodingHandler ( txt_encoding ) ;
if ( conv_hdlr = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" %s: %s %s '%s' \n " ,
" xmlDocDumpFormatMemoryEnc " ,
" Failed to identify encoding handler for " ,
" character set " ,
txt_encoding ) ;
return ;
}
}
}
if ( ( out_buff = xmlAllocOutputBuffer ( conv_hdlr ) ) = = NULL ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDumpFormatMemoryEnc: Failed to allocate output buffer. \n " ) ;
return ;
}
2001-04-10 16:38:06 +00:00
xmlDocContentDumpOutput ( out_buff , out_doc , txt_encoding , format ) ;
2001-02-23 17:55:21 +00:00
xmlOutputBufferFlush ( out_buff ) ;
if ( out_buff - > conv ! = NULL ) {
* doc_txt_len = out_buff - > conv - > use ;
* doc_txt_ptr = xmlStrndup ( out_buff - > conv - > content , * doc_txt_len ) ;
} else {
* doc_txt_len = out_buff - > buffer - > use ;
* doc_txt_ptr = xmlStrndup ( out_buff - > buffer - > content , * doc_txt_len ) ;
}
( void ) xmlOutputBufferClose ( out_buff ) ;
if ( ( * doc_txt_ptr = = NULL ) & & ( * doc_txt_len > 0 ) ) {
* doc_txt_len = 0 ;
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDumpFormatMemoryEnc: %s \n " ,
" Failed to allocate memory for document text representation. " ) ;
}
return ;
}
/**
* xmlDocDumpMemory :
* @ cur : the document
* @ mem : OUT : the memory pointer
2001-10-10 09:45:09 +00:00
* @ size : OUT : the memory length
2001-02-23 17:55:21 +00:00
*
2001-12-17 15:30:10 +00:00
* Dump an XML document in memory and return the # xmlChar * and it ' s size .
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
void
xmlDocDumpMemory ( xmlDocPtr cur , xmlChar * * mem , int * size ) {
xmlDocDumpFormatMemoryEnc ( cur , mem , size , NULL , 0 ) ;
}
/**
* xmlDocDumpFormatMemory :
* @ cur : the document
* @ mem : OUT : the memory pointer
2001-10-10 09:45:09 +00:00
* @ size : OUT : the memory length
2001-02-23 17:55:21 +00:00
* @ format : should formatting spaces been added
*
*
2001-12-17 15:30:10 +00:00
* Dump an XML document in memory and return the # xmlChar * and it ' s size .
2002-09-25 22:25:35 +00:00
* It ' s up to the caller to free the memory with xmlFree ( ) .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2002-03-19 14:36:46 +00:00
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
void
xmlDocDumpFormatMemory ( xmlDocPtr cur , xmlChar * * mem , int * size , int format ) {
xmlDocDumpFormatMemoryEnc ( cur , mem , size , NULL , format ) ;
}
/**
* xmlDocDumpMemoryEnc :
* @ out_doc : Document to generate XML text from
* @ doc_txt_ptr : Memory pointer for allocated XML text
* @ doc_txt_len : Length of the generated XML text
* @ txt_encoding : Character encoding to use when generating XML text
*
* Dump the current DOM tree into memory using the character encoding specified
* by the caller . Note it is up to the caller of this function to free the
2002-09-25 22:25:35 +00:00
* allocated memory with xmlFree ( ) .
2001-02-23 17:55:21 +00:00
*/
void
xmlDocDumpMemoryEnc ( xmlDocPtr out_doc , xmlChar * * doc_txt_ptr ,
int * doc_txt_len , const char * txt_encoding ) {
xmlDocDumpFormatMemoryEnc ( out_doc , doc_txt_ptr , doc_txt_len ,
2001-04-10 16:38:06 +00:00
txt_encoding , 0 ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlGetDocCompressMode :
* @ doc : the document
*
* get the compression ratio for a document , ZLIB based
* Returns 0 ( uncompressed ) to 9 ( max compression )
*/
int
xmlGetDocCompressMode ( xmlDocPtr doc ) {
if ( doc = = NULL ) return ( - 1 ) ;
return ( doc - > compression ) ;
}
/**
* xmlSetDocCompressMode :
* @ doc : the document
* @ mode : the compression ratio
*
* set the compression ratio for a document , ZLIB based
* Correct values : 0 ( uncompressed ) to 9 ( max compression )
*/
void
xmlSetDocCompressMode ( xmlDocPtr doc , int mode ) {
if ( doc = = NULL ) return ;
if ( mode < 0 ) doc - > compression = 0 ;
else if ( mode > 9 ) doc - > compression = 9 ;
else doc - > compression = mode ;
}
/**
* xmlGetCompressMode :
*
* get the default compression mode used , ZLIB based .
* Returns 0 ( uncompressed ) to 9 ( max compression )
*/
int
2002-03-04 17:09:44 +00:00
xmlGetCompressMode ( void )
{
return ( xmlCompressMode ) ;
2001-02-23 17:55:21 +00:00
}
/**
* xmlSetCompressMode :
* @ mode : the compression ratio
*
* set the default compression mode used , ZLIB based
* Correct values : 0 ( uncompressed ) to 9 ( max compression )
*/
void
xmlSetCompressMode ( int mode ) {
if ( mode < 0 ) xmlCompressMode = 0 ;
else if ( mode > 9 ) xmlCompressMode = 9 ;
else xmlCompressMode = mode ;
}
/**
2002-06-10 15:59:44 +00:00
* xmlDocFormatDump :
2001-02-23 17:55:21 +00:00
* @ f : the FILE *
* @ cur : the document
2002-06-10 15:59:44 +00:00
* @ format : should formatting spaces been added
2001-02-23 17:55:21 +00:00
*
* Dump an XML document to an open FILE .
*
2001-12-17 15:30:10 +00:00
* returns : the number of bytes written or - 1 in case of failure .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
int
2002-06-10 15:59:44 +00:00
xmlDocFormatDump ( FILE * f , xmlDocPtr cur , int format ) {
2001-02-23 17:55:21 +00:00
xmlOutputBufferPtr buf ;
const char * encoding ;
xmlCharEncodingHandlerPtr handler = NULL ;
int ret ;
if ( cur = = NULL ) {
# ifdef DEBUG_TREE
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDump : document == NULL \n " ) ;
# endif
return ( - 1 ) ;
}
encoding = ( const char * ) cur - > encoding ;
if ( encoding ! = NULL ) {
xmlCharEncoding enc ;
enc = xmlParseCharEncoding ( encoding ) ;
if ( cur - > charset ! = XML_CHAR_ENCODING_UTF8 ) {
xmlGenericError ( xmlGenericErrorContext ,
" xmlDocDump: document not in UTF8 \n " ) ;
return ( - 1 ) ;
}
if ( enc ! = XML_CHAR_ENCODING_UTF8 ) {
handler = xmlFindCharEncodingHandler ( encoding ) ;
if ( handler = = NULL ) {
xmlFree ( ( char * ) cur - > encoding ) ;
cur - > encoding = NULL ;
}
}
}
buf = xmlOutputBufferCreateFile ( f , handler ) ;
if ( buf = = NULL ) return ( - 1 ) ;
2002-06-10 15:59:44 +00:00
xmlDocContentDumpOutput ( buf , cur , NULL , format ) ;
2001-02-23 17:55:21 +00:00
ret = xmlOutputBufferClose ( buf ) ;
return ( ret ) ;
}
2002-06-10 15:59:44 +00:00
/**
* xmlDocDump :
* @ f : the FILE *
* @ cur : the document
*
* Dump an XML document to an open FILE .
*
* returns : the number of bytes written or - 1 in case of failure .
*/
int
xmlDocDump ( FILE * f , xmlDocPtr cur ) {
return ( xmlDocFormatDump ( f , cur , 0 ) ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlSaveFileTo :
* @ buf : an output I / O buffer
* @ cur : the document
2001-12-17 15:30:10 +00:00
* @ encoding : the encoding if any assuming the I / O layer handles the trancoding
2001-02-23 17:55:21 +00:00
*
* Dump an XML document to an I / O buffer .
*
2001-12-17 15:30:10 +00:00
* returns : the number of bytes written or - 1 in case of failure .
2001-02-23 17:55:21 +00:00
*/
int
2001-11-01 14:31:22 +00:00
xmlSaveFileTo ( xmlOutputBufferPtr buf , xmlDocPtr cur , const char * encoding ) {
2001-02-23 17:55:21 +00:00
int ret ;
if ( buf = = NULL ) return ( 0 ) ;
2001-04-10 16:38:06 +00:00
xmlDocContentDumpOutput ( buf , cur , encoding , 0 ) ;
2001-02-23 17:55:21 +00:00
ret = xmlOutputBufferClose ( buf ) ;
return ( ret ) ;
}
2001-04-28 16:55:50 +00:00
/**
* xmlSaveFormatFileTo :
* @ buf : an output I / O buffer
* @ cur : the document
2001-12-17 15:30:10 +00:00
* @ encoding : the encoding if any assuming the I / O layer handles the trancoding
2001-04-28 16:55:50 +00:00
* @ format : should formatting spaces been added
*
* Dump an XML document to an I / O buffer .
*
2001-12-17 15:30:10 +00:00
* returns : the number of bytes written or - 1 in case of failure .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
* or xmlKeepBlanksDefault ( 0 ) was called
2001-04-28 16:55:50 +00:00
*/
int
2001-11-01 14:31:22 +00:00
xmlSaveFormatFileTo ( xmlOutputBufferPtr buf , xmlDocPtr cur , const char * encoding , int format ) {
2001-04-28 16:55:50 +00:00
int ret ;
if ( buf = = NULL ) return ( 0 ) ;
xmlDocContentDumpOutput ( buf , cur , encoding , format ) ;
ret = xmlOutputBufferClose ( buf ) ;
return ( ret ) ;
}
2001-02-23 17:55:21 +00:00
/**
2002-12-10 15:19:08 +00:00
* xmlSaveFormatFileEnc :
2001-07-23 19:10:52 +00:00
* @ filename : the filename or URL to output
* @ cur : the document being saved
* @ encoding : the name of the encoding to use or NULL .
* @ format : should formatting spaces be added .
2001-12-17 15:30:10 +00:00
*
2002-12-11 14:23:49 +00:00
* Dump an XML document to a file or an URL .
*
2001-12-17 15:30:10 +00:00
* Returns the number of bytes written or - 1 in case of error .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
* or xmlKeepBlanksDefault ( 0 ) was called
2001-02-23 17:55:21 +00:00
*/
int
2001-07-23 19:10:52 +00:00
xmlSaveFormatFileEnc ( const char * filename , xmlDocPtr cur ,
const char * encoding , int format ) {
2001-02-23 17:55:21 +00:00
xmlOutputBufferPtr buf ;
xmlCharEncodingHandlerPtr handler = NULL ;
2001-05-22 15:08:55 +00:00
xmlCharEncoding enc ;
2001-02-23 17:55:21 +00:00
int ret ;
2003-01-14 00:17:42 +00:00
if ( cur = = NULL )
return ( - 1 ) ;
2002-01-13 20:32:08 +00:00
if ( encoding = = NULL )
encoding = ( const char * ) cur - > encoding ;
2001-02-23 17:55:21 +00:00
if ( encoding ! = NULL ) {
enc = xmlParseCharEncoding ( encoding ) ;
if ( cur - > charset ! = XML_CHAR_ENCODING_UTF8 ) {
xmlGenericError ( xmlGenericErrorContext ,
2001-12-17 15:30:10 +00:00
" xmlSaveFormatFileEnc: document not in UTF8 \n " ) ;
2001-02-23 17:55:21 +00:00
return ( - 1 ) ;
}
if ( enc ! = XML_CHAR_ENCODING_UTF8 ) {
handler = xmlFindCharEncodingHandler ( encoding ) ;
2001-05-22 15:08:55 +00:00
if ( handler = = NULL )
2001-02-23 17:55:21 +00:00
return ( - 1 ) ;
}
}
2001-07-23 19:10:52 +00:00
# ifdef HAVE_ZLIB_H
if ( cur - > compression < 0 ) cur - > compression = xmlCompressMode ;
# endif
2001-02-23 17:55:21 +00:00
/*
* save the content to a temp buffer .
*/
2001-07-23 19:10:52 +00:00
buf = xmlOutputBufferCreateFilename ( filename , handler , cur - > compression ) ;
2001-02-23 17:55:21 +00:00
if ( buf = = NULL ) return ( - 1 ) ;
2001-07-23 19:10:52 +00:00
xmlDocContentDumpOutput ( buf , cur , encoding , format ) ;
2001-02-23 17:55:21 +00:00
ret = xmlOutputBufferClose ( buf ) ;
return ( ret ) ;
}
2001-07-23 19:10:52 +00:00
/**
* xmlSaveFileEnc :
* @ filename : the filename ( or URL )
* @ cur : the document
* @ encoding : the name of an encoding ( or NULL )
*
* Dump an XML document , converting it to the given encoding
*
2001-12-17 15:30:10 +00:00
* returns : the number of bytes written or - 1 in case of failure .
2001-07-23 19:10:52 +00:00
*/
int
xmlSaveFileEnc ( const char * filename , xmlDocPtr cur , const char * encoding ) {
return ( xmlSaveFormatFileEnc ( filename , cur , encoding , 0 ) ) ;
}
2001-02-23 17:55:21 +00:00
/**
2001-04-26 18:59:03 +00:00
* xmlSaveFormatFile :
2001-02-23 17:55:21 +00:00
* @ filename : the filename ( or URL )
* @ cur : the document
2001-04-26 18:59:03 +00:00
* @ format : should formatting spaces been added
2001-02-23 17:55:21 +00:00
*
* Dump an XML document to a file . Will use compression if
* compiled in and enabled . If @ filename is " - " the stdout file is
2001-12-17 15:30:10 +00:00
* used . If @ format is set then the document will be indented on output .
2003-01-24 14:14:52 +00:00
* Note that @ format = 1 provide node indenting only if xmlIndentTreeOutput = 1
* or xmlKeepBlanksDefault ( 0 ) was called
2001-04-26 18:59:03 +00:00
*
2001-12-17 15:30:10 +00:00
* returns : the number of bytes written or - 1 in case of failure .
2001-02-23 17:55:21 +00:00
*/
int
2001-04-26 18:59:03 +00:00
xmlSaveFormatFile ( const char * filename , xmlDocPtr cur , int format ) {
2001-07-23 19:10:52 +00:00
return ( xmlSaveFormatFileEnc ( filename , cur , NULL , format ) ) ;
2001-02-23 17:55:21 +00:00
}
2001-04-26 18:59:03 +00:00
/**
* xmlSaveFile :
* @ filename : the filename ( or URL )
* @ cur : the document
*
* Dump an XML document to a file . Will use compression if
* compiled in and enabled . If @ filename is " - " the stdout file is
* used .
2001-12-17 15:30:10 +00:00
* returns : the number of bytes written or - 1 in case of failure .
2001-04-26 18:59:03 +00:00
*/
int
xmlSaveFile ( const char * filename , xmlDocPtr cur ) {
2001-07-23 19:10:52 +00:00
return ( xmlSaveFormatFileEnc ( filename , cur , NULL , 0 ) ) ;
2001-04-26 18:59:03 +00:00
}