mirror of
https://github.com/moparisthebest/curl
synced 2024-11-16 06:25:03 -05:00
f193ab4b59
too little data, and one case of not zero-terminating the returned string. I chose a slightly different patch than the one Peter provided.
164 lines
3.4 KiB
C
164 lines
3.4 KiB
C
|
|
#include <stdio.h>
|
|
#include <ctype.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#define EAT_SPACE(ptr) while( ptr && *ptr && isspace((int)*ptr) ) ptr++
|
|
#define EAT_WORD(ptr) while( ptr && *ptr && !isspace((int)*ptr) && ('>' != *ptr)) ptr++
|
|
|
|
#ifdef DEBUG
|
|
#define show(x) printf x
|
|
#else
|
|
#define show(x)
|
|
#endif
|
|
|
|
static
|
|
char *appendstring(char *string, /* original string */
|
|
char *buffer, /* to append */
|
|
int *stringlen, int *stralloc)
|
|
{
|
|
int len = strlen(buffer);
|
|
int needed_len = len + *stringlen;
|
|
|
|
if(needed_len >= *stralloc) {
|
|
char *newptr;
|
|
long newsize = needed_len*2; /* get twice the needed size */
|
|
|
|
newptr = realloc(string, newsize);
|
|
if(newptr) {
|
|
string = newptr;
|
|
*stralloc = newsize;
|
|
}
|
|
else
|
|
return NULL;
|
|
}
|
|
strcpy(&string[*stringlen], buffer);
|
|
*stringlen += len;
|
|
|
|
return string;
|
|
}
|
|
|
|
char *spitout(FILE *stream, char *main, char *sub, int *size)
|
|
{
|
|
char buffer[8192]; /* big enough for anything */
|
|
char cmain[128]=""; /* current main section */
|
|
char csub[128]=""; /* current sub section */
|
|
char *ptr;
|
|
char *end;
|
|
char display = 0;
|
|
|
|
char *string;
|
|
int stringlen=0;
|
|
int stralloc=256;
|
|
|
|
enum {
|
|
STATE_OUTSIDE,
|
|
STATE_INMAIN,
|
|
STATE_INSUB,
|
|
STATE_ILLEGAL
|
|
} state = STATE_OUTSIDE;
|
|
|
|
string = (char *)malloc(stralloc);
|
|
if(!string)
|
|
return NULL;
|
|
|
|
string[0] = 0; /* zero first byte in case of no data */
|
|
|
|
while(fgets(buffer, sizeof(buffer), stream)) {
|
|
|
|
ptr = buffer;
|
|
|
|
/* pass white spaces */
|
|
EAT_SPACE(ptr);
|
|
|
|
if('<' != *ptr) {
|
|
if(display) {
|
|
show(("=> %s", buffer));
|
|
string = appendstring(string, buffer, &stringlen, &stralloc);
|
|
show(("* %s\n", buffer));
|
|
}
|
|
continue;
|
|
}
|
|
|
|
ptr++;
|
|
EAT_SPACE(ptr);
|
|
|
|
if('/' == *ptr) {
|
|
/* end of a section */
|
|
ptr++;
|
|
EAT_SPACE(ptr);
|
|
|
|
end = ptr;
|
|
EAT_WORD(end);
|
|
*end = 0;
|
|
|
|
if((state == STATE_INSUB) &&
|
|
!strcmp(csub, ptr)) {
|
|
/* this is the end of the currently read sub section */
|
|
state--;
|
|
csub[0]=0; /* no sub anymore */
|
|
display=0;
|
|
}
|
|
else if((state == STATE_INMAIN) &&
|
|
!strcmp(cmain, ptr)) {
|
|
/* this is the end of the currently read main section */
|
|
state--;
|
|
cmain[0]=0; /* no main anymore */
|
|
display=0;
|
|
}
|
|
}
|
|
else if(!display) {
|
|
/* this is the beginning of a section */
|
|
end = ptr;
|
|
EAT_WORD(end);
|
|
|
|
*end = 0;
|
|
switch(state) {
|
|
case STATE_OUTSIDE:
|
|
strcpy(cmain, ptr);
|
|
state = STATE_INMAIN;
|
|
break;
|
|
case STATE_INMAIN:
|
|
strcpy(csub, ptr);
|
|
state = STATE_INSUB;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if(display) {
|
|
string = appendstring(string, buffer, &stringlen, &stralloc);
|
|
show(("* %s\n", buffer));
|
|
}
|
|
|
|
if((STATE_INSUB == state) &&
|
|
!strcmp(cmain, main) &&
|
|
!strcmp(csub, sub)) {
|
|
show(("* (%d bytes) %s\n", stringlen, buffer));
|
|
display = 1; /* start displaying */
|
|
}
|
|
else {
|
|
show(("%d (%s/%s): %s\n", state, cmain, csub, buffer));
|
|
display = 0; /* no display */
|
|
}
|
|
}
|
|
|
|
*size = stringlen;
|
|
return string;
|
|
}
|
|
|
|
#ifdef TEST
|
|
int main(int argc, char **argv)
|
|
{
|
|
if(argc< 3) {
|
|
printf("./moo main sub\n");
|
|
}
|
|
else {
|
|
int size;
|
|
char *buffer = spitout(stdin, argv[1], argv[2], &size);
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|