cgit/ui-clone.c
Lukas Fleischer ed7e3bc822 Return proper HTTP response when accessing info/
Currently, when a user directly accesses the info command of a
repository, we exit cgit without printing anything to stdout, bringing
up error messages like "502 Bad Gateway" or "An error occurred while
reading CGI reply (no response received)". Instead of bailing out, at
least print the HTTP headers, including a reasonable error message.

Reported-by: Janus Troelsen
Signed-off-by: Lukas Fleischer <cgit@cryptocrack.de>
2015-01-15 19:48:44 +01:00

104 lines
2.2 KiB
C

/* ui-clone.c: functions for http cloning, based on
* git's http-backend.c by Shawn O. Pearce
*
* Copyright (C) 2006-2014 cgit Development Team <cgit@lists.zx2c4.com>
*
* Licensed under GNU General Public License v2
* (see COPYING for full license text)
*/
#include "cgit.h"
#include "ui-clone.h"
#include "html.h"
#include "ui-shared.h"
static int print_ref_info(const char *refname, const unsigned char *sha1,
int flags, void *cb_data)
{
struct object *obj;
if (!(obj = parse_object(sha1)))
return 0;
htmlf("%s\t%s\n", sha1_to_hex(sha1), refname);
if (obj->type == OBJ_TAG) {
if (!(obj = deref_tag(obj, refname, 0)))
return 0;
htmlf("%s\t%s^{}\n", sha1_to_hex(obj->sha1), refname);
}
return 0;
}
static void print_pack_info(void)
{
struct packed_git *pack;
int ofs;
ctx.page.mimetype = "text/plain";
ctx.page.filename = "objects/info/packs";
cgit_print_http_headers();
ofs = strlen(ctx.repo->path) + strlen("/objects/pack/");
prepare_packed_git();
for (pack = packed_git; pack; pack = pack->next)
if (pack->pack_local)
htmlf("P %s\n", pack->pack_name + ofs);
}
static void send_file(char *path)
{
struct stat st;
if (stat(path, &st)) {
switch (errno) {
case ENOENT:
html_status(404, "Not found", 0);
break;
case EACCES:
html_status(403, "Forbidden", 0);
break;
default:
html_status(400, "Bad request", 0);
}
return;
}
ctx.page.mimetype = "application/octet-stream";
ctx.page.filename = path;
if (!starts_with(ctx.repo->path, path))
ctx.page.filename += strlen(ctx.repo->path) + 1;
cgit_print_http_headers();
html_include(path);
}
void cgit_clone_info(void)
{
if (!ctx.qry.path || strcmp(ctx.qry.path, "refs")) {
html_status(400, "Bad request", 0);
return;
}
ctx.page.mimetype = "text/plain";
ctx.page.filename = "info/refs";
cgit_print_http_headers();
for_each_ref(print_ref_info, NULL);
}
void cgit_clone_objects(void)
{
if (!ctx.qry.path) {
html_status(400, "Bad request", 0);
return;
}
if (!strcmp(ctx.qry.path, "info/packs")) {
print_pack_info();
return;
}
send_file(git_path("objects/%s", ctx.qry.path));
}
void cgit_clone_head(void)
{
send_file(git_path("%s", "HEAD"));
}