// The contents of this file are subject to the BOINC Public License // Version 1.0 (the "License"); you may not use this file except in // compliance with the License. You may obtain a copy of the License at // http://boinc.berkeley.edu/license_1.0.txt // // Software distributed under the License is distributed on an "AS IS" // basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the // License for the specific language governing rights and limitations // under the License. // // The Original Code is the Berkeley Open Infrastructure for Network Computing. // // The Initial Developer of the Original Code is the SETI@home project. // Portions created by the SETI@home project are Copyright (C) 2002 // University of California at Berkeley. All Rights Reserved. // // Contributor(s): // // A very crude interface for parsing XML files; // assumes all elements are either single-line or // have start and end tags on separate lines. // This is meant to be used ONLY for parsing XML files produced // by the BOINC scheduling server or client. // Could replace this with a more general parser. #ifdef _WIN32 #include #endif #include #include #include #include "error_numbers.h" #include "util.h" #include "parse.h" // return true if the tag appears in the line // bool match_tag(const char* buf, const char* tag) { if (strstr(buf, tag)) return true; return false; } bool match_tag(const std::string &s, const char* tag) { if (s.find(tag) != std::string::npos) return true; return false; } // parse an integer of the form 1234 // return true if it's there // Note: this doesn't check for the end tag // bool parse_int(const char* buf, const char* tag, int& x) { char* p = strstr(buf, tag); if (!p) return false; x = strtol(p+strlen(tag), 0, 0); // this parses 0xabcd correctly return true; } // Same, for doubles // bool parse_double(const char* buf, const char* tag, double& x) { char* p = strstr(buf, tag); if (!p) return false; x = atof(p+strlen(tag)); return true; } // parse a string of the form string // bool parse_str(const char* buf, const char* tag, char* dest, int len) { char* p = strstr(buf, tag); if (!p) return false; p = strchr(p, '>'); char* q = strchr(p+1, '<'); if (!q) return false; *q = 0; safe_strncpy(dest, p+1, len); *q = '<'; return true; } // parse a string of the form name="string" // void parse_attr(const char* buf, const char* name, char* dest, int len) { char* p, *q; strcpy(dest, ""); p = strstr(buf, name); if (!p) return; p = strchr(p, '"'); if (!p) return; q = strchr(p+1, '"'); if (!q) return; *q = 0; safe_strncpy(dest, p+1, len); } void copy_stream(FILE* in, FILE* out) { char buf[1024]; int n, m; while (1) { n = fread(buf, 1, 1024, in); m = fwrite(buf, 1, n, out); if (n < 1024) break; } } // append to a malloc'd string // void strcatdup(char*& p, char* buf) { p = (char*)realloc(p, strlen(p) + strlen(buf)+1); if (!p) { fprintf(stderr, "strcatdup: realloc failed\n"); exit(1); } strcat(p, buf); } // copy from a file to a malloc'd string until the end tag is reached // int dup_element_contents(FILE* in, const char* end_tag, char** pp) { char buf[256]; char* p = strdup(""); while (fgets(buf, 256, in)) { if (strstr(buf, end_tag)) { *pp = p; return 0; } strcatdup(p, buf); } fprintf(stderr, "dup_element_contents(): no end tag\n"); return 1; } // copy from a file to static buffer // int copy_element_contents(FILE* in, const char* end_tag, char* p, int len) { char buf[256]; strcpy(p, ""); while (fgets(buf, 256, in)) { if (strstr(buf, end_tag)) { return 0; } strcat(p, buf); } fprintf(stderr, "copy_element_contents(): no end tag\n"); return 1; } // read a file into a malloc'd string // int read_file_malloc(const char* pathname, char*& str) { char buf[256]; FILE* f; f = fopen(pathname, "r"); if (!f) return -1; str = strdup(""); while (fgets(buf, 256, f)) { strcatdup(str, buf); } fclose(f); return 0; } // replace XML element contents. not currently used // void replace_element(char* buf, char* start, char* end, char* replacement) { char temp[4096], *p, *q; p = strstr(buf, start); p += strlen(start); q = strstr(p, end); safe_strncpy(temp, q, sizeof(temp)); strcpy(p, replacement); strcat(p, temp); } // if the given XML has an element of the form // // ... // // then return the contents of that element. // Otherwise strip out all elements // void extract_venue(const char* in, const char* venue_name, char* out) { char* p, *q; char buf[256]; sprintf(buf, "", venue_name); p = strstr(in, buf); if (p) { p += strlen(buf); strcpy(out, p); q = strstr(out, "\n"); if (!q) break; strcpy(p, q+strlen("\n")); } } } // copy a line from the given string. // kinda like fgets() when you're reading from a string // char* sgets(char* buf, int len, char*& in) { char* p; p = strstr(in, "\n"); if (!p) return NULL; *p = 0; safe_strncpy(buf, in, len); *p = '\n'; in = p+1; return buf; } bool extract_xml_record(const std::string &field, const char *tag, std::string &record) { std::string::size_type start_pos,end_pos,m; char end_tag[256]; sprintf(end_tag,"/%s",tag); std::string::size_type i=0,j=0; // find the end tag do { j=field.find(">",j+1); end_pos=field.rfind(end_tag,j,j-i+1); i=field.rfind("<",j,j-i); } while ((end_pos==std::string::npos) && (j!=std::string::npos)); if (j!=std::string::npos) { end_pos=j; } else { return false; } // find the start tag j--; i=0; do { j=field.rfind(">",j-1); start_pos=field.rfind(tag,j,j-i+1); i=field.rfind("<",j,j-i+1); m=field.rfind("/",j,j-i+1); } while (((m!=std::string::npos) || (start_pos==std::string::npos)) && (i!=std::string::npos)); if (i!=std::string::npos) { start_pos=i; } else { return false; } record=std::string(field,start_pos,end_pos-start_pos+1); return true; }