#include #define _GNU_SOURCE #include #include #ifndef _MSC_VER # include #else # define strncasecmp strnicmp # include #endif #include #include /* read */ #include /* fstat */ #include /* open */ #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 char *_attribute = 0; static int print_filenames = 0; static void free_and_exit(int i); #define USE_BUFFER 0 #define NODE_NAME_LEN 256 #define STRING_LEN 2048 #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-a \tprint this attribute as the output\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 \tfilter sections that contain this vale\n\n"); printf(" To print the contents of the 'type' element of the XML section "); printf("that begins\n at '/printer/output' use the following command:\n\n"); printf("\txmlgrep -r /printer/output -p type sample.xml\n\n"); printf(" To filter 'output' elements under '/printer' that have attribute"); printf(" 'n' set to '1'\n use the following command:\n\n"); printf("\txmlgrep -r /printer -p output -a n -v 1 sample.xml\n\n"); printf(" To filter out sections that contain the 'driver' element with "); printf("'generic' as\n it's value use 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 ret) { if (_root && _root != _static_root) free(_root); if (_element && _element != _static_element) free(_element); if (_value) free(_value); if (_print) free(_print); if (_attribute) free(_attribute); if (_filenames) { unsigned int i; for (i=0; i < _fcount; i++) { if (_filenames[i]) { if (print_filenames) printf("%s\n", _filenames[i]); free(_filenames[i]); } } free(_filenames); } exit(ret); } int parse_option(char **args, int n, int max) { char *opt, *arg = 0; unsigned int alen = 0; unsigned int olen; opt = args[n]; if (strncmp(opt, "--", 2) == 0) opt++; if ((arg = strchr(opt, '=')) != NULL) { *arg++ = 0; } else if (++n < max) { arg = args[n]; #if 0 if (arg && arg[0] == '-') arg = 0; #endif } olen = strlen(opt); if (strncmp(opt, "-help", olen) == 0) { show_help(); } else if (strncmp(opt, "-root", olen) == 0) { if (arg == 0) SHOW_NOVAL(opt); alen = strlen(arg)+1; _root = malloc(alen); memcpy(_root, arg, alen); return 2; } else if (strncmp(opt, "-element", olen) == 0) { if (arg == 0) SHOW_NOVAL(opt); alen = strlen(arg)+1; _element = malloc(alen); memcpy(_element, arg, alen); return 2; } else if (strncmp(opt, "-value", olen) == 0) { if (arg == 0) SHOW_NOVAL(opt); alen = strlen(arg)+1; _value = malloc(alen); memcpy(_value, arg, alen); return 2; } else if (strncmp(opt, "-print", olen) == 0) { if (arg == 0) SHOW_NOVAL(opt); alen = strlen(arg)+1; _print = malloc(alen); memcpy(_print, arg, alen); return 2; } else if (strncmp(opt, "-attribute", olen) == 0) { if (arg == 0) SHOW_NOVAL(opt); alen = strlen(arg)+1; _attribute = malloc(alen); memcpy(_attribute, arg, alen); return 2; } else if (strncmp(opt, "-list-filenames", olen) == 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; } alen = strlen(opt)+1; _filenames[pos] = malloc(alen); memcpy(_filenames[pos], opt, alen); } 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 = xmlNodeGetNum(xid, _print); for (i=0; i%s\n", _filenames[num], _print, _attribute, _value, value, _print); } } else { printf("%s: <%s>%s\n", _filenames[num], _print, value, _print); } } } free(xmid); } else if (xmid && _value) { no_elements = xmlNodeGetNum(xmid, _element); for (i=0; i%s\n", _filenames[num], nodename, _value, nodename); } } } free(xmid); } else if (xmid && _element) { char parentname[NODE_NAME_LEN]; xmlNodeCopyName(xid, (char *)&parentname, NODE_NAME_LEN); no_elements = xmlNodeGetNum(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 = xmlNodeGetNum(xid, elem); for (i=0; i