1 /* This file is part of the YAZ toolkit.
2 * Copyright (C) 1995-2013 Index Data
3 * See the file LICENSE for details.
7 * \brief Implements HTTP decoding
14 #include <yaz/yaz-version.h>
15 #include <yaz/yaz-iconv.h>
16 #include <yaz/matchstr.h>
18 #include <yaz/base64.h>
21 #define strncasecmp _strnicmp
22 #define strcasecmp _stricmp
25 static int decode_headers_content(ODR o, int off, Z_HTTP_Header **headers,
26 char **content_buf, int *content_len)
32 while (i < o->size-1 && o->buf[i] == '\n')
36 if (o->buf[i] == '\r' && i < o->size-1 && o->buf[i+1] == '\n')
41 if (o->buf[i] == '\n')
50 else if (o->buf[i] == ':')
53 *headers = (Z_HTTP_Header *) odr_malloc(o, sizeof(**headers));
54 (*headers)->name = (char*) odr_malloc(o, i - po + 1);
55 memcpy ((*headers)->name, o->buf + po, i - po);
56 (*headers)->name[i - po] = '\0';
58 while (i < o->size-1 && o->buf[i] == ' ')
60 for (po = i; i < o->size-1 && !strchr("\r\n", o->buf[i]); i++)
63 (*headers)->value = (char*) odr_malloc(o, i - po + 1);
64 memcpy ((*headers)->value, o->buf + po, i - po);
65 (*headers)->value[i - po] = '\0';
67 if (!strcasecmp((*headers)->name, "Transfer-Encoding")
69 !strcasecmp((*headers)->value, "chunked"))
71 headers = &(*headers)->next;
72 if (i < o->size-1 && o->buf[i] == '\r')
76 if (o->buf[i] != '\n')
87 /* we know buffer will be smaller than o->size - i*/
88 *content_buf = (char*) odr_malloc(o, o->size - i);
94 for (; i < o->size-2; i++)
95 if (yaz_isdigit(o->buf[i]))
96 chunk_len = chunk_len * 16 +
98 else if (yaz_isupper(o->buf[i]))
99 chunk_len = chunk_len * 16 +
100 (o->buf[i] - ('A'-10));
101 else if (yaz_islower(o->buf[i]))
102 chunk_len = chunk_len * 16 +
103 (o->buf[i] - ('a'-10));
106 /* chunk extension ... */
107 while (o->buf[i] != '\r' && o->buf[i+1] != '\n')
116 i += 2; /* skip CRLF */
119 if (chunk_len < 0 || off + chunk_len > o->size)
125 memcpy (*content_buf + off, o->buf + i, chunk_len);
126 i += chunk_len + 2; /* skip chunk+CRLF */
140 else if (i == o->size)
147 *content_len = o->size - i;
148 *content_buf = (char*) odr_malloc(o, *content_len + 1);
149 memcpy(*content_buf, o->buf + i, *content_len);
150 (*content_buf)[*content_len] = '\0';
156 void z_HTTP_header_add_content_type(ODR o, Z_HTTP_Header **hp,
157 const char *content_type,
160 const char *l = "Content-Type";
163 char *ctype = (char *)
164 odr_malloc(o, strlen(content_type)+strlen(charset) + 15);
165 sprintf(ctype, "%s; charset=%s", content_type, charset);
166 z_HTTP_header_add(o, hp, l, ctype);
169 z_HTTP_header_add(o, hp, l, content_type);
174 * HTTP Basic authentication is described at:
175 * http://tools.ietf.org/html/rfc1945#section-11.1
177 void z_HTTP_header_add_basic_auth(ODR o, Z_HTTP_Header **hp,
178 const char *username, const char *password)
188 len = strlen(username) + strlen(password);
189 tmp = (char *) odr_malloc(o, len+2);
190 sprintf(tmp, "%s:%s", username, password);
191 buf = (char *) odr_malloc(o, (len+1) * 8/6 + 12);
192 strcpy(buf, "Basic ");
193 yaz_base64encode(tmp, &buf[strlen(buf)]);
194 z_HTTP_header_add(o, hp, "Authorization", buf);
198 void z_HTTP_header_add(ODR o, Z_HTTP_Header **hp, const char *n,
203 *hp = (Z_HTTP_Header *) odr_malloc(o, sizeof(**hp));
204 (*hp)->name = odr_strdup(o, n);
205 (*hp)->value = odr_strdup(o, v);
211 static void z_HTTP_header_set(ODR o, Z_HTTP_Header **hp, const char *n,
216 if (!strcmp((*hp)->name, n))
218 (*hp)->value = odr_strdup(o, v);
223 *hp = (Z_HTTP_Header *) odr_malloc(o, sizeof(**hp));
224 (*hp)->name = odr_strdup(o, n);
225 (*hp)->value = odr_strdup(o, v);
230 const char *z_HTTP_header_lookup(const Z_HTTP_Header *hp, const char *n)
232 for (; hp; hp = hp->next)
233 if (!yaz_matchstr(hp->name, n))
239 Z_GDU *z_get_HTTP_Request(ODR o)
241 Z_GDU *p = (Z_GDU *) odr_malloc(o, sizeof(*p));
242 Z_HTTP_Request *hreq;
244 p->which = Z_GDU_HTTP_Request;
245 p->u.HTTP_Request = (Z_HTTP_Request *) odr_malloc(o, sizeof(*hreq));
246 hreq = p->u.HTTP_Request;
248 hreq->content_len = 0;
249 hreq->content_buf = 0;
250 hreq->version = "1.1";
251 hreq->method = "POST";
253 z_HTTP_header_add(o, &hreq->headers, "User-Agent", "YAZ/" YAZ_VERSION);
258 Z_GDU *z_get_HTTP_Request_host_path(ODR odr,
262 Z_GDU *p = z_get_HTTP_Request(odr);
264 p->u.HTTP_Request->path = odr_strdup(odr, path);
268 const char *cp0 = strstr(host, "://");
275 cp1 = strchr(cp0, '/');
277 cp1 = cp0+strlen(cp0);
281 char *h = (char*) odr_malloc(odr, cp1 - cp0 + 1);
282 memcpy (h, cp0, cp1 - cp0);
284 z_HTTP_header_add(odr, &p->u.HTTP_Request->headers,
291 Z_GDU *z_get_HTTP_Request_uri(ODR odr, const char *uri, const char *args,
294 Z_GDU *p = z_get_HTTP_Request(odr);
295 const char *cp0 = strstr(uri, "://");
302 cp1 = strchr(cp0, '/');
304 cp1 = cp0+strlen(cp0);
308 char *h = (char*) odr_malloc(odr, cp1 - cp0 + 1);
309 memcpy (h, cp0, cp1 - cp0);
311 z_HTTP_header_add(odr, &p->u.HTTP_Request->headers,
322 p->u.HTTP_Request->path = odr_malloc(odr, cp1 - uri + strlen(args) + 2);
325 memcpy(p->u.HTTP_Request->path, uri, cp1 - uri);
326 strcpy(p->u.HTTP_Request->path + (cp1 - uri), "/");
329 strcpy(p->u.HTTP_Request->path, "/");
330 strcat(p->u.HTTP_Request->path, args);
334 Z_GDU *z_get_HTTP_Response(ODR o, int code)
336 Z_GDU *p = (Z_GDU *) odr_malloc(o, sizeof(*p));
337 Z_HTTP_Response *hres;
339 p->which = Z_GDU_HTTP_Response;
340 p->u.HTTP_Response = (Z_HTTP_Response *) odr_malloc(o, sizeof(*hres));
341 hres = p->u.HTTP_Response;
343 hres->content_len = 0;
344 hres->content_buf = 0;
346 hres->version = "1.1";
347 z_HTTP_header_add(o, &hres->headers, "Server",
351 hres->content_buf = (char*) odr_malloc(o, 400);
352 sprintf(hres->content_buf,
353 "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01//EN\""
354 " \"http://www.w3.org/TR/html4/strict.dtd\">\n"
357 " <TITLE>YAZ " YAZ_VERSION "</TITLE>\n"
360 " <P><A HREF=\"http://www.indexdata.com/yaz/\">YAZ</A> "
362 " <P>Error: %d</P>\n"
363 " <P>Description: %.50s</P>\n"
366 code, z_HTTP_errmsg(code));
367 hres->content_len = strlen(hres->content_buf);
368 z_HTTP_header_add(o, &hres->headers, "Content-Type", "text/html");
373 const char *z_HTTP_errmsg(int code)
377 else if (code == 400)
378 return "Bad Request";
379 else if (code == 404)
381 else if (code == 405)
382 return "Method Not Allowed";
383 else if (code == 500)
384 return "Internal Error";
386 return "Unknown Error";
389 int yaz_decode_http_response(ODR o, Z_HTTP_Response **hr_p)
392 Z_HTTP_Response *hr = (Z_HTTP_Response *) odr_malloc(o, sizeof(*hr));
399 while (i < o->size-2 && !strchr(" \r\n", o->buf[i]))
401 hr->version = (char *) odr_malloc(o, i - po + 1);
403 memcpy(hr->version, o->buf + po, i - po);
404 hr->version[i-po] = 0;
405 if (o->buf[i] != ' ')
412 while (i < o->size-2 && o->buf[i] >= '0' && o->buf[i] <= '9')
414 hr->code = hr->code*10 + (o->buf[i] - '0');
417 while (i < o->size-1 && o->buf[i] != '\n')
419 return decode_headers_content(o, i, &hr->headers,
420 &hr->content_buf, &hr->content_len);
423 int yaz_decode_http_request(ODR o, Z_HTTP_Request **hr_p)
426 Z_HTTP_Request *hr = (Z_HTTP_Request *) odr_malloc(o, sizeof(*hr));
431 for (i = 0; o->buf[i] != ' '; i++)
432 if (i >= o->size-5 || i > 30)
437 hr->method = (char *) odr_malloc(o, i+1);
438 memcpy (hr->method, o->buf, i);
439 hr->method[i] = '\0';
442 for (i = po; o->buf[i] != ' '; i++)
448 hr->path = (char *) odr_malloc(o, i - po + 1);
449 memcpy (hr->path, o->buf+po, i - po);
450 hr->path[i - po] = '\0';
453 if (i > o->size-5 || memcmp(o->buf+i, "HTTP/", 5))
460 while (i < o->size && !strchr("\r\n", o->buf[i]))
462 hr->version = (char *) odr_malloc(o, i - po + 1);
463 memcpy(hr->version, o->buf + po, i - po);
464 hr->version[i - po] = '\0';
466 if (i < o->size-1 && o->buf[i] == '\r')
468 if (o->buf[i] != '\n')
473 return decode_headers_content(o, i, &hr->headers,
474 &hr->content_buf, &hr->content_len);
477 static void dump_http_package(ODR o, const char *buf, size_t len)
484 odr_printf(o, "%.*s\n", i, buf);
489 odr_printf(o, "%.*s\n", i, buf);
490 odr_printf(o, "(truncated\n", (long) len);
493 else if (buf[i] == 0)
495 odr_printf(o, "%.*s\n", i, buf);
496 odr_printf(o, "(binary data)\n", (long) len);
502 int yaz_encode_http_response(ODR o, Z_HTTP_Response *hr)
508 sprintf(sbuf, "HTTP/%s %d %s\r\n", hr->version,
510 z_HTTP_errmsg(hr->code));
511 odr_write2(o, sbuf, strlen(sbuf));
512 /* apply Content-Length if not already applied */
513 if (!z_HTTP_header_lookup(hr->headers,
517 sprintf(lstr, "Content-Length: %d\r\n",
519 odr_write2(o, lstr, strlen(lstr));
521 for (h = hr->headers; h; h = h->next)
523 odr_write2(o, h->name, strlen(h->name));
524 odr_write2(o, ": ", 2);
525 odr_write2(o, h->value, strlen(h->value));
526 odr_write2(o, "\r\n", 2);
528 odr_write(o, (unsigned char *) "\r\n", 2);
530 odr_write2(o, hr->content_buf, hr->content_len);
531 if (o->direction == ODR_PRINT)
533 odr_printf(o, "-- HTTP response:\n");
534 dump_http_package(o, (const char *) o->buf + top0, o->top - top0);
535 odr_printf(o, "--\n");
540 int yaz_encode_http_request(ODR o, Z_HTTP_Request *hr)
545 odr_write2(o, hr->method, strlen(hr->method));
546 odr_write2(o, " ", 1);
547 odr_write2(o, hr->path, strlen(hr->path));
548 odr_write2(o, " HTTP/", 6);
549 odr_write2(o, hr->version, strlen(hr->version));
550 odr_write2(o, "\r\n", 2);
551 if (hr->content_len &&
552 !z_HTTP_header_lookup(hr->headers,
556 sprintf(lstr, "Content-Length: %d\r\n",
558 odr_write2(o, lstr, strlen(lstr));
560 for (h = hr->headers; h; h = h->next)
562 odr_write2(o, h->name, strlen(h->name));
563 odr_write2(o, ": ", 2);
564 odr_write2(o, h->value, strlen(h->value));
565 odr_write2(o, "\r\n", 2);
567 odr_write2(o, "\r\n", 2);
569 odr_write2(o, hr->content_buf, hr->content_len);
570 if (o->direction == ODR_PRINT)
572 odr_printf(o, "-- HTTP request:\n");
573 dump_http_package(o, (const char *) o->buf + top0, o->top - top0);
574 odr_printf(o, "--\n");
582 * c-file-style: "Stroustrup"
583 * indent-tabs-mode: nil
585 * vim: shiftwidth=4 tabstop=8 expandtab