#include #include #include #include #include "xml.h" static const char *_static_root = "/"; static const char *_static_element = "*"; static unsigned int _fcount = 0; static char **_filenames = 0; static char *_element = 0; static char *_value = 0; static char *_root = 0; static char *_print = 0; static int print_filenames = 0; static void free_and_exit(int i); #define SHOW_NOVAL(opt) \ { \ printf("option '%s' requires a value\n\n", (opt)); \ free_and_exit(-1); \ } void show_help () { printf("usage: xmlgrep [options] [file ...]\n\n"); printf("Options:\n"); printf("\t-h\t\tshow this help message\n"); printf("\t-e \t\tshow sections that contain this element\n"); printf("\t-p \t\tprint this element as the output\n"); printf("\t-r \tspecify the XML search root\n"); printf("\t-v \tshow sections where one of the elements has this "); printf("value\n\n"); printf(" To print the contents of the 'type' element of the XML section "); printf("that begins\n at '/printer/output' one would use the following "); printf("syntax:\n\n\txmlgrep -r /printer/output -p type sample.xml\n\n"); printf(" To filter out sections that contain the 'driver' element with "); printf("'generic' as\n it's value one would issue the following command:"); printf("\n\n\txmlgrep -r /printer/output -e driver -v generic sample.xml"); printf("\n\n"); free_and_exit(0); } void free_and_exit(int i) { if (_root && _root != _static_root) free(_root); if (_element && _element != _static_element) free(_element); if (_value) free(_value); if (_print) free(_print); if (_filenames) { for (i=0; i < _fcount; i++) { if (_filenames[i]) { if (print_filenames) printf("%s\n", _filenames[i]); free(_filenames[i]); } } free(_filenames); } exit(i); } int parse_option(char **args, int n, int max) { char *opt, *arg = 0; int sz; opt = args[n]; if (opt[0] == '-' && opt[1] == '-') opt++; if ((arg = strchr(opt, '=')) != NULL) { *arg++ = 0; } else if (++n < max) { arg = args[n]; #if 0 if (arg && arg[0] == '-') arg = 0; #endif } sz = strlen(opt); if (strncmp(opt, "-help", sz) == 0) { show_help(); } else if (strncmp(opt, "-root", sz) == 0) { if (arg == 0) SHOW_NOVAL(opt); _root = strdup(arg); return 2; } else if (strncmp(opt, "-element", sz) == 0) { if (arg == 0) SHOW_NOVAL(opt); _element = strdup(arg); return 2; } else if (strncmp(opt, "-value", sz) == 0) { if (arg == 0) SHOW_NOVAL(opt); _value = strdup(arg); return 2; } else if (strncmp(opt, "-print", sz) == 0) { if (arg == 0) SHOW_NOVAL(opt); _print = strdup(arg); return 2; } else if (strncmp(opt, "-list-filenames", sz) == 0) { /* undocumented test argument */ print_filenames = 1; return 1; } else if (opt[0] == '-') { printf("Unknown option %s\n", opt); free_and_exit(-1); } else { int pos = _fcount++; if (_filenames == 0) { _filenames = (char **)malloc(sizeof(char*)); } else { char **ptr = (char **)realloc(_filenames, _fcount*sizeof(char*)); if (ptr == 0) { printf("Out of memory.\n\n"); free_and_exit(-1); } _filenames = ptr; } _filenames[pos] = strdup(opt); } return 1; } void walk_the_tree(size_t num, void *xid, char *tree) { unsigned int i, no_elements; if (!tree) /* last node from the tree */ { void *xmid = xmlMarkId(xid); if (xmid && _print) { no_elements = xmlGetNumNodes(xid, _print); for (i=0; i%s\n", _filenames[num], _print, value, _print); } } free(xmid); } else if (xmid && _value) { no_elements = xmlGetNumNodes(xmid, _element); for (i=0; i%s\n", _filenames[num], nodename, _value, nodename); } } } free(xmid); } else if (xmid && _element) { char parentname[64]; xmlCopyNodeName(xid, (char *)&parentname, 64); no_elements = xmlGetNumNodes(xmid, _element); for (i=0; i <%s>%s \n", _filenames[num], parentname, nodename, value, nodename, parentname); } } } } else printf("Error executing xmlMarkId\n"); } else if (xid) /* walk the rest of the tree */ { char *elem, *next; void *xmid; elem = tree; if (*elem == '/') elem++; next = strchr(elem, '/'); xmid = xmlMarkId(xid); if (xmid) { if (next) *next++ = 0; no_elements = xmlGetNumNodes(xid, elem); for (i=0; i