2002-03-04 17:09:44 +00:00
/*
* Canonical XML implementation test program
* ( http : //www.w3.org/TR/2001/REC-xml-c14n-20010315)
*
* See Copyright for the status of this software .
2012-09-11 13:26:36 +08:00
*
2002-03-04 17:09:44 +00:00
* Author : Aleksey Sanin < aleksey @ aleksey . com >
*/
# include "libxml.h"
2003-09-29 13:20:24 +00:00
# if defined(LIBXML_C14N_ENABLED) && defined(LIBXML_OUTPUT_ENABLED)
2002-03-04 17:09:44 +00:00
# include <stdio.h>
# include <string.h>
2022-03-02 00:29:17 +01:00
# include <stdlib.h>
2012-08-15 13:52:09 +08:00
# ifndef STDOUT_FILENO
2002-03-04 17:09:44 +00:00
# ifdef HAVE_UNISTD_H
# include <unistd.h>
2012-01-02 15:49:31 +08:00
# else
# define STDOUT_FILENO fileno(stdout)
# endif /* HAVE_UNISTD_H */
2012-08-15 13:52:09 +08:00
# endif
2018-01-23 17:36:23 +01:00
# ifdef _WIN32
# include <io.h>
# endif
2002-03-04 17:09:44 +00:00
# include <libxml/xmlmemory.h>
# include <libxml/parser.h>
# include <libxml/xpath.h>
# include <libxml/xpathInternals.h>
# include <libxml/c14n.h>
static void usage ( const char * name ) {
fprintf ( stderr ,
" Usage: %s <mode> <xml-file> [<xpath-expr>] [<inclusive-ns-list>] \n " ,
name ) ;
fprintf ( stderr , " where <mode> is one of following: \n " ) ;
fprintf ( stderr ,
2009-07-09 10:26:22 +02:00
" --with-comments \t XML file canonicalization v1.0 w comments \n " ) ;
2002-03-04 17:09:44 +00:00
fprintf ( stderr ,
2009-07-09 10:26:22 +02:00
" --without-comments \t XML file canonicalization v1.0 w/o comments \n " ) ;
2002-03-04 17:09:44 +00:00
fprintf ( stderr ,
2009-07-09 10:26:22 +02:00
" --1-1-with-comments \t XML file canonicalization v1.1 w comments \n " ) ;
2002-03-04 17:09:44 +00:00
fprintf ( stderr ,
2009-07-09 10:26:22 +02:00
" --1-1-without-comments \t XML file canonicalization v1.1 w/o comments \n " ) ;
fprintf ( stderr ,
" --exc-with-comments \t Exclusive XML file canonicalization v1.0 w comments \n " ) ;
fprintf ( stderr ,
" --exc-without-comments \t Exclusive XML file canonicalization v1.0 w/o comments \n " ) ;
2002-03-04 17:09:44 +00:00
}
2002-12-22 10:25:41 +00:00
static xmlXPathObjectPtr
2002-03-04 17:09:44 +00:00
load_xpath_expr ( xmlDocPtr parent_doc , const char * filename ) ;
2002-12-22 10:25:41 +00:00
static xmlChar * * parse_list ( xmlChar * str ) ;
2002-03-04 17:09:44 +00:00
2003-08-05 15:52:22 +00:00
/* static void print_xpath_nodes(xmlNodeSetPtr nodes); */
2002-03-04 17:09:44 +00:00
2012-09-11 13:26:36 +08:00
static int
2009-07-09 10:26:22 +02:00
test_c14n ( const char * xml_filename , int with_comments , int mode ,
2002-03-04 17:09:44 +00:00
const char * xpath_filename , xmlChar * * inclusive_namespaces ) {
xmlDocPtr doc ;
2012-09-11 13:26:36 +08:00
xmlXPathObjectPtr xpath = NULL ;
2002-03-04 17:09:44 +00:00
xmlChar * result = NULL ;
int ret ;
/*
* build an XML tree from a the file ; we need to add default
* attributes and resolve all character and entities references
*/
xmlLoadExtDtdDefaultValue = XML_DETECT_IDS | XML_COMPLETE_ATTRS ;
xmlSubstituteEntitiesDefault ( 1 ) ;
2005-08-23 22:14:02 +00:00
doc = xmlReadFile ( xml_filename , NULL , XML_PARSE_DTDATTR | XML_PARSE_NOENT ) ;
2002-03-04 17:09:44 +00:00
if ( doc = = NULL ) {
fprintf ( stderr , " Error: unable to parse file \" %s \" \n " , xml_filename ) ;
return ( - 1 ) ;
}
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/*
* Check the document is of the right kind
2012-09-11 13:26:36 +08:00
*/
2002-03-04 17:09:44 +00:00
if ( xmlDocGetRootElement ( doc ) = = NULL ) {
fprintf ( stderr , " Error: empty document for file \" %s \" \n " , xml_filename ) ;
xmlFreeDoc ( doc ) ;
return ( - 1 ) ;
}
2012-09-11 13:26:36 +08:00
/*
* load xpath file if specified
2002-03-04 17:09:44 +00:00
*/
if ( xpath_filename ) {
xpath = load_xpath_expr ( doc , xpath_filename ) ;
if ( xpath = = NULL ) {
fprintf ( stderr , " Error: unable to evaluate xpath expression \n " ) ;
2012-09-11 13:26:36 +08:00
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( - 1 ) ;
}
}
/*
* Canonical form
2012-09-11 13:26:36 +08:00
*/
2002-03-04 17:09:44 +00:00
/* fprintf(stderr,"File \"%s\" loaded: start canonization\n", xml_filename); */
2012-09-11 13:26:36 +08:00
ret = xmlC14NDocDumpMemory ( doc ,
( xpath ) ? xpath - > nodesetval : NULL ,
2009-07-09 10:26:22 +02:00
mode , inclusive_namespaces ,
2002-03-04 17:09:44 +00:00
with_comments , & result ) ;
if ( ret > = 0 ) {
if ( result ! = NULL ) {
2011-05-09 12:14:59 +03:00
if ( write ( STDOUT_FILENO , result , ret ) = = - 1 ) {
fprintf ( stderr , " Can't write data \n " ) ;
}
xmlFree ( result ) ;
2002-03-04 17:09:44 +00:00
}
} else {
fprintf ( stderr , " Error: failed to canonicalize XML file \" %s \" (ret=%d) \n " , xml_filename , ret ) ;
if ( result ! = NULL ) xmlFree ( result ) ;
2012-09-11 13:26:36 +08:00
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( - 1 ) ;
}
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/*
* Cleanup
2012-09-11 13:26:36 +08:00
*/
2002-03-04 17:09:44 +00:00
if ( xpath ! = NULL ) xmlXPathFreeObject ( xpath ) ;
2012-09-11 13:26:36 +08:00
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( ret ) ;
}
int main ( int argc , char * * argv ) {
int ret = - 1 ;
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/*
* Init libxml
2012-09-11 13:26:36 +08:00
*/
2002-03-04 17:09:44 +00:00
xmlInitParser ( ) ;
LIBXML_TEST_VERSION
/*
* Parse command line and process file
*/
if ( argc < 3 ) {
fprintf ( stderr , " Error: wrong number of arguments. \n " ) ;
usage ( argv [ 0 ] ) ;
} else if ( strcmp ( argv [ 1 ] , " --with-comments " ) = = 0 ) {
2009-07-09 10:26:22 +02:00
ret = test_c14n ( argv [ 2 ] , 1 , XML_C14N_1_0 , ( argc > 3 ) ? argv [ 3 ] : NULL , NULL ) ;
2002-03-04 17:09:44 +00:00
} else if ( strcmp ( argv [ 1 ] , " --without-comments " ) = = 0 ) {
2009-07-09 10:26:22 +02:00
ret = test_c14n ( argv [ 2 ] , 0 , XML_C14N_1_0 , ( argc > 3 ) ? argv [ 3 ] : NULL , NULL ) ;
} else if ( strcmp ( argv [ 1 ] , " --1-1-with-comments " ) = = 0 ) {
ret = test_c14n ( argv [ 2 ] , 1 , XML_C14N_1_1 , ( argc > 3 ) ? argv [ 3 ] : NULL , NULL ) ;
} else if ( strcmp ( argv [ 1 ] , " --1-1-without-comments " ) = = 0 ) {
ret = test_c14n ( argv [ 2 ] , 0 , XML_C14N_1_1 , ( argc > 3 ) ? argv [ 3 ] : NULL , NULL ) ;
2002-03-04 17:09:44 +00:00
} else if ( strcmp ( argv [ 1 ] , " --exc-with-comments " ) = = 0 ) {
xmlChar * * list ;
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/* load exclusive namespace from command line */
list = ( argc > 4 ) ? parse_list ( ( xmlChar * ) argv [ 4 ] ) : NULL ;
2009-07-09 10:26:22 +02:00
ret = test_c14n ( argv [ 2 ] , 1 , XML_C14N_EXCLUSIVE_1_0 , ( argc > 3 ) ? argv [ 3 ] : NULL , list ) ;
2002-03-04 17:09:44 +00:00
if ( list ! = NULL ) xmlFree ( list ) ;
} else if ( strcmp ( argv [ 1 ] , " --exc-without-comments " ) = = 0 ) {
xmlChar * * list ;
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/* load exclusive namespace from command line */
list = ( argc > 4 ) ? parse_list ( ( xmlChar * ) argv [ 4 ] ) : NULL ;
2009-07-09 10:26:22 +02:00
ret = test_c14n ( argv [ 2 ] , 0 , XML_C14N_EXCLUSIVE_1_0 , ( argc > 3 ) ? argv [ 3 ] : NULL , list ) ;
2002-03-04 17:09:44 +00:00
if ( list ! = NULL ) xmlFree ( list ) ;
} else {
fprintf ( stderr , " Error: bad option. \n " ) ;
usage ( argv [ 0 ] ) ;
}
2012-09-11 13:26:36 +08:00
/*
2002-03-04 17:09:44 +00:00
* Shutdown libxml
*/
xmlCleanupParser ( ) ;
xmlMemoryDump ( ) ;
2009-09-05 14:52:55 +02:00
2002-03-04 17:09:44 +00:00
return ( ( ret > = 0 ) ? 0 : 1 ) ;
}
/*
* Macro used to grow the current buffer .
*/
# define growBufferReentrant() { \
buffer_size * = 2 ; \
buffer = ( xmlChar * * ) \
2009-09-05 14:52:55 +02:00
xmlRealloc ( buffer , buffer_size * sizeof ( xmlChar * ) ) ; \
2002-03-04 17:09:44 +00:00
if ( buffer = = NULL ) { \
perror ( " realloc failed " ) ; \
return ( NULL ) ; \
} \
}
2002-12-10 15:19:08 +00:00
static xmlChar * *
parse_list ( xmlChar * str ) {
2002-03-04 17:09:44 +00:00
xmlChar * * buffer ;
xmlChar * * out = NULL ;
int buffer_size = 0 ;
2002-06-04 04:27:06 +00:00
int len ;
2002-03-04 17:09:44 +00:00
if ( str = = NULL ) {
return ( NULL ) ;
}
2002-09-24 14:13:13 +00:00
len = xmlStrlen ( str ) ;
2002-06-04 04:27:06 +00:00
if ( ( str [ 0 ] = = ' \' ' ) & & ( str [ len - 1 ] = = ' \' ' ) ) {
str [ len - 1 ] = ' \0 ' ;
str + + ;
}
2002-03-04 17:09:44 +00:00
/*
* allocate an translation buffer .
*/
buffer_size = 1000 ;
buffer = ( xmlChar * * ) xmlMalloc ( buffer_size * sizeof ( xmlChar * ) ) ;
if ( buffer = = NULL ) {
perror ( " malloc failed " ) ;
return ( NULL ) ;
}
out = buffer ;
2009-09-05 14:52:55 +02:00
2002-03-04 17:09:44 +00:00
while ( * str ! = ' \0 ' ) {
if ( out - buffer > buffer_size - 10 ) {
int indx = out - buffer ;
growBufferReentrant ( ) ;
out = & buffer [ indx ] ;
}
( * out + + ) = str ;
while ( * str ! = ' , ' & & * str ! = ' \0 ' ) + + str ;
if ( * str = = ' , ' ) * ( str + + ) = ' \0 ' ;
}
( * out ) = NULL ;
return buffer ;
}
2002-12-10 15:19:08 +00:00
static xmlXPathObjectPtr
2002-03-04 17:09:44 +00:00
load_xpath_expr ( xmlDocPtr parent_doc , const char * filename ) {
2012-09-11 13:26:36 +08:00
xmlXPathObjectPtr xpath ;
2002-03-04 17:09:44 +00:00
xmlDocPtr doc ;
xmlChar * expr ;
2012-09-11 13:26:36 +08:00
xmlXPathContextPtr ctx ;
2002-03-04 17:09:44 +00:00
xmlNodePtr node ;
xmlNsPtr ns ;
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/*
* load XPath expr as a file
*/
xmlLoadExtDtdDefaultValue = XML_DETECT_IDS | XML_COMPLETE_ATTRS ;
xmlSubstituteEntitiesDefault ( 1 ) ;
2005-08-23 22:14:02 +00:00
doc = xmlReadFile ( filename , NULL , XML_PARSE_DTDATTR | XML_PARSE_NOENT ) ;
2002-03-04 17:09:44 +00:00
if ( doc = = NULL ) {
fprintf ( stderr , " Error: unable to parse file \" %s \" \n " , filename ) ;
return ( NULL ) ;
}
2012-09-11 13:26:36 +08:00
2002-03-04 17:09:44 +00:00
/*
* Check the document is of the right kind
2012-09-11 13:26:36 +08:00
*/
2002-03-04 17:09:44 +00:00
if ( xmlDocGetRootElement ( doc ) = = NULL ) {
fprintf ( stderr , " Error: empty document for file \" %s \" \n " , filename ) ;
xmlFreeDoc ( doc ) ;
return ( NULL ) ;
}
node = doc - > children ;
while ( node ! = NULL & & ! xmlStrEqual ( node - > name , ( const xmlChar * ) " XPath " ) ) {
node = node - > next ;
}
2012-09-11 13:26:36 +08:00
if ( node = = NULL ) {
2002-03-04 17:09:44 +00:00
fprintf ( stderr , " Error: XPath element expected in the file \" %s \" \n " , filename ) ;
xmlFreeDoc ( doc ) ;
return ( NULL ) ;
}
expr = xmlNodeGetContent ( node ) ;
if ( expr = = NULL ) {
fprintf ( stderr , " Error: XPath content element is NULL \" %s \" \n " , filename ) ;
xmlFreeDoc ( doc ) ;
return ( NULL ) ;
}
ctx = xmlXPathNewContext ( parent_doc ) ;
if ( ctx = = NULL ) {
fprintf ( stderr , " Error: unable to create new context \n " ) ;
2012-09-11 13:26:36 +08:00
xmlFree ( expr ) ;
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( NULL ) ;
}
/*
* Register namespaces
*/
ns = node - > nsDef ;
while ( ns ! = NULL ) {
if ( xmlXPathRegisterNs ( ctx , ns - > prefix , ns - > href ) ! = 0 ) {
fprintf ( stderr , " Error: unable to register NS with prefix= \" %s \" and href= \" %s \" \n " , ns - > prefix , ns - > href ) ;
2012-09-11 13:26:36 +08:00
xmlFree ( expr ) ;
xmlXPathFreeContext ( ctx ) ;
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( NULL ) ;
}
ns = ns - > next ;
}
2012-09-11 13:26:36 +08:00
/*
2002-03-04 17:09:44 +00:00
* Evaluate xpath
*/
xpath = xmlXPathEvalExpression ( expr , ctx ) ;
if ( xpath = = NULL ) {
fprintf ( stderr , " Error: unable to evaluate xpath expression \n " ) ;
2012-09-11 13:26:36 +08:00
xmlFree ( expr ) ;
xmlXPathFreeContext ( ctx ) ;
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( NULL ) ;
}
2002-06-04 04:27:06 +00:00
/* print_xpath_nodes(xpath->nodesetval); */
2002-03-04 17:09:44 +00:00
2012-09-11 13:26:36 +08:00
xmlFree ( expr ) ;
xmlXPathFreeContext ( ctx ) ;
xmlFreeDoc ( doc ) ;
2002-03-04 17:09:44 +00:00
return ( xpath ) ;
}
2003-08-05 15:52:22 +00:00
/*
2002-12-10 15:19:08 +00:00
static void
2002-06-04 04:27:06 +00:00
print_xpath_nodes ( xmlNodeSetPtr nodes ) {
2002-03-04 17:09:44 +00:00
xmlNodePtr cur ;
int i ;
2012-09-11 13:26:36 +08:00
if ( nodes = = NULL ) {
2002-03-04 17:09:44 +00:00
fprintf ( stderr , " Error: no nodes set defined \n " ) ;
return ;
}
2012-09-11 13:26:36 +08:00
2002-06-04 04:27:06 +00:00
fprintf ( stderr , " Nodes Set: \n ----- \n " ) ;
for ( i = 0 ; i < nodes - > nodeNr ; + + i ) {
if ( nodes - > nodeTab [ i ] - > type = = XML_NAMESPACE_DECL ) {
xmlNsPtr ns ;
2012-09-11 13:26:36 +08:00
2002-06-04 04:27:06 +00:00
ns = ( xmlNsPtr ) nodes - > nodeTab [ i ] ;
cur = ( xmlNodePtr ) ns - > next ;
2012-09-11 13:26:36 +08:00
fprintf ( stderr , " namespace \" %s \" = \" %s \" for node %s:%s \n " ,
2002-06-04 04:27:06 +00:00
ns - > prefix , ns - > href ,
( cur - > ns ) ? cur - > ns - > prefix : BAD_CAST " " , cur - > name ) ;
} else if ( nodes - > nodeTab [ i ] - > type = = XML_ELEMENT_NODE ) {
2012-09-11 13:26:36 +08:00
cur = nodes - > nodeTab [ i ] ;
fprintf ( stderr , " element node \" %s:%s \" \n " ,
2002-06-04 04:27:06 +00:00
( cur - > ns ) ? cur - > ns - > prefix : BAD_CAST " " , cur - > name ) ;
} else {
2012-09-11 13:26:36 +08:00
cur = nodes - > nodeTab [ i ] ;
2002-06-04 04:27:06 +00:00
fprintf ( stderr , " node \" %s \" : type %d \n " , cur - > name , cur - > type ) ;
}
2002-03-04 17:09:44 +00:00
}
}
2003-08-05 15:52:22 +00:00
*/
2002-06-04 04:27:06 +00:00
2002-03-04 17:09:44 +00:00
# else
# include <stdio.h>
2003-09-29 13:20:24 +00:00
int main ( int argc ATTRIBUTE_UNUSED , char * * argv ATTRIBUTE_UNUSED ) {
printf ( " %s : XPath/Canonicalization and output support not compiled in \n " , argv [ 0 ] ) ;
2002-03-04 17:09:44 +00:00
return ( 0 ) ;
}
# endif /* LIBXML_C14N_ENABLED */