X-Git-Url: http://git.indexdata.com/?a=blobdiff_plain;f=src%2Fhttp_command.c;h=940b1f143c6b74255feb27cd15c0fa04595e43d8;hb=4b0c3f7b760ec6aaa688c09791b07cfa53d9ec47;hp=e453a9da2327c65f1a91e3ce63ff1c2d55ab9ced;hpb=b9a83dec2858e9aa4247b29b3f8f8cf42a1a786d;p=pazpar2-moved-to-github.git diff --git a/src/http_command.c b/src/http_command.c index e453a9d..940b1f1 100644 --- a/src/http_command.c +++ b/src/http_command.c @@ -1,5 +1,5 @@ /* - * $Id: http_command.c,v 1.2 2006-12-20 22:19:35 adam Exp $ + * $Id: http_command.c,v 1.22 2007-01-15 04:34:28 quinn Exp $ */ #include @@ -11,9 +11,13 @@ #include #include +#if HAVE_CONFIG_H +#include +#endif + #include -#include "command.h" +#include "config.h" #include "util.h" #include "eventl.h" #include "pazpar2.h" @@ -96,7 +100,7 @@ unsigned int make_sessionid() if (gettimeofday(&t, 0) < 0) abort(); res = t.tv_sec; - res = ((res << 8) | (seq & 0xff)) & ((unsigned int) (1 << 31) - 1); + res = ((res << 8) | (seq & 0xff)) & ((1U << 31) - 1); return res; } @@ -144,6 +148,34 @@ static void cmd_init(struct http_channel *c) http_send_response(c); } +// Compares two hitsbytarget nodes by hitcount +static int cmp_ht(const void *p1, const void *p2) +{ + const struct hitsbytarget *h1 = p1; + const struct hitsbytarget *h2 = p2; + return h2->hits - h1->hits; +} + +// This implements functionality somewhat similar to 'bytarget', but in a termlist form +static void targets_termlist(WRBUF wrbuf, struct session *se) +{ + struct hitsbytarget *ht; + int count, i; + + if (!(ht = hitsbytarget(se, &count))) + return; + qsort(ht, count, sizeof(struct hitsbytarget), cmp_ht); + for (i = 0; i < count && i < 15; i++) + { + wrbuf_puts(wrbuf, "\n\n"); + wrbuf_printf(wrbuf, "%s\n", ht[i].id); + wrbuf_printf(wrbuf, "%d\n", ht[i].hits); + wrbuf_printf(wrbuf, "%s\n", ht[i].state); + wrbuf_printf(wrbuf, "%d\n", ht[i].diagnostic); + wrbuf_puts(wrbuf, "\n\n"); + } +} + static void cmd_termlist(struct http_channel *c) { struct http_response *rs = c->response; @@ -152,21 +184,53 @@ static void cmd_termlist(struct http_channel *c) struct termlist_score **p; int len; int i; + char *name = http_argbyname(rq, "name"); + int status; if (!s) return; + + status = session_active_clients(s->psession); + + if (!name) + name = "subject"; + if (strlen(name) > 255) + return; + wrbuf_rewind(c->wrbuf); wrbuf_puts(c->wrbuf, ""); - p = termlist(s->psession, &len); - if (p) - for (i = 0; i < len; i++) + wrbuf_printf(c->wrbuf, "\n%d", status); + while (*name) + { + char tname[256]; + char *tp; + + if (!(tp = strchr(name, ','))) + tp = name + strlen(name); + strncpy(tname, name, tp - name); + tname[tp - name] = '\0'; + + wrbuf_printf(c->wrbuf, "\n\n", tname); + if (!strcmp(tname, "xtargets")) + targets_termlist(c->wrbuf, s->psession); + else { - wrbuf_puts(c->wrbuf, "\n"); - wrbuf_printf(c->wrbuf, "%s", p[i]->term); - wrbuf_printf(c->wrbuf, "%d", p[i]->frequency); - wrbuf_puts(c->wrbuf, ""); + p = termlist(s->psession, tname, &len); + if (p) + for (i = 0; i < len; i++) + { + wrbuf_puts(c->wrbuf, "\n"); + wrbuf_printf(c->wrbuf, "%s", p[i]->term); + wrbuf_printf(c->wrbuf, "%d", p[i]->frequency); + wrbuf_puts(c->wrbuf, ""); + } } + wrbuf_puts(c->wrbuf, "\n"); + name = tp; + if (*name == ',') + name++; + } wrbuf_puts(c->wrbuf, ""); rs->payload = nmem_strdup(rq->channel->nmem, wrbuf_buf(c->wrbuf)); http_send_response(c); @@ -207,15 +271,80 @@ static void cmd_bytarget(struct http_channel *c) http_send_response(c); } -static void show_records(struct http_channel *c) +static void write_metadata(WRBUF w, struct conf_service *service, + struct record_metadata **ml, int full) +{ + int imeta; + + for (imeta = 0; imeta < service->num_metadata; imeta++) + { + struct conf_metadata *cmd = &service->metadata[imeta]; + struct record_metadata *md; + if (!cmd->brief && !full) + continue; + for (md = ml[imeta]; md; md = md->next) + { + wrbuf_printf(w, "", cmd->name); + switch (cmd->type) + { + case Metadata_type_generic: + wrbuf_puts(w, md->data.text); + break; + case Metadata_type_year: + wrbuf_printf(w, "%d", md->data.number.min); + if (md->data.number.min != md->data.number.max) + wrbuf_printf(w, "-%d", md->data.number.max); + break; + default: + wrbuf_puts(w, "[can't represent]"); + } + wrbuf_printf(w, "", cmd->name); + } + } +} + +static void cmd_record(struct http_channel *c) +{ + struct http_response *rs = c->response; + struct http_request *rq = c->request; + struct http_session *s = locate_session(rq, rs); + struct record_cluster *rec; + struct conf_service *service = global_parameters.server->service; + char *idstr = http_argbyname(rq, "id"); + int id; + + if (!s) + return; + if (!idstr) + { + error(rs, "417", "Must supply id", 0); + return; + } + wrbuf_rewind(c->wrbuf); + id = atoi(idstr); + if (!(rec = show_single(s->psession, id))) + { + error(rs, "500", "Record missing", 0); + return; + } + wrbuf_puts(c->wrbuf, "\n"); + wrbuf_printf(c->wrbuf, "%d", rec->recid); + write_metadata(c->wrbuf, service, rec->metadata, 1); + wrbuf_puts(c->wrbuf, "\n"); + rs->payload = nmem_strdup(c->nmem, wrbuf_buf(c->wrbuf)); + http_send_response(c); +} + +static void show_records(struct http_channel *c, int active) { struct http_request *rq = c->request; struct http_response *rs = c->response; struct http_session *s = locate_session(rq, rs); - struct record **rl; - NMEM nmem_show; + struct record_cluster **rl; + struct reclist_sortparms *sp; char *start = http_argbyname(rq, "start"); char *num = http_argbyname(rq, "num"); + char *sort = http_argbyname(rq, "sort"); int startn = 0; int numn = 20; int total; @@ -225,16 +354,27 @@ static void show_records(struct http_channel *c) if (!s) return; + // We haven't counted clients yet if we're called on a block release + if (active < 0) + active = session_active_clients(s->psession); + if (start) startn = atoi(start); if (num) numn = atoi(num); + if (!sort) + sort = "relevance"; + if (!(sp = reclist_parse_sortparms(c->nmem, sort))) + { + error(rs, "500", "Bad sort parameters", 0); + return; + } - nmem_show = nmem_create(); - rl = show(s->psession, startn, &numn, &total, &total_hits, nmem_show); + rl = show(s->psession, sp, startn, &numn, &total, &total_hits, c->nmem); wrbuf_rewind(c->wrbuf); wrbuf_puts(c->wrbuf, "\nOK\n"); + wrbuf_printf(c->wrbuf, "%d\n", active); wrbuf_printf(c->wrbuf, "%d\n", total); wrbuf_printf(c->wrbuf, "%d\n", total_hits); wrbuf_printf(c->wrbuf, "%d\n", startn); @@ -244,27 +384,29 @@ static void show_records(struct http_channel *c) { int ccount; struct record *p; + struct record_cluster *rec = rl[i]; + struct conf_service *service = global_parameters.server->service; wrbuf_puts(c->wrbuf, "\n"); - wrbuf_printf(c->wrbuf, "%s\n", rl[i]->title); - for (ccount = 1, p = rl[i]->next_cluster; p; p = p->next_cluster, ccount++) + write_metadata(c->wrbuf, service, rec->metadata, 0); + for (ccount = 0, p = rl[i]->records; p; p = p->next, ccount++) ; if (ccount > 1) wrbuf_printf(c->wrbuf, "%d\n", ccount); + wrbuf_printf(c->wrbuf, "%d\n", rec->recid); wrbuf_puts(c->wrbuf, "\n"); } wrbuf_puts(c->wrbuf, "\n"); rs->payload = nmem_strdup(c->nmem, wrbuf_buf(c->wrbuf)); http_send_response(c); - nmem_destroy(nmem_show); } static void show_records_ready(void *data) { struct http_channel *c = (struct http_channel *) data; - show_records(c); + show_records(c, -1); } static void cmd_show(struct http_channel *c) @@ -273,13 +415,16 @@ static void cmd_show(struct http_channel *c) struct http_response *rs = c->response; struct http_session *s = locate_session(rq, rs); char *block = http_argbyname(rq, "block"); + int status; if (!s) return; + status = session_active_clients(s->psession); + if (block) { - if (!s->psession->reclist || !s->psession->reclist->num_records) + if (status && (!s->psession->reclist || !s->psession->reclist->num_records)) { session_set_watch(s->psession, SESSION_WATCH_RECORDS, show_records_ready, c); yaz_log(YLOG_DEBUG, "Blocking on cmd_show"); @@ -287,7 +432,7 @@ static void cmd_show(struct http_channel *c) } } - show_records(c); + show_records(c, status); } static void cmd_ping(struct http_channel *c) @@ -333,14 +478,17 @@ static void cmd_stat(struct http_channel *c) struct http_response *rs = c->response; struct http_session *s = locate_session(rq, rs); struct statistics stat; + int clients; if (!s) return; + clients = session_active_clients(s->psession); statistics(s->psession, &stat); wrbuf_rewind(c->wrbuf); wrbuf_puts(c->wrbuf, ""); + wrbuf_printf(c->wrbuf, "%d\n", clients); wrbuf_printf(c->wrbuf, "%d\n", stat.num_hits); wrbuf_printf(c->wrbuf, "%d\n", stat.num_records); wrbuf_printf(c->wrbuf, "%d\n", stat.num_clients); @@ -357,6 +505,25 @@ static void cmd_stat(struct http_channel *c) http_send_response(c); } +static void cmd_info(struct http_channel *c) +{ + char yaz_version_str[20]; + struct http_response *rs = c->response; + + wrbuf_rewind(c->wrbuf); + wrbuf_puts(c->wrbuf, "\n"); + wrbuf_printf(c->wrbuf, " \n"); + wrbuf_printf(c->wrbuf, " %s\n", VERSION); + + yaz_version(yaz_version_str, 0); + wrbuf_printf(c->wrbuf, " %s\n", + YAZ_VERSION, yaz_version_str); + wrbuf_printf(c->wrbuf, " \n"); + + wrbuf_puts(c->wrbuf, ""); + rs->payload = nmem_strdup(c->nmem, wrbuf_buf(c->wrbuf)); + http_send_response(c); +} struct { char *name; @@ -370,6 +537,8 @@ struct { { "termlist", cmd_termlist }, { "exit", cmd_exit }, { "ping", cmd_ping }, + { "record", cmd_record }, + { "info", cmd_info }, {0,0} }; @@ -380,6 +549,10 @@ void http_command(struct http_channel *c) int i; c->response = rs; + + http_addheader(rs, "Expires", "Thu, 19 Nov 1981 08:52:00 GMT"); + http_addheader(rs, "Cache-Control", "no-store, no-cache, must-revalidate, post-check=0, pre-check=0"); + if (!command) { error(rs, "417", "Must supply command", 0);