blob: 7a05d832495e6ab7c3385a46d44fc00a238d9136 [file] [log] [blame]
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001/*
2 * xmlsave.c: Implemetation of the document serializer
3 *
4 * See Copyright for the status of this software.
5 *
6 * daniel@veillard.com
7 */
8
9#define IN_LIBXML
10#include "libxml.h"
11
12#include <string.h>
13#include <libxml/xmlmemory.h>
14#include <libxml/parserInternals.h>
15#include <libxml/tree.h>
16#include <libxml/xmlsave.h>
Daniel Veillard1a8741c2004-03-04 13:40:59 +000017
Daniel Veillard753086a2004-03-28 16:12:44 +000018#define MAX_INDENT 60
19
Daniel Veillard656ce942004-04-30 23:11:45 +000020#include <libxml/HTMLtree.h>
21
Daniel Veillard50cdab52012-07-16 14:52:00 +080022#include "buf.h"
23#include "enc.h"
24#include "save.h"
25
Daniel Veillard1a8741c2004-03-04 13:40:59 +000026/************************************************************************
27 * *
28 * XHTML detection *
29 * *
30 ************************************************************************/
31#define XHTML_STRICT_PUBLIC_ID BAD_CAST \
32 "-//W3C//DTD XHTML 1.0 Strict//EN"
33#define XHTML_STRICT_SYSTEM_ID BAD_CAST \
34 "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"
35#define XHTML_FRAME_PUBLIC_ID BAD_CAST \
36 "-//W3C//DTD XHTML 1.0 Frameset//EN"
37#define XHTML_FRAME_SYSTEM_ID BAD_CAST \
38 "http://www.w3.org/TR/xhtml1/DTD/xhtml1-frameset.dtd"
39#define XHTML_TRANS_PUBLIC_ID BAD_CAST \
40 "-//W3C//DTD XHTML 1.0 Transitional//EN"
41#define XHTML_TRANS_SYSTEM_ID BAD_CAST \
42 "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"
43
44#define XHTML_NS_NAME BAD_CAST "http://www.w3.org/1999/xhtml"
45/**
46 * xmlIsXHTML:
47 * @systemID: the system identifier
48 * @publicID: the public identifier
49 *
50 * Try to find if the document correspond to an XHTML DTD
51 *
52 * Returns 1 if true, 0 if not and -1 in case of error
53 */
54int
55xmlIsXHTML(const xmlChar *systemID, const xmlChar *publicID) {
56 if ((systemID == NULL) && (publicID == NULL))
57 return(-1);
58 if (publicID != NULL) {
59 if (xmlStrEqual(publicID, XHTML_STRICT_PUBLIC_ID)) return(1);
60 if (xmlStrEqual(publicID, XHTML_FRAME_PUBLIC_ID)) return(1);
61 if (xmlStrEqual(publicID, XHTML_TRANS_PUBLIC_ID)) return(1);
62 }
63 if (systemID != NULL) {
64 if (xmlStrEqual(systemID, XHTML_STRICT_SYSTEM_ID)) return(1);
65 if (xmlStrEqual(systemID, XHTML_FRAME_SYSTEM_ID)) return(1);
66 if (xmlStrEqual(systemID, XHTML_TRANS_SYSTEM_ID)) return(1);
67 }
68 return(0);
69}
Daniel Veillard1a8741c2004-03-04 13:40:59 +000070
71#ifdef LIBXML_OUTPUT_ENABLED
72
Daniel Veillardf8e3db02012-09-11 13:26:36 +080073#define TODO \
Daniel Veillard1a8741c2004-03-04 13:40:59 +000074 xmlGenericError(xmlGenericErrorContext, \
75 "Unimplemented block at %s:%d\n", \
76 __FILE__, __LINE__);
77
78struct _xmlSaveCtxt {
79 void *_private;
80 int type;
81 int fd;
82 const xmlChar *filename;
83 const xmlChar *encoding;
84 xmlCharEncodingHandlerPtr handler;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +000085 xmlOutputBufferPtr buf;
86 xmlDocPtr doc;
Daniel Veillard1a8741c2004-03-04 13:40:59 +000087 int options;
88 int level;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +000089 int format;
Daniel Veillard3995bc32004-05-15 18:57:31 +000090 char indent[MAX_INDENT + 1]; /* array for indenting output */
Daniel Veillard753086a2004-03-28 16:12:44 +000091 int indent_nr;
92 int indent_size;
Daniel Veillard3995bc32004-05-15 18:57:31 +000093 xmlCharEncodingOutputFunc escape; /* used for element content */
94 xmlCharEncodingOutputFunc escapeAttr;/* used for attribute content */
Daniel Veillard1a8741c2004-03-04 13:40:59 +000095};
96
97/************************************************************************
98 * *
Daniel Veillardf8e3db02012-09-11 13:26:36 +080099 * Output error handlers *
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000100 * *
101 ************************************************************************/
102/**
103 * xmlSaveErrMemory:
104 * @extra: extra informations
105 *
106 * Handle an out of memory condition
107 */
108static void
109xmlSaveErrMemory(const char *extra)
110{
111 __xmlSimpleError(XML_FROM_OUTPUT, XML_ERR_NO_MEMORY, NULL, NULL, extra);
112}
113
114/**
115 * xmlSaveErr:
116 * @code: the error number
117 * @node: the location of the error.
118 * @extra: extra informations
119 *
120 * Handle an out of memory condition
121 */
122static void
123xmlSaveErr(int code, xmlNodePtr node, const char *extra)
124{
125 const char *msg = NULL;
126
127 switch(code) {
128 case XML_SAVE_NOT_UTF8:
Rob Richards417b74d2006-08-15 23:14:24 +0000129 msg = "string is not in UTF-8\n";
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000130 break;
131 case XML_SAVE_CHAR_INVALID:
Rob Richards417b74d2006-08-15 23:14:24 +0000132 msg = "invalid character value\n";
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000133 break;
134 case XML_SAVE_UNKNOWN_ENCODING:
Rob Richards417b74d2006-08-15 23:14:24 +0000135 msg = "unknown encoding %s\n";
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000136 break;
137 case XML_SAVE_NO_DOCTYPE:
Rob Richards417b74d2006-08-15 23:14:24 +0000138 msg = "document has no DOCTYPE\n";
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000139 break;
140 default:
Rob Richards417b74d2006-08-15 23:14:24 +0000141 msg = "unexpected error number\n";
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000142 }
143 __xmlSimpleError(XML_FROM_OUTPUT, code, node, msg, extra);
144}
145
146/************************************************************************
147 * *
Daniel Veillard83a75e02004-05-14 21:50:42 +0000148 * Special escaping routines *
149 * *
150 ************************************************************************/
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000151static unsigned char *
152xmlSerializeHexCharRef(unsigned char *out, int val) {
153 unsigned char *ptr;
154
155 *out++ = '&';
156 *out++ = '#';
157 *out++ = 'x';
158 if (val < 0x10) ptr = out;
159 else if (val < 0x100) ptr = out + 1;
160 else if (val < 0x1000) ptr = out + 2;
161 else if (val < 0x10000) ptr = out + 3;
162 else if (val < 0x100000) ptr = out + 4;
163 else ptr = out + 5;
164 out = ptr + 1;
165 while (val > 0) {
166 switch (val & 0xF) {
167 case 0: *ptr-- = '0'; break;
168 case 1: *ptr-- = '1'; break;
169 case 2: *ptr-- = '2'; break;
170 case 3: *ptr-- = '3'; break;
171 case 4: *ptr-- = '4'; break;
172 case 5: *ptr-- = '5'; break;
173 case 6: *ptr-- = '6'; break;
174 case 7: *ptr-- = '7'; break;
175 case 8: *ptr-- = '8'; break;
176 case 9: *ptr-- = '9'; break;
177 case 0xA: *ptr-- = 'A'; break;
178 case 0xB: *ptr-- = 'B'; break;
179 case 0xC: *ptr-- = 'C'; break;
180 case 0xD: *ptr-- = 'D'; break;
181 case 0xE: *ptr-- = 'E'; break;
182 case 0xF: *ptr-- = 'F'; break;
183 default: *ptr-- = '0'; break;
184 }
185 val >>= 4;
186 }
187 *out++ = ';';
188 *out = 0;
189 return(out);
190}
191
Daniel Veillard83a75e02004-05-14 21:50:42 +0000192/**
193 * xmlEscapeEntities:
194 * @out: a pointer to an array of bytes to store the result
195 * @outlen: the length of @out
196 * @in: a pointer to an array of unescaped UTF-8 bytes
197 * @inlen: the length of @in
198 *
199 * Take a block of UTF-8 chars in and escape them. Used when there is no
200 * encoding specified.
201 *
202 * Returns 0 if success, or -1 otherwise
203 * The value of @inlen after return is the number of octets consumed
204 * if the return value is positive, else unpredictable.
205 * The value of @outlen after return is the number of octets consumed.
206 */
207static int
208xmlEscapeEntities(unsigned char* out, int *outlen,
209 const xmlChar* in, int *inlen) {
210 unsigned char* outstart = out;
211 const unsigned char* base = in;
212 unsigned char* outend = out + *outlen;
213 const unsigned char* inend;
214 int val;
215
216 inend = in + (*inlen);
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800217
Daniel Veillard83a75e02004-05-14 21:50:42 +0000218 while ((in < inend) && (out < outend)) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800219 if (*in == '<') {
Daniel Veillard83a75e02004-05-14 21:50:42 +0000220 if (outend - out < 4) break;
221 *out++ = '&';
222 *out++ = 'l';
223 *out++ = 't';
224 *out++ = ';';
225 in++;
226 continue;
227 } else if (*in == '>') {
228 if (outend - out < 4) break;
229 *out++ = '&';
230 *out++ = 'g';
231 *out++ = 't';
232 *out++ = ';';
233 in++;
234 continue;
235 } else if (*in == '&') {
236 if (outend - out < 5) break;
237 *out++ = '&';
238 *out++ = 'a';
239 *out++ = 'm';
240 *out++ = 'p';
241 *out++ = ';';
242 in++;
243 continue;
244 } else if (((*in >= 0x20) && (*in < 0x80)) ||
245 (*in == '\n') || (*in == '\t')) {
246 /*
247 * default case, just copy !
248 */
249 *out++ = *in++;
250 continue;
251 } else if (*in >= 0x80) {
252 /*
253 * We assume we have UTF-8 input.
254 */
Daniel Veillard07953482012-01-22 17:42:35 +0800255 if (outend - out < 11) break;
Daniel Veillard83a75e02004-05-14 21:50:42 +0000256
257 if (*in < 0xC0) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000258 xmlSaveErr(XML_SAVE_NOT_UTF8, NULL, NULL);
Daniel Veillard83a75e02004-05-14 21:50:42 +0000259 in++;
260 goto error;
261 } else if (*in < 0xE0) {
262 if (inend - in < 2) break;
263 val = (in[0]) & 0x1F;
264 val <<= 6;
265 val |= (in[1]) & 0x3F;
266 in += 2;
267 } else if (*in < 0xF0) {
268 if (inend - in < 3) break;
269 val = (in[0]) & 0x0F;
270 val <<= 6;
271 val |= (in[1]) & 0x3F;
272 val <<= 6;
273 val |= (in[2]) & 0x3F;
274 in += 3;
275 } else if (*in < 0xF8) {
276 if (inend - in < 4) break;
277 val = (in[0]) & 0x07;
278 val <<= 6;
279 val |= (in[1]) & 0x3F;
280 val <<= 6;
281 val |= (in[2]) & 0x3F;
282 val <<= 6;
283 val |= (in[3]) & 0x3F;
284 in += 4;
285 } else {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000286 xmlSaveErr(XML_SAVE_CHAR_INVALID, NULL, NULL);
Daniel Veillard83a75e02004-05-14 21:50:42 +0000287 in++;
288 goto error;
289 }
290 if (!IS_CHAR(val)) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000291 xmlSaveErr(XML_SAVE_CHAR_INVALID, NULL, NULL);
Daniel Veillard83a75e02004-05-14 21:50:42 +0000292 in++;
293 goto error;
294 }
295
296 /*
297 * We could do multiple things here. Just save as a char ref
298 */
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000299 out = xmlSerializeHexCharRef(out, val);
Daniel Veillard83a75e02004-05-14 21:50:42 +0000300 } else if (IS_BYTE_CHAR(*in)) {
301 if (outend - out < 6) break;
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000302 out = xmlSerializeHexCharRef(out, *in++);
Daniel Veillard83a75e02004-05-14 21:50:42 +0000303 } else {
304 xmlGenericError(xmlGenericErrorContext,
305 "xmlEscapeEntities : char out of range\n");
306 in++;
307 goto error;
308 }
309 }
310 *outlen = out - outstart;
311 *inlen = in - base;
312 return(0);
313error:
314 *outlen = out - outstart;
315 *inlen = in - base;
316 return(-1);
317}
318
319/************************************************************************
320 * *
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000321 * Allocation and deallocation *
322 * *
323 ************************************************************************/
Daniel Veillard753086a2004-03-28 16:12:44 +0000324/**
325 * xmlSaveCtxtInit:
326 * @ctxt: the saving context
327 *
328 * Initialize a saving context
329 */
330static void
331xmlSaveCtxtInit(xmlSaveCtxtPtr ctxt)
332{
333 int i;
William M. Brack12d37ab2005-02-21 13:54:07 +0000334 int len;
Daniel Veillard753086a2004-03-28 16:12:44 +0000335
336 if (ctxt == NULL) return;
Daniel Veillard3995bc32004-05-15 18:57:31 +0000337 if ((ctxt->encoding == NULL) && (ctxt->escape == NULL))
338 ctxt->escape = xmlEscapeEntities;
William M. Brack12d37ab2005-02-21 13:54:07 +0000339 len = xmlStrlen((xmlChar *)xmlTreeIndentString);
340 if ((xmlTreeIndentString == NULL) || (len == 0)) {
Daniel Veillard753086a2004-03-28 16:12:44 +0000341 memset(&ctxt->indent[0], 0, MAX_INDENT + 1);
342 } else {
William M. Brack12d37ab2005-02-21 13:54:07 +0000343 ctxt->indent_size = len;
Daniel Veillard753086a2004-03-28 16:12:44 +0000344 ctxt->indent_nr = MAX_INDENT / ctxt->indent_size;
345 for (i = 0;i < ctxt->indent_nr;i++)
346 memcpy(&ctxt->indent[i * ctxt->indent_size], xmlTreeIndentString,
347 ctxt->indent_size);
348 ctxt->indent[ctxt->indent_nr * ctxt->indent_size] = 0;
349 }
Rob Richards2ce51c02005-09-12 12:16:35 +0000350
Daniel Veillard9a00fd22005-11-09 08:56:26 +0000351 if (xmlSaveNoEmptyTags) {
352 ctxt->options |= XML_SAVE_NO_EMPTY;
353 }
Daniel Veillard753086a2004-03-28 16:12:44 +0000354}
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000355
356/**
357 * xmlFreeSaveCtxt:
358 *
359 * Free a saving context, destroying the ouptut in any remaining buffer
360 */
361static void
362xmlFreeSaveCtxt(xmlSaveCtxtPtr ctxt)
363{
364 if (ctxt == NULL) return;
365 if (ctxt->encoding != NULL)
366 xmlFree((char *) ctxt->encoding);
Daniel Veillarde2161a62004-04-29 17:14:25 +0000367 if (ctxt->buf != NULL)
368 xmlOutputBufferClose(ctxt->buf);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000369 xmlFree(ctxt);
370}
371
372/**
373 * xmlNewSaveCtxt:
374 *
375 * Create a new saving context
376 *
377 * Returns the new structure or NULL in case of error
378 */
379static xmlSaveCtxtPtr
380xmlNewSaveCtxt(const char *encoding, int options)
381{
382 xmlSaveCtxtPtr ret;
383
384 ret = (xmlSaveCtxtPtr) xmlMalloc(sizeof(xmlSaveCtxt));
385 if (ret == NULL) {
386 xmlSaveErrMemory("creating saving context");
387 return ( NULL );
388 }
389 memset(ret, 0, sizeof(xmlSaveCtxt));
Daniel Veillard6fc5db02005-01-16 00:05:58 +0000390
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000391 if (encoding != NULL) {
392 ret->handler = xmlFindCharEncodingHandler(encoding);
393 if (ret->handler == NULL) {
394 xmlSaveErr(XML_SAVE_UNKNOWN_ENCODING, NULL, encoding);
395 xmlFreeSaveCtxt(ret);
396 return(NULL);
397 }
398 ret->encoding = xmlStrdup((const xmlChar *)encoding);
Daniel Veillarddab39b52006-10-16 23:22:10 +0000399 ret->escape = NULL;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000400 }
Daniel Veillard753086a2004-03-28 16:12:44 +0000401 xmlSaveCtxtInit(ret);
402
Rob Richards2ce51c02005-09-12 12:16:35 +0000403 /*
404 * Use the options
405 */
406
Daniel Veillard9a00fd22005-11-09 08:56:26 +0000407 /* Re-check this option as it may already have been set */
408 if ((ret->options & XML_SAVE_NO_EMPTY) && ! (options & XML_SAVE_NO_EMPTY)) {
409 options |= XML_SAVE_NO_EMPTY;
410 }
Rob Richards2ce51c02005-09-12 12:16:35 +0000411
412 ret->options = options;
413 if (options & XML_SAVE_FORMAT)
414 ret->format = 1;
Adam Spraggd2e62312010-11-03 15:33:40 +0100415 else if (options & XML_SAVE_WSNONSIG)
416 ret->format = 2;
Rob Richards2ce51c02005-09-12 12:16:35 +0000417
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000418 return(ret);
419}
420
421/************************************************************************
422 * *
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800423 * Dumping XML tree content to a simple buffer *
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000424 * *
425 ************************************************************************/
426/**
427 * xmlAttrSerializeContent:
428 * @buf: the XML buffer output
429 * @doc: the document
430 * @attr: the attribute pointer
431 *
432 * Serialize the attribute in the buffer
433 */
434static void
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000435xmlAttrSerializeContent(xmlOutputBufferPtr buf, xmlAttrPtr attr)
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000436{
437 xmlNodePtr children;
438
439 children = attr->children;
440 while (children != NULL) {
441 switch (children->type) {
442 case XML_TEXT_NODE:
Daniel Veillard50cdab52012-07-16 14:52:00 +0800443 xmlBufAttrSerializeTxtContent(buf->buffer, attr->doc,
444 attr, children->content);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000445 break;
446 case XML_ENTITY_REF_NODE:
Daniel Veillard50cdab52012-07-16 14:52:00 +0800447 xmlBufAdd(buf->buffer, BAD_CAST "&", 1);
448 xmlBufAdd(buf->buffer, children->name,
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000449 xmlStrlen(children->name));
Daniel Veillard50cdab52012-07-16 14:52:00 +0800450 xmlBufAdd(buf->buffer, BAD_CAST ";", 1);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000451 break;
452 default:
453 /* should not happen unless we have a badly built tree */
454 break;
455 }
456 children = children->next;
457 }
458}
459
Daniel Veillard50cdab52012-07-16 14:52:00 +0800460/**
461 * xmlBufDumpNotationTable:
462 * @buf: an xmlBufPtr output
463 * @table: A notation table
464 *
465 * This will dump the content of the notation table as an XML DTD definition
466 */
467void
468xmlBufDumpNotationTable(xmlBufPtr buf, xmlNotationTablePtr table) {
469 xmlBufferPtr buffer;
470
471 buffer = xmlBufferCreate();
472 if (buffer == NULL) {
473 /*
474 * TODO set the error in buf
475 */
476 return;
477 }
478 xmlDumpNotationTable(buffer, table);
479 xmlBufMergeBuffer(buf, buffer);
480}
481
482/**
483 * xmlBufDumpElementDecl:
484 * @buf: an xmlBufPtr output
485 * @elem: An element table
486 *
487 * This will dump the content of the element declaration as an XML
488 * DTD definition
489 */
490void
491xmlBufDumpElementDecl(xmlBufPtr buf, xmlElementPtr elem) {
492 xmlBufferPtr buffer;
493
494 buffer = xmlBufferCreate();
495 if (buffer == NULL) {
496 /*
497 * TODO set the error in buf
498 */
499 return;
500 }
501 xmlDumpElementDecl(buffer, elem);
502 xmlBufMergeBuffer(buf, buffer);
503}
504
505/**
506 * xmlBufDumpAttributeDecl:
507 * @buf: an xmlBufPtr output
508 * @attr: An attribute declaration
509 *
510 * This will dump the content of the attribute declaration as an XML
511 * DTD definition
512 */
513void
514xmlBufDumpAttributeDecl(xmlBufPtr buf, xmlAttributePtr attr) {
515 xmlBufferPtr buffer;
516
517 buffer = xmlBufferCreate();
518 if (buffer == NULL) {
519 /*
520 * TODO set the error in buf
521 */
522 return;
523 }
524 xmlDumpAttributeDecl(buffer, attr);
525 xmlBufMergeBuffer(buf, buffer);
526}
527
528/**
529 * xmlBufDumpEntityDecl:
530 * @buf: an xmlBufPtr output
531 * @ent: An entity table
532 *
533 * This will dump the content of the entity table as an XML DTD definition
534 */
535void
536xmlBufDumpEntityDecl(xmlBufPtr buf, xmlEntityPtr ent) {
537 xmlBufferPtr buffer;
538
539 buffer = xmlBufferCreate();
540 if (buffer == NULL) {
541 /*
542 * TODO set the error in buf
543 */
544 return;
545 }
546 xmlDumpEntityDecl(buffer, ent);
547 xmlBufMergeBuffer(buf, buffer);
548}
549
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000550/************************************************************************
551 * *
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800552 * Dumping XML tree content to an I/O output buffer *
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000553 * *
554 ************************************************************************/
555
Daniel Veillardda3fee42008-09-01 13:08:57 +0000556static int xmlSaveSwitchEncoding(xmlSaveCtxtPtr ctxt, const char *encoding) {
557 xmlOutputBufferPtr buf = ctxt->buf;
558
559 if ((encoding != NULL) && (buf->encoder == NULL) && (buf->conv == NULL)) {
560 buf->encoder = xmlFindCharEncodingHandler((const char *)encoding);
561 if (buf->encoder == NULL) {
562 xmlSaveErr(XML_SAVE_UNKNOWN_ENCODING, NULL,
563 (const char *)encoding);
564 return(-1);
565 }
Daniel Veillard50cdab52012-07-16 14:52:00 +0800566 buf->conv = xmlBufCreate();
Daniel Veillardda3fee42008-09-01 13:08:57 +0000567 if (buf->conv == NULL) {
568 xmlCharEncCloseFunc(buf->encoder);
569 xmlSaveErrMemory("creating encoding buffer");
570 return(-1);
571 }
572 /*
573 * initialize the state, e.g. if outputting a BOM
574 */
Daniel Veillard50cdab52012-07-16 14:52:00 +0800575 xmlCharEncOutput(buf, 1);
Daniel Veillardda3fee42008-09-01 13:08:57 +0000576 }
577 return(0);
578}
579
580static int xmlSaveClearEncoding(xmlSaveCtxtPtr ctxt) {
581 xmlOutputBufferPtr buf = ctxt->buf;
582 xmlOutputBufferFlush(buf);
583 xmlCharEncCloseFunc(buf->encoder);
Daniel Veillard50cdab52012-07-16 14:52:00 +0800584 xmlBufFree(buf->conv);
Daniel Veillardda3fee42008-09-01 13:08:57 +0000585 buf->encoder = NULL;
586 buf->conv = NULL;
587 return(0);
588}
589
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000590#ifdef LIBXML_HTML_ENABLED
591static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000592xhtmlNodeDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000593#endif
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000594static void xmlNodeListDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur);
595static void xmlNodeDumpOutputInternal(xmlSaveCtxtPtr ctxt, xmlNodePtr cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000596void xmlNsListDumpOutput(xmlOutputBufferPtr buf, xmlNsPtr cur);
Daniel Veillarddab39b52006-10-16 23:22:10 +0000597static int xmlDocContentDumpOutput(xmlSaveCtxtPtr ctxt, xmlDocPtr cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000598
599/**
Adam Spraggd2e62312010-11-03 15:33:40 +0100600 * xmlOutputBufferWriteWSNonSig:
601 * @ctxt: The save context
602 * @extra: Number of extra indents to apply to ctxt->level
603 *
604 * Write out formatting for non-significant whitespace output.
605 */
606static void
607xmlOutputBufferWriteWSNonSig(xmlSaveCtxtPtr ctxt, int extra)
608{
609 int i;
610 if ((ctxt == NULL) || (ctxt->buf == NULL))
611 return;
612 xmlOutputBufferWrite(ctxt->buf, 1, "\n");
613 for (i = 0; i < (ctxt->level + extra); i += ctxt->indent_nr) {
614 xmlOutputBufferWrite(ctxt->buf, ctxt->indent_size *
615 ((ctxt->level + extra - i) > ctxt->indent_nr ?
616 ctxt->indent_nr : (ctxt->level + extra - i)),
617 ctxt->indent);
618 }
619}
620
621/**
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000622 * xmlNsDumpOutput:
623 * @buf: the XML buffer output
624 * @cur: a namespace
Adam Spraggd2e62312010-11-03 15:33:40 +0100625 * @ctxt: the output save context. Optional.
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000626 *
627 * Dump a local Namespace definition.
628 * Should be called in the context of attributes dumps.
Adam Spraggd2e62312010-11-03 15:33:40 +0100629 * If @ctxt is supplied, @buf should be its buffer.
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000630 */
631static void
Adam Spraggd2e62312010-11-03 15:33:40 +0100632xmlNsDumpOutput(xmlOutputBufferPtr buf, xmlNsPtr cur, xmlSaveCtxtPtr ctxt) {
Daniel Veillardce244ad2004-11-05 10:03:46 +0000633 if ((cur == NULL) || (buf == NULL)) return;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000634 if ((cur->type == XML_LOCAL_NAMESPACE) && (cur->href != NULL)) {
635 if (xmlStrEqual(cur->prefix, BAD_CAST "xml"))
636 return;
637
Adam Spraggd2e62312010-11-03 15:33:40 +0100638 if (ctxt != NULL && ctxt->format == 2)
639 xmlOutputBufferWriteWSNonSig(ctxt, 2);
640 else
641 xmlOutputBufferWrite(buf, 1, " ");
642
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000643 /* Within the context of an element attributes */
644 if (cur->prefix != NULL) {
Adam Spraggd2e62312010-11-03 15:33:40 +0100645 xmlOutputBufferWrite(buf, 6, "xmlns:");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000646 xmlOutputBufferWriteString(buf, (const char *)cur->prefix);
647 } else
Adam Spraggd2e62312010-11-03 15:33:40 +0100648 xmlOutputBufferWrite(buf, 5, "xmlns");
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000649 xmlOutputBufferWrite(buf, 1, "=");
Daniel Veillard50cdab52012-07-16 14:52:00 +0800650 xmlBufWriteQuotedString(buf->buffer, cur->href);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000651 }
652}
653
654/**
Adam Spraggd2e62312010-11-03 15:33:40 +0100655 * xmlNsDumpOutputCtxt
656 * @ctxt: the save context
657 * @cur: a namespace
658 *
659 * Dump a local Namespace definition to a save context.
660 * Should be called in the context of attribute dumps.
661 */
662static void
663xmlNsDumpOutputCtxt(xmlSaveCtxtPtr ctxt, xmlNsPtr cur) {
664 xmlNsDumpOutput(ctxt->buf, cur, ctxt);
665}
666
667/**
668 * xmlNsListDumpOutputCtxt
669 * @ctxt: the save context
670 * @cur: the first namespace
671 *
672 * Dump a list of local namespace definitions to a save context.
673 * Should be called in the context of attribute dumps.
674 */
675static void
676xmlNsListDumpOutputCtxt(xmlSaveCtxtPtr ctxt, xmlNsPtr cur) {
677 while (cur != NULL) {
678 xmlNsDumpOutput(ctxt->buf, cur, ctxt);
679 cur = cur->next;
680 }
681}
682
683/**
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000684 * xmlNsListDumpOutput:
685 * @buf: the XML buffer output
686 * @cur: the first namespace
687 *
688 * Dump a list of local Namespace definitions.
689 * Should be called in the context of attributes dumps.
690 */
691void
692xmlNsListDumpOutput(xmlOutputBufferPtr buf, xmlNsPtr cur) {
693 while (cur != NULL) {
Adam Spraggd2e62312010-11-03 15:33:40 +0100694 xmlNsDumpOutput(buf, cur, NULL);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000695 cur = cur->next;
696 }
697}
698
699/**
700 * xmlDtdDumpOutput:
701 * @buf: the XML buffer output
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000702 * @dtd: the pointer to the DTD
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800703 *
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000704 * Dump the XML document DTD, if any.
705 */
706static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000707xmlDtdDumpOutput(xmlSaveCtxtPtr ctxt, xmlDtdPtr dtd) {
708 xmlOutputBufferPtr buf;
709 int format, level;
710 xmlDocPtr doc;
711
712 if (dtd == NULL) return;
713 if ((ctxt == NULL) || (ctxt->buf == NULL))
714 return;
715 buf = ctxt->buf;
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000716 xmlOutputBufferWrite(buf, 10, "<!DOCTYPE ");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000717 xmlOutputBufferWriteString(buf, (const char *)dtd->name);
718 if (dtd->ExternalID != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000719 xmlOutputBufferWrite(buf, 8, " PUBLIC ");
Daniel Veillard50cdab52012-07-16 14:52:00 +0800720 xmlBufWriteQuotedString(buf->buffer, dtd->ExternalID);
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000721 xmlOutputBufferWrite(buf, 1, " ");
Daniel Veillard50cdab52012-07-16 14:52:00 +0800722 xmlBufWriteQuotedString(buf->buffer, dtd->SystemID);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000723 } else if (dtd->SystemID != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000724 xmlOutputBufferWrite(buf, 8, " SYSTEM ");
Daniel Veillard50cdab52012-07-16 14:52:00 +0800725 xmlBufWriteQuotedString(buf->buffer, dtd->SystemID);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000726 }
727 if ((dtd->entities == NULL) && (dtd->elements == NULL) &&
Daniel Veillard41c4a752004-09-08 20:55:38 +0000728 (dtd->attributes == NULL) && (dtd->notations == NULL) &&
729 (dtd->pentities == NULL)) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000730 xmlOutputBufferWrite(buf, 1, ">");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000731 return;
732 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000733 xmlOutputBufferWrite(buf, 3, " [\n");
Daniel Veillard41c4a752004-09-08 20:55:38 +0000734 /*
735 * Dump the notations first they are not in the DTD children list
736 * Do this only on a standalone DTD or on the internal subset though.
737 */
738 if ((dtd->notations != NULL) && ((dtd->doc == NULL) ||
739 (dtd->doc->intSubset == dtd))) {
Daniel Veillard50cdab52012-07-16 14:52:00 +0800740 xmlBufDumpNotationTable(buf->buffer,
741 (xmlNotationTablePtr) dtd->notations);
Daniel Veillardda3b29a2004-08-14 11:15:13 +0000742 }
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000743 format = ctxt->format;
744 level = ctxt->level;
745 doc = ctxt->doc;
746 ctxt->format = 0;
747 ctxt->level = -1;
748 ctxt->doc = dtd->doc;
749 xmlNodeListDumpOutput(ctxt, dtd->children);
750 ctxt->format = format;
751 ctxt->level = level;
752 ctxt->doc = doc;
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000753 xmlOutputBufferWrite(buf, 2, "]>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000754}
755
756/**
757 * xmlAttrDumpOutput:
758 * @buf: the XML buffer output
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000759 * @cur: the attribute pointer
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000760 *
761 * Dump an XML attribute
762 */
763static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000764xmlAttrDumpOutput(xmlSaveCtxtPtr ctxt, xmlAttrPtr cur) {
765 xmlOutputBufferPtr buf;
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000766
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000767 if (cur == NULL) return;
768 buf = ctxt->buf;
Daniel Veillardce244ad2004-11-05 10:03:46 +0000769 if (buf == NULL) return;
Adam Spraggd2e62312010-11-03 15:33:40 +0100770 if (ctxt->format == 2)
771 xmlOutputBufferWriteWSNonSig(ctxt, 2);
772 else
773 xmlOutputBufferWrite(buf, 1, " ");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000774 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
775 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000776 xmlOutputBufferWrite(buf, 1, ":");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000777 }
778 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000779 xmlOutputBufferWrite(buf, 2, "=\"");
780 xmlAttrSerializeContent(buf, cur);
781 xmlOutputBufferWrite(buf, 1, "\"");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000782}
783
784/**
785 * xmlAttrListDumpOutput:
786 * @buf: the XML buffer output
787 * @doc: the document
788 * @cur: the first attribute pointer
789 * @encoding: an optional encoding string
790 *
791 * Dump a list of XML attributes
792 */
793static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000794xmlAttrListDumpOutput(xmlSaveCtxtPtr ctxt, xmlAttrPtr cur) {
795 if (cur == NULL) return;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000796 while (cur != NULL) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000797 xmlAttrDumpOutput(ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000798 cur = cur->next;
799 }
800}
801
802
803
804/**
805 * xmlNodeListDumpOutput:
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000806 * @cur: the first node
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000807 *
808 * Dump an XML node list, recursive behaviour, children are printed too.
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000809 */
810static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000811xmlNodeListDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000812 xmlOutputBufferPtr buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000813
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000814 if (cur == NULL) return;
815 buf = ctxt->buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000816 while (cur != NULL) {
Adam Spragg8b877132010-11-01 14:24:56 +0100817 if ((ctxt->format == 1) && (xmlIndentTreeOutput) &&
Daniel Veillardbd444842007-03-20 08:47:29 +0000818 ((cur->type == XML_ELEMENT_NODE) ||
819 (cur->type == XML_COMMENT_NODE) ||
820 (cur->type == XML_PI_NODE)))
Daniel Veillard753086a2004-03-28 16:12:44 +0000821 xmlOutputBufferWrite(buf, ctxt->indent_size *
Daniel Veillardf8e3db02012-09-11 13:26:36 +0800822 (ctxt->level > ctxt->indent_nr ?
Daniel Veillard753086a2004-03-28 16:12:44 +0000823 ctxt->indent_nr : ctxt->level),
824 ctxt->indent);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000825 xmlNodeDumpOutputInternal(ctxt, cur);
Adam Spragg8b877132010-11-01 14:24:56 +0100826 if (ctxt->format == 1) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000827 xmlOutputBufferWrite(buf, 1, "\n");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000828 }
829 cur = cur->next;
830 }
831}
832
Daniel Veillardda3fee42008-09-01 13:08:57 +0000833#ifdef LIBXML_HTML_ENABLED
834/**
835 * xmlNodeDumpOutputInternal:
836 * @cur: the current node
837 *
838 * Dump an HTML node, recursive behaviour, children are printed too.
839 */
840static int
841htmlNodeDumpOutputInternal(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) {
842 const xmlChar *oldenc = NULL;
843 const xmlChar *oldctxtenc = ctxt->encoding;
844 const xmlChar *encoding = ctxt->encoding;
845 xmlOutputBufferPtr buf = ctxt->buf;
846 int switched_encoding = 0;
847 xmlDocPtr doc;
848
849 xmlInitParser();
850
Daniel Veillard141ebfa2009-09-02 14:58:13 +0200851 doc = cur->doc;
852 if (doc != NULL) {
Daniel Veillardda3fee42008-09-01 13:08:57 +0000853 oldenc = doc->encoding;
854 if (ctxt->encoding != NULL) {
855 doc->encoding = BAD_CAST ctxt->encoding;
856 } else if (doc->encoding != NULL) {
857 encoding = doc->encoding;
858 }
859 }
860
861 if ((encoding != NULL) && (doc != NULL))
862 htmlSetMetaEncoding(doc, (const xmlChar *) encoding);
863 if ((encoding == NULL) && (doc != NULL))
864 encoding = htmlGetMetaEncoding(doc);
865 if (encoding == NULL)
866 encoding = BAD_CAST "HTML";
867 if ((encoding != NULL) && (oldctxtenc == NULL) &&
868 (buf->encoder == NULL) && (buf->conv == NULL)) {
869 if (xmlSaveSwitchEncoding(ctxt, (const char*) encoding) < 0) {
870 doc->encoding = oldenc;
871 return(-1);
872 }
873 switched_encoding = 1;
874 }
875 if (ctxt->options & XML_SAVE_FORMAT)
876 htmlNodeDumpFormatOutput(buf, doc, cur,
877 (const char *)encoding, 1);
878 else
879 htmlNodeDumpFormatOutput(buf, doc, cur,
880 (const char *)encoding, 0);
881 /*
882 * Restore the state of the saving context at the end of the document
883 */
884 if ((switched_encoding) && (oldctxtenc == NULL)) {
885 xmlSaveClearEncoding(ctxt);
886 }
887 if (doc != NULL)
888 doc->encoding = oldenc;
889 return(0);
890}
891#endif
892
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000893/**
894 * xmlNodeDumpOutputInternal:
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000895 * @cur: the current node
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000896 *
897 * Dump an XML node, recursive behaviour, children are printed too.
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000898 */
899static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000900xmlNodeDumpOutputInternal(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) {
Daniel Veillard753086a2004-03-28 16:12:44 +0000901 int format;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000902 xmlNodePtr tmp;
903 xmlChar *start, *end;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000904 xmlOutputBufferPtr buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000905
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000906 if (cur == NULL) return;
907 buf = ctxt->buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000908 if (cur->type == XML_XINCLUDE_START)
909 return;
910 if (cur->type == XML_XINCLUDE_END)
911 return;
Daniel Veillardce244ad2004-11-05 10:03:46 +0000912 if ((cur->type == XML_DOCUMENT_NODE) ||
913 (cur->type == XML_HTML_DOCUMENT_NODE)) {
914 xmlDocContentDumpOutput(ctxt, (xmlDocPtr) cur);
915 return;
916 }
Daniel Veillardda3fee42008-09-01 13:08:57 +0000917#ifdef LIBXML_HTML_ENABLED
Daniel Veillard856d9282008-09-25 14:31:40 +0000918 if (ctxt->options & XML_SAVE_XHTML) {
919 xhtmlNodeDumpOutput(ctxt, cur);
920 return;
921 }
922 if (((cur->type != XML_NAMESPACE_DECL) && (cur->doc != NULL) &&
923 (cur->doc->type == XML_HTML_DOCUMENT_NODE) &&
924 ((ctxt->options & XML_SAVE_AS_XML) == 0)) ||
925 (ctxt->options & XML_SAVE_AS_HTML)) {
Daniel Veillardda3fee42008-09-01 13:08:57 +0000926 htmlNodeDumpOutputInternal(ctxt, cur);
927 return;
928 }
929#endif
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000930 if (cur->type == XML_DTD_NODE) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000931 xmlDtdDumpOutput(ctxt, (xmlDtdPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000932 return;
933 }
934 if (cur->type == XML_DOCUMENT_FRAG_NODE) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +0000935 xmlNodeListDumpOutput(ctxt, cur->children);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000936 return;
937 }
938 if (cur->type == XML_ELEMENT_DECL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +0800939 xmlBufDumpElementDecl(buf->buffer, (xmlElementPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000940 return;
941 }
942 if (cur->type == XML_ATTRIBUTE_DECL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +0800943 xmlBufDumpAttributeDecl(buf->buffer, (xmlAttributePtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000944 return;
945 }
946 if (cur->type == XML_ENTITY_DECL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +0800947 xmlBufDumpEntityDecl(buf->buffer, (xmlEntityPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000948 return;
949 }
950 if (cur->type == XML_TEXT_NODE) {
951 if (cur->content != NULL) {
William M. Brack4e1c2db2005-02-11 10:58:55 +0000952 if (cur->name != xmlStringTextNoenc) {
Daniel Veillard3995bc32004-05-15 18:57:31 +0000953 xmlOutputBufferWriteEscape(buf, cur->content, ctxt->escape);
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000954 } else {
955 /*
956 * Disable escaping, needed for XSLT
957 */
958 xmlOutputBufferWriteString(buf, (const char *) cur->content);
959 }
960 }
961
962 return;
963 }
964 if (cur->type == XML_PI_NODE) {
965 if (cur->content != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000966 xmlOutputBufferWrite(buf, 2, "<?");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000967 xmlOutputBufferWriteString(buf, (const char *)cur->name);
968 if (cur->content != NULL) {
Adam Spraggd2e62312010-11-03 15:33:40 +0100969 if (ctxt->format == 2)
970 xmlOutputBufferWriteWSNonSig(ctxt, 0);
971 else
972 xmlOutputBufferWrite(buf, 1, " ");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000973 xmlOutputBufferWriteString(buf, (const char *)cur->content);
974 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000975 xmlOutputBufferWrite(buf, 2, "?>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000976 } else {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000977 xmlOutputBufferWrite(buf, 2, "<?");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000978 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Adam Spraggd2e62312010-11-03 15:33:40 +0100979 if (ctxt->format == 2)
980 xmlOutputBufferWriteWSNonSig(ctxt, 0);
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000981 xmlOutputBufferWrite(buf, 2, "?>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000982 }
983 return;
984 }
985 if (cur->type == XML_COMMENT_NODE) {
986 if (cur->content != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000987 xmlOutputBufferWrite(buf, 4, "<!--");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000988 xmlOutputBufferWriteString(buf, (const char *)cur->content);
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000989 xmlOutputBufferWrite(buf, 3, "-->");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000990 }
991 return;
992 }
993 if (cur->type == XML_ENTITY_REF_NODE) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000994 xmlOutputBufferWrite(buf, 1, "&");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000995 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Daniel Veillard7a6361f2004-05-15 16:37:50 +0000996 xmlOutputBufferWrite(buf, 1, ";");
Daniel Veillard1a8741c2004-03-04 13:40:59 +0000997 return;
998 }
999 if (cur->type == XML_CDATA_SECTION_NODE) {
Daniel Veillardd0d2f092008-03-07 16:50:21 +00001000 if (cur->content == NULL || *cur->content == '\0') {
1001 xmlOutputBufferWrite(buf, 12, "<![CDATA[]]>");
Daniel Veillard7cd517c2005-05-20 18:47:22 +00001002 } else {
1003 start = end = cur->content;
1004 while (*end != '\0') {
1005 if ((*end == ']') && (*(end + 1) == ']') &&
1006 (*(end + 2) == '>')) {
1007 end = end + 2;
1008 xmlOutputBufferWrite(buf, 9, "<![CDATA[");
1009 xmlOutputBufferWrite(buf, end - start, (const char *)start);
1010 xmlOutputBufferWrite(buf, 3, "]]>");
1011 start = end;
1012 }
1013 end++;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001014 }
Daniel Veillard7cd517c2005-05-20 18:47:22 +00001015 if (start != end) {
1016 xmlOutputBufferWrite(buf, 9, "<![CDATA[");
1017 xmlOutputBufferWriteString(buf, (const char *)start);
1018 xmlOutputBufferWrite(buf, 3, "]]>");
1019 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001020 }
1021 return;
1022 }
1023 if (cur->type == XML_ATTRIBUTE_NODE) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001024 xmlAttrDumpOutput(ctxt, (xmlAttrPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001025 return;
1026 }
1027 if (cur->type == XML_NAMESPACE_DECL) {
Adam Spraggd2e62312010-11-03 15:33:40 +01001028 xmlNsDumpOutputCtxt(ctxt, (xmlNsPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001029 return;
1030 }
1031
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001032 format = ctxt->format;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001033 if (format == 1) {
1034 tmp = cur->children;
1035 while (tmp != NULL) {
1036 if ((tmp->type == XML_TEXT_NODE) ||
1037 (tmp->type == XML_CDATA_SECTION_NODE) ||
1038 (tmp->type == XML_ENTITY_REF_NODE)) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001039 ctxt->format = 0;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001040 break;
1041 }
1042 tmp = tmp->next;
1043 }
1044 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001045 xmlOutputBufferWrite(buf, 1, "<");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001046 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
1047 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001048 xmlOutputBufferWrite(buf, 1, ":");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001049 }
1050
1051 xmlOutputBufferWriteString(buf, (const char *)cur->name);
1052 if (cur->nsDef)
Adam Spraggd2e62312010-11-03 15:33:40 +01001053 xmlNsListDumpOutputCtxt(ctxt, cur->nsDef);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001054 if (cur->properties != NULL)
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001055 xmlAttrListDumpOutput(ctxt, cur->properties);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001056
1057 if (((cur->type == XML_ELEMENT_NODE) || (cur->content == NULL)) &&
Rob Richards2ce51c02005-09-12 12:16:35 +00001058 (cur->children == NULL) && ((ctxt->options & XML_SAVE_NO_EMPTY) == 0)) {
Adam Spraggd2e62312010-11-03 15:33:40 +01001059 if (ctxt->format == 2)
1060 xmlOutputBufferWriteWSNonSig(ctxt, 0);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001061 xmlOutputBufferWrite(buf, 2, "/>");
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001062 ctxt->format = format;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001063 return;
1064 }
Adam Spraggd2e62312010-11-03 15:33:40 +01001065 if (ctxt->format == 2)
1066 xmlOutputBufferWriteWSNonSig(ctxt, 1);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001067 xmlOutputBufferWrite(buf, 1, ">");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001068 if ((cur->type != XML_ELEMENT_NODE) && (cur->content != NULL)) {
Daniel Veillard3995bc32004-05-15 18:57:31 +00001069 xmlOutputBufferWriteEscape(buf, cur->content, ctxt->escape);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001070 }
1071 if (cur->children != NULL) {
Adam Spragg8b877132010-11-01 14:24:56 +01001072 if (ctxt->format == 1) xmlOutputBufferWrite(buf, 1, "\n");
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001073 if (ctxt->level >= 0) ctxt->level++;
1074 xmlNodeListDumpOutput(ctxt, cur->children);
1075 if (ctxt->level > 0) ctxt->level--;
Adam Spragg8b877132010-11-01 14:24:56 +01001076 if ((xmlIndentTreeOutput) && (ctxt->format == 1))
Daniel Veillard753086a2004-03-28 16:12:44 +00001077 xmlOutputBufferWrite(buf, ctxt->indent_size *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001078 (ctxt->level > ctxt->indent_nr ?
Daniel Veillard753086a2004-03-28 16:12:44 +00001079 ctxt->indent_nr : ctxt->level),
1080 ctxt->indent);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001081 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001082 xmlOutputBufferWrite(buf, 2, "</");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001083 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
1084 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001085 xmlOutputBufferWrite(buf, 1, ":");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001086 }
1087
1088 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Adam Spraggd2e62312010-11-03 15:33:40 +01001089 if (ctxt->format == 2)
1090 xmlOutputBufferWriteWSNonSig(ctxt, 0);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001091 xmlOutputBufferWrite(buf, 1, ">");
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001092 ctxt->format = format;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001093}
1094
1095/**
1096 * xmlDocContentDumpOutput:
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001097 * @cur: the document
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001098 *
1099 * Dump an XML document.
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001100 */
Daniel Veillarddab39b52006-10-16 23:22:10 +00001101static int
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001102xmlDocContentDumpOutput(xmlSaveCtxtPtr ctxt, xmlDocPtr cur) {
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001103#ifdef LIBXML_HTML_ENABLED
1104 xmlDtdPtr dtd;
1105 int is_xhtml = 0;
1106#endif
1107 const xmlChar *oldenc = cur->encoding;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001108 const xmlChar *oldctxtenc = ctxt->encoding;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001109 const xmlChar *encoding = ctxt->encoding;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001110 xmlCharEncodingOutputFunc oldescape = ctxt->escape;
1111 xmlCharEncodingOutputFunc oldescapeAttr = ctxt->escapeAttr;
1112 xmlOutputBufferPtr buf = ctxt->buf;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001113 xmlCharEncoding enc;
Daniel Veillardda3fee42008-09-01 13:08:57 +00001114 int switched_encoding = 0;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001115
1116 xmlInitParser();
1117
Daniel Veillardda3fee42008-09-01 13:08:57 +00001118 if ((cur->type != XML_HTML_DOCUMENT_NODE) &&
1119 (cur->type != XML_DOCUMENT_NODE))
1120 return(-1);
1121
Daniel Veillarddab39b52006-10-16 23:22:10 +00001122 if (ctxt->encoding != NULL) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001123 cur->encoding = BAD_CAST ctxt->encoding;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001124 } else if (cur->encoding != NULL) {
1125 encoding = cur->encoding;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001126 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001127
Daniel Veillard856d9282008-09-25 14:31:40 +00001128 if (((cur->type == XML_HTML_DOCUMENT_NODE) &&
1129 ((ctxt->options & XML_SAVE_AS_XML) == 0) &&
1130 ((ctxt->options & XML_SAVE_XHTML) == 0)) ||
1131 (ctxt->options & XML_SAVE_AS_HTML)) {
Daniel Veillardda3fee42008-09-01 13:08:57 +00001132#ifdef LIBXML_HTML_ENABLED
1133 if (encoding != NULL)
1134 htmlSetMetaEncoding(cur, (const xmlChar *) encoding);
1135 if (encoding == NULL)
1136 encoding = htmlGetMetaEncoding(cur);
1137 if (encoding == NULL)
1138 encoding = BAD_CAST "HTML";
1139 if ((encoding != NULL) && (oldctxtenc == NULL) &&
1140 (buf->encoder == NULL) && (buf->conv == NULL)) {
1141 if (xmlSaveSwitchEncoding(ctxt, (const char*) encoding) < 0) {
1142 cur->encoding = oldenc;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001143 return(-1);
1144 }
Daniel Veillarddab39b52006-10-16 23:22:10 +00001145 }
Daniel Veillardda3fee42008-09-01 13:08:57 +00001146 if (ctxt->options & XML_SAVE_FORMAT)
1147 htmlDocContentDumpFormatOutput(buf, cur,
1148 (const char *)encoding, 1);
Daniel Veillard100e1802005-08-08 14:44:11 +00001149 else
Daniel Veillardda3fee42008-09-01 13:08:57 +00001150 htmlDocContentDumpFormatOutput(buf, cur,
1151 (const char *)encoding, 0);
1152 if (ctxt->encoding != NULL)
1153 cur->encoding = oldenc;
1154 return(0);
1155#else
1156 return(-1);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001157#endif
Daniel Veillard856d9282008-09-25 14:31:40 +00001158 } else if ((cur->type == XML_DOCUMENT_NODE) ||
1159 (ctxt->options & XML_SAVE_AS_XML) ||
1160 (ctxt->options & XML_SAVE_XHTML)) {
Daniel Veillardda3fee42008-09-01 13:08:57 +00001161 enc = xmlParseCharEncoding((const char*) encoding);
1162 if ((encoding != NULL) && (oldctxtenc == NULL) &&
1163 (buf->encoder == NULL) && (buf->conv == NULL) &&
1164 ((ctxt->options & XML_SAVE_NO_DECL) == 0)) {
1165 if ((enc != XML_CHAR_ENCODING_UTF8) &&
1166 (enc != XML_CHAR_ENCODING_NONE) &&
1167 (enc != XML_CHAR_ENCODING_ASCII)) {
1168 /*
1169 * we need to switch to this encoding but just for this
1170 * document since we output the XMLDecl the conversion
1171 * must be done to not generate not well formed documents.
1172 */
1173 if (xmlSaveSwitchEncoding(ctxt, (const char*) encoding) < 0) {
1174 cur->encoding = oldenc;
1175 return(-1);
1176 }
1177 switched_encoding = 1;
1178 }
1179 if (ctxt->escape == xmlEscapeEntities)
1180 ctxt->escape = NULL;
1181 if (ctxt->escapeAttr == xmlEscapeEntities)
1182 ctxt->escapeAttr = NULL;
1183 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001184
Daniel Veillardda3fee42008-09-01 13:08:57 +00001185
1186 /*
1187 * Save the XML declaration
1188 */
1189 if ((ctxt->options & XML_SAVE_NO_DECL) == 0) {
1190 xmlOutputBufferWrite(buf, 14, "<?xml version=");
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001191 if (cur->version != NULL)
Daniel Veillard50cdab52012-07-16 14:52:00 +08001192 xmlBufWriteQuotedString(buf->buffer, cur->version);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001193 else
Daniel Veillardda3fee42008-09-01 13:08:57 +00001194 xmlOutputBufferWrite(buf, 5, "\"1.0\"");
1195 if (encoding != NULL) {
1196 xmlOutputBufferWrite(buf, 10, " encoding=");
Daniel Veillard50cdab52012-07-16 14:52:00 +08001197 xmlBufWriteQuotedString(buf->buffer, (xmlChar *) encoding);
Daniel Veillardda3fee42008-09-01 13:08:57 +00001198 }
1199 switch (cur->standalone) {
1200 case 0:
1201 xmlOutputBufferWrite(buf, 16, " standalone=\"no\"");
1202 break;
1203 case 1:
1204 xmlOutputBufferWrite(buf, 17, " standalone=\"yes\"");
1205 break;
1206 }
1207 xmlOutputBufferWrite(buf, 3, "?>\n");
1208 }
1209
1210#ifdef LIBXML_HTML_ENABLED
Daniel Veillard856d9282008-09-25 14:31:40 +00001211 if (ctxt->options & XML_SAVE_XHTML)
1212 is_xhtml = 1;
Daniel Veillardda3fee42008-09-01 13:08:57 +00001213 if ((ctxt->options & XML_SAVE_NO_XHTML) == 0) {
1214 dtd = xmlGetIntSubset(cur);
1215 if (dtd != NULL) {
1216 is_xhtml = xmlIsXHTML(dtd->SystemID, dtd->ExternalID);
1217 if (is_xhtml < 0) is_xhtml = 0;
1218 }
1219 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001220#endif
Daniel Veillardda3fee42008-09-01 13:08:57 +00001221 if (cur->children != NULL) {
1222 xmlNodePtr child = cur->children;
1223
1224 while (child != NULL) {
1225 ctxt->level = 0;
1226#ifdef LIBXML_HTML_ENABLED
1227 if (is_xhtml)
1228 xhtmlNodeDumpOutput(ctxt, child);
1229 else
1230#endif
1231 xmlNodeDumpOutputInternal(ctxt, child);
1232 xmlOutputBufferWrite(buf, 1, "\n");
1233 child = child->next;
1234 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001235 }
1236 }
Daniel Veillardda3fee42008-09-01 13:08:57 +00001237
Daniel Veillarddab39b52006-10-16 23:22:10 +00001238 /*
1239 * Restore the state of the saving context at the end of the document
1240 */
Daniel Veillardda3fee42008-09-01 13:08:57 +00001241 if ((switched_encoding) && (oldctxtenc == NULL)) {
1242 xmlSaveClearEncoding(ctxt);
Daniel Veillarddab39b52006-10-16 23:22:10 +00001243 ctxt->escape = oldescape;
1244 ctxt->escapeAttr = oldescapeAttr;
1245 }
Daniel Veillardda3fee42008-09-01 13:08:57 +00001246 cur->encoding = oldenc;
Daniel Veillarddab39b52006-10-16 23:22:10 +00001247 return(0);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001248}
1249
1250#ifdef LIBXML_HTML_ENABLED
1251/************************************************************************
1252 * *
1253 * Functions specific to XHTML serialization *
1254 * *
1255 ************************************************************************/
1256
1257/**
1258 * xhtmlIsEmpty:
1259 * @node: the node
1260 *
1261 * Check if a node is an empty xhtml node
1262 *
1263 * Returns 1 if the node is an empty node, 0 if not and -1 in case of error
1264 */
1265static int
1266xhtmlIsEmpty(xmlNodePtr node) {
1267 if (node == NULL)
1268 return(-1);
1269 if (node->type != XML_ELEMENT_NODE)
1270 return(0);
1271 if ((node->ns != NULL) && (!xmlStrEqual(node->ns->href, XHTML_NS_NAME)))
1272 return(0);
1273 if (node->children != NULL)
1274 return(0);
1275 switch (node->name[0]) {
1276 case 'a':
1277 if (xmlStrEqual(node->name, BAD_CAST "area"))
1278 return(1);
1279 return(0);
1280 case 'b':
1281 if (xmlStrEqual(node->name, BAD_CAST "br"))
1282 return(1);
1283 if (xmlStrEqual(node->name, BAD_CAST "base"))
1284 return(1);
1285 if (xmlStrEqual(node->name, BAD_CAST "basefont"))
1286 return(1);
1287 return(0);
1288 case 'c':
1289 if (xmlStrEqual(node->name, BAD_CAST "col"))
1290 return(1);
1291 return(0);
1292 case 'f':
1293 if (xmlStrEqual(node->name, BAD_CAST "frame"))
1294 return(1);
1295 return(0);
1296 case 'h':
1297 if (xmlStrEqual(node->name, BAD_CAST "hr"))
1298 return(1);
1299 return(0);
1300 case 'i':
1301 if (xmlStrEqual(node->name, BAD_CAST "img"))
1302 return(1);
1303 if (xmlStrEqual(node->name, BAD_CAST "input"))
1304 return(1);
1305 if (xmlStrEqual(node->name, BAD_CAST "isindex"))
1306 return(1);
1307 return(0);
1308 case 'l':
1309 if (xmlStrEqual(node->name, BAD_CAST "link"))
1310 return(1);
1311 return(0);
1312 case 'm':
1313 if (xmlStrEqual(node->name, BAD_CAST "meta"))
1314 return(1);
1315 return(0);
1316 case 'p':
1317 if (xmlStrEqual(node->name, BAD_CAST "param"))
1318 return(1);
1319 return(0);
1320 }
1321 return(0);
1322}
1323
1324/**
1325 * xhtmlAttrListDumpOutput:
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001326 * @cur: the first attribute pointer
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001327 *
1328 * Dump a list of XML attributes
1329 */
1330static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001331xhtmlAttrListDumpOutput(xmlSaveCtxtPtr ctxt, xmlAttrPtr cur) {
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001332 xmlAttrPtr xml_lang = NULL;
1333 xmlAttrPtr lang = NULL;
1334 xmlAttrPtr name = NULL;
1335 xmlAttrPtr id = NULL;
1336 xmlNodePtr parent;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001337 xmlOutputBufferPtr buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001338
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001339 if (cur == NULL) return;
1340 buf = ctxt->buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001341 parent = cur->parent;
1342 while (cur != NULL) {
1343 if ((cur->ns == NULL) && (xmlStrEqual(cur->name, BAD_CAST "id")))
1344 id = cur;
1345 else
1346 if ((cur->ns == NULL) && (xmlStrEqual(cur->name, BAD_CAST "name")))
1347 name = cur;
1348 else
1349 if ((cur->ns == NULL) && (xmlStrEqual(cur->name, BAD_CAST "lang")))
1350 lang = cur;
1351 else
1352 if ((cur->ns != NULL) && (xmlStrEqual(cur->name, BAD_CAST "lang")) &&
1353 (xmlStrEqual(cur->ns->prefix, BAD_CAST "xml")))
1354 xml_lang = cur;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001355 else if ((cur->ns == NULL) &&
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001356 ((cur->children == NULL) ||
1357 (cur->children->content == NULL) ||
1358 (cur->children->content[0] == 0)) &&
1359 (htmlIsBooleanAttr(cur->name))) {
1360 if (cur->children != NULL)
1361 xmlFreeNode(cur->children);
1362 cur->children = xmlNewText(cur->name);
1363 if (cur->children != NULL)
1364 cur->children->parent = (xmlNodePtr) cur;
1365 }
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001366 xmlAttrDumpOutput(ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001367 cur = cur->next;
1368 }
1369 /*
1370 * C.8
1371 */
1372 if ((name != NULL) && (id == NULL)) {
1373 if ((parent != NULL) && (parent->name != NULL) &&
1374 ((xmlStrEqual(parent->name, BAD_CAST "a")) ||
1375 (xmlStrEqual(parent->name, BAD_CAST "p")) ||
1376 (xmlStrEqual(parent->name, BAD_CAST "div")) ||
1377 (xmlStrEqual(parent->name, BAD_CAST "img")) ||
1378 (xmlStrEqual(parent->name, BAD_CAST "map")) ||
1379 (xmlStrEqual(parent->name, BAD_CAST "applet")) ||
1380 (xmlStrEqual(parent->name, BAD_CAST "form")) ||
1381 (xmlStrEqual(parent->name, BAD_CAST "frame")) ||
1382 (xmlStrEqual(parent->name, BAD_CAST "iframe")))) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001383 xmlOutputBufferWrite(buf, 5, " id=\"");
1384 xmlAttrSerializeContent(buf, name);
1385 xmlOutputBufferWrite(buf, 1, "\"");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001386 }
1387 }
1388 /*
1389 * C.7.
1390 */
1391 if ((lang != NULL) && (xml_lang == NULL)) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001392 xmlOutputBufferWrite(buf, 11, " xml:lang=\"");
1393 xmlAttrSerializeContent(buf, lang);
1394 xmlOutputBufferWrite(buf, 1, "\"");
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001395 } else
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001396 if ((xml_lang != NULL) && (lang == NULL)) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001397 xmlOutputBufferWrite(buf, 7, " lang=\"");
1398 xmlAttrSerializeContent(buf, xml_lang);
1399 xmlOutputBufferWrite(buf, 1, "\"");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001400 }
1401}
1402
1403/**
1404 * xhtmlNodeListDumpOutput:
1405 * @buf: the XML buffer output
1406 * @doc: the XHTML document
1407 * @cur: the first node
1408 * @level: the imbrication level for indenting
1409 * @format: is formatting allowed
1410 * @encoding: an optional encoding string
1411 *
1412 * Dump an XML node list, recursive behaviour, children are printed too.
1413 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
1414 * or xmlKeepBlanksDefault(0) was called
1415 */
1416static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001417xhtmlNodeListDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001418 xmlOutputBufferPtr buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001419
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001420 if (cur == NULL) return;
1421 buf = ctxt->buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001422 while (cur != NULL) {
Adam Spragg8b877132010-11-01 14:24:56 +01001423 if ((ctxt->format == 1) && (xmlIndentTreeOutput) &&
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001424 (cur->type == XML_ELEMENT_NODE))
Daniel Veillard753086a2004-03-28 16:12:44 +00001425 xmlOutputBufferWrite(buf, ctxt->indent_size *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001426 (ctxt->level > ctxt->indent_nr ?
Daniel Veillard753086a2004-03-28 16:12:44 +00001427 ctxt->indent_nr : ctxt->level),
1428 ctxt->indent);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001429 xhtmlNodeDumpOutput(ctxt, cur);
Adam Spragg8b877132010-11-01 14:24:56 +01001430 if (ctxt->format == 1) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001431 xmlOutputBufferWrite(buf, 1, "\n");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001432 }
1433 cur = cur->next;
1434 }
1435}
1436
1437/**
1438 * xhtmlNodeDumpOutput:
1439 * @buf: the XML buffer output
1440 * @doc: the XHTML document
1441 * @cur: the current node
1442 * @level: the imbrication level for indenting
1443 * @format: is formatting allowed
1444 * @encoding: an optional encoding string
1445 *
1446 * Dump an XHTML node, recursive behaviour, children are printed too.
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001447 */
1448static void
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001449xhtmlNodeDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) {
Rob Richards31f73022005-08-26 15:33:26 +00001450 int format, addmeta = 0;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001451 xmlNodePtr tmp;
1452 xmlChar *start, *end;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001453 xmlOutputBufferPtr buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001454
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001455 if (cur == NULL) return;
Daniel Veillard60071ae2005-09-12 00:03:43 +00001456 if ((cur->type == XML_DOCUMENT_NODE) ||
1457 (cur->type == XML_HTML_DOCUMENT_NODE)) {
1458 xmlDocContentDumpOutput(ctxt, (xmlDocPtr) cur);
1459 return;
1460 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001461 if (cur->type == XML_XINCLUDE_START)
1462 return;
1463 if (cur->type == XML_XINCLUDE_END)
1464 return;
Daniel Veillard3e62adb2012-08-09 14:24:02 +08001465 if (cur->type == XML_NAMESPACE_DECL) {
1466 xmlNsDumpOutputCtxt(ctxt, (xmlNsPtr) cur);
1467 return;
1468 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001469 if (cur->type == XML_DTD_NODE) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001470 xmlDtdDumpOutput(ctxt, (xmlDtdPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001471 return;
1472 }
Rob Richards2e2691b2005-10-21 14:45:16 +00001473 if (cur->type == XML_DOCUMENT_FRAG_NODE) {
1474 xhtmlNodeListDumpOutput(ctxt, cur->children);
1475 return;
1476 }
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001477 buf = ctxt->buf;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001478 if (cur->type == XML_ELEMENT_DECL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +08001479 xmlBufDumpElementDecl(buf->buffer, (xmlElementPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001480 return;
1481 }
1482 if (cur->type == XML_ATTRIBUTE_DECL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +08001483 xmlBufDumpAttributeDecl(buf->buffer, (xmlAttributePtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001484 return;
1485 }
1486 if (cur->type == XML_ENTITY_DECL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +08001487 xmlBufDumpEntityDecl(buf->buffer, (xmlEntityPtr) cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001488 return;
1489 }
1490 if (cur->type == XML_TEXT_NODE) {
1491 if (cur->content != NULL) {
1492 if ((cur->name == xmlStringText) ||
1493 (cur->name != xmlStringTextNoenc)) {
Daniel Veillard3995bc32004-05-15 18:57:31 +00001494 xmlOutputBufferWriteEscape(buf, cur->content, ctxt->escape);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001495 } else {
1496 /*
1497 * Disable escaping, needed for XSLT
1498 */
1499 xmlOutputBufferWriteString(buf, (const char *) cur->content);
1500 }
1501 }
1502
1503 return;
1504 }
1505 if (cur->type == XML_PI_NODE) {
1506 if (cur->content != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001507 xmlOutputBufferWrite(buf, 2, "<?");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001508 xmlOutputBufferWriteString(buf, (const char *)cur->name);
1509 if (cur->content != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001510 xmlOutputBufferWrite(buf, 1, " ");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001511 xmlOutputBufferWriteString(buf, (const char *)cur->content);
1512 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001513 xmlOutputBufferWrite(buf, 2, "?>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001514 } else {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001515 xmlOutputBufferWrite(buf, 2, "<?");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001516 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001517 xmlOutputBufferWrite(buf, 2, "?>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001518 }
1519 return;
1520 }
1521 if (cur->type == XML_COMMENT_NODE) {
1522 if (cur->content != NULL) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001523 xmlOutputBufferWrite(buf, 4, "<!--");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001524 xmlOutputBufferWriteString(buf, (const char *)cur->content);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001525 xmlOutputBufferWrite(buf, 3, "-->");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001526 }
1527 return;
1528 }
1529 if (cur->type == XML_ENTITY_REF_NODE) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001530 xmlOutputBufferWrite(buf, 1, "&");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001531 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001532 xmlOutputBufferWrite(buf, 1, ";");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001533 return;
1534 }
1535 if (cur->type == XML_CDATA_SECTION_NODE) {
Daniel Veillardd0d2f092008-03-07 16:50:21 +00001536 if (cur->content == NULL || *cur->content == '\0') {
1537 xmlOutputBufferWrite(buf, 12, "<![CDATA[]]>");
1538 } else {
1539 start = end = cur->content;
1540 while (*end != '\0') {
1541 if (*end == ']' && *(end + 1) == ']' && *(end + 2) == '>') {
1542 end = end + 2;
1543 xmlOutputBufferWrite(buf, 9, "<![CDATA[");
1544 xmlOutputBufferWrite(buf, end - start, (const char *)start);
1545 xmlOutputBufferWrite(buf, 3, "]]>");
1546 start = end;
1547 }
1548 end++;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001549 }
Daniel Veillardd0d2f092008-03-07 16:50:21 +00001550 if (start != end) {
1551 xmlOutputBufferWrite(buf, 9, "<![CDATA[");
1552 xmlOutputBufferWriteString(buf, (const char *)start);
1553 xmlOutputBufferWrite(buf, 3, "]]>");
1554 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001555 }
1556 return;
1557 }
Daniel Veillarda76a81f2007-10-10 08:28:18 +00001558 if (cur->type == XML_ATTRIBUTE_NODE) {
1559 xmlAttrDumpOutput(ctxt, (xmlAttrPtr) cur);
1560 return;
1561 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001562
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001563 format = ctxt->format;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001564 if (format == 1) {
1565 tmp = cur->children;
1566 while (tmp != NULL) {
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001567 if ((tmp->type == XML_TEXT_NODE) ||
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001568 (tmp->type == XML_ENTITY_REF_NODE)) {
1569 format = 0;
1570 break;
1571 }
1572 tmp = tmp->next;
1573 }
1574 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001575 xmlOutputBufferWrite(buf, 1, "<");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001576 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
1577 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001578 xmlOutputBufferWrite(buf, 1, ":");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001579 }
1580
1581 xmlOutputBufferWriteString(buf, (const char *)cur->name);
1582 if (cur->nsDef)
Adam Spraggd2e62312010-11-03 15:33:40 +01001583 xmlNsListDumpOutputCtxt(ctxt, cur->nsDef);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001584 if ((xmlStrEqual(cur->name, BAD_CAST "html") &&
1585 (cur->ns == NULL) && (cur->nsDef == NULL))) {
1586 /*
1587 * 3.1.1. Strictly Conforming Documents A.3.1.1 3/
1588 */
1589 xmlOutputBufferWriteString(buf,
1590 " xmlns=\"http://www.w3.org/1999/xhtml\"");
1591 }
1592 if (cur->properties != NULL)
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001593 xhtmlAttrListDumpOutput(ctxt, cur->properties);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001594
Nick Wellnhofer359e7502017-11-13 21:13:46 +01001595 if ((cur->type == XML_ELEMENT_NODE) &&
1596 (cur->parent != NULL) &&
1597 (cur->parent->parent == (xmlNodePtr) cur->doc) &&
1598 xmlStrEqual(cur->name, BAD_CAST"head") &&
1599 xmlStrEqual(cur->parent->name, BAD_CAST"html")) {
Rob Richards31f73022005-08-26 15:33:26 +00001600
Nick Wellnhofer359e7502017-11-13 21:13:46 +01001601 tmp = cur->children;
1602 while (tmp != NULL) {
1603 if (xmlStrEqual(tmp->name, BAD_CAST"meta")) {
1604 xmlChar *httpequiv;
Rob Richards31f73022005-08-26 15:33:26 +00001605
Nick Wellnhofer359e7502017-11-13 21:13:46 +01001606 httpequiv = xmlGetProp(tmp, BAD_CAST"http-equiv");
1607 if (httpequiv != NULL) {
1608 if (xmlStrcasecmp(httpequiv, BAD_CAST"Content-Type") == 0) {
1609 xmlFree(httpequiv);
1610 break;
1611 }
1612 xmlFree(httpequiv);
1613 }
1614 }
1615 tmp = tmp->next;
1616 }
1617 if (tmp == NULL)
1618 addmeta = 1;
1619 }
Rob Richards31f73022005-08-26 15:33:26 +00001620
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001621 if ((cur->type == XML_ELEMENT_NODE) && (cur->children == NULL)) {
1622 if (((cur->ns == NULL) || (cur->ns->prefix == NULL)) &&
Rob Richards31f73022005-08-26 15:33:26 +00001623 ((xhtmlIsEmpty(cur) == 1) && (addmeta == 0))) {
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001624 /*
1625 * C.2. Empty Elements
1626 */
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001627 xmlOutputBufferWrite(buf, 3, " />");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001628 } else {
Rob Richards31f73022005-08-26 15:33:26 +00001629 if (addmeta == 1) {
1630 xmlOutputBufferWrite(buf, 1, ">");
Adam Spragg8b877132010-11-01 14:24:56 +01001631 if (ctxt->format == 1) {
Rob Richards2ce51c02005-09-12 12:16:35 +00001632 xmlOutputBufferWrite(buf, 1, "\n");
1633 if (xmlIndentTreeOutput)
1634 xmlOutputBufferWrite(buf, ctxt->indent_size *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001635 (ctxt->level + 1 > ctxt->indent_nr ?
Rob Richards2ce51c02005-09-12 12:16:35 +00001636 ctxt->indent_nr : ctxt->level + 1), ctxt->indent);
1637 }
Rob Richards31f73022005-08-26 15:33:26 +00001638 xmlOutputBufferWriteString(buf,
1639 "<meta http-equiv=\"Content-Type\" content=\"text/html; charset=");
1640 if (ctxt->encoding) {
1641 xmlOutputBufferWriteString(buf, (const char *)ctxt->encoding);
1642 } else {
1643 xmlOutputBufferWrite(buf, 5, "UTF-8");
1644 }
Rob Richards2ce51c02005-09-12 12:16:35 +00001645 xmlOutputBufferWrite(buf, 4, "\" />");
Adam Spragg8b877132010-11-01 14:24:56 +01001646 if (ctxt->format == 1)
Rob Richards2ce51c02005-09-12 12:16:35 +00001647 xmlOutputBufferWrite(buf, 1, "\n");
1648 } else {
1649 xmlOutputBufferWrite(buf, 1, ">");
Rob Richards31f73022005-08-26 15:33:26 +00001650 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001651 /*
1652 * C.3. Element Minimization and Empty Element Content
1653 */
Rob Richards2ce51c02005-09-12 12:16:35 +00001654 xmlOutputBufferWrite(buf, 2, "</");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001655 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
1656 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001657 xmlOutputBufferWrite(buf, 1, ":");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001658 }
1659 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001660 xmlOutputBufferWrite(buf, 1, ">");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001661 }
1662 return;
1663 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001664 xmlOutputBufferWrite(buf, 1, ">");
Rob Richards31f73022005-08-26 15:33:26 +00001665 if (addmeta == 1) {
Adam Spragg8b877132010-11-01 14:24:56 +01001666 if (ctxt->format == 1) {
Rob Richards2ce51c02005-09-12 12:16:35 +00001667 xmlOutputBufferWrite(buf, 1, "\n");
1668 if (xmlIndentTreeOutput)
1669 xmlOutputBufferWrite(buf, ctxt->indent_size *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001670 (ctxt->level + 1 > ctxt->indent_nr ?
Rob Richards2ce51c02005-09-12 12:16:35 +00001671 ctxt->indent_nr : ctxt->level + 1), ctxt->indent);
1672 }
Rob Richards31f73022005-08-26 15:33:26 +00001673 xmlOutputBufferWriteString(buf,
1674 "<meta http-equiv=\"Content-Type\" content=\"text/html; charset=");
1675 if (ctxt->encoding) {
1676 xmlOutputBufferWriteString(buf, (const char *)ctxt->encoding);
1677 } else {
1678 xmlOutputBufferWrite(buf, 5, "UTF-8");
1679 }
1680 xmlOutputBufferWrite(buf, 4, "\" />");
1681 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001682 if ((cur->type != XML_ELEMENT_NODE) && (cur->content != NULL)) {
Daniel Veillard3995bc32004-05-15 18:57:31 +00001683 xmlOutputBufferWriteEscape(buf, cur->content, ctxt->escape);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001684 }
1685
Kasimier T. Buchcik7b4e2e22006-07-13 13:07:11 +00001686#if 0
1687 /*
1688 * This was removed due to problems with HTML processors.
1689 * See bug #345147.
Daniel Veillarddab39b52006-10-16 23:22:10 +00001690 */
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001691 /*
1692 * 4.8. Script and Style elements
1693 */
1694 if ((cur->type == XML_ELEMENT_NODE) &&
1695 ((xmlStrEqual(cur->name, BAD_CAST "script")) ||
1696 (xmlStrEqual(cur->name, BAD_CAST "style"))) &&
1697 ((cur->ns == NULL) ||
1698 (xmlStrEqual(cur->ns->href, XHTML_NS_NAME)))) {
1699 xmlNodePtr child = cur->children;
1700
1701 while (child != NULL) {
Daniel Veillarddbd61052005-09-12 14:03:26 +00001702 if (child->type == XML_TEXT_NODE) {
1703 if ((xmlStrchr(child->content, '<') == NULL) &&
1704 (xmlStrchr(child->content, '&') == NULL) &&
1705 (xmlStrstr(child->content, BAD_CAST "]]>") == NULL)) {
1706 /* Nothing to escape, so just output as is... */
1707 /* FIXME: Should we do something about "--" also? */
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001708 int level = ctxt->level;
1709 int indent = ctxt->format;
1710
1711 ctxt->level = 0;
1712 ctxt->format = 0;
Daniel Veillarddbd61052005-09-12 14:03:26 +00001713 xmlOutputBufferWriteString(buf, (const char *) child->content);
1714 /* (We cannot use xhtmlNodeDumpOutput() here because
1715 * we wish to leave '>' unescaped!) */
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001716 ctxt->level = level;
1717 ctxt->format = indent;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001718 } else {
Daniel Veillarddbd61052005-09-12 14:03:26 +00001719 /* We must use a CDATA section. Unfortunately,
1720 * this will break CSS and JavaScript when read by
1721 * a browser in HTML4-compliant mode. :-( */
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001722 start = end = child->content;
1723 while (*end != '\0') {
1724 if (*end == ']' &&
1725 *(end + 1) == ']' &&
1726 *(end + 2) == '>') {
1727 end = end + 2;
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001728 xmlOutputBufferWrite(buf, 9, "<![CDATA[");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001729 xmlOutputBufferWrite(buf, end - start,
1730 (const char *)start);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001731 xmlOutputBufferWrite(buf, 3, "]]>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001732 start = end;
1733 }
1734 end++;
1735 }
1736 if (start != end) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001737 xmlOutputBufferWrite(buf, 9, "<![CDATA[");
1738 xmlOutputBufferWrite(buf, end - start,
1739 (const char *)start);
1740 xmlOutputBufferWrite(buf, 3, "]]>");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001741 }
1742 }
1743 } else {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001744 int level = ctxt->level;
1745 int indent = ctxt->format;
1746
1747 ctxt->level = 0;
1748 ctxt->format = 0;
1749 xhtmlNodeDumpOutput(ctxt, child);
1750 ctxt->level = level;
1751 ctxt->format = indent;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001752 }
1753 child = child->next;
1754 }
Kasimier T. Buchcik7b4e2e22006-07-13 13:07:11 +00001755 }
1756#endif
1757
1758 if (cur->children != NULL) {
Daniel Veillardf0244ce2004-05-09 23:48:39 +00001759 int indent = ctxt->format;
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001760
Adam Spragg8b877132010-11-01 14:24:56 +01001761 if (format == 1) xmlOutputBufferWrite(buf, 1, "\n");
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001762 if (ctxt->level >= 0) ctxt->level++;
Daniel Veillardf0244ce2004-05-09 23:48:39 +00001763 ctxt->format = format;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001764 xhtmlNodeListDumpOutput(ctxt, cur->children);
1765 if (ctxt->level > 0) ctxt->level--;
Daniel Veillardf0244ce2004-05-09 23:48:39 +00001766 ctxt->format = indent;
Adam Spragg8b877132010-11-01 14:24:56 +01001767 if ((xmlIndentTreeOutput) && (format == 1))
Daniel Veillard753086a2004-03-28 16:12:44 +00001768 xmlOutputBufferWrite(buf, ctxt->indent_size *
Daniel Veillardf8e3db02012-09-11 13:26:36 +08001769 (ctxt->level > ctxt->indent_nr ?
Daniel Veillard753086a2004-03-28 16:12:44 +00001770 ctxt->indent_nr : ctxt->level),
1771 ctxt->indent);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001772 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001773 xmlOutputBufferWrite(buf, 2, "</");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001774 if ((cur->ns != NULL) && (cur->ns->prefix != NULL)) {
1775 xmlOutputBufferWriteString(buf, (const char *)cur->ns->prefix);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001776 xmlOutputBufferWrite(buf, 1, ":");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001777 }
1778
1779 xmlOutputBufferWriteString(buf, (const char *)cur->name);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00001780 xmlOutputBufferWrite(buf, 1, ">");
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001781}
1782#endif
1783
1784/************************************************************************
1785 * *
1786 * Public entry points *
1787 * *
1788 ************************************************************************/
1789
1790/**
1791 * xmlSaveToFd:
1792 * @fd: a file descriptor number
1793 * @encoding: the encoding name to use or NULL
1794 * @options: a set of xmlSaveOptions
1795 *
1796 * Create a document saving context serializing to a file descriptor
1797 * with the encoding and the options given.
1798 *
1799 * Returns a new serialization context or NULL in case of error.
1800 */
1801xmlSaveCtxtPtr
1802xmlSaveToFd(int fd, const char *encoding, int options)
1803{
1804 xmlSaveCtxtPtr ret;
1805
1806 ret = xmlNewSaveCtxt(encoding, options);
1807 if (ret == NULL) return(NULL);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001808 ret->buf = xmlOutputBufferCreateFd(fd, ret->handler);
1809 if (ret->buf == NULL) {
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001810 xmlFreeSaveCtxt(ret);
1811 return(NULL);
1812 }
1813 return(ret);
1814}
1815
1816/**
1817 * xmlSaveToFilename:
1818 * @filename: a file name or an URL
1819 * @encoding: the encoding name to use or NULL
1820 * @options: a set of xmlSaveOptions
1821 *
1822 * Create a document saving context serializing to a filename or possibly
1823 * to an URL (but this is less reliable) with the encoding and the options
1824 * given.
1825 *
1826 * Returns a new serialization context or NULL in case of error.
1827 */
1828xmlSaveCtxtPtr
1829xmlSaveToFilename(const char *filename, const char *encoding, int options)
1830{
1831 xmlSaveCtxtPtr ret;
1832 int compression = 0; /* TODO handle compression option */
1833
1834 ret = xmlNewSaveCtxt(encoding, options);
1835 if (ret == NULL) return(NULL);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001836 ret->buf = xmlOutputBufferCreateFilename(filename, ret->handler,
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001837 compression);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001838 if (ret->buf == NULL) {
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001839 xmlFreeSaveCtxt(ret);
1840 return(NULL);
1841 }
1842 return(ret);
1843}
1844
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001845/**
1846 * xmlSaveToBuffer:
1847 * @buffer: a buffer
1848 * @encoding: the encoding name to use or NULL
1849 * @options: a set of xmlSaveOptions
1850 *
1851 * Create a document saving context serializing to a buffer
1852 * with the encoding and the options given
1853 *
1854 * Returns a new serialization context or NULL in case of error.
Daniel Veillard9a00fd22005-11-09 08:56:26 +00001855 */
1856
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001857xmlSaveCtxtPtr
1858xmlSaveToBuffer(xmlBufferPtr buffer, const char *encoding, int options)
1859{
Daniel Veillard9a00fd22005-11-09 08:56:26 +00001860 xmlSaveCtxtPtr ret;
1861 xmlOutputBufferPtr out_buff;
1862 xmlCharEncodingHandlerPtr handler;
1863
1864 ret = xmlNewSaveCtxt(encoding, options);
1865 if (ret == NULL) return(NULL);
1866
1867 if (encoding != NULL) {
1868 handler = xmlFindCharEncodingHandler(encoding);
1869 if (handler == NULL) {
1870 xmlFree(ret);
1871 return(NULL);
1872 }
1873 } else
1874 handler = NULL;
1875 out_buff = xmlOutputBufferCreateBuffer(buffer, handler);
1876 if (out_buff == NULL) {
1877 xmlFree(ret);
1878 if (handler) xmlCharEncCloseFunc(handler);
1879 return(NULL);
1880 }
1881
1882 ret->buf = out_buff;
1883 return(ret);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001884}
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001885
1886/**
1887 * xmlSaveToIO:
1888 * @iowrite: an I/O write function
1889 * @ioclose: an I/O close function
1890 * @ioctx: an I/O handler
1891 * @encoding: the encoding name to use or NULL
1892 * @options: a set of xmlSaveOptions
1893 *
1894 * Create a document saving context serializing to a file descriptor
1895 * with the encoding and the options given
1896 *
1897 * Returns a new serialization context or NULL in case of error.
1898 */
1899xmlSaveCtxtPtr
1900xmlSaveToIO(xmlOutputWriteCallback iowrite,
1901 xmlOutputCloseCallback ioclose,
1902 void *ioctx, const char *encoding, int options)
1903{
1904 xmlSaveCtxtPtr ret;
1905
1906 ret = xmlNewSaveCtxt(encoding, options);
1907 if (ret == NULL) return(NULL);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001908 ret->buf = xmlOutputBufferCreateIO(iowrite, ioclose, ioctx, ret->handler);
1909 if (ret->buf == NULL) {
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001910 xmlFreeSaveCtxt(ret);
1911 return(NULL);
1912 }
1913 return(ret);
1914}
1915
1916/**
1917 * xmlSaveDoc:
1918 * @ctxt: a document saving context
1919 * @doc: a document
1920 *
1921 * Save a full document to a saving context
Daniel Veillard377e1a92004-04-16 16:30:05 +00001922 * TODO: The function is not fully implemented yet as it does not return the
1923 * byte count but 0 instead
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001924 *
1925 * Returns the number of byte written or -1 in case of error
1926 */
1927long
1928xmlSaveDoc(xmlSaveCtxtPtr ctxt, xmlDocPtr doc)
1929{
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001930 long ret = 0;
1931
Daniel Veillardce682bc2004-11-05 17:22:25 +00001932 if ((ctxt == NULL) || (doc == NULL)) return(-1);
Daniel Veillarddab39b52006-10-16 23:22:10 +00001933 if (xmlDocContentDumpOutput(ctxt, doc) < 0)
1934 return(-1);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001935 return(ret);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001936}
1937
1938/**
1939 * xmlSaveTree:
1940 * @ctxt: a document saving context
Daniel Veillard681e9042006-09-29 09:16:00 +00001941 * @node: the top node of the subtree to save
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001942 *
1943 * Save a subtree starting at the node parameter to a saving context
Daniel Veillard377e1a92004-04-16 16:30:05 +00001944 * TODO: The function is not fully implemented yet as it does not return the
1945 * byte count but 0 instead
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001946 *
1947 * Returns the number of byte written or -1 in case of error
1948 */
1949long
1950xmlSaveTree(xmlSaveCtxtPtr ctxt, xmlNodePtr node)
1951{
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001952 long ret = 0;
1953
Daniel Veillardce682bc2004-11-05 17:22:25 +00001954 if ((ctxt == NULL) || (node == NULL)) return(-1);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001955 xmlNodeDumpOutputInternal(ctxt, node);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001956 return(ret);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001957}
1958
1959/**
1960 * xmlSaveFlush:
1961 * @ctxt: a document saving context
1962 *
1963 * Flush a document saving context, i.e. make sure that all bytes have
1964 * been output.
1965 *
1966 * Returns the number of byte written or -1 in case of error.
1967 */
1968int
1969xmlSaveFlush(xmlSaveCtxtPtr ctxt)
1970{
1971 if (ctxt == NULL) return(-1);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00001972 if (ctxt->buf == NULL) return(-1);
1973 return(xmlOutputBufferFlush(ctxt->buf));
Daniel Veillard1a8741c2004-03-04 13:40:59 +00001974}
1975
1976/**
1977 * xmlSaveClose:
1978 * @ctxt: a document saving context
1979 *
1980 * Close a document saving context, i.e. make sure that all bytes have
1981 * been output and free the associated data.
1982 *
1983 * Returns the number of byte written or -1 in case of error.
1984 */
1985int
1986xmlSaveClose(xmlSaveCtxtPtr ctxt)
1987{
1988 int ret;
1989
1990 if (ctxt == NULL) return(-1);
1991 ret = xmlSaveFlush(ctxt);
1992 xmlFreeSaveCtxt(ctxt);
1993 return(ret);
1994}
1995
Daniel Veillard3995bc32004-05-15 18:57:31 +00001996/**
1997 * xmlSaveSetEscape:
1998 * @ctxt: a document saving context
1999 * @escape: the escaping function
2000 *
2001 * Set a custom escaping function to be used for text in element content
2002 *
2003 * Returns 0 if successful or -1 in case of error.
2004 */
2005int
2006xmlSaveSetEscape(xmlSaveCtxtPtr ctxt, xmlCharEncodingOutputFunc escape)
2007{
2008 if (ctxt == NULL) return(-1);
2009 ctxt->escape = escape;
2010 return(0);
2011}
2012
2013/**
2014 * xmlSaveSetAttrEscape:
2015 * @ctxt: a document saving context
2016 * @escape: the escaping function
2017 *
2018 * Set a custom escaping function to be used for text in attribute content
2019 *
2020 * Returns 0 if successful or -1 in case of error.
2021 */
2022int
2023xmlSaveSetAttrEscape(xmlSaveCtxtPtr ctxt, xmlCharEncodingOutputFunc escape)
2024{
2025 if (ctxt == NULL) return(-1);
2026 ctxt->escapeAttr = escape;
2027 return(0);
2028}
2029
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002030/************************************************************************
2031 * *
2032 * Public entry points based on buffers *
2033 * *
2034 ************************************************************************/
Daniel Veillard50cdab52012-07-16 14:52:00 +08002035
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002036/**
Daniel Veillard50cdab52012-07-16 14:52:00 +08002037 * xmlBufAttrSerializeTxtContent:
2038 * @buf: and xmlBufPtr output
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002039 * @doc: the document
2040 * @attr: the attribute node
2041 * @string: the text content
2042 *
Daniel Veillard50cdab52012-07-16 14:52:00 +08002043 * Serialize text attribute values to an xmlBufPtr
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002044 */
2045void
Daniel Veillard50cdab52012-07-16 14:52:00 +08002046xmlBufAttrSerializeTxtContent(xmlBufPtr buf, xmlDocPtr doc,
2047 xmlAttrPtr attr, const xmlChar * string)
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002048{
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002049 xmlChar *base, *cur;
2050
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002051 if (string == NULL)
2052 return;
2053 base = cur = (xmlChar *) string;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002054 while (*cur != 0) {
2055 if (*cur == '\n') {
2056 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002057 xmlBufAdd(buf, base, cur - base);
2058 xmlBufAdd(buf, BAD_CAST "&#10;", 5);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002059 cur++;
2060 base = cur;
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002061 } else if (*cur == '\r') {
2062 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002063 xmlBufAdd(buf, base, cur - base);
2064 xmlBufAdd(buf, BAD_CAST "&#13;", 5);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002065 cur++;
2066 base = cur;
2067 } else if (*cur == '\t') {
2068 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002069 xmlBufAdd(buf, base, cur - base);
2070 xmlBufAdd(buf, BAD_CAST "&#9;", 4);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002071 cur++;
2072 base = cur;
2073 } else if (*cur == '"') {
2074 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002075 xmlBufAdd(buf, base, cur - base);
2076 xmlBufAdd(buf, BAD_CAST "&quot;", 6);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002077 cur++;
2078 base = cur;
2079 } else if (*cur == '<') {
2080 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002081 xmlBufAdd(buf, base, cur - base);
2082 xmlBufAdd(buf, BAD_CAST "&lt;", 4);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002083 cur++;
2084 base = cur;
2085 } else if (*cur == '>') {
2086 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002087 xmlBufAdd(buf, base, cur - base);
2088 xmlBufAdd(buf, BAD_CAST "&gt;", 4);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002089 cur++;
2090 base = cur;
2091 } else if (*cur == '&') {
2092 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002093 xmlBufAdd(buf, base, cur - base);
2094 xmlBufAdd(buf, BAD_CAST "&amp;", 5);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002095 cur++;
2096 base = cur;
Daniel Veillardc97750d2016-05-23 13:39:13 +08002097 } else if ((*cur >= 0x80) && (cur[1] != 0) &&
2098 ((doc == NULL) || (doc->encoding == NULL))) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002099 /*
2100 * We assume we have UTF-8 content.
2101 */
Daniel Veillard07953482012-01-22 17:42:35 +08002102 unsigned char tmp[12];
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002103 int val = 0, l = 1;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002104
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002105 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002106 xmlBufAdd(buf, base, cur - base);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002107 if (*cur < 0xC0) {
2108 xmlSaveErr(XML_SAVE_NOT_UTF8, (xmlNodePtr) attr, NULL);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002109 xmlSerializeHexCharRef(tmp, *cur);
Daniel Veillard50cdab52012-07-16 14:52:00 +08002110 xmlBufAdd(buf, (xmlChar *) tmp, -1);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002111 cur++;
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002112 base = cur;
2113 continue;
2114 } else if (*cur < 0xE0) {
2115 val = (cur[0]) & 0x1F;
2116 val <<= 6;
2117 val |= (cur[1]) & 0x3F;
2118 l = 2;
Daniel Veillardc97750d2016-05-23 13:39:13 +08002119 } else if ((*cur < 0xF0) && (cur [2] != 0)) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002120 val = (cur[0]) & 0x0F;
2121 val <<= 6;
2122 val |= (cur[1]) & 0x3F;
2123 val <<= 6;
2124 val |= (cur[2]) & 0x3F;
2125 l = 3;
Daniel Veillardc97750d2016-05-23 13:39:13 +08002126 } else if ((*cur < 0xF8) && (cur [2] != 0) && (cur[3] != 0)) {
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002127 val = (cur[0]) & 0x07;
2128 val <<= 6;
2129 val |= (cur[1]) & 0x3F;
2130 val <<= 6;
2131 val |= (cur[2]) & 0x3F;
2132 val <<= 6;
2133 val |= (cur[3]) & 0x3F;
2134 l = 4;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002135 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002136 if ((l == 1) || (!IS_CHAR(val))) {
2137 xmlSaveErr(XML_SAVE_CHAR_INVALID, (xmlNodePtr) attr, NULL);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002138 xmlSerializeHexCharRef(tmp, *cur);
Daniel Veillard50cdab52012-07-16 14:52:00 +08002139 xmlBufAdd(buf, (xmlChar *) tmp, -1);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002140 cur++;
2141 base = cur;
2142 continue;
2143 }
2144 /*
2145 * We could do multiple things here. Just save
2146 * as a char ref
2147 */
2148 xmlSerializeHexCharRef(tmp, val);
Daniel Veillard50cdab52012-07-16 14:52:00 +08002149 xmlBufAdd(buf, (xmlChar *) tmp, -1);
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002150 cur += l;
2151 base = cur;
2152 } else {
2153 cur++;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002154 }
Daniel Veillard7a6361f2004-05-15 16:37:50 +00002155 }
2156 if (base != cur)
Daniel Veillard50cdab52012-07-16 14:52:00 +08002157 xmlBufAdd(buf, base, cur - base);
2158}
2159
2160/**
2161 * xmlAttrSerializeTxtContent:
2162 * @buf: the XML buffer output
2163 * @doc: the document
2164 * @attr: the attribute node
2165 * @string: the text content
2166 *
2167 * Serialize text attribute values to an xml simple buffer
2168 */
2169void
2170xmlAttrSerializeTxtContent(xmlBufferPtr buf, xmlDocPtr doc,
2171 xmlAttrPtr attr, const xmlChar * string)
2172{
2173 xmlBufPtr buffer;
2174
2175 if ((buf == NULL) || (string == NULL))
2176 return;
2177 buffer = xmlBufFromBuffer(buf);
2178 if (buffer == NULL)
2179 return;
2180 xmlBufAttrSerializeTxtContent(buffer, doc, attr, string);
2181 xmlBufBackToBuffer(buffer);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002182}
2183
2184/**
2185 * xmlNodeDump:
2186 * @buf: the XML buffer output
2187 * @doc: the document
2188 * @cur: the current node
2189 * @level: the imbrication level for indenting
2190 * @format: is formatting allowed
2191 *
2192 * Dump an XML node, recursive behaviour,children are printed too.
2193 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2194 * or xmlKeepBlanksDefault(0) was called
Daniel Veillard50cdab52012-07-16 14:52:00 +08002195 * Since this is using xmlBuffer structures it is limited to 2GB and somehow
2196 * deprecated, use xmlBufNodeDump() instead.
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002197 *
2198 * Returns the number of bytes written to the buffer or -1 in case of error
2199 */
2200int
2201xmlNodeDump(xmlBufferPtr buf, xmlDocPtr doc, xmlNodePtr cur, int level,
2202 int format)
2203{
Daniel Veillard50cdab52012-07-16 14:52:00 +08002204 xmlBufPtr buffer;
2205 int ret;
2206
2207 if ((buf == NULL) || (cur == NULL))
2208 return(-1);
2209 buffer = xmlBufFromBuffer(buf);
2210 if (buffer == NULL)
2211 return(-1);
2212 ret = xmlBufNodeDump(buffer, doc, cur, level, format);
2213 xmlBufBackToBuffer(buffer);
2214 if (ret > INT_MAX)
2215 return(-1);
2216 return((int) ret);
2217}
2218
2219/**
2220 * xmlBufNodeDump:
2221 * @buf: the XML buffer output
2222 * @doc: the document
2223 * @cur: the current node
2224 * @level: the imbrication level for indenting
2225 * @format: is formatting allowed
2226 *
2227 * Dump an XML node, recursive behaviour,children are printed too.
2228 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2229 * or xmlKeepBlanksDefault(0) was called
2230 *
2231 * Returns the number of bytes written to the buffer, in case of error 0
2232 * is returned or @buf stores the error
2233 */
2234
2235size_t
2236xmlBufNodeDump(xmlBufPtr buf, xmlDocPtr doc, xmlNodePtr cur, int level,
2237 int format)
2238{
2239 size_t use;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002240 int ret;
2241 xmlOutputBufferPtr outbuf;
Daniel Veillard23922c52013-02-11 11:52:44 +08002242 int oldalloc;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002243
2244 xmlInitParser();
2245
2246 if (cur == NULL) {
2247#ifdef DEBUG_TREE
2248 xmlGenericError(xmlGenericErrorContext,
2249 "xmlNodeDump : node == NULL\n");
2250#endif
2251 return (-1);
2252 }
2253 if (buf == NULL) {
2254#ifdef DEBUG_TREE
2255 xmlGenericError(xmlGenericErrorContext,
2256 "xmlNodeDump : buf == NULL\n");
2257#endif
2258 return (-1);
2259 }
2260 outbuf = (xmlOutputBufferPtr) xmlMalloc(sizeof(xmlOutputBuffer));
2261 if (outbuf == NULL) {
2262 xmlSaveErrMemory("creating buffer");
2263 return (-1);
2264 }
2265 memset(outbuf, 0, (size_t) sizeof(xmlOutputBuffer));
2266 outbuf->buffer = buf;
2267 outbuf->encoder = NULL;
2268 outbuf->writecallback = NULL;
2269 outbuf->closecallback = NULL;
2270 outbuf->context = NULL;
2271 outbuf->written = 0;
2272
Daniel Veillard50cdab52012-07-16 14:52:00 +08002273 use = xmlBufUse(buf);
Daniel Veillard23922c52013-02-11 11:52:44 +08002274 oldalloc = xmlBufGetAllocationScheme(buf);
2275 xmlBufSetAllocationScheme(buf, XML_BUFFER_ALLOC_DOUBLEIT);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002276 xmlNodeDumpOutput(outbuf, doc, cur, level, format, NULL);
Daniel Veillard23922c52013-02-11 11:52:44 +08002277 xmlBufSetAllocationScheme(buf, oldalloc);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002278 xmlFree(outbuf);
Daniel Veillard50cdab52012-07-16 14:52:00 +08002279 ret = xmlBufUse(buf) - use;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002280 return (ret);
2281}
2282
2283/**
2284 * xmlElemDump:
2285 * @f: the FILE * for the output
2286 * @doc: the document
2287 * @cur: the current node
2288 *
2289 * Dump an XML/HTML node, recursive behaviour, children are printed too.
2290 */
2291void
2292xmlElemDump(FILE * f, xmlDocPtr doc, xmlNodePtr cur)
2293{
2294 xmlOutputBufferPtr outbuf;
2295
2296 xmlInitParser();
2297
2298 if (cur == NULL) {
2299#ifdef DEBUG_TREE
2300 xmlGenericError(xmlGenericErrorContext,
2301 "xmlElemDump : cur == NULL\n");
2302#endif
2303 return;
2304 }
2305#ifdef DEBUG_TREE
2306 if (doc == NULL) {
2307 xmlGenericError(xmlGenericErrorContext,
2308 "xmlElemDump : doc == NULL\n");
2309 }
2310#endif
2311
2312 outbuf = xmlOutputBufferCreateFile(f, NULL);
2313 if (outbuf == NULL)
2314 return;
2315 if ((doc != NULL) && (doc->type == XML_HTML_DOCUMENT_NODE)) {
2316#ifdef LIBXML_HTML_ENABLED
2317 htmlNodeDumpOutput(outbuf, doc, cur, NULL);
2318#else
2319 xmlSaveErr(XML_ERR_INTERNAL_ERROR, cur, "HTML support not compiled in\n");
2320#endif /* LIBXML_HTML_ENABLED */
2321 } else
2322 xmlNodeDumpOutput(outbuf, doc, cur, 0, 1, NULL);
2323 xmlOutputBufferClose(outbuf);
2324}
2325
2326/************************************************************************
2327 * *
2328 * Saving functions front-ends *
2329 * *
2330 ************************************************************************/
2331
2332/**
2333 * xmlNodeDumpOutput:
2334 * @buf: the XML buffer output
2335 * @doc: the document
2336 * @cur: the current node
2337 * @level: the imbrication level for indenting
2338 * @format: is formatting allowed
2339 * @encoding: an optional encoding string
2340 *
2341 * Dump an XML node, recursive behaviour, children are printed too.
2342 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2343 * or xmlKeepBlanksDefault(0) was called
2344 */
2345void
2346xmlNodeDumpOutput(xmlOutputBufferPtr buf, xmlDocPtr doc, xmlNodePtr cur,
2347 int level, int format, const char *encoding)
2348{
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002349 xmlSaveCtxt ctxt;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002350#ifdef LIBXML_HTML_ENABLED
2351 xmlDtdPtr dtd;
2352 int is_xhtml = 0;
2353#endif
2354
2355 xmlInitParser();
2356
Daniel Veillardce244ad2004-11-05 10:03:46 +00002357 if ((buf == NULL) || (cur == NULL)) return;
2358
Daniel Veillard64354ea2005-03-31 15:22:56 +00002359 if (encoding == NULL)
2360 encoding = "UTF-8";
2361
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002362 memset(&ctxt, 0, sizeof(ctxt));
2363 ctxt.doc = doc;
2364 ctxt.buf = buf;
2365 ctxt.level = level;
Adam Spragg8b877132010-11-01 14:24:56 +01002366 ctxt.format = format ? 1 : 0;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002367 ctxt.encoding = (const xmlChar *) encoding;
Daniel Veillard753086a2004-03-28 16:12:44 +00002368 xmlSaveCtxtInit(&ctxt);
Daniel Veillard856d9282008-09-25 14:31:40 +00002369 ctxt.options |= XML_SAVE_AS_XML;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002370
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002371#ifdef LIBXML_HTML_ENABLED
2372 dtd = xmlGetIntSubset(doc);
2373 if (dtd != NULL) {
Daniel Veillard33b20b72005-09-12 21:43:20 +00002374 is_xhtml = xmlIsXHTML(dtd->SystemID, dtd->ExternalID);
2375 if (is_xhtml < 0)
2376 is_xhtml = 0;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002377 }
2378
2379 if (is_xhtml)
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002380 xhtmlNodeDumpOutput(&ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002381 else
2382#endif
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002383 xmlNodeDumpOutputInternal(&ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002384}
2385
2386/**
2387 * xmlDocDumpFormatMemoryEnc:
2388 * @out_doc: Document to generate XML text from
2389 * @doc_txt_ptr: Memory pointer for allocated XML text
2390 * @doc_txt_len: Length of the generated XML text
2391 * @txt_encoding: Character encoding to use when generating XML text
2392 * @format: should formatting spaces been added
2393 *
2394 * Dump the current DOM tree into memory using the character encoding specified
2395 * by the caller. Note it is up to the caller of this function to free the
2396 * allocated memory with xmlFree().
2397 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2398 * or xmlKeepBlanksDefault(0) was called
2399 */
2400
2401void
2402xmlDocDumpFormatMemoryEnc(xmlDocPtr out_doc, xmlChar **doc_txt_ptr,
2403 int * doc_txt_len, const char * txt_encoding,
2404 int format) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002405 xmlSaveCtxt ctxt;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002406 int dummy = 0;
2407 xmlOutputBufferPtr out_buff = NULL;
2408 xmlCharEncodingHandlerPtr conv_hdlr = NULL;
2409
2410 if (doc_txt_len == NULL) {
2411 doc_txt_len = &dummy; /* Continue, caller just won't get length */
2412 }
2413
2414 if (doc_txt_ptr == NULL) {
2415 *doc_txt_len = 0;
2416 return;
2417 }
2418
2419 *doc_txt_ptr = NULL;
2420 *doc_txt_len = 0;
2421
2422 if (out_doc == NULL) {
2423 /* No document, no output */
2424 return;
2425 }
2426
2427 /*
2428 * Validate the encoding value, if provided.
2429 * This logic is copied from xmlSaveFileEnc.
2430 */
2431
2432 if (txt_encoding == NULL)
2433 txt_encoding = (const char *) out_doc->encoding;
2434 if (txt_encoding != NULL) {
2435 conv_hdlr = xmlFindCharEncodingHandler(txt_encoding);
2436 if ( conv_hdlr == NULL ) {
2437 xmlSaveErr(XML_SAVE_UNKNOWN_ENCODING, (xmlNodePtr) out_doc,
2438 txt_encoding);
2439 return;
2440 }
2441 }
2442
2443 if ((out_buff = xmlAllocOutputBuffer(conv_hdlr)) == NULL ) {
2444 xmlSaveErrMemory("creating buffer");
2445 return;
2446 }
2447
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002448 memset(&ctxt, 0, sizeof(ctxt));
2449 ctxt.doc = out_doc;
2450 ctxt.buf = out_buff;
2451 ctxt.level = 0;
Adam Spragg8b877132010-11-01 14:24:56 +01002452 ctxt.format = format ? 1 : 0;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002453 ctxt.encoding = (const xmlChar *) txt_encoding;
Daniel Veillard753086a2004-03-28 16:12:44 +00002454 xmlSaveCtxtInit(&ctxt);
Daniel Veillard856d9282008-09-25 14:31:40 +00002455 ctxt.options |= XML_SAVE_AS_XML;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002456 xmlDocContentDumpOutput(&ctxt, out_doc);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002457 xmlOutputBufferFlush(out_buff);
2458 if (out_buff->conv != NULL) {
Daniel Veillard50cdab52012-07-16 14:52:00 +08002459 *doc_txt_len = xmlBufUse(out_buff->conv);
2460 *doc_txt_ptr = xmlStrndup(xmlBufContent(out_buff->conv), *doc_txt_len);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002461 } else {
Daniel Veillard50cdab52012-07-16 14:52:00 +08002462 *doc_txt_len = xmlBufUse(out_buff->buffer);
2463 *doc_txt_ptr = xmlStrndup(xmlBufContent(out_buff->buffer),*doc_txt_len);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002464 }
2465 (void)xmlOutputBufferClose(out_buff);
2466
2467 if ((*doc_txt_ptr == NULL) && (*doc_txt_len > 0)) {
2468 *doc_txt_len = 0;
2469 xmlSaveErrMemory("creating output");
2470 }
2471
2472 return;
2473}
2474
2475/**
2476 * xmlDocDumpMemory:
2477 * @cur: the document
2478 * @mem: OUT: the memory pointer
2479 * @size: OUT: the memory length
2480 *
2481 * Dump an XML document in memory and return the #xmlChar * and it's size
2482 * in bytes. It's up to the caller to free the memory with xmlFree().
2483 * The resulting byte array is zero terminated, though the last 0 is not
2484 * included in the returned size.
2485 */
2486void
2487xmlDocDumpMemory(xmlDocPtr cur, xmlChar**mem, int *size) {
2488 xmlDocDumpFormatMemoryEnc(cur, mem, size, NULL, 0);
2489}
2490
2491/**
2492 * xmlDocDumpFormatMemory:
2493 * @cur: the document
2494 * @mem: OUT: the memory pointer
2495 * @size: OUT: the memory length
2496 * @format: should formatting spaces been added
2497 *
2498 *
2499 * Dump an XML document in memory and return the #xmlChar * and it's size.
2500 * It's up to the caller to free the memory with xmlFree().
2501 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2502 * or xmlKeepBlanksDefault(0) was called
2503 */
2504void
2505xmlDocDumpFormatMemory(xmlDocPtr cur, xmlChar**mem, int *size, int format) {
2506 xmlDocDumpFormatMemoryEnc(cur, mem, size, NULL, format);
2507}
2508
2509/**
2510 * xmlDocDumpMemoryEnc:
2511 * @out_doc: Document to generate XML text from
2512 * @doc_txt_ptr: Memory pointer for allocated XML text
2513 * @doc_txt_len: Length of the generated XML text
2514 * @txt_encoding: Character encoding to use when generating XML text
2515 *
2516 * Dump the current DOM tree into memory using the character encoding specified
2517 * by the caller. Note it is up to the caller of this function to free the
2518 * allocated memory with xmlFree().
2519 */
2520
2521void
2522xmlDocDumpMemoryEnc(xmlDocPtr out_doc, xmlChar **doc_txt_ptr,
2523 int * doc_txt_len, const char * txt_encoding) {
2524 xmlDocDumpFormatMemoryEnc(out_doc, doc_txt_ptr, doc_txt_len,
2525 txt_encoding, 0);
2526}
2527
2528/**
2529 * xmlDocFormatDump:
2530 * @f: the FILE*
2531 * @cur: the document
2532 * @format: should formatting spaces been added
2533 *
2534 * Dump an XML document to an open FILE.
2535 *
2536 * returns: the number of bytes written or -1 in case of failure.
2537 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2538 * or xmlKeepBlanksDefault(0) was called
2539 */
2540int
2541xmlDocFormatDump(FILE *f, xmlDocPtr cur, int format) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002542 xmlSaveCtxt ctxt;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002543 xmlOutputBufferPtr buf;
2544 const char * encoding;
2545 xmlCharEncodingHandlerPtr handler = NULL;
2546 int ret;
2547
2548 if (cur == NULL) {
2549#ifdef DEBUG_TREE
2550 xmlGenericError(xmlGenericErrorContext,
2551 "xmlDocDump : document == NULL\n");
2552#endif
2553 return(-1);
2554 }
2555 encoding = (const char *) cur->encoding;
2556
2557 if (encoding != NULL) {
Daniel Veillard3814a362007-07-26 11:41:46 +00002558 handler = xmlFindCharEncodingHandler(encoding);
2559 if (handler == NULL) {
2560 xmlFree((char *) cur->encoding);
2561 cur->encoding = NULL;
2562 encoding = NULL;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002563 }
Daniel Veillard3814a362007-07-26 11:41:46 +00002564 }
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002565 buf = xmlOutputBufferCreateFile(f, handler);
2566 if (buf == NULL) return(-1);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002567 memset(&ctxt, 0, sizeof(ctxt));
2568 ctxt.doc = cur;
2569 ctxt.buf = buf;
2570 ctxt.level = 0;
Adam Spragg8b877132010-11-01 14:24:56 +01002571 ctxt.format = format ? 1 : 0;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002572 ctxt.encoding = (const xmlChar *) encoding;
Daniel Veillard753086a2004-03-28 16:12:44 +00002573 xmlSaveCtxtInit(&ctxt);
Daniel Veillard856d9282008-09-25 14:31:40 +00002574 ctxt.options |= XML_SAVE_AS_XML;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002575 xmlDocContentDumpOutput(&ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002576
2577 ret = xmlOutputBufferClose(buf);
2578 return(ret);
2579}
2580
2581/**
2582 * xmlDocDump:
2583 * @f: the FILE*
2584 * @cur: the document
2585 *
2586 * Dump an XML document to an open FILE.
2587 *
2588 * returns: the number of bytes written or -1 in case of failure.
2589 */
2590int
2591xmlDocDump(FILE *f, xmlDocPtr cur) {
2592 return(xmlDocFormatDump (f, cur, 0));
2593}
2594
2595/**
2596 * xmlSaveFileTo:
2597 * @buf: an output I/O buffer
2598 * @cur: the document
2599 * @encoding: the encoding if any assuming the I/O layer handles the trancoding
2600 *
2601 * Dump an XML document to an I/O buffer.
Daniel Veillard3d97e662004-11-04 10:49:00 +00002602 * Warning ! This call xmlOutputBufferClose() on buf which is not available
2603 * after this call.
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002604 *
2605 * returns: the number of bytes written or -1 in case of failure.
2606 */
2607int
2608xmlSaveFileTo(xmlOutputBufferPtr buf, xmlDocPtr cur, const char *encoding) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002609 xmlSaveCtxt ctxt;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002610 int ret;
2611
Daniel Veillard3d97e662004-11-04 10:49:00 +00002612 if (buf == NULL) return(-1);
2613 if (cur == NULL) {
2614 xmlOutputBufferClose(buf);
2615 return(-1);
2616 }
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002617 memset(&ctxt, 0, sizeof(ctxt));
2618 ctxt.doc = cur;
2619 ctxt.buf = buf;
2620 ctxt.level = 0;
2621 ctxt.format = 0;
2622 ctxt.encoding = (const xmlChar *) encoding;
Daniel Veillard753086a2004-03-28 16:12:44 +00002623 xmlSaveCtxtInit(&ctxt);
Daniel Veillard856d9282008-09-25 14:31:40 +00002624 ctxt.options |= XML_SAVE_AS_XML;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002625 xmlDocContentDumpOutput(&ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002626 ret = xmlOutputBufferClose(buf);
2627 return(ret);
2628}
2629
2630/**
2631 * xmlSaveFormatFileTo:
2632 * @buf: an output I/O buffer
2633 * @cur: the document
2634 * @encoding: the encoding if any assuming the I/O layer handles the trancoding
2635 * @format: should formatting spaces been added
2636 *
2637 * Dump an XML document to an I/O buffer.
Daniel Veillard3d97e662004-11-04 10:49:00 +00002638 * Warning ! This call xmlOutputBufferClose() on buf which is not available
2639 * after this call.
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002640 *
2641 * returns: the number of bytes written or -1 in case of failure.
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002642 */
2643int
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002644xmlSaveFormatFileTo(xmlOutputBufferPtr buf, xmlDocPtr cur,
2645 const char *encoding, int format)
2646{
2647 xmlSaveCtxt ctxt;
2648 int ret;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002649
Daniel Veillard3d97e662004-11-04 10:49:00 +00002650 if (buf == NULL) return(-1);
Daniel Veillardce244ad2004-11-05 10:03:46 +00002651 if ((cur == NULL) ||
2652 ((cur->type != XML_DOCUMENT_NODE) &&
2653 (cur->type != XML_HTML_DOCUMENT_NODE))) {
Daniel Veillard3d97e662004-11-04 10:49:00 +00002654 xmlOutputBufferClose(buf);
2655 return(-1);
2656 }
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002657 memset(&ctxt, 0, sizeof(ctxt));
2658 ctxt.doc = cur;
2659 ctxt.buf = buf;
2660 ctxt.level = 0;
Adam Spragg8b877132010-11-01 14:24:56 +01002661 ctxt.format = format ? 1 : 0;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002662 ctxt.encoding = (const xmlChar *) encoding;
Daniel Veillard753086a2004-03-28 16:12:44 +00002663 xmlSaveCtxtInit(&ctxt);
Daniel Veillard856d9282008-09-25 14:31:40 +00002664 ctxt.options |= XML_SAVE_AS_XML;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002665 xmlDocContentDumpOutput(&ctxt, cur);
2666 ret = xmlOutputBufferClose(buf);
2667 return (ret);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002668}
2669
2670/**
2671 * xmlSaveFormatFileEnc:
2672 * @filename: the filename or URL to output
2673 * @cur: the document being saved
2674 * @encoding: the name of the encoding to use or NULL.
2675 * @format: should formatting spaces be added.
2676 *
2677 * Dump an XML document to a file or an URL.
2678 *
2679 * Returns the number of bytes written or -1 in case of error.
2680 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2681 * or xmlKeepBlanksDefault(0) was called
2682 */
2683int
2684xmlSaveFormatFileEnc( const char * filename, xmlDocPtr cur,
2685 const char * encoding, int format ) {
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002686 xmlSaveCtxt ctxt;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002687 xmlOutputBufferPtr buf;
2688 xmlCharEncodingHandlerPtr handler = NULL;
2689 int ret;
2690
2691 if (cur == NULL)
2692 return(-1);
2693
2694 if (encoding == NULL)
2695 encoding = (const char *) cur->encoding;
2696
2697 if (encoding != NULL) {
2698
2699 handler = xmlFindCharEncodingHandler(encoding);
2700 if (handler == NULL)
2701 return(-1);
2702 }
2703
Nick Wellnhofercb5541c2017-11-13 17:08:38 +01002704#ifdef LIBXML_ZLIB_ENABLED
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002705 if (cur->compression < 0) cur->compression = xmlGetCompressMode();
2706#endif
Daniel Veillardf8e3db02012-09-11 13:26:36 +08002707 /*
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002708 * save the content to a temp buffer.
2709 */
2710 buf = xmlOutputBufferCreateFilename(filename, handler, cur->compression);
2711 if (buf == NULL) return(-1);
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002712 memset(&ctxt, 0, sizeof(ctxt));
2713 ctxt.doc = cur;
2714 ctxt.buf = buf;
2715 ctxt.level = 0;
Adam Spragg8b877132010-11-01 14:24:56 +01002716 ctxt.format = format ? 1 : 0;
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002717 ctxt.encoding = (const xmlChar *) encoding;
Daniel Veillard753086a2004-03-28 16:12:44 +00002718 xmlSaveCtxtInit(&ctxt);
Daniel Veillard856d9282008-09-25 14:31:40 +00002719 ctxt.options |= XML_SAVE_AS_XML;
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002720
Daniel Veillard32b7cdb2004-03-15 13:46:37 +00002721 xmlDocContentDumpOutput(&ctxt, cur);
Daniel Veillard1a8741c2004-03-04 13:40:59 +00002722
2723 ret = xmlOutputBufferClose(buf);
2724 return(ret);
2725}
2726
2727
2728/**
2729 * xmlSaveFileEnc:
2730 * @filename: the filename (or URL)
2731 * @cur: the document
2732 * @encoding: the name of an encoding (or NULL)
2733 *
2734 * Dump an XML document, converting it to the given encoding
2735 *
2736 * returns: the number of bytes written or -1 in case of failure.
2737 */
2738int
2739xmlSaveFileEnc(const char *filename, xmlDocPtr cur, const char *encoding) {
2740 return ( xmlSaveFormatFileEnc( filename, cur, encoding, 0 ) );
2741}
2742
2743/**
2744 * xmlSaveFormatFile:
2745 * @filename: the filename (or URL)
2746 * @cur: the document
2747 * @format: should formatting spaces been added
2748 *
2749 * Dump an XML document to a file. Will use compression if
2750 * compiled in and enabled. If @filename is "-" the stdout file is
2751 * used. If @format is set then the document will be indented on output.
2752 * Note that @format = 1 provide node indenting only if xmlIndentTreeOutput = 1
2753 * or xmlKeepBlanksDefault(0) was called
2754 *
2755 * returns: the number of bytes written or -1 in case of failure.
2756 */
2757int
2758xmlSaveFormatFile(const char *filename, xmlDocPtr cur, int format) {
2759 return ( xmlSaveFormatFileEnc( filename, cur, NULL, format ) );
2760}
2761
2762/**
2763 * xmlSaveFile:
2764 * @filename: the filename (or URL)
2765 * @cur: the document
2766 *
2767 * Dump an XML document to a file. Will use compression if
2768 * compiled in and enabled. If @filename is "-" the stdout file is
2769 * used.
2770 * returns: the number of bytes written or -1 in case of failure.
2771 */
2772int
2773xmlSaveFile(const char *filename, xmlDocPtr cur) {
2774 return(xmlSaveFormatFileEnc(filename, cur, NULL, 0));
2775}
2776
2777#endif /* LIBXML_OUTPUT_ENABLED */
2778
Daniel Veillard5d4644e2005-04-01 13:11:58 +00002779#define bottom_xmlsave
2780#include "elfgcchack.h"