summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--ui-log.c11
-rw-r--r--ui-shared.c1
2 files changed, 8 insertions, 4 deletions
diff --git a/ui-log.c b/ui-log.c
index 0947604..4441d1d 100644
--- a/ui-log.c
+++ b/ui-log.c
@@ -1,234 +1,237 @@
1/* ui-log.c: functions for log output 1/* ui-log.c: functions for log output
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#include "html.h" 10#include "html.h"
11#include "ui-shared.h" 11#include "ui-shared.h"
12 12
13int files, add_lines, rem_lines; 13int files, add_lines, rem_lines;
14 14
15void count_lines(char *line, int size) 15void count_lines(char *line, int size)
16{ 16{
17 if (size <= 0) 17 if (size <= 0)
18 return; 18 return;
19 19
20 if (line[0] == '+') 20 if (line[0] == '+')
21 add_lines++; 21 add_lines++;
22 22
23 else if (line[0] == '-') 23 else if (line[0] == '-')
24 rem_lines++; 24 rem_lines++;
25} 25}
26 26
27void inspect_files(struct diff_filepair *pair) 27void inspect_files(struct diff_filepair *pair)
28{ 28{
29 unsigned long old_size = 0; 29 unsigned long old_size = 0;
30 unsigned long new_size = 0; 30 unsigned long new_size = 0;
31 int binary = 0; 31 int binary = 0;
32 32
33 files++; 33 files++;
34 if (ctx.repo->enable_log_linecount) 34 if (ctx.repo->enable_log_linecount)
35 cgit_diff_files(pair->one->sha1, pair->two->sha1, &old_size, 35 cgit_diff_files(pair->one->sha1, pair->two->sha1, &old_size,
36 &new_size, &binary, count_lines); 36 &new_size, &binary, count_lines);
37} 37}
38 38
39void show_commit_decorations(struct commit *commit) 39void show_commit_decorations(struct commit *commit)
40{ 40{
41 struct name_decoration *deco; 41 struct name_decoration *deco;
42 static char buf[1024]; 42 static char buf[1024];
43 43
44 buf[sizeof(buf) - 1] = 0; 44 buf[sizeof(buf) - 1] = 0;
45 deco = lookup_decoration(&name_decoration, &commit->object); 45 deco = lookup_decoration(&name_decoration, &commit->object);
46 while (deco) { 46 while (deco) {
47 if (!prefixcmp(deco->name, "refs/heads/")) { 47 if (!prefixcmp(deco->name, "refs/heads/")) {
48 strncpy(buf, deco->name + 11, sizeof(buf) - 1); 48 strncpy(buf, deco->name + 11, sizeof(buf) - 1);
49 cgit_log_link(buf, NULL, "branch-deco", buf, NULL, NULL, 49 cgit_log_link(buf, NULL, "branch-deco", buf, NULL, NULL,
50 0, NULL, NULL, ctx.qry.showmsg); 50 0, NULL, NULL, ctx.qry.showmsg);
51 } 51 }
52 else if (!prefixcmp(deco->name, "tag: refs/tags/")) { 52 else if (!prefixcmp(deco->name, "tag: refs/tags/")) {
53 strncpy(buf, deco->name + 15, sizeof(buf) - 1); 53 strncpy(buf, deco->name + 15, sizeof(buf) - 1);
54 cgit_tag_link(buf, NULL, "tag-deco", ctx.qry.head, buf); 54 cgit_tag_link(buf, NULL, "tag-deco", ctx.qry.head, buf);
55 } 55 }
56 else if (!prefixcmp(deco->name, "refs/tags/")) { 56 else if (!prefixcmp(deco->name, "refs/tags/")) {
57 strncpy(buf, deco->name + 10, sizeof(buf) - 1); 57 strncpy(buf, deco->name + 10, sizeof(buf) - 1);
58 cgit_tag_link(buf, NULL, "tag-deco", ctx.qry.head, buf); 58 cgit_tag_link(buf, NULL, "tag-deco", ctx.qry.head, buf);
59 } 59 }
60 else if (!prefixcmp(deco->name, "refs/remotes/")) { 60 else if (!prefixcmp(deco->name, "refs/remotes/")) {
61 strncpy(buf, deco->name + 13, sizeof(buf) - 1); 61 strncpy(buf, deco->name + 13, sizeof(buf) - 1);
62 cgit_log_link(buf, NULL, "remote-deco", NULL, 62 cgit_log_link(buf, NULL, "remote-deco", NULL,
63 sha1_to_hex(commit->object.sha1), NULL, 63 sha1_to_hex(commit->object.sha1), NULL,
64 0, NULL, NULL, ctx.qry.showmsg); 64 0, NULL, NULL, ctx.qry.showmsg);
65 } 65 }
66 else { 66 else {
67 strncpy(buf, deco->name, sizeof(buf) - 1); 67 strncpy(buf, deco->name, sizeof(buf) - 1);
68 cgit_commit_link(buf, NULL, "deco", ctx.qry.head, 68 cgit_commit_link(buf, NULL, "deco", ctx.qry.head,
69 sha1_to_hex(commit->object.sha1), 0); 69 sha1_to_hex(commit->object.sha1), 0);
70 } 70 }
71 deco = deco->next; 71 deco = deco->next;
72 } 72 }
73} 73}
74 74
75void print_commit(struct commit *commit) 75void print_commit(struct commit *commit)
76{ 76{
77 struct commitinfo *info; 77 struct commitinfo *info;
78 char *tmp; 78 char *tmp;
79 int cols = 2; 79 int cols = 2;
80 80
81 info = cgit_parse_commit(commit); 81 info = cgit_parse_commit(commit);
82 htmlf("<tr%s><td>", 82 htmlf("<tr%s><td>",
83 ctx.qry.showmsg ? " class='logheader'" : ""); 83 ctx.qry.showmsg ? " class='logheader'" : "");
84 tmp = fmt("id=%s", sha1_to_hex(commit->object.sha1)); 84 tmp = fmt("id=%s", sha1_to_hex(commit->object.sha1));
85 tmp = cgit_pageurl(ctx.repo->url, "commit", tmp); 85 tmp = cgit_pageurl(ctx.repo->url, "commit", tmp);
86 html_link_open(tmp, NULL, NULL); 86 html_link_open(tmp, NULL, NULL);
87 cgit_print_age(commit->date, TM_WEEK * 2, FMT_SHORTDATE); 87 cgit_print_age(commit->date, TM_WEEK * 2, FMT_SHORTDATE);
88 html_link_close(); 88 html_link_close();
89 htmlf("</td><td%s>", 89 htmlf("</td><td%s>",
90 ctx.qry.showmsg ? " class='logsubject'" : ""); 90 ctx.qry.showmsg ? " class='logsubject'" : "");
91 cgit_commit_link(info->subject, NULL, NULL, ctx.qry.head, 91 cgit_commit_link(info->subject, NULL, NULL, ctx.qry.head,
92 sha1_to_hex(commit->object.sha1), 0); 92 sha1_to_hex(commit->object.sha1), 0);
93 show_commit_decorations(commit); 93 show_commit_decorations(commit);
94 html("</td><td>"); 94 html("</td><td>");
95 html_txt(info->author); 95 html_txt(info->author);
96 if (ctx.repo->enable_log_filecount) { 96 if (ctx.repo->enable_log_filecount) {
97 files = 0; 97 files = 0;
98 add_lines = 0; 98 add_lines = 0;
99 rem_lines = 0; 99 rem_lines = 0;
100 cgit_diff_commit(commit, inspect_files); 100 cgit_diff_commit(commit, inspect_files);
101 html("</td><td>"); 101 html("</td><td>");
102 htmlf("%d", files); 102 htmlf("%d", files);
103 if (ctx.repo->enable_log_linecount) { 103 if (ctx.repo->enable_log_linecount) {
104 html("</td><td>"); 104 html("</td><td>");
105 htmlf("-%d/+%d", rem_lines, add_lines); 105 htmlf("-%d/+%d", rem_lines, add_lines);
106 } 106 }
107 } 107 }
108 html("</td></tr>\n"); 108 html("</td></tr>\n");
109 if (ctx.qry.showmsg) { 109 if (ctx.qry.showmsg) {
110 if (ctx.repo->enable_log_filecount) { 110 if (ctx.repo->enable_log_filecount) {
111 cols++; 111 cols++;
112 if (ctx.repo->enable_log_linecount) 112 if (ctx.repo->enable_log_linecount)
113 cols++; 113 cols++;
114 } 114 }
115 htmlf("<tr class='nohover'><td/><td colspan='%d' class='logmsg'>", 115 htmlf("<tr class='nohover'><td/><td colspan='%d' class='logmsg'>",
116 cols); 116 cols);
117 html_txt(info->msg); 117 html_txt(info->msg);
118 html("</td></tr>\n"); 118 html("</td></tr>\n");
119 } 119 }
120 cgit_free_commitinfo(info); 120 cgit_free_commitinfo(info);
121} 121}
122 122
123static const char *disambiguate_ref(const char *ref) 123static const char *disambiguate_ref(const char *ref)
124{ 124{
125 unsigned char sha1[20]; 125 unsigned char sha1[20];
126 const char *longref; 126 const char *longref;
127 127
128 longref = fmt("refs/heads/%s", ref); 128 longref = fmt("refs/heads/%s", ref);
129 if (get_sha1(longref, sha1) == 0) 129 if (get_sha1(longref, sha1) == 0)
130 return longref; 130 return longref;
131 131
132 return ref; 132 return ref;
133} 133}
134 134
135void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern, 135void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern,
136 char *path, int pager) 136 char *path, int pager)
137{ 137{
138 struct rev_info rev; 138 struct rev_info rev;
139 struct commit *commit; 139 struct commit *commit;
140 const char *argv[] = {NULL, NULL, NULL, NULL, NULL}; 140 const char *argv[] = {NULL, NULL, NULL, NULL, NULL};
141 int argc = 2; 141 int argc = 2;
142 int i, columns = 3; 142 int i, columns = 3;
143 143
144 if (!tip) 144 if (!tip)
145 tip = ctx.qry.head; 145 tip = ctx.qry.head;
146 146
147 argv[1] = disambiguate_ref(tip); 147 argv[1] = disambiguate_ref(tip);
148 148
149 if (grep && pattern && (!strcmp(grep, "grep") || 149 if (grep && pattern) {
150 !strcmp(grep, "author") || 150 if (!strcmp(grep, "grep") || !strcmp(grep, "author") ||
151 !strcmp(grep, "committer"))) 151 !strcmp(grep, "committer"))
152 argv[argc++] = fmt("--%s=%s", grep, pattern); 152 argv[argc++] = fmt("--%s=%s", grep, pattern);
153 if (!strcmp(grep, "range"))
154 argv[1] = pattern;
155 }
153 156
154 if (path) { 157 if (path) {
155 argv[argc++] = "--"; 158 argv[argc++] = "--";
156 argv[argc++] = path; 159 argv[argc++] = path;
157 } 160 }
158 init_revisions(&rev, NULL); 161 init_revisions(&rev, NULL);
159 rev.abbrev = DEFAULT_ABBREV; 162 rev.abbrev = DEFAULT_ABBREV;
160 rev.commit_format = CMIT_FMT_DEFAULT; 163 rev.commit_format = CMIT_FMT_DEFAULT;
161 rev.verbose_header = 1; 164 rev.verbose_header = 1;
162 rev.show_root_diff = 0; 165 rev.show_root_diff = 0;
163 setup_revisions(argc, argv, &rev, NULL); 166 setup_revisions(argc, argv, &rev, NULL);
164 load_ref_decorations(DECORATE_FULL_REFS); 167 load_ref_decorations(DECORATE_FULL_REFS);
165 rev.show_decorations = 1; 168 rev.show_decorations = 1;
166 rev.grep_filter.regflags |= REG_ICASE; 169 rev.grep_filter.regflags |= REG_ICASE;
167 compile_grep_patterns(&rev.grep_filter); 170 compile_grep_patterns(&rev.grep_filter);
168 prepare_revision_walk(&rev); 171 prepare_revision_walk(&rev);
169 172
170 if (pager) 173 if (pager)
171 html("<table class='list nowrap'>"); 174 html("<table class='list nowrap'>");
172 175
173 html("<tr class='nohover'><th class='left'>Age</th>" 176 html("<tr class='nohover'><th class='left'>Age</th>"
174 "<th class='left'>Commit message"); 177 "<th class='left'>Commit message");
175 if (pager) { 178 if (pager) {
176 html(" ("); 179 html(" (");
177 cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL, 180 cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL,
178 NULL, ctx.qry.head, ctx.qry.sha1, 181 NULL, ctx.qry.head, ctx.qry.sha1,
179 ctx.qry.path, ctx.qry.ofs, ctx.qry.grep, 182 ctx.qry.path, ctx.qry.ofs, ctx.qry.grep,
180 ctx.qry.search, ctx.qry.showmsg ? 0 : 1); 183 ctx.qry.search, ctx.qry.showmsg ? 0 : 1);
181 html(")"); 184 html(")");
182 } 185 }
183 html("</th><th class='left'>Author</th>"); 186 html("</th><th class='left'>Author</th>");
184 if (ctx.repo->enable_log_filecount) { 187 if (ctx.repo->enable_log_filecount) {
185 html("<th class='left'>Files</th>"); 188 html("<th class='left'>Files</th>");
186 columns++; 189 columns++;
187 if (ctx.repo->enable_log_linecount) { 190 if (ctx.repo->enable_log_linecount) {
188 html("<th class='left'>Lines</th>"); 191 html("<th class='left'>Lines</th>");
189 columns++; 192 columns++;
190 } 193 }
191 } 194 }
192 html("</tr>\n"); 195 html("</tr>\n");
193 196
194 if (ofs<0) 197 if (ofs<0)
195 ofs = 0; 198 ofs = 0;
196 199
197 for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; i++) { 200 for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; i++) {
198 free(commit->buffer); 201 free(commit->buffer);
199 commit->buffer = NULL; 202 commit->buffer = NULL;
200 free_commit_list(commit->parents); 203 free_commit_list(commit->parents);
201 commit->parents = NULL; 204 commit->parents = NULL;
202 } 205 }
203 206
204 for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; i++) { 207 for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; i++) {
205 print_commit(commit); 208 print_commit(commit);
206 free(commit->buffer); 209 free(commit->buffer);
207 commit->buffer = NULL; 210 commit->buffer = NULL;
208 free_commit_list(commit->parents); 211 free_commit_list(commit->parents);
209 commit->parents = NULL; 212 commit->parents = NULL;
210 } 213 }
211 if (pager) { 214 if (pager) {
212 htmlf("</table><div class='pager'>", 215 htmlf("</table><div class='pager'>",
213 columns); 216 columns);
214 if (ofs > 0) { 217 if (ofs > 0) {
215 cgit_log_link("[prev]", NULL, NULL, ctx.qry.head, 218 cgit_log_link("[prev]", NULL, NULL, ctx.qry.head,
216 ctx.qry.sha1, ctx.qry.path, 219 ctx.qry.sha1, ctx.qry.path,
217 ofs - cnt, ctx.qry.grep, 220 ofs - cnt, ctx.qry.grep,
218 ctx.qry.search, ctx.qry.showmsg); 221 ctx.qry.search, ctx.qry.showmsg);
219 html("&nbsp;"); 222 html("&nbsp;");
220 } 223 }
221 if ((commit = get_revision(&rev)) != NULL) { 224 if ((commit = get_revision(&rev)) != NULL) {
222 cgit_log_link("[next]", NULL, NULL, ctx.qry.head, 225 cgit_log_link("[next]", NULL, NULL, ctx.qry.head,
223 ctx.qry.sha1, ctx.qry.path, 226 ctx.qry.sha1, ctx.qry.path,
224 ofs + cnt, ctx.qry.grep, 227 ofs + cnt, ctx.qry.grep,
225 ctx.qry.search, ctx.qry.showmsg); 228 ctx.qry.search, ctx.qry.showmsg);
226 } 229 }
227 html("</div>"); 230 html("</div>");
228 } else if ((commit = get_revision(&rev)) != NULL) { 231 } else if ((commit = get_revision(&rev)) != NULL) {
229 html("<tr class='nohover'><td colspan='3'>"); 232 html("<tr class='nohover'><td colspan='3'>");
230 cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL, NULL, 0, 233 cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL, NULL, 0,
231 NULL, NULL, ctx.qry.showmsg); 234 NULL, NULL, ctx.qry.showmsg);
232 html("</td></tr>\n"); 235 html("</td></tr>\n");
233 } 236 }
234} 237}
diff --git a/ui-shared.c b/ui-shared.c
index 8827fff..0f65474 100644
--- a/ui-shared.c
+++ b/ui-shared.c
@@ -357,444 +357,445 @@ void cgit_snapshot_link(char *name, char *title, char *class, char *head,
357} 357}
358 358
359void cgit_diff_link(char *name, char *title, char *class, char *head, 359void cgit_diff_link(char *name, char *title, char *class, char *head,
360 char *new_rev, char *old_rev, char *path, 360 char *new_rev, char *old_rev, char *path,
361 int toggle_ssdiff) 361 int toggle_ssdiff)
362{ 362{
363 char *delim; 363 char *delim;
364 364
365 delim = repolink(title, class, "diff", head, path); 365 delim = repolink(title, class, "diff", head, path);
366 if (new_rev && ctx.qry.head != NULL && strcmp(new_rev, ctx.qry.head)) { 366 if (new_rev && ctx.qry.head != NULL && strcmp(new_rev, ctx.qry.head)) {
367 html(delim); 367 html(delim);
368 html("id="); 368 html("id=");
369 html_url_arg(new_rev); 369 html_url_arg(new_rev);
370 delim = "&amp;"; 370 delim = "&amp;";
371 } 371 }
372 if (old_rev) { 372 if (old_rev) {
373 html(delim); 373 html(delim);
374 html("id2="); 374 html("id2=");
375 html_url_arg(old_rev); 375 html_url_arg(old_rev);
376 delim = "&amp;"; 376 delim = "&amp;";
377 } 377 }
378 if ((ctx.qry.ssdiff && !toggle_ssdiff) || (!ctx.qry.ssdiff && toggle_ssdiff)) { 378 if ((ctx.qry.ssdiff && !toggle_ssdiff) || (!ctx.qry.ssdiff && toggle_ssdiff)) {
379 html(delim); 379 html(delim);
380 html("ss=1"); 380 html("ss=1");
381 } 381 }
382 html("'>"); 382 html("'>");
383 html_txt(name); 383 html_txt(name);
384 html("</a>"); 384 html("</a>");
385} 385}
386 386
387void cgit_patch_link(char *name, char *title, char *class, char *head, 387void cgit_patch_link(char *name, char *title, char *class, char *head,
388 char *rev) 388 char *rev)
389{ 389{
390 reporevlink("patch", name, title, class, head, rev, NULL); 390 reporevlink("patch", name, title, class, head, rev, NULL);
391} 391}
392 392
393void cgit_stats_link(char *name, char *title, char *class, char *head, 393void cgit_stats_link(char *name, char *title, char *class, char *head,
394 char *path) 394 char *path)
395{ 395{
396 reporevlink("stats", name, title, class, head, NULL, path); 396 reporevlink("stats", name, title, class, head, NULL, path);
397} 397}
398 398
399void cgit_object_link(struct object *obj) 399void cgit_object_link(struct object *obj)
400{ 400{
401 char *page, *shortrev, *fullrev, *name; 401 char *page, *shortrev, *fullrev, *name;
402 402
403 fullrev = sha1_to_hex(obj->sha1); 403 fullrev = sha1_to_hex(obj->sha1);
404 shortrev = xstrdup(fullrev); 404 shortrev = xstrdup(fullrev);
405 shortrev[10] = '\0'; 405 shortrev[10] = '\0';
406 if (obj->type == OBJ_COMMIT) { 406 if (obj->type == OBJ_COMMIT) {
407 cgit_commit_link(fmt("commit %s...", shortrev), NULL, NULL, 407 cgit_commit_link(fmt("commit %s...", shortrev), NULL, NULL,
408 ctx.qry.head, fullrev, 0); 408 ctx.qry.head, fullrev, 0);
409 return; 409 return;
410 } else if (obj->type == OBJ_TREE) 410 } else if (obj->type == OBJ_TREE)
411 page = "tree"; 411 page = "tree";
412 else if (obj->type == OBJ_TAG) 412 else if (obj->type == OBJ_TAG)
413 page = "tag"; 413 page = "tag";
414 else 414 else
415 page = "blob"; 415 page = "blob";
416 name = fmt("%s %s...", typename(obj->type), shortrev); 416 name = fmt("%s %s...", typename(obj->type), shortrev);
417 reporevlink(page, name, NULL, NULL, ctx.qry.head, fullrev, NULL); 417 reporevlink(page, name, NULL, NULL, ctx.qry.head, fullrev, NULL);
418} 418}
419 419
420void cgit_print_date(time_t secs, char *format, int local_time) 420void cgit_print_date(time_t secs, char *format, int local_time)
421{ 421{
422 char buf[64]; 422 char buf[64];
423 struct tm *time; 423 struct tm *time;
424 424
425 if (!secs) 425 if (!secs)
426 return; 426 return;
427 if(local_time) 427 if(local_time)
428 time = localtime(&secs); 428 time = localtime(&secs);
429 else 429 else
430 time = gmtime(&secs); 430 time = gmtime(&secs);
431 strftime(buf, sizeof(buf)-1, format, time); 431 strftime(buf, sizeof(buf)-1, format, time);
432 html_txt(buf); 432 html_txt(buf);
433} 433}
434 434
435void cgit_print_age(time_t t, time_t max_relative, char *format) 435void cgit_print_age(time_t t, time_t max_relative, char *format)
436{ 436{
437 time_t now, secs; 437 time_t now, secs;
438 438
439 if (!t) 439 if (!t)
440 return; 440 return;
441 time(&now); 441 time(&now);
442 secs = now - t; 442 secs = now - t;
443 443
444 if (secs > max_relative && max_relative >= 0) { 444 if (secs > max_relative && max_relative >= 0) {
445 cgit_print_date(t, format, ctx.cfg.local_time); 445 cgit_print_date(t, format, ctx.cfg.local_time);
446 return; 446 return;
447 } 447 }
448 448
449 if (secs < TM_HOUR * 2) { 449 if (secs < TM_HOUR * 2) {
450 htmlf("<span class='age-mins'>%.0f min.</span>", 450 htmlf("<span class='age-mins'>%.0f min.</span>",
451 secs * 1.0 / TM_MIN); 451 secs * 1.0 / TM_MIN);
452 return; 452 return;
453 } 453 }
454 if (secs < TM_DAY * 2) { 454 if (secs < TM_DAY * 2) {
455 htmlf("<span class='age-hours'>%.0f hours</span>", 455 htmlf("<span class='age-hours'>%.0f hours</span>",
456 secs * 1.0 / TM_HOUR); 456 secs * 1.0 / TM_HOUR);
457 return; 457 return;
458 } 458 }
459 if (secs < TM_WEEK * 2) { 459 if (secs < TM_WEEK * 2) {
460 htmlf("<span class='age-days'>%.0f days</span>", 460 htmlf("<span class='age-days'>%.0f days</span>",
461 secs * 1.0 / TM_DAY); 461 secs * 1.0 / TM_DAY);
462 return; 462 return;
463 } 463 }
464 if (secs < TM_MONTH * 2) { 464 if (secs < TM_MONTH * 2) {
465 htmlf("<span class='age-weeks'>%.0f weeks</span>", 465 htmlf("<span class='age-weeks'>%.0f weeks</span>",
466 secs * 1.0 / TM_WEEK); 466 secs * 1.0 / TM_WEEK);
467 return; 467 return;
468 } 468 }
469 if (secs < TM_YEAR * 2) { 469 if (secs < TM_YEAR * 2) {
470 htmlf("<span class='age-months'>%.0f months</span>", 470 htmlf("<span class='age-months'>%.0f months</span>",
471 secs * 1.0 / TM_MONTH); 471 secs * 1.0 / TM_MONTH);
472 return; 472 return;
473 } 473 }
474 htmlf("<span class='age-years'>%.0f years</span>", 474 htmlf("<span class='age-years'>%.0f years</span>",
475 secs * 1.0 / TM_YEAR); 475 secs * 1.0 / TM_YEAR);
476} 476}
477 477
478void cgit_print_http_headers(struct cgit_context *ctx) 478void cgit_print_http_headers(struct cgit_context *ctx)
479{ 479{
480 if (ctx->env.no_http && !strcmp(ctx->env.no_http, "1")) 480 if (ctx->env.no_http && !strcmp(ctx->env.no_http, "1"))
481 return; 481 return;
482 482
483 if (ctx->page.status) 483 if (ctx->page.status)
484 htmlf("Status: %d %s\n", ctx->page.status, ctx->page.statusmsg); 484 htmlf("Status: %d %s\n", ctx->page.status, ctx->page.statusmsg);
485 if (ctx->page.mimetype && ctx->page.charset) 485 if (ctx->page.mimetype && ctx->page.charset)
486 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype, 486 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype,
487 ctx->page.charset); 487 ctx->page.charset);
488 else if (ctx->page.mimetype) 488 else if (ctx->page.mimetype)
489 htmlf("Content-Type: %s\n", ctx->page.mimetype); 489 htmlf("Content-Type: %s\n", ctx->page.mimetype);
490 if (ctx->page.size) 490 if (ctx->page.size)
491 htmlf("Content-Length: %ld\n", ctx->page.size); 491 htmlf("Content-Length: %ld\n", ctx->page.size);
492 if (ctx->page.filename) 492 if (ctx->page.filename)
493 htmlf("Content-Disposition: inline; filename=\"%s\"\n", 493 htmlf("Content-Disposition: inline; filename=\"%s\"\n",
494 ctx->page.filename); 494 ctx->page.filename);
495 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified)); 495 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified));
496 htmlf("Expires: %s\n", http_date(ctx->page.expires)); 496 htmlf("Expires: %s\n", http_date(ctx->page.expires));
497 if (ctx->page.etag) 497 if (ctx->page.etag)
498 htmlf("ETag: \"%s\"\n", ctx->page.etag); 498 htmlf("ETag: \"%s\"\n", ctx->page.etag);
499 html("\n"); 499 html("\n");
500 if (ctx->env.request_method && !strcmp(ctx->env.request_method, "HEAD")) 500 if (ctx->env.request_method && !strcmp(ctx->env.request_method, "HEAD"))
501 exit(0); 501 exit(0);
502} 502}
503 503
504void cgit_print_docstart(struct cgit_context *ctx) 504void cgit_print_docstart(struct cgit_context *ctx)
505{ 505{
506 if (ctx->cfg.embedded) { 506 if (ctx->cfg.embedded) {
507 if (ctx->cfg.header) 507 if (ctx->cfg.header)
508 html_include(ctx->cfg.header); 508 html_include(ctx->cfg.header);
509 return; 509 return;
510 } 510 }
511 511
512 char *host = cgit_hosturl(); 512 char *host = cgit_hosturl();
513 html(cgit_doctype); 513 html(cgit_doctype);
514 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n"); 514 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n");
515 html("<head>\n"); 515 html("<head>\n");
516 html("<title>"); 516 html("<title>");
517 html_txt(ctx->page.title); 517 html_txt(ctx->page.title);
518 html("</title>\n"); 518 html("</title>\n");
519 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version); 519 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version);
520 if (ctx->cfg.robots && *ctx->cfg.robots) 520 if (ctx->cfg.robots && *ctx->cfg.robots)
521 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots); 521 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots);
522 html("<link rel='stylesheet' type='text/css' href='"); 522 html("<link rel='stylesheet' type='text/css' href='");
523 html_attr(ctx->cfg.css); 523 html_attr(ctx->cfg.css);
524 html("'/>\n"); 524 html("'/>\n");
525 if (ctx->cfg.favicon) { 525 if (ctx->cfg.favicon) {
526 html("<link rel='shortcut icon' href='"); 526 html("<link rel='shortcut icon' href='");
527 html_attr(ctx->cfg.favicon); 527 html_attr(ctx->cfg.favicon);
528 html("'/>\n"); 528 html("'/>\n");
529 } 529 }
530 if (host && ctx->repo) { 530 if (host && ctx->repo) {
531 html("<link rel='alternate' title='Atom feed' href='"); 531 html("<link rel='alternate' title='Atom feed' href='");
532 html(cgit_httpscheme()); 532 html(cgit_httpscheme());
533 html_attr(cgit_hosturl()); 533 html_attr(cgit_hosturl());
534 html_attr(cgit_fileurl(ctx->repo->url, "atom", ctx->qry.path, 534 html_attr(cgit_fileurl(ctx->repo->url, "atom", ctx->qry.path,
535 fmt("h=%s", ctx->qry.head))); 535 fmt("h=%s", ctx->qry.head)));
536 html("' type='application/atom+xml'/>\n"); 536 html("' type='application/atom+xml'/>\n");
537 } 537 }
538 if (ctx->cfg.head_include) 538 if (ctx->cfg.head_include)
539 html_include(ctx->cfg.head_include); 539 html_include(ctx->cfg.head_include);
540 html("</head>\n"); 540 html("</head>\n");
541 html("<body>\n"); 541 html("<body>\n");
542 if (ctx->cfg.header) 542 if (ctx->cfg.header)
543 html_include(ctx->cfg.header); 543 html_include(ctx->cfg.header);
544} 544}
545 545
546void cgit_print_docend() 546void cgit_print_docend()
547{ 547{
548 html("</div> <!-- class=content -->\n"); 548 html("</div> <!-- class=content -->\n");
549 if (ctx.cfg.embedded) { 549 if (ctx.cfg.embedded) {
550 html("</div> <!-- id=cgit -->\n"); 550 html("</div> <!-- id=cgit -->\n");
551 if (ctx.cfg.footer) 551 if (ctx.cfg.footer)
552 html_include(ctx.cfg.footer); 552 html_include(ctx.cfg.footer);
553 return; 553 return;
554 } 554 }
555 if (ctx.cfg.footer) 555 if (ctx.cfg.footer)
556 html_include(ctx.cfg.footer); 556 html_include(ctx.cfg.footer);
557 else { 557 else {
558 htmlf("<div class='footer'>generated by cgit %s at ", 558 htmlf("<div class='footer'>generated by cgit %s at ",
559 cgit_version); 559 cgit_version);
560 cgit_print_date(time(NULL), FMT_LONGDATE, ctx.cfg.local_time); 560 cgit_print_date(time(NULL), FMT_LONGDATE, ctx.cfg.local_time);
561 html("</div>\n"); 561 html("</div>\n");
562 } 562 }
563 html("</div> <!-- id=cgit -->\n"); 563 html("</div> <!-- id=cgit -->\n");
564 html("</body>\n</html>\n"); 564 html("</body>\n</html>\n");
565} 565}
566 566
567int print_branch_option(const char *refname, const unsigned char *sha1, 567int print_branch_option(const char *refname, const unsigned char *sha1,
568 int flags, void *cb_data) 568 int flags, void *cb_data)
569{ 569{
570 char *name = (char *)refname; 570 char *name = (char *)refname;
571 html_option(name, name, ctx.qry.head); 571 html_option(name, name, ctx.qry.head);
572 return 0; 572 return 0;
573} 573}
574 574
575int print_archive_ref(const char *refname, const unsigned char *sha1, 575int print_archive_ref(const char *refname, const unsigned char *sha1,
576 int flags, void *cb_data) 576 int flags, void *cb_data)
577{ 577{
578 struct tag *tag; 578 struct tag *tag;
579 struct taginfo *info; 579 struct taginfo *info;
580 struct object *obj; 580 struct object *obj;
581 char buf[256], *url; 581 char buf[256], *url;
582 unsigned char fileid[20]; 582 unsigned char fileid[20];
583 int *header = (int *)cb_data; 583 int *header = (int *)cb_data;
584 584
585 if (prefixcmp(refname, "refs/archives")) 585 if (prefixcmp(refname, "refs/archives"))
586 return 0; 586 return 0;
587 strncpy(buf, refname+14, sizeof(buf)); 587 strncpy(buf, refname+14, sizeof(buf));
588 obj = parse_object(sha1); 588 obj = parse_object(sha1);
589 if (!obj) 589 if (!obj)
590 return 1; 590 return 1;
591 if (obj->type == OBJ_TAG) { 591 if (obj->type == OBJ_TAG) {
592 tag = lookup_tag(sha1); 592 tag = lookup_tag(sha1);
593 if (!tag || parse_tag(tag) || !(info = cgit_parse_tag(tag))) 593 if (!tag || parse_tag(tag) || !(info = cgit_parse_tag(tag)))
594 return 0; 594 return 0;
595 hashcpy(fileid, tag->tagged->sha1); 595 hashcpy(fileid, tag->tagged->sha1);
596 } else if (obj->type != OBJ_BLOB) { 596 } else if (obj->type != OBJ_BLOB) {
597 return 0; 597 return 0;
598 } else { 598 } else {
599 hashcpy(fileid, sha1); 599 hashcpy(fileid, sha1);
600 } 600 }
601 if (!*header) { 601 if (!*header) {
602 html("<h1>download</h1>\n"); 602 html("<h1>download</h1>\n");
603 *header = 1; 603 *header = 1;
604 } 604 }
605 url = cgit_pageurl(ctx.qry.repo, "blob", 605 url = cgit_pageurl(ctx.qry.repo, "blob",
606 fmt("id=%s&amp;path=%s", sha1_to_hex(fileid), 606 fmt("id=%s&amp;path=%s", sha1_to_hex(fileid),
607 buf)); 607 buf));
608 html_link_open(url, NULL, "menu"); 608 html_link_open(url, NULL, "menu");
609 html_txt(strlpart(buf, 20)); 609 html_txt(strlpart(buf, 20));
610 html_link_close(); 610 html_link_close();
611 return 0; 611 return 0;
612} 612}
613 613
614void cgit_add_hidden_formfields(int incl_head, int incl_search, char *page) 614void cgit_add_hidden_formfields(int incl_head, int incl_search, char *page)
615{ 615{
616 char *url; 616 char *url;
617 617
618 if (!ctx.cfg.virtual_root) { 618 if (!ctx.cfg.virtual_root) {
619 url = fmt("%s/%s", ctx.qry.repo, page); 619 url = fmt("%s/%s", ctx.qry.repo, page);
620 if (ctx.qry.path) 620 if (ctx.qry.path)
621 url = fmt("%s/%s", url, ctx.qry.path); 621 url = fmt("%s/%s", url, ctx.qry.path);
622 html_hidden("url", url); 622 html_hidden("url", url);
623 } 623 }
624 624
625 if (incl_head && ctx.qry.head && ctx.repo->defbranch && 625 if (incl_head && ctx.qry.head && ctx.repo->defbranch &&
626 strcmp(ctx.qry.head, ctx.repo->defbranch)) 626 strcmp(ctx.qry.head, ctx.repo->defbranch))
627 html_hidden("h", ctx.qry.head); 627 html_hidden("h", ctx.qry.head);
628 628
629 if (ctx.qry.sha1) 629 if (ctx.qry.sha1)
630 html_hidden("id", ctx.qry.sha1); 630 html_hidden("id", ctx.qry.sha1);
631 if (ctx.qry.sha2) 631 if (ctx.qry.sha2)
632 html_hidden("id2", ctx.qry.sha2); 632 html_hidden("id2", ctx.qry.sha2);
633 if (ctx.qry.showmsg) 633 if (ctx.qry.showmsg)
634 html_hidden("showmsg", "1"); 634 html_hidden("showmsg", "1");
635 635
636 if (incl_search) { 636 if (incl_search) {
637 if (ctx.qry.grep) 637 if (ctx.qry.grep)
638 html_hidden("qt", ctx.qry.grep); 638 html_hidden("qt", ctx.qry.grep);
639 if (ctx.qry.search) 639 if (ctx.qry.search)
640 html_hidden("q", ctx.qry.search); 640 html_hidden("q", ctx.qry.search);
641 } 641 }
642} 642}
643 643
644const char *fallback_cmd = "repolist"; 644const char *fallback_cmd = "repolist";
645 645
646char *hc(struct cgit_cmd *cmd, const char *page) 646char *hc(struct cgit_cmd *cmd, const char *page)
647{ 647{
648 return (strcmp(cmd ? cmd->name : fallback_cmd, page) ? NULL : "active"); 648 return (strcmp(cmd ? cmd->name : fallback_cmd, page) ? NULL : "active");
649} 649}
650 650
651static void print_header(struct cgit_context *ctx) 651static void print_header(struct cgit_context *ctx)
652{ 652{
653 html("<table id='header'>\n"); 653 html("<table id='header'>\n");
654 html("<tr>\n"); 654 html("<tr>\n");
655 655
656 if (ctx->cfg.logo && ctx->cfg.logo[0] != 0) { 656 if (ctx->cfg.logo && ctx->cfg.logo[0] != 0) {
657 html("<td class='logo' rowspan='2'><a href='"); 657 html("<td class='logo' rowspan='2'><a href='");
658 if (ctx->cfg.logo_link) 658 if (ctx->cfg.logo_link)
659 html_attr(ctx->cfg.logo_link); 659 html_attr(ctx->cfg.logo_link);
660 else 660 else
661 html_attr(cgit_rooturl()); 661 html_attr(cgit_rooturl());
662 html("'><img src='"); 662 html("'><img src='");
663 html_attr(ctx->cfg.logo); 663 html_attr(ctx->cfg.logo);
664 html("' alt='cgit logo'/></a></td>\n"); 664 html("' alt='cgit logo'/></a></td>\n");
665 } 665 }
666 666
667 html("<td class='main'>"); 667 html("<td class='main'>");
668 if (ctx->repo) { 668 if (ctx->repo) {
669 cgit_index_link("index", NULL, NULL, NULL, 0); 669 cgit_index_link("index", NULL, NULL, NULL, 0);
670 html(" : "); 670 html(" : ");
671 cgit_summary_link(ctx->repo->name, ctx->repo->name, NULL, NULL); 671 cgit_summary_link(ctx->repo->name, ctx->repo->name, NULL, NULL);
672 html("</td><td class='form'>"); 672 html("</td><td class='form'>");
673 html("<form method='get' action=''>\n"); 673 html("<form method='get' action=''>\n");
674 cgit_add_hidden_formfields(0, 1, ctx->qry.page); 674 cgit_add_hidden_formfields(0, 1, ctx->qry.page);
675 html("<select name='h' onchange='this.form.submit();'>\n"); 675 html("<select name='h' onchange='this.form.submit();'>\n");
676 for_each_branch_ref(print_branch_option, ctx->qry.head); 676 for_each_branch_ref(print_branch_option, ctx->qry.head);
677 html("</select> "); 677 html("</select> ");
678 html("<input type='submit' name='' value='switch'/>"); 678 html("<input type='submit' name='' value='switch'/>");
679 html("</form>"); 679 html("</form>");
680 } else 680 } else
681 html_txt(ctx->cfg.root_title); 681 html_txt(ctx->cfg.root_title);
682 html("</td></tr>\n"); 682 html("</td></tr>\n");
683 683
684 html("<tr><td class='sub'>"); 684 html("<tr><td class='sub'>");
685 if (ctx->repo) { 685 if (ctx->repo) {
686 html_txt(ctx->repo->desc); 686 html_txt(ctx->repo->desc);
687 html("</td><td class='sub right'>"); 687 html("</td><td class='sub right'>");
688 html_txt(ctx->repo->owner); 688 html_txt(ctx->repo->owner);
689 } else { 689 } else {
690 if (ctx->cfg.root_desc) 690 if (ctx->cfg.root_desc)
691 html_txt(ctx->cfg.root_desc); 691 html_txt(ctx->cfg.root_desc);
692 else if (ctx->cfg.index_info) 692 else if (ctx->cfg.index_info)
693 html_include(ctx->cfg.index_info); 693 html_include(ctx->cfg.index_info);
694 } 694 }
695 html("</td></tr></table>\n"); 695 html("</td></tr></table>\n");
696} 696}
697 697
698void cgit_print_pageheader(struct cgit_context *ctx) 698void cgit_print_pageheader(struct cgit_context *ctx)
699{ 699{
700 struct cgit_cmd *cmd = cgit_get_cmd(ctx); 700 struct cgit_cmd *cmd = cgit_get_cmd(ctx);
701 701
702 if (!cmd && ctx->repo) 702 if (!cmd && ctx->repo)
703 fallback_cmd = "summary"; 703 fallback_cmd = "summary";
704 704
705 html("<div id='cgit'>"); 705 html("<div id='cgit'>");
706 if (!ctx->cfg.noheader) 706 if (!ctx->cfg.noheader)
707 print_header(ctx); 707 print_header(ctx);
708 708
709 html("<table class='tabs'><tr><td>\n"); 709 html("<table class='tabs'><tr><td>\n");
710 if (ctx->repo) { 710 if (ctx->repo) {
711 cgit_summary_link("summary", NULL, hc(cmd, "summary"), 711 cgit_summary_link("summary", NULL, hc(cmd, "summary"),
712 ctx->qry.head); 712 ctx->qry.head);
713 cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head, 713 cgit_refs_link("refs", NULL, hc(cmd, "refs"), ctx->qry.head,
714 ctx->qry.sha1, NULL); 714 ctx->qry.sha1, NULL);
715 cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head, 715 cgit_log_link("log", NULL, hc(cmd, "log"), ctx->qry.head,
716 NULL, NULL, 0, NULL, NULL, ctx->qry.showmsg); 716 NULL, NULL, 0, NULL, NULL, ctx->qry.showmsg);
717 cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head, 717 cgit_tree_link("tree", NULL, hc(cmd, "tree"), ctx->qry.head,
718 ctx->qry.sha1, NULL); 718 ctx->qry.sha1, NULL);
719 cgit_commit_link("commit", NULL, hc(cmd, "commit"), 719 cgit_commit_link("commit", NULL, hc(cmd, "commit"),
720 ctx->qry.head, ctx->qry.sha1, 0); 720 ctx->qry.head, ctx->qry.sha1, 0);
721 cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head, 721 cgit_diff_link("diff", NULL, hc(cmd, "diff"), ctx->qry.head,
722 ctx->qry.sha1, ctx->qry.sha2, NULL, 0); 722 ctx->qry.sha1, ctx->qry.sha2, NULL, 0);
723 if (ctx->repo->max_stats) 723 if (ctx->repo->max_stats)
724 cgit_stats_link("stats", NULL, hc(cmd, "stats"), 724 cgit_stats_link("stats", NULL, hc(cmd, "stats"),
725 ctx->qry.head, NULL); 725 ctx->qry.head, NULL);
726 if (ctx->repo->readme) 726 if (ctx->repo->readme)
727 reporevlink("about", "about", NULL, 727 reporevlink("about", "about", NULL,
728 hc(cmd, "about"), ctx->qry.head, NULL, 728 hc(cmd, "about"), ctx->qry.head, NULL,
729 NULL); 729 NULL);
730 html("</td><td class='form'>"); 730 html("</td><td class='form'>");
731 html("<form class='right' method='get' action='"); 731 html("<form class='right' method='get' action='");
732 if (ctx->cfg.virtual_root) 732 if (ctx->cfg.virtual_root)
733 html_url_path(cgit_fileurl(ctx->qry.repo, "log", 733 html_url_path(cgit_fileurl(ctx->qry.repo, "log",
734 ctx->qry.path, NULL)); 734 ctx->qry.path, NULL));
735 html("'>\n"); 735 html("'>\n");
736 cgit_add_hidden_formfields(1, 0, "log"); 736 cgit_add_hidden_formfields(1, 0, "log");
737 html("<select name='qt'>\n"); 737 html("<select name='qt'>\n");
738 html_option("grep", "log msg", ctx->qry.grep); 738 html_option("grep", "log msg", ctx->qry.grep);
739 html_option("author", "author", ctx->qry.grep); 739 html_option("author", "author", ctx->qry.grep);
740 html_option("committer", "committer", ctx->qry.grep); 740 html_option("committer", "committer", ctx->qry.grep);
741 html_option("range", "range", ctx->qry.grep);
741 html("</select>\n"); 742 html("</select>\n");
742 html("<input class='txt' type='text' size='10' name='q' value='"); 743 html("<input class='txt' type='text' size='10' name='q' value='");
743 html_attr(ctx->qry.search); 744 html_attr(ctx->qry.search);
744 html("'/>\n"); 745 html("'/>\n");
745 html("<input type='submit' value='search'/>\n"); 746 html("<input type='submit' value='search'/>\n");
746 html("</form>\n"); 747 html("</form>\n");
747 } else { 748 } else {
748 site_link(NULL, "index", NULL, hc(cmd, "repolist"), NULL, 0); 749 site_link(NULL, "index", NULL, hc(cmd, "repolist"), NULL, 0);
749 if (ctx->cfg.root_readme) 750 if (ctx->cfg.root_readme)
750 site_link("about", "about", NULL, hc(cmd, "about"), 751 site_link("about", "about", NULL, hc(cmd, "about"),
751 NULL, 0); 752 NULL, 0);
752 html("</td><td class='form'>"); 753 html("</td><td class='form'>");
753 html("<form method='get' action='"); 754 html("<form method='get' action='");
754 html_attr(cgit_rooturl()); 755 html_attr(cgit_rooturl());
755 html("'>\n"); 756 html("'>\n");
756 html("<input type='text' name='q' size='10' value='"); 757 html("<input type='text' name='q' size='10' value='");
757 html_attr(ctx->qry.search); 758 html_attr(ctx->qry.search);
758 html("'/>\n"); 759 html("'/>\n");
759 html("<input type='submit' value='search'/>\n"); 760 html("<input type='submit' value='search'/>\n");
760 html("</form>"); 761 html("</form>");
761 } 762 }
762 html("</td></tr></table>\n"); 763 html("</td></tr></table>\n");
763 html("<div class='content'>"); 764 html("<div class='content'>");
764} 765}
765 766
766void cgit_print_filemode(unsigned short mode) 767void cgit_print_filemode(unsigned short mode)
767{ 768{
768 if (S_ISDIR(mode)) 769 if (S_ISDIR(mode))
769 html("d"); 770 html("d");
770 else if (S_ISLNK(mode)) 771 else if (S_ISLNK(mode))
771 html("l"); 772 html("l");
772 else if (S_ISGITLINK(mode)) 773 else if (S_ISGITLINK(mode))
773 html("m"); 774 html("m");
774 else 775 else
775 html("-"); 776 html("-");
776 html_fileperm(mode >> 6); 777 html_fileperm(mode >> 6);
777 html_fileperm(mode >> 3); 778 html_fileperm(mode >> 3);
778 html_fileperm(mode); 779 html_fileperm(mode);
779} 780}
780 781
781void cgit_print_snapshot_links(const char *repo, const char *head, 782void cgit_print_snapshot_links(const char *repo, const char *head,
782 const char *hex, int snapshots) 783 const char *hex, int snapshots)
783{ 784{
784 const struct cgit_snapshot_format* f; 785 const struct cgit_snapshot_format* f;
785 char *prefix; 786 char *prefix;
786 char *filename; 787 char *filename;
787 unsigned char sha1[20]; 788 unsigned char sha1[20];
788 789
789 if (get_sha1(fmt("refs/tags/%s", hex), sha1) == 0 && 790 if (get_sha1(fmt("refs/tags/%s", hex), sha1) == 0 &&
790 (hex[0] == 'v' || hex[0] == 'V') && isdigit(hex[1])) 791 (hex[0] == 'v' || hex[0] == 'V') && isdigit(hex[1]))
791 hex++; 792 hex++;
792 prefix = xstrdup(fmt("%s-%s", cgit_repobasename(repo), hex)); 793 prefix = xstrdup(fmt("%s-%s", cgit_repobasename(repo), hex));
793 for (f = cgit_snapshot_formats; f->suffix; f++) { 794 for (f = cgit_snapshot_formats; f->suffix; f++) {
794 if (!(snapshots & f->bit)) 795 if (!(snapshots & f->bit))
795 continue; 796 continue;
796 filename = fmt("%s%s", prefix, f->suffix); 797 filename = fmt("%s%s", prefix, f->suffix);
797 cgit_snapshot_link(filename, NULL, NULL, NULL, NULL, filename); 798 cgit_snapshot_link(filename, NULL, NULL, NULL, NULL, filename);
798 html("<br/>"); 799 html("<br/>");
799 } 800 }
800} 801}