blob: 66cb19cdbd1058e37e60157c40a18d1a782bb3ff [file] [log] [blame]
Lars Hjemlie5da4bc2008-08-06 10:53:50 +02001/* ui-plain.c: functions for output of plain blobs by path
2 *
3 * Copyright (C) 2008 Lars Hjemli
4 *
5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text)
7 */
8
9#include "cgit.h"
10#include "html.h"
11#include "ui-shared.h"
12
13char *curr_rev;
14char *match_path;
15int match;
16
17static void print_object(const unsigned char *sha1, const char *path)
18{
19 enum object_type type;
Lars Hjemlic4d46c72009-02-13 20:43:30 +010020 char *buf, *ext;
Ramsay Jonesbdd4a562008-11-04 19:22:08 +000021 unsigned long size;
Lars Hjemlic4d46c72009-02-13 20:43:30 +010022 struct string_list_item *mime;
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020023
24 type = sha1_object_info(sha1, &size);
25 if (type == OBJ_BAD) {
Lars Hjemli885096c2008-08-06 22:57:44 +020026 html_status(404, "Not found", 0);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020027 return;
28 }
29
30 buf = read_sha1_file(sha1, &type, &size);
31 if (!buf) {
Lars Hjemli885096c2008-08-06 22:57:44 +020032 html_status(404, "Not found", 0);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020033 return;
34 }
Lars Hjemlic4d46c72009-02-13 20:43:30 +010035 ctx.page.mimetype = NULL;
36 ext = strrchr(path, '.');
37 if (ext && *(++ext)) {
38 mime = string_list_lookup(ext, &ctx.cfg.mimetypes);
39 if (mime)
40 ctx.page.mimetype = (char *)mime->util;
41 }
42 if (!ctx.page.mimetype) {
43 if (buffer_is_binary(buf, size))
44 ctx.page.mimetype = "application/octet-stream";
45 else
46 ctx.page.mimetype = "text/plain";
47 }
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020048 ctx.page.filename = fmt("%s", path);
49 ctx.page.size = size;
Lars Hjemli488a2142009-02-19 22:38:36 +010050 ctx.page.etag = sha1_to_hex(sha1);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020051 cgit_print_http_headers(&ctx);
52 html_raw(buf, size);
53 match = 1;
54}
55
56static int walk_tree(const unsigned char *sha1, const char *base, int baselen,
57 const char *pathname, unsigned mode, int stage,
58 void *cbdata)
59{
Lars Hjemli135b2312008-09-02 00:53:04 +020060 if (S_ISDIR(mode))
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020061 return READ_TREE_RECURSIVE;
62
Lars Hjemlib5072522009-09-06 19:33:10 +020063 if (S_ISREG(mode) && !strncmp(base, match_path, baselen) &&
64 !strcmp(pathname, match_path + baselen))
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020065 print_object(sha1, pathname);
66
67 return 0;
68}
69
70void cgit_print_plain(struct cgit_context *ctx)
71{
72 const char *rev = ctx->qry.sha1;
73 unsigned char sha1[20];
74 struct commit *commit;
75 const char *paths[] = {ctx->qry.path, NULL};
76
77 if (!rev)
78 rev = ctx->qry.head;
79
80 curr_rev = xstrdup(rev);
81 if (get_sha1(rev, sha1)) {
Lars Hjemli885096c2008-08-06 22:57:44 +020082 html_status(404, "Not found", 0);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020083 return;
84 }
85 commit = lookup_commit_reference(sha1);
86 if (!commit || parse_commit(commit)) {
Lars Hjemli885096c2008-08-06 22:57:44 +020087 html_status(404, "Not found", 0);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020088 return;
89 }
90 match_path = ctx->qry.path;
Martins Polakovs24538b02009-12-10 17:34:42 +015991 read_tree_recursive(commit->tree, "", 0, 0, paths, walk_tree, NULL);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020092 if (!match)
Lars Hjemli885096c2008-08-06 22:57:44 +020093 html_status(404, "Not found", 0);
Lars Hjemlie5da4bc2008-08-06 10:53:50 +020094}