summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c3
-rw-r--r--cgit.h6
-rw-r--r--cgitrc5
-rw-r--r--shared.c4
-rw-r--r--ui-commit.c2
-rw-r--r--ui-snapshot.c43
6 files changed, 48 insertions, 15 deletions
diff --git a/cgit.c b/cgit.c
index 7b55b7b..8795bbc 100644
--- a/cgit.c
+++ b/cgit.c
@@ -1,201 +1,202 @@
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.%s.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 cache_safe_filename(cgit_querystring)));
32 item->ttl = cgit_cache_repo_ttl; 32 item->ttl = cgit_cache_repo_ttl;
33 } else { 33 } else {
34 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,
35 cache_safe_filename(cgit_repo->url), 35 cache_safe_filename(cgit_repo->url),
36 cgit_query_page, 36 cgit_query_page,
37 cache_safe_filename(cgit_querystring))); 37 cache_safe_filename(cgit_querystring)));
38 if (cgit_query_has_symref) 38 if (cgit_query_has_symref)
39 item->ttl = cgit_cache_dynamic_ttl; 39 item->ttl = cgit_cache_dynamic_ttl;
40 else if (cgit_query_has_sha1) 40 else if (cgit_query_has_sha1)
41 item->ttl = cgit_cache_static_ttl; 41 item->ttl = cgit_cache_static_ttl;
42 else 42 else
43 item->ttl = cgit_cache_repo_ttl; 43 item->ttl = cgit_cache_repo_ttl;
44 } 44 }
45 return 1; 45 return 1;
46} 46}
47 47
48static void cgit_print_repo_page(struct cacheitem *item) 48static void cgit_print_repo_page(struct cacheitem *item)
49{ 49{
50 char *title; 50 char *title;
51 int show_search; 51 int show_search;
52 52
53 if (!cgit_query_head) 53 if (!cgit_query_head)
54 cgit_query_head = cgit_repo->defbranch; 54 cgit_query_head = cgit_repo->defbranch;
55 55
56 if (chdir(cgit_repo->path)) { 56 if (chdir(cgit_repo->path)) {
57 title = fmt("%s - %s", cgit_root_title, "Bad request"); 57 title = fmt("%s - %s", cgit_root_title, "Bad request");
58 cgit_print_docstart(title, item); 58 cgit_print_docstart(title, item);
59 cgit_print_pageheader(title, 0); 59 cgit_print_pageheader(title, 0);
60 cgit_print_error(fmt("Unable to scan repository: %s", 60 cgit_print_error(fmt("Unable to scan repository: %s",
61 strerror(errno))); 61 strerror(errno)));
62 cgit_print_docend(); 62 cgit_print_docend();
63 return; 63 return;
64 } 64 }
65 65
66 title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc); 66 title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc);
67 show_search = 0; 67 show_search = 0;
68 setenv("GIT_DIR", cgit_repo->path, 1); 68 setenv("GIT_DIR", cgit_repo->path, 1);
69 69
70 if ((cgit_cmd == CMD_SNAPSHOT) && cgit_repo->snapshots) { 70 if ((cgit_cmd == CMD_SNAPSHOT) && cgit_repo->snapshots) {
71 cgit_print_snapshot(item, cgit_query_sha1, 71 cgit_print_snapshot(item, cgit_query_sha1,
72 cgit_repobasename(cgit_repo->url), 72 cgit_repobasename(cgit_repo->url),
73 cgit_query_name); 73 cgit_query_name,
74 cgit_repo->snapshots );
74 return; 75 return;
75 } 76 }
76 77
77 if (cgit_cmd == CMD_BLOB) { 78 if (cgit_cmd == CMD_BLOB) {
78 cgit_print_blob(item, cgit_query_sha1, cgit_query_path); 79 cgit_print_blob(item, cgit_query_sha1, cgit_query_path);
79 return; 80 return;
80 } 81 }
81 82
82 show_search = (cgit_cmd == CMD_LOG); 83 show_search = (cgit_cmd == CMD_LOG);
83 cgit_print_docstart(title, item); 84 cgit_print_docstart(title, item);
84 if (!cgit_cmd) { 85 if (!cgit_cmd) {
85 cgit_print_pageheader("summary", show_search); 86 cgit_print_pageheader("summary", show_search);
86 cgit_print_summary(); 87 cgit_print_summary();
87 cgit_print_docend(); 88 cgit_print_docend();
88 return; 89 return;
89 } 90 }
90 91
91 cgit_print_pageheader(cgit_query_page, show_search); 92 cgit_print_pageheader(cgit_query_page, show_search);
92 93
93 switch(cgit_cmd) { 94 switch(cgit_cmd) {
94 case CMD_LOG: 95 case CMD_LOG:
95 cgit_print_log(cgit_query_sha1, cgit_query_ofs, 96 cgit_print_log(cgit_query_sha1, cgit_query_ofs,
96 cgit_max_commit_count, cgit_query_search, 97 cgit_max_commit_count, cgit_query_search,
97 cgit_query_path, 1); 98 cgit_query_path, 1);
98 break; 99 break;
99 case CMD_TREE: 100 case CMD_TREE:
100 cgit_print_tree(cgit_query_sha1, cgit_query_path); 101 cgit_print_tree(cgit_query_sha1, cgit_query_path);
101 break; 102 break;
102 case CMD_COMMIT: 103 case CMD_COMMIT:
103 cgit_print_commit(cgit_query_sha1); 104 cgit_print_commit(cgit_query_sha1);
104 break; 105 break;
105 case CMD_DIFF: 106 case CMD_DIFF:
106 cgit_print_diff(cgit_query_sha1, cgit_query_sha2); 107 cgit_print_diff(cgit_query_sha1, cgit_query_sha2);
107 break; 108 break;
108 default: 109 default:
109 cgit_print_error("Invalid request"); 110 cgit_print_error("Invalid request");
110 } 111 }
111 cgit_print_docend(); 112 cgit_print_docend();
112} 113}
113 114
114static void cgit_fill_cache(struct cacheitem *item, int use_cache) 115static void cgit_fill_cache(struct cacheitem *item, int use_cache)
115{ 116{
116 static char buf[PATH_MAX]; 117 static char buf[PATH_MAX];
117 int stdout2; 118 int stdout2;
118 119
119 getcwd(buf, sizeof(buf)); 120 getcwd(buf, sizeof(buf));
120 item->st.st_mtime = time(NULL); 121 item->st.st_mtime = time(NULL);
121 122
122 if (use_cache) { 123 if (use_cache) {
123 stdout2 = chk_positive(dup(STDOUT_FILENO), 124 stdout2 = chk_positive(dup(STDOUT_FILENO),
124 "Preserving STDOUT"); 125 "Preserving STDOUT");
125 chk_zero(close(STDOUT_FILENO), "Closing STDOUT"); 126 chk_zero(close(STDOUT_FILENO), "Closing STDOUT");
126 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)"); 127 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)");
127 } 128 }
128 129
129 if (cgit_repo) 130 if (cgit_repo)
130 cgit_print_repo_page(item); 131 cgit_print_repo_page(item);
131 else 132 else
132 cgit_print_repolist(item); 133 cgit_print_repolist(item);
133 134
134 if (use_cache) { 135 if (use_cache) {
135 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT"); 136 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT");
136 chk_positive(dup2(stdout2, STDOUT_FILENO), 137 chk_positive(dup2(stdout2, STDOUT_FILENO),
137 "Restoring original STDOUT"); 138 "Restoring original STDOUT");
138 chk_zero(close(stdout2), "Closing temporary STDOUT"); 139 chk_zero(close(stdout2), "Closing temporary STDOUT");
139 } 140 }
140 141
141 chdir(buf); 142 chdir(buf);
142} 143}
143 144
144static void cgit_check_cache(struct cacheitem *item) 145static void cgit_check_cache(struct cacheitem *item)
145{ 146{
146 int i = 0; 147 int i = 0;
147 148
148 top: 149 top:
149 if (++i > cgit_max_lock_attempts) { 150 if (++i > cgit_max_lock_attempts) {
150 die("cgit_refresh_cache: unable to lock %s: %s", 151 die("cgit_refresh_cache: unable to lock %s: %s",
151 item->name, strerror(errno)); 152 item->name, strerror(errno));
152 } 153 }
153 if (!cache_exist(item)) { 154 if (!cache_exist(item)) {
154 if (!cache_lock(item)) { 155 if (!cache_lock(item)) {
155 sleep(1); 156 sleep(1);
156 goto top; 157 goto top;
157 } 158 }
158 if (!cache_exist(item)) { 159 if (!cache_exist(item)) {
159 cgit_fill_cache(item, 1); 160 cgit_fill_cache(item, 1);
160 cache_unlock(item); 161 cache_unlock(item);
161 } else { 162 } else {
162 cache_cancel_lock(item); 163 cache_cancel_lock(item);
163 } 164 }
164 } else if (cache_expired(item) && cache_lock(item)) { 165 } else if (cache_expired(item) && cache_lock(item)) {
165 if (cache_expired(item)) { 166 if (cache_expired(item)) {
166 cgit_fill_cache(item, 1); 167 cgit_fill_cache(item, 1);
167 cache_unlock(item); 168 cache_unlock(item);
168 } else { 169 } else {
169 cache_cancel_lock(item); 170 cache_cancel_lock(item);
170 } 171 }
171 } 172 }
172} 173}
173 174
174static void cgit_print_cache(struct cacheitem *item) 175static void cgit_print_cache(struct cacheitem *item)
175{ 176{
176 static char buf[4096]; 177 static char buf[4096];
177 ssize_t i; 178 ssize_t i;
178 179
179 int fd = open(item->name, O_RDONLY); 180 int fd = open(item->name, O_RDONLY);
180 if (fd<0) 181 if (fd<0)
181 die("Unable to open cached file %s", item->name); 182 die("Unable to open cached file %s", item->name);
182 183
183 while((i=read(fd, buf, sizeof(buf))) > 0) 184 while((i=read(fd, buf, sizeof(buf))) > 0)
184 write(STDOUT_FILENO, buf, i); 185 write(STDOUT_FILENO, buf, i);
185 186
186 close(fd); 187 close(fd);
187} 188}
188 189
189static void cgit_parse_args(int argc, const char **argv) 190static void cgit_parse_args(int argc, const char **argv)
190{ 191{
191 int i; 192 int i;
192 193
193 for (i = 1; i < argc; i++) { 194 for (i = 1; i < argc; i++) {
194 if (!strncmp(argv[i], "--cache=", 8)) { 195 if (!strncmp(argv[i], "--cache=", 8)) {
195 cgit_cache_root = xstrdup(argv[i]+8); 196 cgit_cache_root = xstrdup(argv[i]+8);
196 } 197 }
197 if (!strcmp(argv[i], "--nocache")) { 198 if (!strcmp(argv[i], "--nocache")) {
198 cgit_nocache = 1; 199 cgit_nocache = 1;
199 } 200 }
200 if (!strncmp(argv[i], "--query=", 8)) { 201 if (!strncmp(argv[i], "--query=", 8)) {
201 cgit_querystring = xstrdup(argv[i]+8); 202 cgit_querystring = xstrdup(argv[i]+8);
diff --git a/cgit.h b/cgit.h
index 1dbf901..ea61be7 100644
--- a/cgit.h
+++ b/cgit.h
@@ -110,132 +110,134 @@ extern char *cgit_logo;
110extern char *cgit_index_header; 110extern char *cgit_index_header;
111extern char *cgit_logo_link; 111extern char *cgit_logo_link;
112extern char *cgit_module_link; 112extern char *cgit_module_link;
113extern char *cgit_agefile; 113extern char *cgit_agefile;
114extern char *cgit_virtual_root; 114extern char *cgit_virtual_root;
115extern char *cgit_script_name; 115extern char *cgit_script_name;
116extern char *cgit_cache_root; 116extern char *cgit_cache_root;
117extern char *cgit_repo_group; 117extern char *cgit_repo_group;
118 118
119extern int cgit_nocache; 119extern int cgit_nocache;
120extern int cgit_snapshots; 120extern int cgit_snapshots;
121extern int cgit_enable_index_links; 121extern int cgit_enable_index_links;
122extern int cgit_enable_log_filecount; 122extern int cgit_enable_log_filecount;
123extern int cgit_enable_log_linecount; 123extern int cgit_enable_log_linecount;
124extern int cgit_max_lock_attempts; 124extern int cgit_max_lock_attempts;
125extern int cgit_cache_root_ttl; 125extern int cgit_cache_root_ttl;
126extern int cgit_cache_repo_ttl; 126extern int cgit_cache_repo_ttl;
127extern int cgit_cache_dynamic_ttl; 127extern int cgit_cache_dynamic_ttl;
128extern int cgit_cache_static_ttl; 128extern int cgit_cache_static_ttl;
129extern int cgit_cache_max_create_time; 129extern int cgit_cache_max_create_time;
130extern int cgit_summary_log; 130extern int cgit_summary_log;
131 131
132extern int cgit_max_msg_len; 132extern int cgit_max_msg_len;
133extern int cgit_max_repodesc_len; 133extern int cgit_max_repodesc_len;
134extern int cgit_max_commit_count; 134extern int cgit_max_commit_count;
135 135
136extern int cgit_query_has_symref; 136extern int cgit_query_has_symref;
137extern int cgit_query_has_sha1; 137extern int cgit_query_has_sha1;
138 138
139extern char *cgit_querystring; 139extern char *cgit_querystring;
140extern char *cgit_query_repo; 140extern char *cgit_query_repo;
141extern char *cgit_query_page; 141extern char *cgit_query_page;
142extern char *cgit_query_search; 142extern char *cgit_query_search;
143extern char *cgit_query_head; 143extern char *cgit_query_head;
144extern char *cgit_query_sha1; 144extern char *cgit_query_sha1;
145extern char *cgit_query_sha2; 145extern char *cgit_query_sha2;
146extern char *cgit_query_path; 146extern char *cgit_query_path;
147extern char *cgit_query_name; 147extern char *cgit_query_name;
148extern int cgit_query_ofs; 148extern int cgit_query_ofs;
149 149
150extern int htmlfd; 150extern int htmlfd;
151 151
152extern int cgit_get_cmd_index(const char *cmd); 152extern int cgit_get_cmd_index(const char *cmd);
153extern struct repoinfo *cgit_get_repoinfo(const char *url); 153extern struct repoinfo *cgit_get_repoinfo(const char *url);
154extern void cgit_global_config_cb(const char *name, const char *value); 154extern void cgit_global_config_cb(const char *name, const char *value);
155extern void cgit_repo_config_cb(const char *name, const char *value); 155extern void cgit_repo_config_cb(const char *name, const char *value);
156extern void cgit_querystring_cb(const char *name, const char *value); 156extern void cgit_querystring_cb(const char *name, const char *value);
157 157
158extern int chk_zero(int result, char *msg); 158extern int chk_zero(int result, char *msg);
159extern int chk_positive(int result, char *msg); 159extern int chk_positive(int result, char *msg);
160extern int chk_non_negative(int result, char *msg); 160extern int chk_non_negative(int result, char *msg);
161 161
162extern int hextoint(char c); 162extern int hextoint(char c);
163extern char *trim_end(const char *str, char c); 163extern char *trim_end(const char *str, char c);
164 164
165extern void *cgit_free_commitinfo(struct commitinfo *info); 165extern void *cgit_free_commitinfo(struct commitinfo *info);
166 166
167extern int cgit_diff_files(const unsigned char *old_sha1, 167extern int cgit_diff_files(const unsigned char *old_sha1,
168 const unsigned char *new_sha1, 168 const unsigned char *new_sha1,
169 linediff_fn fn); 169 linediff_fn fn);
170 170
171extern void cgit_diff_tree(const unsigned char *old_sha1, 171extern void cgit_diff_tree(const unsigned char *old_sha1,
172 const unsigned char *new_sha1, 172 const unsigned char *new_sha1,
173 filepair_fn fn); 173 filepair_fn fn);
174 174
175extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); 175extern void cgit_diff_commit(struct commit *commit, filepair_fn fn);
176 176
177extern char *fmt(const char *format,...); 177extern char *fmt(const char *format,...);
178 178
179extern void html(const char *txt); 179extern void html(const char *txt);
180extern void htmlf(const char *format,...); 180extern void htmlf(const char *format,...);
181extern void html_txt(char *txt); 181extern void html_txt(char *txt);
182extern void html_ntxt(int len, char *txt); 182extern void html_ntxt(int len, char *txt);
183extern void html_attr(char *txt); 183extern void html_attr(char *txt);
184extern void html_hidden(char *name, char *value); 184extern void html_hidden(char *name, char *value);
185extern void html_link_open(char *url, char *title, char *class); 185extern void html_link_open(char *url, char *title, char *class);
186extern void html_link_close(void); 186extern void html_link_close(void);
187extern void html_filemode(unsigned short mode); 187extern void html_filemode(unsigned short mode);
188extern int html_include(const char *filename); 188extern int html_include(const char *filename);
189 189
190extern int cgit_read_config(const char *filename, configfn fn); 190extern int cgit_read_config(const char *filename, configfn fn);
191extern int cgit_parse_query(char *txt, configfn fn); 191extern int cgit_parse_query(char *txt, configfn fn);
192extern struct commitinfo *cgit_parse_commit(struct commit *commit); 192extern struct commitinfo *cgit_parse_commit(struct commit *commit);
193extern struct taginfo *cgit_parse_tag(struct tag *tag); 193extern struct taginfo *cgit_parse_tag(struct tag *tag);
194extern void cgit_parse_url(const char *url); 194extern void cgit_parse_url(const char *url);
195 195
196extern char *cache_safe_filename(const char *unsafe); 196extern char *cache_safe_filename(const char *unsafe);
197extern int cache_lock(struct cacheitem *item); 197extern int cache_lock(struct cacheitem *item);
198extern int cache_unlock(struct cacheitem *item); 198extern int cache_unlock(struct cacheitem *item);
199extern int cache_cancel_lock(struct cacheitem *item); 199extern int cache_cancel_lock(struct cacheitem *item);
200extern int cache_exist(struct cacheitem *item); 200extern int cache_exist(struct cacheitem *item);
201extern int cache_expired(struct cacheitem *item); 201extern int cache_expired(struct cacheitem *item);
202 202
203extern char *cgit_repourl(const char *reponame); 203extern char *cgit_repourl(const char *reponame);
204extern char *cgit_fileurl(const char *reponame, const char *pagename, 204extern char *cgit_fileurl(const char *reponame, const char *pagename,
205 const char *filename, const char *query); 205 const char *filename, const char *query);
206extern char *cgit_pageurl(const char *reponame, const char *pagename, 206extern char *cgit_pageurl(const char *reponame, const char *pagename,
207 const char *query); 207 const char *query);
208 208
209extern const char *cgit_repobasename(const char *reponame); 209extern const char *cgit_repobasename(const char *reponame);
210 210
211extern void cgit_tree_link(char *name, char *title, char *class, char *head, 211extern void cgit_tree_link(char *name, char *title, char *class, char *head,
212 char *rev, char *path); 212 char *rev, char *path);
213extern void cgit_log_link(char *name, char *title, char *class, char *head, 213extern void cgit_log_link(char *name, char *title, char *class, char *head,
214 char *rev, char *path, int ofs); 214 char *rev, char *path, int ofs);
215extern void cgit_commit_link(char *name, char *title, char *class, char *head, 215extern void cgit_commit_link(char *name, char *title, char *class, char *head,
216 char *rev); 216 char *rev);
217extern void cgit_diff_link(char *name, char *title, char *class, char *head, 217extern void cgit_diff_link(char *name, char *title, char *class, char *head,
218 char *new_rev, char *old_rev, char *path); 218 char *new_rev, char *old_rev, char *path);
219 219
220extern void cgit_print_error(char *msg); 220extern void cgit_print_error(char *msg);
221extern void cgit_print_date(time_t secs, char *format); 221extern void cgit_print_date(time_t secs, char *format);
222extern void cgit_print_age(time_t t, time_t max_relative, char *format); 222extern void cgit_print_age(time_t t, time_t max_relative, char *format);
223extern void cgit_print_docstart(char *title, struct cacheitem *item); 223extern void cgit_print_docstart(char *title, struct cacheitem *item);
224extern void cgit_print_docend(); 224extern void cgit_print_docend();
225extern void cgit_print_pageheader(char *title, int show_search); 225extern void cgit_print_pageheader(char *title, int show_search);
226extern void cgit_print_snapshot_start(const char *mimetype, 226extern void cgit_print_snapshot_start(const char *mimetype,
227 const char *filename, 227 const char *filename,
228 struct cacheitem *item); 228 struct cacheitem *item);
229 229
230extern void cgit_print_repolist(struct cacheitem *item); 230extern void cgit_print_repolist(struct cacheitem *item);
231extern void cgit_print_summary(); 231extern void cgit_print_summary();
232extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path, int pager); 232extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path, int pager);
233extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path); 233extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path);
234extern void cgit_print_tree(const char *rev, char *path); 234extern void cgit_print_tree(const char *rev, char *path);
235extern void cgit_print_commit(char *hex); 235extern void cgit_print_commit(char *hex);
236extern void cgit_print_diff(const char *new_hex, const char *old_hex); 236extern void cgit_print_diff(const char *new_hex, const char *old_hex);
237extern void cgit_print_snapshot(struct cacheitem *item, const char *hex, 237extern void cgit_print_snapshot(struct cacheitem *item, const char *hex,
238 const char *prefix, const char *filename); 238 const char *prefix, const char *filename,
239extern void cgit_print_snapshot_links(const char *repo, const char *hex); 239 int snapshot);
240extern void cgit_print_snapshot_links(const char *repo, const char *hex,int snapshots);
241extern int cgit_parse_snapshots_mask(const char *str);
240 242
241#endif /* CGIT_H */ 243#endif /* CGIT_H */
diff --git a/cgitrc b/cgitrc
index 40877f8..1040997 100644
--- a/cgitrc
+++ b/cgitrc
@@ -1,120 +1,121 @@
1## 1##
2## cgitrc: template for /etc/cgitrc 2## cgitrc: template for /etc/cgitrc
3## 3##
4 4
5 5
6## Uncomment and set to 1 to deactivate caching of generated pages. Mostly 6## Uncomment and set to 1 to deactivate caching of generated pages. Mostly
7## usefull for testing. 7## usefull for testing.
8#nocache=0 8#nocache=0
9 9
10 10
11## Enable/disable snapshots by default. This can be overridden per repo 11## Set allowed snapshot types by default. Can be overridden per repo
12# can be any combination of zip/tar.gz/tar.bz2/tar
12#snapshots=0 13#snapshots=0
13 14
14 15
15## Enable/disable extra links to summary/log/tree per repo on index page 16## Enable/disable extra links to summary/log/tree per repo on index page
16#enable-index-links=0 17#enable-index-links=0
17 18
18 19
19## Enable/disable display of 'number of files changed' in log view 20## Enable/disable display of 'number of files changed' in log view
20#enable-log-filecount=0 21#enable-log-filecount=0
21 22
22 23
23## Enable/disable display of 'number of lines changed' in log view 24## Enable/disable display of 'number of lines changed' in log view
24#enable-log-linecount=0 25#enable-log-linecount=0
25 26
26 27
27## Enable/disable display of HEAD shortlog in summary view. Set it to maximum 28## Enable/disable display of HEAD shortlog in summary view. Set it to maximum
28## number of commits that should be displayed 29## number of commits that should be displayed
29#summary-log=0 30#summary-log=0
30 31
31 32
32## Specify a root for virtual urls. This makes cgit generate urls like 33## Specify a root for virtual urls. This makes cgit generate urls like
33## 34##
34## http://localhost/git/repo/log/?id=master 35## http://localhost/git/repo/log/?id=master
35## 36##
36## instead of 37## instead of
37## 38##
38## http://localhost/cgit/cgit.cgi?r=repo&p=log&id=master 39## http://localhost/cgit/cgit.cgi?r=repo&p=log&id=master
39## 40##
40## For this to work with apache, rewrite rules must be added to httpd.conf, 41## For this to work with apache, rewrite rules must be added to httpd.conf,
41## possibly looking something like this: 42## possibly looking something like this:
42## 43##
43## RewriteRule ^/git/$ /cgit/cgit.cgi [L,QSA] 44## RewriteRule ^/git/$ /cgit/cgit.cgi [L,QSA]
44## RewriteRule ^/git/([^/]+)/$ /cgit/cgit.cgi?r=$1 [L,QSA] 45## RewriteRule ^/git/([^/]+)/$ /cgit/cgit.cgi?r=$1 [L,QSA]
45## RewriteRule ^/git/([^/]+)/([^/]+)/$ /cgit/cgit.cgi?r=$1&p=$2 [L,QSA] 46## RewriteRule ^/git/([^/]+)/([^/]+)/$ /cgit/cgit.cgi?r=$1&p=$2 [L,QSA]
46## 47##
47## This setting is disabled by default. 48## This setting is disabled by default.
48#virtual-root=/git 49#virtual-root=/git
49 50
50 51
51## Set the title printed on the root page 52## Set the title printed on the root page
52#root-title=Git repository browser 53#root-title=Git repository browser
53 54
54 55
55## Link to css file 56## Link to css file
56#css=/cgit/cgit.css 57#css=/cgit/cgit.css
57 58
58 59
59## Link to logo file 60## Link to logo file
60#logo=/cgit/git-logo.png 61#logo=/cgit/git-logo.png
61 62
62 63
63## Url loaded when clicking the logo 64## Url loaded when clicking the logo
64#logo-link=http://www.kernel.org/pub/software/scm/git/docs/ 65#logo-link=http://www.kernel.org/pub/software/scm/git/docs/
65 66
66 67
67## Url loaded when clicking a submodule link 68## Url loaded when clicking a submodule link
68#module-link=./?repo=%s&page=commit&id=%s 69#module-link=./?repo=%s&page=commit&id=%s
69 70
70 71
71## Number of chars shown of repo description (in repolist view) 72## Number of chars shown of repo description (in repolist view)
72#max-repodesc-length=60 73#max-repodesc-length=60
73 74
74 75
75## Number of chars shown of commit subject message (in log view) 76## Number of chars shown of commit subject message (in log view)
76#max-message-length=60 77#max-message-length=60
77 78
78 79
79## Number of commits per page in log view 80## Number of commits per page in log view
80#max-commit-count=50 81#max-commit-count=50
81 82
82 83
83## Root of cached output 84## Root of cached output
84#cache-root=/var/cache/cgit 85#cache-root=/var/cache/cgit
85 86
86 87
87## Include another config-file 88## Include another config-file
88#include=/var/cgit/repolist 89#include=/var/cgit/repolist
89 90
90## 91##
91## Time-To-Live settings: specifies how long (in minutes) different pages 92## Time-To-Live settings: specifies how long (in minutes) different pages
92## should be cached (0 for instant expiration, -1 for immortal pages) 93## should be cached (0 for instant expiration, -1 for immortal pages)
93## 94##
94 95
95## ttl for root page 96## ttl for root page
96#cache-root-ttl=5 97#cache-root-ttl=5
97 98
98## ttl for repo summary page 99## ttl for repo summary page
99#cache-repo-ttl=5 100#cache-repo-ttl=5
100 101
101## ttl for other dynamic pages 102## ttl for other dynamic pages
102#cache-dynamic-ttl=5 103#cache-dynamic-ttl=5
103 104
104## ttl for static pages (addressed by SHA-1) 105## ttl for static pages (addressed by SHA-1)
105#cache-static-ttl=-1 106#cache-static-ttl=-1
106 107
107 108
108 109
109## Example repository entry. Required values are repo.url and repo.path (each 110## Example repository entry. Required values are repo.url and repo.path (each
110## repository section must start with repo.url). 111## repository section must start with repo.url).
111#repo.url=cgit 112#repo.url=cgit
112#repo.name=cgit 113#repo.name=cgit
113#repo.desc=the caching cgi for git 114#repo.desc=the caching cgi for git
114#repo.path=/pub/git/cgit 115#repo.path=/pub/git/cgit
115#repo.owner=Lars Hjemli 116#repo.owner=Lars Hjemli
116 #repo.snapshots=1 # override a sitewide snapshot-setting 117 #repo.snapshots=tar.bz2 # override a sitewide snapshot-setting
117 #repo.enable-log-filecount=0 # override the default filecount setting 118 #repo.enable-log-filecount=0 # override the default filecount setting
118 #repo.enable-log-linecount=0 # override the default linecount setting 119 #repo.enable-log-linecount=0 # override the default linecount setting
119 #repo.module-link=/git/%s/commit/?id=%s # override the standard module-link 120 #repo.module-link=/git/%s/commit/?id=%s # override the standard module-link
120 #repo.readme=info/web/readme # specify a file to include on summary page 121 #repo.readme=info/web/readme # specify a file to include on summary page
diff --git a/shared.c b/shared.c
index 65fc8b2..ccbde27 100644
--- a/shared.c
+++ b/shared.c
@@ -30,299 +30,299 @@ int cgit_nocache = 0;
30int cgit_snapshots = 0; 30int cgit_snapshots = 0;
31int cgit_enable_index_links = 0; 31int cgit_enable_index_links = 0;
32int cgit_enable_log_filecount = 0; 32int cgit_enable_log_filecount = 0;
33int cgit_enable_log_linecount = 0; 33int cgit_enable_log_linecount = 0;
34int cgit_max_lock_attempts = 5; 34int cgit_max_lock_attempts = 5;
35int cgit_cache_root_ttl = 5; 35int cgit_cache_root_ttl = 5;
36int cgit_cache_repo_ttl = 5; 36int cgit_cache_repo_ttl = 5;
37int cgit_cache_dynamic_ttl = 5; 37int cgit_cache_dynamic_ttl = 5;
38int cgit_cache_static_ttl = -1; 38int cgit_cache_static_ttl = -1;
39int cgit_cache_max_create_time = 5; 39int cgit_cache_max_create_time = 5;
40int cgit_summary_log = 0; 40int cgit_summary_log = 0;
41 41
42int cgit_max_msg_len = 60; 42int cgit_max_msg_len = 60;
43int cgit_max_repodesc_len = 60; 43int cgit_max_repodesc_len = 60;
44int cgit_max_commit_count = 50; 44int cgit_max_commit_count = 50;
45 45
46int cgit_query_has_symref = 0; 46int cgit_query_has_symref = 0;
47int cgit_query_has_sha1 = 0; 47int cgit_query_has_sha1 = 0;
48 48
49char *cgit_querystring = NULL; 49char *cgit_querystring = NULL;
50char *cgit_query_repo = NULL; 50char *cgit_query_repo = NULL;
51char *cgit_query_page = NULL; 51char *cgit_query_page = NULL;
52char *cgit_query_head = NULL; 52char *cgit_query_head = NULL;
53char *cgit_query_search = NULL; 53char *cgit_query_search = NULL;
54char *cgit_query_sha1 = NULL; 54char *cgit_query_sha1 = NULL;
55char *cgit_query_sha2 = NULL; 55char *cgit_query_sha2 = NULL;
56char *cgit_query_path = NULL; 56char *cgit_query_path = NULL;
57char *cgit_query_name = NULL; 57char *cgit_query_name = NULL;
58int cgit_query_ofs = 0; 58int cgit_query_ofs = 0;
59 59
60int htmlfd = 0; 60int htmlfd = 0;
61 61
62 62
63int cgit_get_cmd_index(const char *cmd) 63int cgit_get_cmd_index(const char *cmd)
64{ 64{
65 static char *cmds[] = {"log", "commit", "diff", "tree", "blob", 65 static char *cmds[] = {"log", "commit", "diff", "tree", "blob",
66 "snapshot", NULL}; 66 "snapshot", NULL};
67 int i; 67 int i;
68 68
69 for(i = 0; cmds[i]; i++) 69 for(i = 0; cmds[i]; i++)
70 if (!strcmp(cmd, cmds[i])) 70 if (!strcmp(cmd, cmds[i]))
71 return i + 1; 71 return i + 1;
72 return 0; 72 return 0;
73} 73}
74 74
75int chk_zero(int result, char *msg) 75int chk_zero(int result, char *msg)
76{ 76{
77 if (result != 0) 77 if (result != 0)
78 die("%s: %s", msg, strerror(errno)); 78 die("%s: %s", msg, strerror(errno));
79 return result; 79 return result;
80} 80}
81 81
82int chk_positive(int result, char *msg) 82int chk_positive(int result, char *msg)
83{ 83{
84 if (result <= 0) 84 if (result <= 0)
85 die("%s: %s", msg, strerror(errno)); 85 die("%s: %s", msg, strerror(errno));
86 return result; 86 return result;
87} 87}
88 88
89int chk_non_negative(int result, char *msg) 89int chk_non_negative(int result, char *msg)
90{ 90{
91 if (result < 0) 91 if (result < 0)
92 die("%s: %s",msg, strerror(errno)); 92 die("%s: %s",msg, strerror(errno));
93 return result; 93 return result;
94} 94}
95 95
96struct repoinfo *add_repo(const char *url) 96struct repoinfo *add_repo(const char *url)
97{ 97{
98 struct repoinfo *ret; 98 struct repoinfo *ret;
99 99
100 if (++cgit_repolist.count > cgit_repolist.length) { 100 if (++cgit_repolist.count > cgit_repolist.length) {
101 if (cgit_repolist.length == 0) 101 if (cgit_repolist.length == 0)
102 cgit_repolist.length = 8; 102 cgit_repolist.length = 8;
103 else 103 else
104 cgit_repolist.length *= 2; 104 cgit_repolist.length *= 2;
105 cgit_repolist.repos = xrealloc(cgit_repolist.repos, 105 cgit_repolist.repos = xrealloc(cgit_repolist.repos,
106 cgit_repolist.length * 106 cgit_repolist.length *
107 sizeof(struct repoinfo)); 107 sizeof(struct repoinfo));
108 } 108 }
109 109
110 ret = &cgit_repolist.repos[cgit_repolist.count-1]; 110 ret = &cgit_repolist.repos[cgit_repolist.count-1];
111 ret->url = xstrdup(url); 111 ret->url = xstrdup(url);
112 ret->name = ret->url; 112 ret->name = ret->url;
113 ret->path = NULL; 113 ret->path = NULL;
114 ret->desc = NULL; 114 ret->desc = NULL;
115 ret->owner = NULL; 115 ret->owner = NULL;
116 ret->group = cgit_repo_group; 116 ret->group = cgit_repo_group;
117 ret->defbranch = "master"; 117 ret->defbranch = "master";
118 ret->snapshots = cgit_snapshots; 118 ret->snapshots = cgit_snapshots;
119 ret->enable_log_filecount = cgit_enable_log_filecount; 119 ret->enable_log_filecount = cgit_enable_log_filecount;
120 ret->enable_log_linecount = cgit_enable_log_linecount; 120 ret->enable_log_linecount = cgit_enable_log_linecount;
121 ret->module_link = cgit_module_link; 121 ret->module_link = cgit_module_link;
122 ret->readme = NULL; 122 ret->readme = NULL;
123 return ret; 123 return ret;
124} 124}
125 125
126struct repoinfo *cgit_get_repoinfo(const char *url) 126struct repoinfo *cgit_get_repoinfo(const char *url)
127{ 127{
128 int i; 128 int i;
129 struct repoinfo *repo; 129 struct repoinfo *repo;
130 130
131 for (i=0; i<cgit_repolist.count; i++) { 131 for (i=0; i<cgit_repolist.count; i++) {
132 repo = &cgit_repolist.repos[i]; 132 repo = &cgit_repolist.repos[i];
133 if (!strcmp(repo->url, url)) 133 if (!strcmp(repo->url, url))
134 return repo; 134 return repo;
135 } 135 }
136 return NULL; 136 return NULL;
137} 137}
138 138
139void cgit_global_config_cb(const char *name, const char *value) 139void cgit_global_config_cb(const char *name, const char *value)
140{ 140{
141 if (!strcmp(name, "root-title")) 141 if (!strcmp(name, "root-title"))
142 cgit_root_title = xstrdup(value); 142 cgit_root_title = xstrdup(value);
143 else if (!strcmp(name, "css")) 143 else if (!strcmp(name, "css"))
144 cgit_css = xstrdup(value); 144 cgit_css = xstrdup(value);
145 else if (!strcmp(name, "logo")) 145 else if (!strcmp(name, "logo"))
146 cgit_logo = xstrdup(value); 146 cgit_logo = xstrdup(value);
147 else if (!strcmp(name, "index-header")) 147 else if (!strcmp(name, "index-header"))
148 cgit_index_header = xstrdup(value); 148 cgit_index_header = xstrdup(value);
149 else if (!strcmp(name, "logo-link")) 149 else if (!strcmp(name, "logo-link"))
150 cgit_logo_link = xstrdup(value); 150 cgit_logo_link = xstrdup(value);
151 else if (!strcmp(name, "module-link")) 151 else if (!strcmp(name, "module-link"))
152 cgit_module_link = xstrdup(value); 152 cgit_module_link = xstrdup(value);
153 else if (!strcmp(name, "virtual-root")) 153 else if (!strcmp(name, "virtual-root"))
154 cgit_virtual_root = xstrdup(value); 154 cgit_virtual_root = xstrdup(value);
155 else if (!strcmp(name, "nocache")) 155 else if (!strcmp(name, "nocache"))
156 cgit_nocache = atoi(value); 156 cgit_nocache = atoi(value);
157 else if (!strcmp(name, "snapshots")) 157 else if (!strcmp(name, "snapshots"))
158 cgit_snapshots = atoi(value); 158 cgit_snapshots = cgit_parse_snapshots_mask(value);
159 else if (!strcmp(name, "enable-index-links")) 159 else if (!strcmp(name, "enable-index-links"))
160 cgit_enable_index_links = atoi(value); 160 cgit_enable_index_links = atoi(value);
161 else if (!strcmp(name, "enable-log-filecount")) 161 else if (!strcmp(name, "enable-log-filecount"))
162 cgit_enable_log_filecount = atoi(value); 162 cgit_enable_log_filecount = atoi(value);
163 else if (!strcmp(name, "enable-log-linecount")) 163 else if (!strcmp(name, "enable-log-linecount"))
164 cgit_enable_log_linecount = atoi(value); 164 cgit_enable_log_linecount = atoi(value);
165 else if (!strcmp(name, "cache-root")) 165 else if (!strcmp(name, "cache-root"))
166 cgit_cache_root = xstrdup(value); 166 cgit_cache_root = xstrdup(value);
167 else if (!strcmp(name, "cache-root-ttl")) 167 else if (!strcmp(name, "cache-root-ttl"))
168 cgit_cache_root_ttl = atoi(value); 168 cgit_cache_root_ttl = atoi(value);
169 else if (!strcmp(name, "cache-repo-ttl")) 169 else if (!strcmp(name, "cache-repo-ttl"))
170 cgit_cache_repo_ttl = atoi(value); 170 cgit_cache_repo_ttl = atoi(value);
171 else if (!strcmp(name, "cache-static-ttl")) 171 else if (!strcmp(name, "cache-static-ttl"))
172 cgit_cache_static_ttl = atoi(value); 172 cgit_cache_static_ttl = atoi(value);
173 else if (!strcmp(name, "cache-dynamic-ttl")) 173 else if (!strcmp(name, "cache-dynamic-ttl"))
174 cgit_cache_dynamic_ttl = atoi(value); 174 cgit_cache_dynamic_ttl = atoi(value);
175 else if (!strcmp(name, "max-message-length")) 175 else if (!strcmp(name, "max-message-length"))
176 cgit_max_msg_len = atoi(value); 176 cgit_max_msg_len = atoi(value);
177 else if (!strcmp(name, "max-repodesc-length")) 177 else if (!strcmp(name, "max-repodesc-length"))
178 cgit_max_repodesc_len = atoi(value); 178 cgit_max_repodesc_len = atoi(value);
179 else if (!strcmp(name, "max-commit-count")) 179 else if (!strcmp(name, "max-commit-count"))
180 cgit_max_commit_count = atoi(value); 180 cgit_max_commit_count = atoi(value);
181 else if (!strcmp(name, "summary-log")) 181 else if (!strcmp(name, "summary-log"))
182 cgit_summary_log = atoi(value); 182 cgit_summary_log = atoi(value);
183 else if (!strcmp(name, "agefile")) 183 else if (!strcmp(name, "agefile"))
184 cgit_agefile = xstrdup(value); 184 cgit_agefile = xstrdup(value);
185 else if (!strcmp(name, "repo.group")) 185 else if (!strcmp(name, "repo.group"))
186 cgit_repo_group = xstrdup(value); 186 cgit_repo_group = xstrdup(value);
187 else if (!strcmp(name, "repo.url")) 187 else if (!strcmp(name, "repo.url"))
188 cgit_repo = add_repo(value); 188 cgit_repo = add_repo(value);
189 else if (!strcmp(name, "repo.name")) 189 else if (!strcmp(name, "repo.name"))
190 cgit_repo->name = xstrdup(value); 190 cgit_repo->name = xstrdup(value);
191 else if (cgit_repo && !strcmp(name, "repo.path")) 191 else if (cgit_repo && !strcmp(name, "repo.path"))
192 cgit_repo->path = xstrdup(value); 192 cgit_repo->path = xstrdup(value);
193 else if (cgit_repo && !strcmp(name, "repo.desc")) 193 else if (cgit_repo && !strcmp(name, "repo.desc"))
194 cgit_repo->desc = xstrdup(value); 194 cgit_repo->desc = xstrdup(value);
195 else if (cgit_repo && !strcmp(name, "repo.owner")) 195 else if (cgit_repo && !strcmp(name, "repo.owner"))
196 cgit_repo->owner = xstrdup(value); 196 cgit_repo->owner = xstrdup(value);
197 else if (cgit_repo && !strcmp(name, "repo.defbranch")) 197 else if (cgit_repo && !strcmp(name, "repo.defbranch"))
198 cgit_repo->defbranch = xstrdup(value); 198 cgit_repo->defbranch = xstrdup(value);
199 else if (cgit_repo && !strcmp(name, "repo.snapshots")) 199 else if (cgit_repo && !strcmp(name, "repo.snapshots"))
200 cgit_repo->snapshots = cgit_snapshots * atoi(value); 200 cgit_repo->snapshots = cgit_snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */
201 else if (cgit_repo && !strcmp(name, "repo.enable-log-filecount")) 201 else if (cgit_repo && !strcmp(name, "repo.enable-log-filecount"))
202 cgit_repo->enable_log_filecount = cgit_enable_log_filecount * atoi(value); 202 cgit_repo->enable_log_filecount = cgit_enable_log_filecount * atoi(value);
203 else if (cgit_repo && !strcmp(name, "repo.enable-log-linecount")) 203 else if (cgit_repo && !strcmp(name, "repo.enable-log-linecount"))
204 cgit_repo->enable_log_linecount = cgit_enable_log_linecount * atoi(value); 204 cgit_repo->enable_log_linecount = cgit_enable_log_linecount * atoi(value);
205 else if (cgit_repo && !strcmp(name, "repo.module-link")) 205 else if (cgit_repo && !strcmp(name, "repo.module-link"))
206 cgit_repo->module_link= xstrdup(value); 206 cgit_repo->module_link= xstrdup(value);
207 else if (cgit_repo && !strcmp(name, "repo.readme") && value != NULL) { 207 else if (cgit_repo && !strcmp(name, "repo.readme") && value != NULL) {
208 if (*value == '/') 208 if (*value == '/')
209 cgit_repo->readme = xstrdup(value); 209 cgit_repo->readme = xstrdup(value);
210 else 210 else
211 cgit_repo->readme = xstrdup(fmt("%s/%s", cgit_repo->path, value)); 211 cgit_repo->readme = xstrdup(fmt("%s/%s", cgit_repo->path, value));
212 } else if (!strcmp(name, "include")) 212 } else if (!strcmp(name, "include"))
213 cgit_read_config(value, cgit_global_config_cb); 213 cgit_read_config(value, cgit_global_config_cb);
214} 214}
215 215
216void cgit_querystring_cb(const char *name, const char *value) 216void cgit_querystring_cb(const char *name, const char *value)
217{ 217{
218 if (!strcmp(name,"r")) { 218 if (!strcmp(name,"r")) {
219 cgit_query_repo = xstrdup(value); 219 cgit_query_repo = xstrdup(value);
220 cgit_repo = cgit_get_repoinfo(value); 220 cgit_repo = cgit_get_repoinfo(value);
221 } else if (!strcmp(name, "p")) { 221 } else if (!strcmp(name, "p")) {
222 cgit_query_page = xstrdup(value); 222 cgit_query_page = xstrdup(value);
223 cgit_cmd = cgit_get_cmd_index(value); 223 cgit_cmd = cgit_get_cmd_index(value);
224 } else if (!strcmp(name, "url")) { 224 } else if (!strcmp(name, "url")) {
225 cgit_parse_url(value); 225 cgit_parse_url(value);
226 } else if (!strcmp(name, "q")) { 226 } else if (!strcmp(name, "q")) {
227 cgit_query_search = xstrdup(value); 227 cgit_query_search = xstrdup(value);
228 } else if (!strcmp(name, "h")) { 228 } else if (!strcmp(name, "h")) {
229 cgit_query_head = xstrdup(value); 229 cgit_query_head = xstrdup(value);
230 cgit_query_has_symref = 1; 230 cgit_query_has_symref = 1;
231 } else if (!strcmp(name, "id")) { 231 } else if (!strcmp(name, "id")) {
232 cgit_query_sha1 = xstrdup(value); 232 cgit_query_sha1 = xstrdup(value);
233 cgit_query_has_sha1 = 1; 233 cgit_query_has_sha1 = 1;
234 } else if (!strcmp(name, "id2")) { 234 } else if (!strcmp(name, "id2")) {
235 cgit_query_sha2 = xstrdup(value); 235 cgit_query_sha2 = xstrdup(value);
236 cgit_query_has_sha1 = 1; 236 cgit_query_has_sha1 = 1;
237 } else if (!strcmp(name, "ofs")) { 237 } else if (!strcmp(name, "ofs")) {
238 cgit_query_ofs = atoi(value); 238 cgit_query_ofs = atoi(value);
239 } else if (!strcmp(name, "path")) { 239 } else if (!strcmp(name, "path")) {
240 cgit_query_path = trim_end(value, '/'); 240 cgit_query_path = trim_end(value, '/');
241 } else if (!strcmp(name, "name")) { 241 } else if (!strcmp(name, "name")) {
242 cgit_query_name = xstrdup(value); 242 cgit_query_name = xstrdup(value);
243 } 243 }
244} 244}
245 245
246void *cgit_free_commitinfo(struct commitinfo *info) 246void *cgit_free_commitinfo(struct commitinfo *info)
247{ 247{
248 free(info->author); 248 free(info->author);
249 free(info->author_email); 249 free(info->author_email);
250 free(info->committer); 250 free(info->committer);
251 free(info->committer_email); 251 free(info->committer_email);
252 free(info->subject); 252 free(info->subject);
253 free(info); 253 free(info);
254 return NULL; 254 return NULL;
255} 255}
256 256
257int hextoint(char c) 257int hextoint(char c)
258{ 258{
259 if (c >= 'a' && c <= 'f') 259 if (c >= 'a' && c <= 'f')
260 return 10 + c - 'a'; 260 return 10 + c - 'a';
261 else if (c >= 'A' && c <= 'F') 261 else if (c >= 'A' && c <= 'F')
262 return 10 + c - 'A'; 262 return 10 + c - 'A';
263 else if (c >= '0' && c <= '9') 263 else if (c >= '0' && c <= '9')
264 return c - '0'; 264 return c - '0';
265 else 265 else
266 return -1; 266 return -1;
267} 267}
268 268
269char *trim_end(const char *str, char c) 269char *trim_end(const char *str, char c)
270{ 270{
271 int len; 271 int len;
272 char *s, *t; 272 char *s, *t;
273 273
274 if (str == NULL) 274 if (str == NULL)
275 return NULL; 275 return NULL;
276 t = (char *)str; 276 t = (char *)str;
277 len = strlen(t); 277 len = strlen(t);
278 while(len > 0 && t[len - 1] == c) 278 while(len > 0 && t[len - 1] == c)
279 len--; 279 len--;
280 280
281 if (len == 0) 281 if (len == 0)
282 return NULL; 282 return NULL;
283 283
284 c = t[len]; 284 c = t[len];
285 t[len] = '\0'; 285 t[len] = '\0';
286 s = xstrdup(t); 286 s = xstrdup(t);
287 t[len] = c; 287 t[len] = c;
288 return s; 288 return s;
289} 289}
290 290
291void cgit_diff_tree_cb(struct diff_queue_struct *q, 291void cgit_diff_tree_cb(struct diff_queue_struct *q,
292 struct diff_options *options, void *data) 292 struct diff_options *options, void *data)
293{ 293{
294 int i; 294 int i;
295 295
296 for (i = 0; i < q->nr; i++) { 296 for (i = 0; i < q->nr; i++) {
297 if (q->queue[i]->status == 'U') 297 if (q->queue[i]->status == 'U')
298 continue; 298 continue;
299 ((filepair_fn)data)(q->queue[i]); 299 ((filepair_fn)data)(q->queue[i]);
300 } 300 }
301} 301}
302 302
303static int load_mmfile(mmfile_t *file, const unsigned char *sha1) 303static int load_mmfile(mmfile_t *file, const unsigned char *sha1)
304{ 304{
305 enum object_type type; 305 enum object_type type;
306 306
307 if (is_null_sha1(sha1)) { 307 if (is_null_sha1(sha1)) {
308 file->ptr = (char *)""; 308 file->ptr = (char *)"";
309 file->size = 0; 309 file->size = 0;
310 } else { 310 } else {
311 file->ptr = read_sha1_file(sha1, &type, &file->size); 311 file->ptr = read_sha1_file(sha1, &type, &file->size);
312 } 312 }
313 return 1; 313 return 1;
314} 314}
315 315
316/* 316/*
317 * Receive diff-buffers from xdiff and concatenate them as 317 * Receive diff-buffers from xdiff and concatenate them as
318 * needed across multiple callbacks. 318 * needed across multiple callbacks.
319 * 319 *
320 * This is basically a copy of xdiff-interface.c/xdiff_outf(), 320 * This is basically a copy of xdiff-interface.c/xdiff_outf(),
321 * ripped from git and modified to use globals instead of 321 * ripped from git and modified to use globals instead of
322 * a special callback-struct. 322 * a special callback-struct.
323 */ 323 */
324char *diffbuf = NULL; 324char *diffbuf = NULL;
325int buflen = 0; 325int buflen = 0;
326 326
327int filediff_cb(void *priv, mmbuffer_t *mb, int nbuf) 327int filediff_cb(void *priv, mmbuffer_t *mb, int nbuf)
328{ 328{
diff --git a/ui-commit.c b/ui-commit.c
index bf5e6dc..50e9e11 100644
--- a/ui-commit.c
+++ b/ui-commit.c
@@ -71,155 +71,155 @@ void print_fileinfo(struct fileinfo *info)
71 !is_null_sha1(info->old_sha1) && 71 !is_null_sha1(info->old_sha1) &&
72 !is_null_sha1(info->new_sha1)) { 72 !is_null_sha1(info->new_sha1)) {
73 html("<span class='modechange'>["); 73 html("<span class='modechange'>[");
74 html_filemode(info->old_mode); 74 html_filemode(info->old_mode);
75 html("]</span>"); 75 html("]</span>");
76 } 76 }
77 htmlf("</td><td class='%s'>", class); 77 htmlf("</td><td class='%s'>", class);
78 cgit_tree_link(info->new_path, NULL, NULL, cgit_query_head, curr_rev, 78 cgit_tree_link(info->new_path, NULL, NULL, cgit_query_head, curr_rev,
79 info->new_path); 79 info->new_path);
80 if (info->status == DIFF_STATUS_COPIED || info->status == DIFF_STATUS_RENAMED) 80 if (info->status == DIFF_STATUS_COPIED || info->status == DIFF_STATUS_RENAMED)
81 htmlf(" (%s from %s)", 81 htmlf(" (%s from %s)",
82 info->status == DIFF_STATUS_COPIED ? "copied" : "renamed", 82 info->status == DIFF_STATUS_COPIED ? "copied" : "renamed",
83 info->old_path); 83 info->old_path);
84 html("</td><td class='right'>"); 84 html("</td><td class='right'>");
85 htmlf("%d", info->added + info->removed); 85 htmlf("%d", info->added + info->removed);
86 html("</td><td class='graph'>"); 86 html("</td><td class='graph'>");
87 htmlf("<table width='%d%%'><tr>", (max_changes > 100 ? 100 : max_changes)); 87 htmlf("<table width='%d%%'><tr>", (max_changes > 100 ? 100 : max_changes));
88 htmlf("<td class='add' style='width: %.1f%%;'/>", 88 htmlf("<td class='add' style='width: %.1f%%;'/>",
89 info->added * 100.0 / max_changes); 89 info->added * 100.0 / max_changes);
90 htmlf("<td class='rem' style='width: %.1f%%;'/>", 90 htmlf("<td class='rem' style='width: %.1f%%;'/>",
91 info->removed * 100.0 / max_changes); 91 info->removed * 100.0 / max_changes);
92 htmlf("<td class='none' style='width: %.1f%%;'/>", 92 htmlf("<td class='none' style='width: %.1f%%;'/>",
93 (max_changes - info->removed - info->added) * 100.0 / max_changes); 93 (max_changes - info->removed - info->added) * 100.0 / max_changes);
94 html("</tr></table></td></tr>\n"); 94 html("</tr></table></td></tr>\n");
95} 95}
96 96
97void cgit_count_diff_lines(char *line, int len) 97void cgit_count_diff_lines(char *line, int len)
98{ 98{
99 if (line && (len > 0)) { 99 if (line && (len > 0)) {
100 if (line[0] == '+') 100 if (line[0] == '+')
101 lines_added++; 101 lines_added++;
102 else if (line[0] == '-') 102 else if (line[0] == '-')
103 lines_removed++; 103 lines_removed++;
104 } 104 }
105} 105}
106 106
107void inspect_filepair(struct diff_filepair *pair) 107void inspect_filepair(struct diff_filepair *pair)
108{ 108{
109 files++; 109 files++;
110 lines_added = 0; 110 lines_added = 0;
111 lines_removed = 0; 111 lines_removed = 0;
112 cgit_diff_files(pair->one->sha1, pair->two->sha1, cgit_count_diff_lines); 112 cgit_diff_files(pair->one->sha1, pair->two->sha1, cgit_count_diff_lines);
113 if (files >= slots) { 113 if (files >= slots) {
114 if (slots == 0) 114 if (slots == 0)
115 slots = 4; 115 slots = 4;
116 else 116 else
117 slots = slots * 2; 117 slots = slots * 2;
118 items = xrealloc(items, slots * sizeof(struct fileinfo)); 118 items = xrealloc(items, slots * sizeof(struct fileinfo));
119 } 119 }
120 items[files-1].status = pair->status; 120 items[files-1].status = pair->status;
121 hashcpy(items[files-1].old_sha1, pair->one->sha1); 121 hashcpy(items[files-1].old_sha1, pair->one->sha1);
122 hashcpy(items[files-1].new_sha1, pair->two->sha1); 122 hashcpy(items[files-1].new_sha1, pair->two->sha1);
123 items[files-1].old_mode = pair->one->mode; 123 items[files-1].old_mode = pair->one->mode;
124 items[files-1].new_mode = pair->two->mode; 124 items[files-1].new_mode = pair->two->mode;
125 items[files-1].old_path = xstrdup(pair->one->path); 125 items[files-1].old_path = xstrdup(pair->one->path);
126 items[files-1].new_path = xstrdup(pair->two->path); 126 items[files-1].new_path = xstrdup(pair->two->path);
127 items[files-1].added = lines_added; 127 items[files-1].added = lines_added;
128 items[files-1].removed = lines_removed; 128 items[files-1].removed = lines_removed;
129 if (lines_added + lines_removed > max_changes) 129 if (lines_added + lines_removed > max_changes)
130 max_changes = lines_added + lines_removed; 130 max_changes = lines_added + lines_removed;
131 total_adds += lines_added; 131 total_adds += lines_added;
132 total_rems += lines_removed; 132 total_rems += lines_removed;
133} 133}
134 134
135 135
136void cgit_print_commit(char *hex) 136void cgit_print_commit(char *hex)
137{ 137{
138 struct commit *commit, *parent; 138 struct commit *commit, *parent;
139 struct commitinfo *info; 139 struct commitinfo *info;
140 struct commit_list *p; 140 struct commit_list *p;
141 unsigned char sha1[20]; 141 unsigned char sha1[20];
142 char *tmp; 142 char *tmp;
143 int i; 143 int i;
144 144
145 if (!hex) 145 if (!hex)
146 hex = cgit_query_head; 146 hex = cgit_query_head;
147 curr_rev = hex; 147 curr_rev = hex;
148 148
149 if (get_sha1(hex, sha1)) { 149 if (get_sha1(hex, sha1)) {
150 cgit_print_error(fmt("Bad object id: %s", hex)); 150 cgit_print_error(fmt("Bad object id: %s", hex));
151 return; 151 return;
152 } 152 }
153 commit = lookup_commit_reference(sha1); 153 commit = lookup_commit_reference(sha1);
154 if (!commit) { 154 if (!commit) {
155 cgit_print_error(fmt("Bad commit reference: %s", hex)); 155 cgit_print_error(fmt("Bad commit reference: %s", hex));
156 return; 156 return;
157 } 157 }
158 info = cgit_parse_commit(commit); 158 info = cgit_parse_commit(commit);
159 159
160 html("<table class='commit-info'>\n"); 160 html("<table class='commit-info'>\n");
161 html("<tr><th>author</th><td>"); 161 html("<tr><th>author</th><td>");
162 html_txt(info->author); 162 html_txt(info->author);
163 html(" "); 163 html(" ");
164 html_txt(info->author_email); 164 html_txt(info->author_email);
165 html("</td><td class='right'>"); 165 html("</td><td class='right'>");
166 cgit_print_date(info->author_date, FMT_LONGDATE); 166 cgit_print_date(info->author_date, FMT_LONGDATE);
167 html("</td></tr>\n"); 167 html("</td></tr>\n");
168 html("<tr><th>committer</th><td>"); 168 html("<tr><th>committer</th><td>");
169 html_txt(info->committer); 169 html_txt(info->committer);
170 html(" "); 170 html(" ");
171 html_txt(info->committer_email); 171 html_txt(info->committer_email);
172 html("</td><td class='right'>"); 172 html("</td><td class='right'>");
173 cgit_print_date(info->committer_date, FMT_LONGDATE); 173 cgit_print_date(info->committer_date, FMT_LONGDATE);
174 html("</td></tr>\n"); 174 html("</td></tr>\n");
175 html("<tr><th>tree</th><td colspan='2' class='sha1'>"); 175 html("<tr><th>tree</th><td colspan='2' class='sha1'>");
176 tmp = xstrdup(hex); 176 tmp = xstrdup(hex);
177 cgit_tree_link(sha1_to_hex(commit->tree->object.sha1), NULL, NULL, 177 cgit_tree_link(sha1_to_hex(commit->tree->object.sha1), NULL, NULL,
178 cgit_query_head, tmp, NULL); 178 cgit_query_head, tmp, NULL);
179 html("</td></tr>\n"); 179 html("</td></tr>\n");
180 for (p = commit->parents; p ; p = p->next) { 180 for (p = commit->parents; p ; p = p->next) {
181 parent = lookup_commit_reference(p->item->object.sha1); 181 parent = lookup_commit_reference(p->item->object.sha1);
182 if (!parent) { 182 if (!parent) {
183 html("<tr><td colspan='3'>"); 183 html("<tr><td colspan='3'>");
184 cgit_print_error("Error reading parent commit"); 184 cgit_print_error("Error reading parent commit");
185 html("</td></tr>"); 185 html("</td></tr>");
186 continue; 186 continue;
187 } 187 }
188 html("<tr><th>parent</th>" 188 html("<tr><th>parent</th>"
189 "<td colspan='2' class='sha1'>"); 189 "<td colspan='2' class='sha1'>");
190 cgit_commit_link(sha1_to_hex(p->item->object.sha1), NULL, NULL, 190 cgit_commit_link(sha1_to_hex(p->item->object.sha1), NULL, NULL,
191 cgit_query_head, sha1_to_hex(p->item->object.sha1)); 191 cgit_query_head, sha1_to_hex(p->item->object.sha1));
192 html(" ("); 192 html(" (");
193 cgit_diff_link("diff", NULL, NULL, cgit_query_head, hex, 193 cgit_diff_link("diff", NULL, NULL, cgit_query_head, hex,
194 sha1_to_hex(p->item->object.sha1), NULL); 194 sha1_to_hex(p->item->object.sha1), NULL);
195 html(")</td></tr>"); 195 html(")</td></tr>");
196 } 196 }
197 if (cgit_repo->snapshots) { 197 if (cgit_repo->snapshots) {
198 html("<tr><th>download</th><td colspan='2' class='sha1'>"); 198 html("<tr><th>download</th><td colspan='2' class='sha1'>");
199 cgit_print_snapshot_links(cgit_query_repo,hex); 199 cgit_print_snapshot_links(cgit_query_repo,hex,cgit_repo->snapshots);
200 html("</td></tr>"); 200 html("</td></tr>");
201 } 201 }
202 html("</table>\n"); 202 html("</table>\n");
203 html("<div class='commit-subject'>"); 203 html("<div class='commit-subject'>");
204 html_txt(info->subject); 204 html_txt(info->subject);
205 html("</div>"); 205 html("</div>");
206 html("<div class='commit-msg'>"); 206 html("<div class='commit-msg'>");
207 html_txt(info->msg); 207 html_txt(info->msg);
208 html("</div>"); 208 html("</div>");
209 if (!(commit->parents && commit->parents->next && commit->parents->next->next)) { 209 if (!(commit->parents && commit->parents->next && commit->parents->next->next)) {
210 html("<div class='diffstat-header'>Diffstat</div>"); 210 html("<div class='diffstat-header'>Diffstat</div>");
211 html("<table class='diffstat'>"); 211 html("<table class='diffstat'>");
212 max_changes = 0; 212 max_changes = 0;
213 cgit_diff_commit(commit, inspect_filepair); 213 cgit_diff_commit(commit, inspect_filepair);
214 for(i = 0; i<files; i++) 214 for(i = 0; i<files; i++)
215 print_fileinfo(&items[i]); 215 print_fileinfo(&items[i]);
216 html("</table>"); 216 html("</table>");
217 html("<div class='diffstat-summary'>"); 217 html("<div class='diffstat-summary'>");
218 htmlf("%d files changed, %d insertions, %d deletions (", 218 htmlf("%d files changed, %d insertions, %d deletions (",
219 files, total_adds, total_rems); 219 files, total_adds, total_rems);
220 cgit_diff_link("show diff", NULL, NULL, cgit_query_head, hex, 220 cgit_diff_link("show diff", NULL, NULL, cgit_query_head, hex,
221 NULL, NULL); 221 NULL, NULL);
222 html(")</div>"); 222 html(")</div>");
223 } 223 }
224 cgit_free_commitinfo(info); 224 cgit_free_commitinfo(info);
225} 225}
diff --git a/ui-snapshot.c b/ui-snapshot.c
index 053fd48..d6be55b 100644
--- a/ui-snapshot.c
+++ b/ui-snapshot.c
@@ -1,116 +1,145 @@
1/* ui-snapshot.c: generate snapshot of a commit 1/* ui-snapshot.c: generate snapshot of a commit
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 write_compressed_tar_archive(struct archiver_args *args,const char *filter) 11static int write_compressed_tar_archive(struct archiver_args *args,const char *filter)
12{ 12{
13 int rw[2]; 13 int rw[2];
14 pid_t gzpid; 14 pid_t gzpid;
15 int stdout2; 15 int stdout2;
16 int status; 16 int status;
17 int rv; 17 int rv;
18 18
19 stdout2 = chk_non_negative(dup(STDIN_FILENO), "Preserving STDOUT before compressing"); 19 stdout2 = chk_non_negative(dup(STDIN_FILENO), "Preserving STDOUT before compressing");
20 chk_zero(pipe(rw), "Opening pipe from compressor subprocess"); 20 chk_zero(pipe(rw), "Opening pipe from compressor subprocess");
21 gzpid = chk_non_negative(fork(), "Forking compressor subprocess"); 21 gzpid = chk_non_negative(fork(), "Forking compressor subprocess");
22 if(gzpid==0) { 22 if(gzpid==0) {
23 /* child */ 23 /* child */
24 chk_zero(close(rw[1]), "Closing write end of pipe in child"); 24 chk_zero(close(rw[1]), "Closing write end of pipe in child");
25 chk_zero(close(STDIN_FILENO), "Closing STDIN"); 25 chk_zero(close(STDIN_FILENO), "Closing STDIN");
26 chk_non_negative(dup2(rw[0],STDIN_FILENO), "Redirecting compressor input to stdin"); 26 chk_non_negative(dup2(rw[0],STDIN_FILENO), "Redirecting compressor input to stdin");
27 execlp(filter,filter,NULL); 27 execlp(filter,filter,NULL);
28 _exit(-1); 28 _exit(-1);
29 } 29 }
30 /* parent */ 30 /* parent */
31 chk_zero(close(rw[0]), "Closing read end of pipe"); 31 chk_zero(close(rw[0]), "Closing read end of pipe");
32 chk_non_negative(dup2(rw[1],STDOUT_FILENO), "Redirecting output to compressor"); 32 chk_non_negative(dup2(rw[1],STDOUT_FILENO), "Redirecting output to compressor");
33 33
34 rv = write_tar_archive(args); 34 rv = write_tar_archive(args);
35 35
36 chk_zero(close(STDOUT_FILENO), "Closing STDOUT redirected to compressor"); 36 chk_zero(close(STDOUT_FILENO), "Closing STDOUT redirected to compressor");
37 chk_non_negative(dup2(stdout2,STDOUT_FILENO), "Restoring uncompressed STDOUT"); 37 chk_non_negative(dup2(stdout2,STDOUT_FILENO), "Restoring uncompressed STDOUT");
38 chk_zero(close(stdout2), "Closing uncompressed STDOUT"); 38 chk_zero(close(stdout2), "Closing uncompressed STDOUT");
39 chk_zero(close(rw[1]), "Closing write end of pipe in parent"); 39 chk_zero(close(rw[1]), "Closing write end of pipe in parent");
40 chk_positive(waitpid(gzpid,&status,0), "Waiting on compressor process"); 40 chk_positive(waitpid(gzpid,&status,0), "Waiting on compressor process");
41 if(! ( WIFEXITED(status) && WEXITSTATUS(status)==0 ) ) 41 if(! ( WIFEXITED(status) && WEXITSTATUS(status)==0 ) )
42 cgit_print_error("Failed to compress archive"); 42 cgit_print_error("Failed to compress archive");
43 43
44 return rv; 44 return rv;
45} 45}
46 46
47static int write_tar_gzip_archive(struct archiver_args *args) 47static int write_tar_gzip_archive(struct archiver_args *args)
48{ 48{
49 return write_compressed_tar_archive(args,"gzip"); 49 return write_compressed_tar_archive(args,"gzip");
50} 50}
51static int write_tar_bzip2_archive(struct archiver_args *args) 51static int write_tar_bzip2_archive(struct archiver_args *args)
52{ 52{
53 return write_compressed_tar_archive(args,"bzip2"); 53 return write_compressed_tar_archive(args,"bzip2");
54} 54}
55 55
56static const struct snapshot_archive_t { 56static const struct snapshot_archive_t {
57 const char *suffix; 57 const char *suffix;
58 const char *mimetype; 58 const char *mimetype;
59 write_archive_fn_t write_func; 59 write_archive_fn_t write_func;
60 int bit;
60 }snapshot_archives[] = { 61 }snapshot_archives[] = {
61 { ".zip", "application/x-zip", write_zip_archive }, 62 { ".zip", "application/x-zip", write_zip_archive, 0x1 },
62 { ".tar.gz", "application/x-tar", write_tar_gzip_archive }, 63 { ".tar.gz", "application/x-tar", write_tar_gzip_archive, 0x2 },
63 { ".tar.bz2", "application/x-tar", write_tar_bzip2_archive }, 64 { ".tar.bz2", "application/x-tar", write_tar_bzip2_archive, 0x4 },
64 { ".tar", "application/x-tar", write_tar_archive } 65 { ".tar", "application/x-tar", write_tar_archive, 0x8 }
65}; 66};
66 67
67void cgit_print_snapshot(struct cacheitem *item, const char *hex, 68void cgit_print_snapshot(struct cacheitem *item, const char *hex,
68 const char *prefix, const char *filename) 69 const char *prefix, const char *filename,
70 int snapshots)
69{ 71{
70 int fnl = strlen(filename); 72 int fnl = strlen(filename);
71 int f; 73 int f;
72 for(f=0;f<(sizeof(snapshot_archives)/sizeof(*snapshot_archives));++f) { 74 for(f=0;f<(sizeof(snapshot_archives)/sizeof(*snapshot_archives));++f) {
73 const struct snapshot_archive_t* sat = &snapshot_archives[f]; 75 const struct snapshot_archive_t* sat = &snapshot_archives[f];
74 int sl = strlen(sat->suffix); 76 int sl;
77 if(!(snapshots&sat->bit)) continue;
78 sl = strlen(sat->suffix);
75 if(fnl<sl || strcmp(&filename[fnl-sl],sat->suffix)) 79 if(fnl<sl || strcmp(&filename[fnl-sl],sat->suffix))
76 continue; 80 continue;
77 81
78 struct archiver_args args; 82 struct archiver_args args;
79 struct commit *commit; 83 struct commit *commit;
80 unsigned char sha1[20]; 84 unsigned char sha1[20];
81 85
82 if(get_sha1(hex, sha1)) { 86 if(get_sha1(hex, sha1)) {
83 cgit_print_error(fmt("Bad object id: %s", hex)); 87 cgit_print_error(fmt("Bad object id: %s", hex));
84 return; 88 return;
85 } 89 }
86 commit = lookup_commit_reference(sha1); 90 commit = lookup_commit_reference(sha1);
87 91
88 if(!commit) { 92 if(!commit) {
89 cgit_print_error(fmt("Not a commit reference: %s", hex)); 93 cgit_print_error(fmt("Not a commit reference: %s", hex));
90 return;; 94 return;;
91 } 95 }
92 96
93 memset(&args,0,sizeof(args)); 97 memset(&args,0,sizeof(args));
94 args.base = fmt("%s/", prefix); 98 args.base = fmt("%s/", prefix);
95 args.tree = commit->tree; 99 args.tree = commit->tree;
96 100
97 cgit_print_snapshot_start(sat->mimetype, filename, item); 101 cgit_print_snapshot_start(sat->mimetype, filename, item);
98 (*sat->write_func)(&args); 102 (*sat->write_func)(&args);
99 return; 103 return;
100 } 104 }
101 cgit_print_error(fmt("Unsupported snapshot format: %s", filename)); 105 cgit_print_error(fmt("Unsupported snapshot format: %s", filename));
102} 106}
103 107
104void cgit_print_snapshot_links(const char *repo,const char *hex) 108void cgit_print_snapshot_links(const char *repo,const char *hex,int snapshots)
105{ 109{
106 char *filename; 110 char *filename;
107 int f; 111 int f;
108 for(f=0;f<(sizeof(snapshot_archives)/sizeof(*snapshot_archives));++f) { 112 for(f=0;f<(sizeof(snapshot_archives)/sizeof(*snapshot_archives));++f) {
109 const struct snapshot_archive_t* sat = &snapshot_archives[f]; 113 const struct snapshot_archive_t* sat = &snapshot_archives[f];
114 if(!(snapshots&sat->bit)) continue;
110 filename = fmt("%s-%s%s",cgit_repobasename(repo),hex,sat->suffix); 115 filename = fmt("%s-%s%s",cgit_repobasename(repo),hex,sat->suffix);
111 htmlf("<a href='%s'>%s</a><br/>", 116 htmlf("<a href='%s'>%s</a><br/>",
112 cgit_fileurl(repo,"snapshot",filename, 117 cgit_fileurl(repo,"snapshot",filename,
113 fmt("id=%s&amp;name=%s",hex,filename)), filename); 118 fmt("id=%s&amp;name=%s",hex,filename)), filename);
114 } 119 }
115} 120}
121
122int cgit_parse_snapshots_mask(const char *str)
123{
124 static const char *delim = " \t,:/|;";
125 int f, tl, rv = 0;
126 /* favor legacy setting */
127 if(atoi(str)) return 1;
128 for(;;) {
129 str += strspn(str,delim);
130 tl = strcspn(str,delim);
131 if(!tl)
132 break;
133 for(f=0;f<(sizeof(snapshot_archives)/sizeof(*snapshot_archives));++f) {
134 const struct snapshot_archive_t* sat = &snapshot_archives[f];
135 if(! ( strncmp(sat->suffix,str,tl) && strncmp(sat->suffix+1,str,tl-1) ) ) {
136 rv |= sat->bit;
137 break;
138 }
139 }
140 str += tl;
141 }
142 return rv;
143}
144
116/* vim:set sw=8: */ 145/* vim:set sw=8: */