summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2009-02-19 22:24:15 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2009-02-19 22:24:15 (UTC)
commit3ff58ddd51bcbcbc9b7649bad1a39aa98af4b49f (patch) (unidiff)
tree4ff1b54dfb2fc025771f39528165e9e46d93a0e0
parent488a214a81a25c6397c56822ed1713f51dddc520 (diff)
downloadcgit-3ff58ddd51bcbcbc9b7649bad1a39aa98af4b49f.zip
cgit-3ff58ddd51bcbcbc9b7649bad1a39aa98af4b49f.tar.gz
cgit-3ff58ddd51bcbcbc9b7649bad1a39aa98af4b49f.tar.bz2
Add support for HEAD requests
This is a quick 'n dirty hack which makes cgit honor HEAD requests. Signed-off-by: Lars Hjemli <hjemli@gmail.com>
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c3
-rw-r--r--ui-shared.c4
2 files changed, 7 insertions, 0 deletions
diff --git a/cgit.c b/cgit.c
index 411e421..dc1c2e4 100644
--- a/cgit.c
+++ b/cgit.c
@@ -409,79 +409,82 @@ static void cgit_parse_args(int argc, const char **argv)
409 exit(0); 409 exit(0);
410 } 410 }
411} 411}
412 412
413static int calc_ttl() 413static int calc_ttl()
414{ 414{
415 if (!ctx.repo) 415 if (!ctx.repo)
416 return ctx.cfg.cache_root_ttl; 416 return ctx.cfg.cache_root_ttl;
417 417
418 if (!ctx.qry.page) 418 if (!ctx.qry.page)
419 return ctx.cfg.cache_repo_ttl; 419 return ctx.cfg.cache_repo_ttl;
420 420
421 if (ctx.qry.has_symref) 421 if (ctx.qry.has_symref)
422 return ctx.cfg.cache_dynamic_ttl; 422 return ctx.cfg.cache_dynamic_ttl;
423 423
424 if (ctx.qry.has_sha1) 424 if (ctx.qry.has_sha1)
425 return ctx.cfg.cache_static_ttl; 425 return ctx.cfg.cache_static_ttl;
426 426
427 return ctx.cfg.cache_repo_ttl; 427 return ctx.cfg.cache_repo_ttl;
428} 428}
429 429
430int main(int argc, const char **argv) 430int main(int argc, const char **argv)
431{ 431{
432 const char *cgit_config_env = getenv("CGIT_CONFIG"); 432 const char *cgit_config_env = getenv("CGIT_CONFIG");
433 const char *method = getenv("REQUEST_METHOD");
433 const char *path; 434 const char *path;
434 char *qry; 435 char *qry;
435 int err, ttl; 436 int err, ttl;
436 437
437 prepare_context(&ctx); 438 prepare_context(&ctx);
438 cgit_repolist.length = 0; 439 cgit_repolist.length = 0;
439 cgit_repolist.count = 0; 440 cgit_repolist.count = 0;
440 cgit_repolist.repos = NULL; 441 cgit_repolist.repos = NULL;
441 442
442 if (getenv("SCRIPT_NAME")) 443 if (getenv("SCRIPT_NAME"))
443 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME")); 444 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME"));
444 if (getenv("QUERY_STRING")) 445 if (getenv("QUERY_STRING"))
445 ctx.qry.raw = xstrdup(getenv("QUERY_STRING")); 446 ctx.qry.raw = xstrdup(getenv("QUERY_STRING"));
446 cgit_parse_args(argc, argv); 447 cgit_parse_args(argc, argv);
447 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG, 448 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG,
448 config_cb); 449 config_cb);
449 ctx.repo = NULL; 450 ctx.repo = NULL;
450 http_parse_querystring(ctx.qry.raw, querystring_cb); 451 http_parse_querystring(ctx.qry.raw, querystring_cb);
451 452
452 /* If virtual-root isn't specified in cgitrc, lets pretend 453 /* If virtual-root isn't specified in cgitrc, lets pretend
453 * that virtual-root equals SCRIPT_NAME. 454 * that virtual-root equals SCRIPT_NAME.
454 */ 455 */
455 if (!ctx.cfg.virtual_root) 456 if (!ctx.cfg.virtual_root)
456 ctx.cfg.virtual_root = ctx.cfg.script_name; 457 ctx.cfg.virtual_root = ctx.cfg.script_name;
457 458
458 /* If no url parameter is specified on the querystring, lets 459 /* If no url parameter is specified on the querystring, lets
459 * use PATH_INFO as url. This allows cgit to work with virtual 460 * use PATH_INFO as url. This allows cgit to work with virtual
460 * urls without the need for rewriterules in the webserver (as 461 * urls without the need for rewriterules in the webserver (as
461 * long as PATH_INFO is included in the cache lookup key). 462 * long as PATH_INFO is included in the cache lookup key).
462 */ 463 */
463 path = getenv("PATH_INFO"); 464 path = getenv("PATH_INFO");
464 if (!ctx.qry.url && path) { 465 if (!ctx.qry.url && path) {
465 if (path[0] == '/') 466 if (path[0] == '/')
466 path++; 467 path++;
467 ctx.qry.url = xstrdup(path); 468 ctx.qry.url = xstrdup(path);
468 if (ctx.qry.raw) { 469 if (ctx.qry.raw) {
469 qry = ctx.qry.raw; 470 qry = ctx.qry.raw;
470 ctx.qry.raw = xstrdup(fmt("%s?%s", path, qry)); 471 ctx.qry.raw = xstrdup(fmt("%s?%s", path, qry));
471 free(qry); 472 free(qry);
472 } else 473 } else
473 ctx.qry.raw = ctx.qry.url; 474 ctx.qry.raw = ctx.qry.url;
474 cgit_parse_url(ctx.qry.url); 475 cgit_parse_url(ctx.qry.url);
475 } 476 }
476 477
477 ttl = calc_ttl(); 478 ttl = calc_ttl();
478 ctx.page.expires += ttl*60; 479 ctx.page.expires += ttl*60;
480 if (method && !strcmp(method, "HEAD"))
481 ctx.cfg.nocache = 1;
479 if (ctx.cfg.nocache) 482 if (ctx.cfg.nocache)
480 ctx.cfg.cache_size = 0; 483 ctx.cfg.cache_size = 0;
481 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root, 484 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root,
482 ctx.qry.raw, ttl, process_request, &ctx); 485 ctx.qry.raw, ttl, process_request, &ctx);
483 if (err) 486 if (err)
484 cgit_print_error(fmt("Error processing page: %s (%d)", 487 cgit_print_error(fmt("Error processing page: %s (%d)",
485 strerror(err), err)); 488 strerror(err), err));
486 return err; 489 return err;
487} 490}
diff --git a/ui-shared.c b/ui-shared.c
index 86a7d29..0dafc83 100644
--- a/ui-shared.c
+++ b/ui-shared.c
@@ -435,63 +435,67 @@ void cgit_print_age(time_t t, time_t max_relative, char *format)
435 secs * 1.0 / TM_HOUR); 435 secs * 1.0 / TM_HOUR);
436 return; 436 return;
437 } 437 }
438 if (secs < TM_WEEK * 2) { 438 if (secs < TM_WEEK * 2) {
439 htmlf("<span class='age-days'>%.0f days</span>", 439 htmlf("<span class='age-days'>%.0f days</span>",
440 secs * 1.0 / TM_DAY); 440 secs * 1.0 / TM_DAY);
441 return; 441 return;
442 } 442 }
443 if (secs < TM_MONTH * 2) { 443 if (secs < TM_MONTH * 2) {
444 htmlf("<span class='age-weeks'>%.0f weeks</span>", 444 htmlf("<span class='age-weeks'>%.0f weeks</span>",
445 secs * 1.0 / TM_WEEK); 445 secs * 1.0 / TM_WEEK);
446 return; 446 return;
447 } 447 }
448 if (secs < TM_YEAR * 2) { 448 if (secs < TM_YEAR * 2) {
449 htmlf("<span class='age-months'>%.0f months</span>", 449 htmlf("<span class='age-months'>%.0f months</span>",
450 secs * 1.0 / TM_MONTH); 450 secs * 1.0 / TM_MONTH);
451 return; 451 return;
452 } 452 }
453 htmlf("<span class='age-years'>%.0f years</span>", 453 htmlf("<span class='age-years'>%.0f years</span>",
454 secs * 1.0 / TM_YEAR); 454 secs * 1.0 / TM_YEAR);
455} 455}
456 456
457void cgit_print_http_headers(struct cgit_context *ctx) 457void cgit_print_http_headers(struct cgit_context *ctx)
458{ 458{
459 const char *method = getenv("REQUEST_METHOD");
460
459 if (ctx->page.mimetype && ctx->page.charset) 461 if (ctx->page.mimetype && ctx->page.charset)
460 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype, 462 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype,
461 ctx->page.charset); 463 ctx->page.charset);
462 else if (ctx->page.mimetype) 464 else if (ctx->page.mimetype)
463 htmlf("Content-Type: %s\n", ctx->page.mimetype); 465 htmlf("Content-Type: %s\n", ctx->page.mimetype);
464 if (ctx->page.size) 466 if (ctx->page.size)
465 htmlf("Content-Length: %ld\n", ctx->page.size); 467 htmlf("Content-Length: %ld\n", ctx->page.size);
466 if (ctx->page.filename) 468 if (ctx->page.filename)
467 htmlf("Content-Disposition: inline; filename=\"%s\"\n", 469 htmlf("Content-Disposition: inline; filename=\"%s\"\n",
468 ctx->page.filename); 470 ctx->page.filename);
469 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified)); 471 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified));
470 htmlf("Expires: %s\n", http_date(ctx->page.expires)); 472 htmlf("Expires: %s\n", http_date(ctx->page.expires));
471 if (ctx->page.etag) 473 if (ctx->page.etag)
472 htmlf("ETag: \"%s\"\n", ctx->page.etag); 474 htmlf("ETag: \"%s\"\n", ctx->page.etag);
473 html("\n"); 475 html("\n");
476 if (method && !strcmp(method, "HEAD"))
477 exit(0);
474} 478}
475 479
476void cgit_print_docstart(struct cgit_context *ctx) 480void cgit_print_docstart(struct cgit_context *ctx)
477{ 481{
478 char *host = cgit_hosturl(); 482 char *host = cgit_hosturl();
479 html(cgit_doctype); 483 html(cgit_doctype);
480 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n"); 484 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n");
481 html("<head>\n"); 485 html("<head>\n");
482 html("<title>"); 486 html("<title>");
483 html_txt(ctx->page.title); 487 html_txt(ctx->page.title);
484 html("</title>\n"); 488 html("</title>\n");
485 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version); 489 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version);
486 if (ctx->cfg.robots && *ctx->cfg.robots) 490 if (ctx->cfg.robots && *ctx->cfg.robots)
487 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots); 491 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots);
488 html("<link rel='stylesheet' type='text/css' href='"); 492 html("<link rel='stylesheet' type='text/css' href='");
489 html_attr(ctx->cfg.css); 493 html_attr(ctx->cfg.css);
490 html("'/>\n"); 494 html("'/>\n");
491 if (ctx->cfg.favicon) { 495 if (ctx->cfg.favicon) {
492 html("<link rel='shortcut icon' href='"); 496 html("<link rel='shortcut icon' href='");
493 html_attr(ctx->cfg.favicon); 497 html_attr(ctx->cfg.favicon);
494 html("'/>\n"); 498 html("'/>\n");
495 } 499 }
496 if (host && ctx->repo) { 500 if (host && ctx->repo) {
497 html("<link rel='alternate' title='Atom feed' href='http://"); 501 html("<link rel='alternate' title='Atom feed' href='http://");