summaryrefslogtreecommitdiffabout
path: root/cgit.c
Unidiff
Diffstat (limited to 'cgit.c') (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c10
1 files changed, 6 insertions, 4 deletions
diff --git a/cgit.c b/cgit.c
index 11dff76..d9a03c2 100644
--- a/cgit.c
+++ b/cgit.c
@@ -1,247 +1,249 @@
1/* cgit.c: cgi for the git scm 1/* cgit.c: cgi for the git scm
2 * 2 *
3 * Copyright (C) 2006 Lars Hjemli 3 * Copyright (C) 2006 Lars Hjemli
4 * 4 *
5 * Licensed under GNU General Public License v2 5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text) 6 * (see COPYING for full license text)
7 */ 7 */
8 8
9#include "cgit.h" 9#include "cgit.h"
10 10
11static int cgit_prepare_cache(struct cacheitem *item) 11static int cgit_prepare_cache(struct cacheitem *item)
12{ 12{
13 if (!cgit_repo && cgit_query_repo) { 13 if (!cgit_repo && cgit_query_repo) {
14 char *title = fmt("%s - %s", cgit_root_title, "Bad request"); 14 char *title = fmt("%s - %s", cgit_root_title, "Bad request");
15 cgit_print_docstart(title, item); 15 cgit_print_docstart(title, item);
16 cgit_print_pageheader(title, 0); 16 cgit_print_pageheader(title, 0);
17 cgit_print_error(fmt("Unknown repo: %s", cgit_query_repo)); 17 cgit_print_error(fmt("Unknown repo: %s", cgit_query_repo));
18 cgit_print_docend(); 18 cgit_print_docend();
19 return 0; 19 return 0;
20 } 20 }
21 21
22 if (!cgit_repo) { 22 if (!cgit_repo) {
23 item->name = xstrdup(fmt("%s/index.html", cgit_cache_root)); 23 item->name = xstrdup(fmt("%s/index.html", cgit_cache_root));
24 item->ttl = cgit_cache_root_ttl; 24 item->ttl = cgit_cache_root_ttl;
25 return 1; 25 return 1;
26 } 26 }
27 27
28 if (!cgit_cmd) { 28 if (!cgit_cmd) {
29 item->name = xstrdup(fmt("%s/%s/index.html", cgit_cache_root, 29 item->name = xstrdup(fmt("%s/%s/index.%s.html", cgit_cache_root,
30 cache_safe_filename(cgit_repo->url))); 30 cache_safe_filename(cgit_repo->url),
31 cache_safe_filename(cgit_querystring)));
31 item->ttl = cgit_cache_repo_ttl; 32 item->ttl = cgit_cache_repo_ttl;
32 } else { 33 } else {
33 item->name = xstrdup(fmt("%s/%s/%s/%s.html", cgit_cache_root, 34 item->name = xstrdup(fmt("%s/%s/%s/%s.html", cgit_cache_root,
34 cache_safe_filename(cgit_repo->url), cgit_query_page, 35 cache_safe_filename(cgit_repo->url),
35 cache_safe_filename(cgit_querystring))); 36 cgit_query_page,
37 cache_safe_filename(cgit_querystring)));
36 if (cgit_query_has_symref) 38 if (cgit_query_has_symref)
37 item->ttl = cgit_cache_dynamic_ttl; 39 item->ttl = cgit_cache_dynamic_ttl;
38 else if (cgit_query_has_sha1) 40 else if (cgit_query_has_sha1)
39 item->ttl = cgit_cache_static_ttl; 41 item->ttl = cgit_cache_static_ttl;
40 else 42 else
41 item->ttl = cgit_cache_repo_ttl; 43 item->ttl = cgit_cache_repo_ttl;
42 } 44 }
43 return 1; 45 return 1;
44} 46}
45 47
46static void cgit_print_repo_page(struct cacheitem *item) 48static void cgit_print_repo_page(struct cacheitem *item)
47{ 49{
48 char *title; 50 char *title;
49 int show_search; 51 int show_search;
50 52
51 if (!cgit_query_head) 53 if (!cgit_query_head)
52 cgit_query_head = cgit_repo->defbranch; 54 cgit_query_head = cgit_repo->defbranch;
53 55
54 if (chdir(cgit_repo->path)) { 56 if (chdir(cgit_repo->path)) {
55 title = fmt("%s - %s", cgit_root_title, "Bad request"); 57 title = fmt("%s - %s", cgit_root_title, "Bad request");
56 cgit_print_docstart(title, item); 58 cgit_print_docstart(title, item);
57 cgit_print_pageheader(title, 0); 59 cgit_print_pageheader(title, 0);
58 cgit_print_error(fmt("Unable to scan repository: %s", 60 cgit_print_error(fmt("Unable to scan repository: %s",
59 strerror(errno))); 61 strerror(errno)));
60 cgit_print_docend(); 62 cgit_print_docend();
61 return; 63 return;
62 } 64 }
63 65
64 title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc); 66 title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc);
65 show_search = 0; 67 show_search = 0;
66 setenv("GIT_DIR", cgit_repo->path, 1); 68 setenv("GIT_DIR", cgit_repo->path, 1);
67 69
68 if ((cgit_cmd == CMD_SNAPSHOT) && cgit_repo->snapshots) { 70 if ((cgit_cmd == CMD_SNAPSHOT) && cgit_repo->snapshots) {
69 cgit_print_snapshot(item, cgit_query_sha1, "zip", 71 cgit_print_snapshot(item, cgit_query_sha1, "zip",
70 cgit_repo->url, cgit_query_name); 72 cgit_repo->url, cgit_query_name);
71 return; 73 return;
72 } 74 }
73 75
74 if (cgit_cmd == CMD_BLOB) { 76 if (cgit_cmd == CMD_BLOB) {
75 cgit_print_blob(item, cgit_query_sha1, cgit_query_path); 77 cgit_print_blob(item, cgit_query_sha1, cgit_query_path);
76 return; 78 return;
77 } 79 }
78 80
79 show_search = (cgit_cmd == CMD_LOG); 81 show_search = (cgit_cmd == CMD_LOG);
80 cgit_print_docstart(title, item); 82 cgit_print_docstart(title, item);
81 if (!cgit_cmd) { 83 if (!cgit_cmd) {
82 cgit_print_pageheader("summary", show_search); 84 cgit_print_pageheader("summary", show_search);
83 cgit_print_summary(); 85 cgit_print_summary();
84 cgit_print_docend(); 86 cgit_print_docend();
85 return; 87 return;
86 } 88 }
87 89
88 cgit_print_pageheader(cgit_query_page, show_search); 90 cgit_print_pageheader(cgit_query_page, show_search);
89 91
90 switch(cgit_cmd) { 92 switch(cgit_cmd) {
91 case CMD_LOG: 93 case CMD_LOG:
92 cgit_print_log(cgit_query_sha1, cgit_query_ofs, 94 cgit_print_log(cgit_query_sha1, cgit_query_ofs,
93 cgit_max_commit_count, cgit_query_search, 95 cgit_max_commit_count, cgit_query_search,
94 cgit_query_path, 1); 96 cgit_query_path, 1);
95 break; 97 break;
96 case CMD_TREE: 98 case CMD_TREE:
97 cgit_print_tree(cgit_query_sha1, cgit_query_path); 99 cgit_print_tree(cgit_query_sha1, cgit_query_path);
98 break; 100 break;
99 case CMD_COMMIT: 101 case CMD_COMMIT:
100 cgit_print_commit(cgit_query_sha1); 102 cgit_print_commit(cgit_query_sha1);
101 break; 103 break;
102 case CMD_DIFF: 104 case CMD_DIFF:
103 cgit_print_diff(cgit_query_sha1, cgit_query_sha2); 105 cgit_print_diff(cgit_query_sha1, cgit_query_sha2);
104 break; 106 break;
105 default: 107 default:
106 cgit_print_error("Invalid request"); 108 cgit_print_error("Invalid request");
107 } 109 }
108 cgit_print_docend(); 110 cgit_print_docend();
109} 111}
110 112
111static void cgit_fill_cache(struct cacheitem *item, int use_cache) 113static void cgit_fill_cache(struct cacheitem *item, int use_cache)
112{ 114{
113 static char buf[PATH_MAX]; 115 static char buf[PATH_MAX];
114 int stdout2; 116 int stdout2;
115 117
116 getcwd(buf, sizeof(buf)); 118 getcwd(buf, sizeof(buf));
117 item->st.st_mtime = time(NULL); 119 item->st.st_mtime = time(NULL);
118 120
119 if (use_cache) { 121 if (use_cache) {
120 stdout2 = chk_positive(dup(STDOUT_FILENO), 122 stdout2 = chk_positive(dup(STDOUT_FILENO),
121 "Preserving STDOUT"); 123 "Preserving STDOUT");
122 chk_zero(close(STDOUT_FILENO), "Closing STDOUT"); 124 chk_zero(close(STDOUT_FILENO), "Closing STDOUT");
123 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)"); 125 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)");
124 } 126 }
125 127
126 if (cgit_repo) 128 if (cgit_repo)
127 cgit_print_repo_page(item); 129 cgit_print_repo_page(item);
128 else 130 else
129 cgit_print_repolist(item); 131 cgit_print_repolist(item);
130 132
131 if (use_cache) { 133 if (use_cache) {
132 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT"); 134 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT");
133 chk_positive(dup2(stdout2, STDOUT_FILENO), 135 chk_positive(dup2(stdout2, STDOUT_FILENO),
134 "Restoring original STDOUT"); 136 "Restoring original STDOUT");
135 chk_zero(close(stdout2), "Closing temporary STDOUT"); 137 chk_zero(close(stdout2), "Closing temporary STDOUT");
136 } 138 }
137 139
138 chdir(buf); 140 chdir(buf);
139} 141}
140 142
141static void cgit_check_cache(struct cacheitem *item) 143static void cgit_check_cache(struct cacheitem *item)
142{ 144{
143 int i = 0; 145 int i = 0;
144 146
145 top: 147 top:
146 if (++i > cgit_max_lock_attempts) { 148 if (++i > cgit_max_lock_attempts) {
147 die("cgit_refresh_cache: unable to lock %s: %s", 149 die("cgit_refresh_cache: unable to lock %s: %s",
148 item->name, strerror(errno)); 150 item->name, strerror(errno));
149 } 151 }
150 if (!cache_exist(item)) { 152 if (!cache_exist(item)) {
151 if (!cache_lock(item)) { 153 if (!cache_lock(item)) {
152 sleep(1); 154 sleep(1);
153 goto top; 155 goto top;
154 } 156 }
155 if (!cache_exist(item)) { 157 if (!cache_exist(item)) {
156 cgit_fill_cache(item, 1); 158 cgit_fill_cache(item, 1);
157 cache_unlock(item); 159 cache_unlock(item);
158 } else { 160 } else {
159 cache_cancel_lock(item); 161 cache_cancel_lock(item);
160 } 162 }
161 } else if (cache_expired(item) && cache_lock(item)) { 163 } else if (cache_expired(item) && cache_lock(item)) {
162 if (cache_expired(item)) { 164 if (cache_expired(item)) {
163 cgit_fill_cache(item, 1); 165 cgit_fill_cache(item, 1);
164 cache_unlock(item); 166 cache_unlock(item);
165 } else { 167 } else {
166 cache_cancel_lock(item); 168 cache_cancel_lock(item);
167 } 169 }
168 } 170 }
169} 171}
170 172
171static void cgit_print_cache(struct cacheitem *item) 173static void cgit_print_cache(struct cacheitem *item)
172{ 174{
173 static char buf[4096]; 175 static char buf[4096];
174 ssize_t i; 176 ssize_t i;
175 177
176 int fd = open(item->name, O_RDONLY); 178 int fd = open(item->name, O_RDONLY);
177 if (fd<0) 179 if (fd<0)
178 die("Unable to open cached file %s", item->name); 180 die("Unable to open cached file %s", item->name);
179 181
180 while((i=read(fd, buf, sizeof(buf))) > 0) 182 while((i=read(fd, buf, sizeof(buf))) > 0)
181 write(STDOUT_FILENO, buf, i); 183 write(STDOUT_FILENO, buf, i);
182 184
183 close(fd); 185 close(fd);
184} 186}
185 187
186static void cgit_parse_args(int argc, const char **argv) 188static void cgit_parse_args(int argc, const char **argv)
187{ 189{
188 int i; 190 int i;
189 191
190 for (i = 1; i < argc; i++) { 192 for (i = 1; i < argc; i++) {
191 if (!strncmp(argv[i], "--cache=", 8)) { 193 if (!strncmp(argv[i], "--cache=", 8)) {
192 cgit_cache_root = xstrdup(argv[i]+8); 194 cgit_cache_root = xstrdup(argv[i]+8);
193 } 195 }
194 if (!strcmp(argv[i], "--nocache")) { 196 if (!strcmp(argv[i], "--nocache")) {
195 cgit_nocache = 1; 197 cgit_nocache = 1;
196 } 198 }
197 if (!strncmp(argv[i], "--query=", 8)) { 199 if (!strncmp(argv[i], "--query=", 8)) {
198 cgit_querystring = xstrdup(argv[i]+8); 200 cgit_querystring = xstrdup(argv[i]+8);
199 } 201 }
200 if (!strncmp(argv[i], "--repo=", 7)) { 202 if (!strncmp(argv[i], "--repo=", 7)) {
201 cgit_query_repo = xstrdup(argv[i]+7); 203 cgit_query_repo = xstrdup(argv[i]+7);
202 } 204 }
203 if (!strncmp(argv[i], "--page=", 7)) { 205 if (!strncmp(argv[i], "--page=", 7)) {
204 cgit_query_page = xstrdup(argv[i]+7); 206 cgit_query_page = xstrdup(argv[i]+7);
205 } 207 }
206 if (!strncmp(argv[i], "--head=", 7)) { 208 if (!strncmp(argv[i], "--head=", 7)) {
207 cgit_query_head = xstrdup(argv[i]+7); 209 cgit_query_head = xstrdup(argv[i]+7);
208 cgit_query_has_symref = 1; 210 cgit_query_has_symref = 1;
209 } 211 }
210 if (!strncmp(argv[i], "--sha1=", 7)) { 212 if (!strncmp(argv[i], "--sha1=", 7)) {
211 cgit_query_sha1 = xstrdup(argv[i]+7); 213 cgit_query_sha1 = xstrdup(argv[i]+7);
212 cgit_query_has_sha1 = 1; 214 cgit_query_has_sha1 = 1;
213 } 215 }
214 if (!strncmp(argv[i], "--ofs=", 6)) { 216 if (!strncmp(argv[i], "--ofs=", 6)) {
215 cgit_query_ofs = atoi(argv[i]+6); 217 cgit_query_ofs = atoi(argv[i]+6);
216 } 218 }
217 } 219 }
218} 220}
219 221
220int main(int argc, const char **argv) 222int main(int argc, const char **argv)
221{ 223{
222 struct cacheitem item; 224 struct cacheitem item;
223 225
224 htmlfd = STDOUT_FILENO; 226 htmlfd = STDOUT_FILENO;
225 item.st.st_mtime = time(NULL); 227 item.st.st_mtime = time(NULL);
226 cgit_repolist.length = 0; 228 cgit_repolist.length = 0;
227 cgit_repolist.count = 0; 229 cgit_repolist.count = 0;
228 cgit_repolist.repos = NULL; 230 cgit_repolist.repos = NULL;
229 231
230 cgit_read_config(CGIT_CONFIG, cgit_global_config_cb); 232 cgit_read_config(CGIT_CONFIG, cgit_global_config_cb);
231 cgit_repo = NULL; 233 cgit_repo = NULL;
232 if (getenv("SCRIPT_NAME")) 234 if (getenv("SCRIPT_NAME"))
233 cgit_script_name = xstrdup(getenv("SCRIPT_NAME")); 235 cgit_script_name = xstrdup(getenv("SCRIPT_NAME"));
234 if (getenv("QUERY_STRING")) 236 if (getenv("QUERY_STRING"))
235 cgit_querystring = xstrdup(getenv("QUERY_STRING")); 237 cgit_querystring = xstrdup(getenv("QUERY_STRING"));
236 cgit_parse_args(argc, argv); 238 cgit_parse_args(argc, argv);
237 cgit_parse_query(cgit_querystring, cgit_querystring_cb); 239 cgit_parse_query(cgit_querystring, cgit_querystring_cb);
238 if (!cgit_prepare_cache(&item)) 240 if (!cgit_prepare_cache(&item))
239 return 0; 241 return 0;
240 if (cgit_nocache) { 242 if (cgit_nocache) {
241 cgit_fill_cache(&item, 0); 243 cgit_fill_cache(&item, 0);
242 } else { 244 } else {
243 cgit_check_cache(&item); 245 cgit_check_cache(&item);
244 cgit_print_cache(&item); 246 cgit_print_cache(&item);
245 } 247 }
246 return 0; 248 return 0;
247} 249}