]>
git.cameronkatri.com Git - cgit.git/blob - ui-plain.c
1 /* ui-plain.c: functions for output of plain blobs by path
3 * Copyright (C) 2008 Lars Hjemli
5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text)
11 #include "ui-shared.h"
16 static void print_object(const unsigned char *sha1
, const char *path
)
18 enum object_type type
;
21 struct string_list_item
*mime
;
23 type
= sha1_object_info(sha1
, &size
);
24 if (type
== OBJ_BAD
) {
25 html_status(404, "Not found", 0);
29 buf
= read_sha1_file(sha1
, &type
, &size
);
31 html_status(404, "Not found", 0);
34 ctx
.page
.mimetype
= NULL
;
35 ext
= strrchr(path
, '.');
36 if (ext
&& *(++ext
)) {
37 mime
= string_list_lookup(&ctx
.cfg
.mimetypes
, ext
);
39 ctx
.page
.mimetype
= (char *)mime
->util
;
41 if (!ctx
.page
.mimetype
) {
42 if (buffer_is_binary(buf
, size
))
43 ctx
.page
.mimetype
= "application/octet-stream";
45 ctx
.page
.mimetype
= "text/plain";
47 ctx
.page
.filename
= fmt("%s", path
);
49 ctx
.page
.etag
= sha1_to_hex(sha1
);
50 cgit_print_http_headers(&ctx
);
55 static char *buildpath(const char *base
, int baselen
, const char *path
)
58 return fmt("%.*s%s/", baselen
, base
, path
);
60 return fmt("%.*s/", baselen
, base
);
63 static void print_dir(const unsigned char *sha1
, const char *base
,
64 int baselen
, const char *path
)
66 char *fullpath
, *slash
;
69 fullpath
= buildpath(base
, baselen
, path
);
70 slash
= (fullpath
[0] == '/' ? "" : "/");
71 ctx
.page
.etag
= sha1_to_hex(sha1
);
72 cgit_print_http_headers(&ctx
);
73 htmlf("<html><head><title>%s", slash
);
75 htmlf("</title></head>\n<body>\n<h2>%s", slash
);
77 html("</h2>\n<ul>\n");
78 len
= strlen(fullpath
);
80 fullpath
[len
- 1] = 0;
81 slash
= strrchr(fullpath
, '/');
87 cgit_plain_link("../", NULL
, NULL
, ctx
.qry
.head
, ctx
.qry
.sha1
,
94 static void print_dir_entry(const unsigned char *sha1
, const char *base
,
95 int baselen
, const char *path
, unsigned mode
)
99 fullpath
= buildpath(base
, baselen
, path
);
101 fullpath
[strlen(fullpath
) - 1] = 0;
103 cgit_plain_link(path
, NULL
, NULL
, ctx
.qry
.head
, ctx
.qry
.sha1
,
109 static void print_dir_tail(void)
111 html(" </ul>\n</body></html>\n");
114 static int walk_tree(const unsigned char *sha1
, const char *base
, int baselen
,
115 const char *pathname
, unsigned mode
, int stage
,
118 if (baselen
== match_baselen
) {
120 print_object(sha1
, pathname
);
121 else if (S_ISDIR(mode
)) {
122 print_dir(sha1
, base
, baselen
, pathname
);
123 return READ_TREE_RECURSIVE
;
126 else if (baselen
> match_baselen
)
127 print_dir_entry(sha1
, base
, baselen
, pathname
, mode
);
128 else if (S_ISDIR(mode
))
129 return READ_TREE_RECURSIVE
;
134 static int basedir_len(const char *path
)
136 char *p
= strrchr(path
, '/');
142 void cgit_print_plain(struct cgit_context
*ctx
)
144 const char *rev
= ctx
->qry
.sha1
;
145 unsigned char sha1
[20];
146 struct commit
*commit
;
147 const char *paths
[] = {ctx
->qry
.path
, NULL
};
152 if (get_sha1(rev
, sha1
)) {
153 html_status(404, "Not found", 0);
156 commit
= lookup_commit_reference(sha1
);
157 if (!commit
|| parse_commit(commit
)) {
158 html_status(404, "Not found", 0);
164 print_dir(commit
->tree
->object
.sha1
, "", 0, "");
167 match_baselen
= basedir_len(paths
[0]);
168 read_tree_recursive(commit
->tree
, "", 0, 0, paths
, walk_tree
, NULL
);
170 html_status(404, "Not found", 0);