X-Git-Url: http://lists.indexdata.com/cgi-bin?a=blobdiff_plain;f=src%2Fhttp_command.c;h=fccd6e570bf55d704c776b435abdf7fd20a23cc0;hb=8a203ad98e5cd0dc77fcf7e3193bc14c285c635a;hp=bc1bee50eadbfa4c51906519c978794b267368ba;hpb=5feca6b29788149a93b90c8c97394cb42e97a1e8;p=pazpar2-moved-to-github.git diff --git a/src/http_command.c b/src/http_command.c index bc1bee5..fccd6e5 100644 --- a/src/http_command.c +++ b/src/http_command.c @@ -1,5 +1,5 @@ /* - * $Id: http_command.c,v 1.12 2007-01-08 19:41:56 quinn Exp $ + * $Id: http_command.c,v 1.31 2007-04-10 00:53:24 quinn Exp $ */ #include @@ -17,6 +17,7 @@ #include +#include "config.h" #include "util.h" #include "eventl.h" #include "pazpar2.h" @@ -31,6 +32,7 @@ struct http_session { struct session *psession; unsigned int session_id; int timestamp; + NMEM nmem; struct http_session *next; }; @@ -46,10 +48,13 @@ static void session_timeout(IOCHAN i, int event) struct http_session *http_session_create() { - struct http_session *r = xmalloc(sizeof(*r)); - r->psession = new_session(); + NMEM nmem = nmem_create(); + struct http_session *r = nmem_malloc(nmem, sizeof(*r)); + + r->psession = new_session(nmem); r->session_id = 0; r->timestamp = 0; + r->nmem = nmem; r->next = session_list; session_list = r; r->timeout_iochan = iochan_create(-1, session_timeout, 0); @@ -72,7 +77,7 @@ void http_session_destroy(struct http_session *s) } iochan_destroy(s->timeout_iochan); destroy_session(s->psession); - xfree(s); + nmem_destroy(s->nmem); } static void error(struct http_response *rs, char *code, char *msg, char *txt) @@ -99,7 +104,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; } @@ -156,7 +161,7 @@ static int cmp_ht(const void *p1, const void *p2) } // This implements functionality somewhat similar to 'bytarget', but in a termlist form -static void targets_termlist(WRBUF wrbuf, struct session *se) +static void targets_termlist(WRBUF wrbuf, struct session *se, int num) { struct hitsbytarget *ht; int count, i; @@ -164,10 +169,11 @@ static void targets_termlist(WRBUF wrbuf, struct session *se) if (!(ht = hitsbytarget(se, &count))) return; qsort(ht, count, sizeof(struct hitsbytarget), cmp_ht); - for (i = 0; i < count && i < 15; i++) + for (i = 0; i < count && i < num && ht[i].hits > 0; i++) { wrbuf_puts(wrbuf, "\n\n"); - wrbuf_printf(wrbuf, "%s\n", ht[i].id); + wrbuf_printf(wrbuf, "%s\n", ht[i].id); + wrbuf_printf(wrbuf, "%s\n", ht[i].name); wrbuf_printf(wrbuf, "%d\n", ht[i].hits); wrbuf_printf(wrbuf, "%s\n", ht[i].state); wrbuf_printf(wrbuf, "%d\n", ht[i].diagnostic); @@ -184,6 +190,8 @@ static void cmd_termlist(struct http_channel *c) int len; int i; char *name = http_argbyname(rq, "name"); + char *nums = http_argbyname(rq, "num"); + int num = 15; int status; if (!s) @@ -195,6 +203,8 @@ static void cmd_termlist(struct http_channel *c) name = "subject"; if (strlen(name) > 255) return; + if (nums) + num = atoi(nums); wrbuf_rewind(c->wrbuf); @@ -212,12 +222,12 @@ static void cmd_termlist(struct http_channel *c) wrbuf_printf(c->wrbuf, "\n\n", tname); if (!strcmp(tname, "xtargets")) - targets_termlist(c->wrbuf, s->psession); + targets_termlist(c->wrbuf, s->psession, num); else { p = termlist(s->psession, tname, &len); if (p) - for (i = 0; i < len; i++) + for (i = 0; i < len && i < num; i++) { wrbuf_puts(c->wrbuf, "\n"); wrbuf_printf(c->wrbuf, "%s", p[i]->term); @@ -231,7 +241,7 @@ static void cmd_termlist(struct http_channel *c) name++; } wrbuf_puts(c->wrbuf, ""); - rs->payload = nmem_strdup(rq->channel->nmem, wrbuf_buf(c->wrbuf)); + rs->payload = nmem_strdup(rq->channel->nmem, wrbuf_cstr(c->wrbuf)); http_send_response(c); } @@ -266,7 +276,83 @@ static void cmd_bytarget(struct http_channel *c) } wrbuf_puts(c->wrbuf, ""); - rs->payload = nmem_strdup(c->nmem, wrbuf_buf(c->wrbuf)); + rs->payload = nmem_strdup(c->nmem, wrbuf_cstr(c->wrbuf)); + http_send_response(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 write_subrecord(struct record *r, WRBUF w, struct conf_service *service) +{ + wrbuf_printf(w, "\n", + r->client->database->url, + r->client->database->name ? r->client->database->name : ""); + write_metadata(w, service, r->metadata, 1); + wrbuf_puts(w, "\n"); +} + +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 record *r; + 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); + for (r = rec->records; r; r = r->next) + write_subrecord(r, c->wrbuf, service); + wrbuf_puts(c->wrbuf, "\n"); + rs->payload = nmem_strdup(c->nmem, wrbuf_cstr(c->wrbuf)); http_send_response(c); } @@ -276,9 +362,10 @@ static void show_records(struct http_channel *c, int active) struct http_response *rs = c->response; struct http_session *s = locate_session(rq, rs); struct record_cluster **rl; - NMEM nmem_show; + 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; @@ -296,9 +383,15 @@ static void show_records(struct http_channel *c, int active) 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"); @@ -314,42 +407,20 @@ static void show_records(struct http_channel *c, int active) struct record *p; struct record_cluster *rec = rl[i]; struct conf_service *service = global_parameters.server->service; - int imeta; wrbuf_puts(c->wrbuf, "\n"); - for (imeta = 0; imeta < service->num_metadata; imeta++) - { - struct conf_metadata *cmd = &service->metadata[imeta]; - struct record_metadata *md; - if (!rec->metadata[imeta]) - continue; - if (!cmd->brief) - continue; - for (md = rec->metadata[imeta]; md; md = md->next) - { - wrbuf_printf(c->wrbuf, "", cmd->name); - switch (cmd->type) - { - case Metadata_type_generic: - wrbuf_puts(c->wrbuf, md->data.text); - break; - default: - wrbuf_puts(c->wrbuf, "[Can't represent]"); - } - wrbuf_printf(c->wrbuf, "", cmd->name); - } - } + 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)); + rs->payload = nmem_strdup(c->nmem, wrbuf_cstr(c->wrbuf)); http_send_response(c); - nmem_destroy(nmem_show); } static void show_records_ready(void *data) @@ -402,6 +473,7 @@ static void cmd_search(struct http_channel *c) struct http_response *rs = c->response; struct http_session *s = locate_session(rq, rs); char *query = http_argbyname(rq, "query"); + char *filter = http_argbyname(rq, "filter"); char *res; if (!s) @@ -411,7 +483,7 @@ static void cmd_search(struct http_channel *c) error(rs, "417", "Must supply query", 0); return; } - res = search(s->psession, query); + res = search(s->psession, query, filter); if (res) { error(rs, "417", res, res); @@ -451,10 +523,29 @@ static void cmd_stat(struct http_channel *c) wrbuf_printf(c->wrbuf, "%d\n", stat.num_failed); wrbuf_printf(c->wrbuf, "%d\n", stat.num_error); wrbuf_puts(c->wrbuf, ""); - rs->payload = nmem_strdup(c->nmem, wrbuf_buf(c->wrbuf)); + rs->payload = nmem_strdup(c->nmem, wrbuf_cstr(c->wrbuf)); 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_cstr(c->wrbuf)); + http_send_response(c); +} struct { char *name; @@ -468,6 +559,8 @@ struct { { "termlist", cmd_termlist }, { "exit", cmd_exit }, { "ping", cmd_ping }, + { "record", cmd_record }, + { "info", cmd_info }, {0,0} }; @@ -478,6 +571,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);