mirror of
https://gitlab.gnome.org/GNOME/libxml2.git
synced 2025-10-23 01:52:48 +03:00
* doc/examples/xpath1.c: added a test template * doc/examples/xpath2.c doc/examples/xpath2.res doc/examples/*: added a new example, and make valgrind target Daniel
171 lines
4.2 KiB
C
171 lines
4.2 KiB
C
/**
|
|
* section: XPath
|
|
* synopsis: Load a document, locate subelements with XPath, modify
|
|
* said elements and save the resulting document.
|
|
* purpose: Shows how to make a full round-trip from a load/edit/save
|
|
* usage: xpath2 <xml-file> <xpath-expr> <new-value>
|
|
* test: xpath2 test3.xml '//child2' child2 > xpath2.tmp ; diff xpath2.tmp xpath2.res ; rm xpath2.tmp
|
|
* author: Aleksey Sanin and Daniel Veillard
|
|
* copy: see Copyright for the status of this software.
|
|
*/
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <assert.h>
|
|
|
|
#include <libxml/tree.h>
|
|
#include <libxml/parser.h>
|
|
#include <libxml/xpath.h>
|
|
#include <libxml/xpathInternals.h>
|
|
|
|
#ifdef LIBXML_XPATH_ENABLED
|
|
|
|
|
|
static void usage(const char *name);
|
|
static int example4(const char *filename, const xmlChar * xpathExpr,
|
|
const xmlChar * value);
|
|
static void update_xpath_nodes(xmlNodeSetPtr nodes, const xmlChar * value);
|
|
|
|
|
|
int
|
|
main(int argc, char **argv) {
|
|
/* Parse command line and process file */
|
|
if (argc != 4) {
|
|
fprintf(stderr, "Error: wrong number of arguments.\n");
|
|
usage(argv[0]);
|
|
return(-1);
|
|
}
|
|
|
|
/* Init libxml */
|
|
xmlInitParser();
|
|
LIBXML_TEST_VERSION
|
|
|
|
/* Do the main job */
|
|
if (example4(argv[1], BAD_CAST argv[2], BAD_CAST argv[3])) {
|
|
usage(argv[0]);
|
|
return(-1);
|
|
}
|
|
|
|
/* Shutdown libxml */
|
|
xmlCleanupParser();
|
|
|
|
/*
|
|
* this is to debug memory for regression tests
|
|
*/
|
|
xmlMemoryDump();
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* usage:
|
|
* @name: the program name.
|
|
*
|
|
* Prints usage information.
|
|
*/
|
|
void
|
|
usage(const char *name) {
|
|
assert(name);
|
|
|
|
fprintf(stderr, "Usage: %s <xml-file> <xpath-expr> <value>\n", name);
|
|
}
|
|
|
|
/**
|
|
* example4:
|
|
* @filename: the input XML filename.
|
|
* @xpathExpr: the xpath expression for evaluation.
|
|
* @value: the new node content.
|
|
*
|
|
* Parses input XML file, evaluates XPath expression and update the nodes
|
|
* then print the result.
|
|
*
|
|
* Returns 0 on success and a negative value otherwise.
|
|
*/
|
|
static int
|
|
example4(const char* filename, const xmlChar* xpathExpr, const xmlChar* value) {
|
|
xmlDocPtr doc;
|
|
xmlXPathContextPtr xpathCtx;
|
|
xmlXPathObjectPtr xpathObj;
|
|
|
|
assert(filename);
|
|
assert(xpathExpr);
|
|
assert(value);
|
|
|
|
/* Load XML document */
|
|
doc = xmlParseFile(filename);
|
|
if (doc == NULL) {
|
|
fprintf(stderr, "Error: unable to parse file \"%s\"\n", filename);
|
|
return(-1);
|
|
}
|
|
|
|
/* Create xpath evaluation context */
|
|
xpathCtx = xmlXPathNewContext(doc);
|
|
if(xpathCtx == NULL) {
|
|
fprintf(stderr,"Error: unable to create new XPath context\n");
|
|
xmlFreeDoc(doc);
|
|
return(-1);
|
|
}
|
|
|
|
/* Evaluate xpath expression */
|
|
xpathObj = xmlXPathEvalExpression(xpathExpr, xpathCtx);
|
|
if(xpathObj == NULL) {
|
|
fprintf(stderr,"Error: unable to evaluate xpath expression \"%s\"\n", xpathExpr);
|
|
xmlXPathFreeContext(xpathCtx);
|
|
xmlFreeDoc(doc);
|
|
return(-1);
|
|
}
|
|
|
|
/* update selected nodes */
|
|
update_xpath_nodes(xpathObj->nodesetval, value);
|
|
|
|
|
|
/* Cleanup of XPath data */
|
|
xmlXPathFreeObject(xpathObj);
|
|
xmlXPathFreeContext(xpathCtx);
|
|
|
|
/* dump the resulting document */
|
|
xmlDocDump(stdout, doc);
|
|
|
|
|
|
/* free the document */
|
|
xmlFreeDoc(doc);
|
|
|
|
return(0);
|
|
}
|
|
|
|
/**
|
|
* update_xpath_nodes:
|
|
* @nodes: the nodes set.
|
|
* @value: the new value for the node(s)
|
|
*
|
|
* Prints the @nodes content to @output.
|
|
*/
|
|
static void
|
|
update_xpath_nodes(xmlNodeSetPtr nodes, const xmlChar* value) {
|
|
int size;
|
|
int i;
|
|
|
|
assert(value);
|
|
size = (nodes) ? nodes->nodeNr : 0;
|
|
|
|
/*
|
|
* NOTE: the nodes are processed in reverse order, i.e. reverse document
|
|
* order because xmlNodeSetContent can actually free up descendant
|
|
* of the node and such nodes may have been selected too ! Handling
|
|
* in reverse order ensure that descendant are accessed first, before
|
|
* they get removed. Mixing XPath and modifications on a tree must be
|
|
* done carefully !
|
|
*/
|
|
for(i = size - 1; i >= 0; i--) {
|
|
assert(nodes->nodeTab[i]);
|
|
|
|
xmlNodeSetContent(nodes->nodeTab[i], value);
|
|
}
|
|
}
|
|
|
|
#else
|
|
int main(void) {
|
|
fprintf(stderr, "XPath support not compiled in\n");
|
|
exit(1);
|
|
}
|
|
#endif
|