json harder than I thought

This commit is contained in:
Ron Pedde 2006-03-28 07:49:39 +00:00
parent 8c39e0294e
commit a102c38763
2 changed files with 74 additions and 20 deletions

View File

@ -21,6 +21,7 @@
#include "mp3-scanner.h" #include "mp3-scanner.h"
#include "rend.h" #include "rend.h"
#include "webserver.h" #include "webserver.h"
#include "xml-rpc.h"
/* typedefs */ /* typedefs */
typedef struct tag_xmlstack { typedef struct tag_xmlstack {
@ -31,6 +32,7 @@ typedef struct tag_xmlstack {
typedef struct tag_xmlstruct { typedef struct tag_xmlstruct {
WS_CONNINFO *pwsc; WS_CONNINFO *pwsc;
int stack_level; int stack_level;
int flags;
XMLSTACK stack; XMLSTACK stack;
} XMLSTRUCT; } XMLSTRUCT;
@ -38,12 +40,6 @@ typedef struct tag_xmlstruct {
void xml_get_stats(WS_CONNINFO *pwsc); void xml_get_stats(WS_CONNINFO *pwsc);
char *xml_entity_encode(char *original); char *xml_entity_encode(char *original);
XMLSTRUCT *xml_init(WS_CONNINFO *pwsc, int emit_header);
void xml_push(XMLSTRUCT *pxml, char *term);
void xml_pop(XMLSTRUCT *pxml);
void xml_output(XMLSTRUCT *pxml, char *section, char *fmt, ...);
void xml_deinit(XMLSTRUCT *pxml);
/** /**
* create an xml response structure, a helper struct for * create an xml response structure, a helper struct for
* building xml responses. * building xml responses.
@ -52,7 +48,7 @@ void xml_deinit(XMLSTRUCT *pxml);
* @param emit_header whether or not to throw out html headers and xml header * @param emit_header whether or not to throw out html headers and xml header
* @returns XMLSTRUCT on success, or NULL if failure * @returns XMLSTRUCT on success, or NULL if failure
*/ */
XMLSTRUCT *xml_init(WS_CONNINFO *pwsc, int emit_header) { XMLSTRUCT *xml_init(WS_CONNINFO *pwsc, int emit_header, int flags) {
XMLSTRUCT *pxml; XMLSTRUCT *pxml;
pxml=(XMLSTRUCT*)malloc(sizeof(XMLSTRUCT)); pxml=(XMLSTRUCT*)malloc(sizeof(XMLSTRUCT));
@ -63,13 +59,20 @@ XMLSTRUCT *xml_init(WS_CONNINFO *pwsc, int emit_header) {
memset(pxml,0,sizeof(XMLSTRUCT)); memset(pxml,0,sizeof(XMLSTRUCT));
pxml->pwsc = pwsc; pxml->pwsc = pwsc;
pxml->flags = flags;
if(emit_header) { if(emit_header) {
ws_addresponseheader(pwsc,"Content-Type","text/xml; charset=utf-8"); if(flags & XML_FLAG_JSON) {
ws_addresponseheader(pwsc,"Content-Type","text/json");
} else {
ws_addresponseheader(pwsc,"Content-Type","text/xml; charset=utf-8");
}
ws_writefd(pwsc,"HTTP/1.0 200 OK\r\n"); ws_writefd(pwsc,"HTTP/1.0 200 OK\r\n");
ws_emitheaders(pwsc); ws_emitheaders(pwsc);
ws_writefd(pwsc,"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>"); if(!(flags & XML_FLAG_JSON)) {
ws_writefd(pwsc,"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>");
}
} }
return pxml; return pxml;
@ -89,9 +92,16 @@ void xml_push(XMLSTRUCT *pxml, char *term) {
pstack->tag=strdup(term); pstack->tag=strdup(term);
pxml->stack.next=pstack; pxml->stack.next=pstack;
pxml->stack_level++; if(pxml->flags & XML_FLAG_READABLE)
ws_writefd(pxml->pwsc,"\n%*s",pxml->stack_level,"");
ws_writefd(pxml->pwsc,"<%s>",term); if(pxml->flags & XML_FLAG_JSON) {
ws_writefd(pxml->pwsc,"{ \"%s\": ",term);
} else {
ws_writefd(pxml->pwsc,"<%s>",term);
}
pxml->stack_level++;
} }
/** /**
@ -110,11 +120,22 @@ void xml_pop(XMLSTRUCT *pxml) {
pxml->stack.next = pstack->next; pxml->stack.next = pstack->next;
ws_writefd(pxml->pwsc,"</%s>",pstack->tag); pxml->stack_level--;
if(pxml->flags & XML_FLAG_READABLE)
ws_writefd(pxml->pwsc,"\n%*s",pxml->stack_level,"");
if(pxml->flags & XML_FLAG_JSON) {
ws_writefd(pxml->pwsc,"}",pstack->tag);
} else {
ws_writefd(pxml->pwsc,"</%s>",pstack->tag);
}
if(pxml->flags & XML_FLAG_READABLE)
ws_writefd(pxml->pwsc,"%*s",pxml->stack_level,"");
free(pstack->tag); free(pstack->tag);
free(pstack); free(pstack);
pxml->stack_level--;
} }
/** /**
@ -124,20 +145,32 @@ void xml_output(XMLSTRUCT *pxml, char *section, char *fmt, ...) {
va_list ap; va_list ap;
char buf[256]; char buf[256];
char *output; char *output;
int oldflags;
va_start(ap, fmt); va_start(ap, fmt);
vsnprintf(buf, sizeof(buf), fmt, ap); vsnprintf(buf, sizeof(buf), fmt, ap);
va_end(ap); va_end(ap);
output = xml_entity_encode(buf);
if(section) { if(section) {
xml_push(pxml,section); xml_push(pxml,section);
} }
ws_writefd(pxml->pwsc,"%s",output);
output = xml_entity_encode(buf);
if(pxml->flags & XML_FLAG_JSON) {
ws_writefd(pxml->pwsc,"\"%s\" ",output);
} else {
ws_writefd(pxml->pwsc,"%s",output);
}
free(output); free(output);
oldflags = pxml->flags;
pxml->flags = pxml->flags & ~XML_FLAG_READABLE;
if(section) { if(section) {
xml_pop(pxml); xml_pop(pxml);
} }
pxml->flags = oldflags;
} }
/** /**
@ -187,6 +220,7 @@ void xml_handle(WS_CONNINFO *pwsc) {
*/ */
void xml_get_stats(WS_CONNINFO *pwsc) { void xml_get_stats(WS_CONNINFO *pwsc) {
int r_secs, r_days, r_hours, r_mins; int r_secs, r_days, r_hours, r_mins;
int flag=0;
char buf[80]; char buf[80];
WS_CONNINFO *pci; WS_CONNINFO *pci;
SCAN_STATUS *pss; SCAN_STATUS *pss;
@ -194,13 +228,20 @@ void xml_get_stats(WS_CONNINFO *pwsc) {
int count; int count;
XMLSTRUCT *pxml; XMLSTRUCT *pxml;
pxml=xml_init(pwsc,1); /* check output... */
if(ws_getvar(pwsc,"output") == NULL) {
flag = XML_FLAG_NONE;
} else if(strcasecmp(ws_getvar(pwsc,"output"),"json") == 0) {
flag = XML_FLAG_JSON | XML_FLAG_READABLE;
} else if(strcasecmp(ws_getvar(pwsc,"output"),"readable") == 0) {
flag = XML_FLAG_READABLE;
}
pxml=xml_init(pwsc,1,flag);
xml_push(pxml,"status"); xml_push(pxml,"status");
xml_push(pxml,"service_status"); xml_push(pxml,"service_status");
xml_push(pxml,"service"); xml_push(pxml,"service");
xml_output(pxml,"name","Rendezvous"); xml_output(pxml,"name","Rendezvous");
#ifndef WITHOUT_MDNS #ifndef WITHOUT_MDNS

View File

@ -7,6 +7,19 @@
#include "webserver.h" #include "webserver.h"
#define XML_FLAG_NONE 0
#define XML_FLAG_JSON 1
#define XML_FLAG_READABLE 2
struct tag_xmlstruct;
extern void xml_handle(WS_CONNINFO *pwsc); extern void xml_handle(WS_CONNINFO *pwsc);
extern struct tag_xmlstruct *xml_init(WS_CONNINFO *pwsc, int emit_header,
int flags);
extern void xml_push(struct tag_xmlstruct *pxml, char *term);
extern void xml_pop(struct tag_xmlstruct *pxml);
extern void xml_output(struct tag_xmlstruct *pxml, char *section, char *fmt, ...);
extern void xml_deinit(struct tag_xmlstruct *pxml);
#endif /* _XMLRPC_H_ */ #endif /* _XMLRPC_H_ */