summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c2
-rw-r--r--cgitrc.5.txt2
2 files changed, 2 insertions, 2 deletions
diff --git a/cgit.c b/cgit.c
index 7b228af..bf46b5a 100644
--- a/cgit.c
+++ b/cgit.c
@@ -1,538 +1,538 @@
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#include "cache.h" 10#include "cache.h"
11#include "cmd.h" 11#include "cmd.h"
12#include "configfile.h" 12#include "configfile.h"
13#include "html.h" 13#include "html.h"
14#include "ui-shared.h" 14#include "ui-shared.h"
15#include "ui-stats.h" 15#include "ui-stats.h"
16#include "scan-tree.h" 16#include "scan-tree.h"
17 17
18const char *cgit_version = CGIT_VERSION; 18const char *cgit_version = CGIT_VERSION;
19 19
20void add_mimetype(const char *name, const char *value) 20void add_mimetype(const char *name, const char *value)
21{ 21{
22 struct string_list_item *item; 22 struct string_list_item *item;
23 23
24 item = string_list_insert(xstrdup(name), &ctx.cfg.mimetypes); 24 item = string_list_insert(xstrdup(name), &ctx.cfg.mimetypes);
25 item->util = xstrdup(value); 25 item->util = xstrdup(value);
26} 26}
27 27
28struct cgit_filter *new_filter(const char *cmd, int extra_args) 28struct cgit_filter *new_filter(const char *cmd, int extra_args)
29{ 29{
30 struct cgit_filter *f; 30 struct cgit_filter *f;
31 31
32 if (!cmd || !cmd[0]) 32 if (!cmd || !cmd[0])
33 return NULL; 33 return NULL;
34 34
35 f = xmalloc(sizeof(struct cgit_filter)); 35 f = xmalloc(sizeof(struct cgit_filter));
36 f->cmd = xstrdup(cmd); 36 f->cmd = xstrdup(cmd);
37 f->argv = xmalloc((2 + extra_args) * sizeof(char *)); 37 f->argv = xmalloc((2 + extra_args) * sizeof(char *));
38 f->argv[0] = f->cmd; 38 f->argv[0] = f->cmd;
39 f->argv[1] = NULL; 39 f->argv[1] = NULL;
40 return f; 40 return f;
41} 41}
42 42
43void config_cb(const char *name, const char *value) 43void config_cb(const char *name, const char *value)
44{ 44{
45 if (!strcmp(name, "root-title")) 45 if (!strcmp(name, "root-title"))
46 ctx.cfg.root_title = xstrdup(value); 46 ctx.cfg.root_title = xstrdup(value);
47 else if (!strcmp(name, "root-desc")) 47 else if (!strcmp(name, "root-desc"))
48 ctx.cfg.root_desc = xstrdup(value); 48 ctx.cfg.root_desc = xstrdup(value);
49 else if (!strcmp(name, "root-readme")) 49 else if (!strcmp(name, "root-readme"))
50 ctx.cfg.root_readme = xstrdup(value); 50 ctx.cfg.root_readme = xstrdup(value);
51 else if (!strcmp(name, "css")) 51 else if (!strcmp(name, "css"))
52 ctx.cfg.css = xstrdup(value); 52 ctx.cfg.css = xstrdup(value);
53 else if (!strcmp(name, "favicon")) 53 else if (!strcmp(name, "favicon"))
54 ctx.cfg.favicon = xstrdup(value); 54 ctx.cfg.favicon = xstrdup(value);
55 else if (!strcmp(name, "footer")) 55 else if (!strcmp(name, "footer"))
56 ctx.cfg.footer = xstrdup(value); 56 ctx.cfg.footer = xstrdup(value);
57 else if (!strcmp(name, "head-include")) 57 else if (!strcmp(name, "head-include"))
58 ctx.cfg.head_include = xstrdup(value); 58 ctx.cfg.head_include = xstrdup(value);
59 else if (!strcmp(name, "header")) 59 else if (!strcmp(name, "header"))
60 ctx.cfg.header = xstrdup(value); 60 ctx.cfg.header = xstrdup(value);
61 else if (!strcmp(name, "logo")) 61 else if (!strcmp(name, "logo"))
62 ctx.cfg.logo = xstrdup(value); 62 ctx.cfg.logo = xstrdup(value);
63 else if (!strcmp(name, "index-header")) 63 else if (!strcmp(name, "index-header"))
64 ctx.cfg.index_header = xstrdup(value); 64 ctx.cfg.index_header = xstrdup(value);
65 else if (!strcmp(name, "index-info")) 65 else if (!strcmp(name, "index-info"))
66 ctx.cfg.index_info = xstrdup(value); 66 ctx.cfg.index_info = xstrdup(value);
67 else if (!strcmp(name, "logo-link")) 67 else if (!strcmp(name, "logo-link"))
68 ctx.cfg.logo_link = xstrdup(value); 68 ctx.cfg.logo_link = xstrdup(value);
69 else if (!strcmp(name, "module-link")) 69 else if (!strcmp(name, "module-link"))
70 ctx.cfg.module_link = xstrdup(value); 70 ctx.cfg.module_link = xstrdup(value);
71 else if (!strcmp(name, "virtual-root")) { 71 else if (!strcmp(name, "virtual-root")) {
72 ctx.cfg.virtual_root = trim_end(value, '/'); 72 ctx.cfg.virtual_root = trim_end(value, '/');
73 if (!ctx.cfg.virtual_root && (!strcmp(value, "/"))) 73 if (!ctx.cfg.virtual_root && (!strcmp(value, "/")))
74 ctx.cfg.virtual_root = ""; 74 ctx.cfg.virtual_root = "";
75 } else if (!strcmp(name, "nocache")) 75 } else if (!strcmp(name, "nocache"))
76 ctx.cfg.nocache = atoi(value); 76 ctx.cfg.nocache = atoi(value);
77 else if (!strcmp(name, "noplainemail")) 77 else if (!strcmp(name, "noplainemail"))
78 ctx.cfg.noplainemail = atoi(value); 78 ctx.cfg.noplainemail = atoi(value);
79 else if (!strcmp(name, "noheader")) 79 else if (!strcmp(name, "noheader"))
80 ctx.cfg.noheader = atoi(value); 80 ctx.cfg.noheader = atoi(value);
81 else if (!strcmp(name, "snapshots")) 81 else if (!strcmp(name, "snapshots"))
82 ctx.cfg.snapshots = cgit_parse_snapshots_mask(value); 82 ctx.cfg.snapshots = cgit_parse_snapshots_mask(value);
83 else if (!strcmp(name, "enable-index-links")) 83 else if (!strcmp(name, "enable-index-links"))
84 ctx.cfg.enable_index_links = atoi(value); 84 ctx.cfg.enable_index_links = atoi(value);
85 else if (!strcmp(name, "enable-log-filecount")) 85 else if (!strcmp(name, "enable-log-filecount"))
86 ctx.cfg.enable_log_filecount = atoi(value); 86 ctx.cfg.enable_log_filecount = atoi(value);
87 else if (!strcmp(name, "enable-log-linecount")) 87 else if (!strcmp(name, "enable-log-linecount"))
88 ctx.cfg.enable_log_linecount = atoi(value); 88 ctx.cfg.enable_log_linecount = atoi(value);
89 else if (!strcmp(name, "max-stats")) 89 else if (!strcmp(name, "max-stats"))
90 ctx.cfg.max_stats = cgit_find_stats_period(value, NULL); 90 ctx.cfg.max_stats = cgit_find_stats_period(value, NULL);
91 else if (!strcmp(name, "cache-size")) 91 else if (!strcmp(name, "cache-size"))
92 ctx.cfg.cache_size = atoi(value); 92 ctx.cfg.cache_size = atoi(value);
93 else if (!strcmp(name, "cache-root")) 93 else if (!strcmp(name, "cache-root"))
94 ctx.cfg.cache_root = xstrdup(value); 94 ctx.cfg.cache_root = xstrdup(value);
95 else if (!strcmp(name, "cache-root-ttl")) 95 else if (!strcmp(name, "cache-root-ttl"))
96 ctx.cfg.cache_root_ttl = atoi(value); 96 ctx.cfg.cache_root_ttl = atoi(value);
97 else if (!strcmp(name, "cache-repo-ttl")) 97 else if (!strcmp(name, "cache-repo-ttl"))
98 ctx.cfg.cache_repo_ttl = atoi(value); 98 ctx.cfg.cache_repo_ttl = atoi(value);
99 else if (!strcmp(name, "cache-static-ttl")) 99 else if (!strcmp(name, "cache-static-ttl"))
100 ctx.cfg.cache_static_ttl = atoi(value); 100 ctx.cfg.cache_static_ttl = atoi(value);
101 else if (!strcmp(name, "cache-dynamic-ttl")) 101 else if (!strcmp(name, "cache-dynamic-ttl"))
102 ctx.cfg.cache_dynamic_ttl = atoi(value); 102 ctx.cfg.cache_dynamic_ttl = atoi(value);
103 else if (!strcmp(name, "about-filter")) 103 else if (!strcmp(name, "about-filter"))
104 ctx.cfg.about_filter = new_filter(value, 0); 104 ctx.cfg.about_filter = new_filter(value, 0);
105 else if (!strcmp(name, "commit-filter")) 105 else if (!strcmp(name, "commit-filter"))
106 ctx.cfg.commit_filter = new_filter(value, 0); 106 ctx.cfg.commit_filter = new_filter(value, 0);
107 else if (!strcmp(name, "embedded")) 107 else if (!strcmp(name, "embedded"))
108 ctx.cfg.embedded = atoi(value); 108 ctx.cfg.embedded = atoi(value);
109 else if (!strcmp(name, "max-message-length")) 109 else if (!strcmp(name, "max-message-length"))
110 ctx.cfg.max_msg_len = atoi(value); 110 ctx.cfg.max_msg_len = atoi(value);
111 else if (!strcmp(name, "max-repodesc-length")) 111 else if (!strcmp(name, "max-repodesc-length"))
112 ctx.cfg.max_repodesc_len = atoi(value); 112 ctx.cfg.max_repodesc_len = atoi(value);
113 else if (!strcmp(name, "max-repo-count")) 113 else if (!strcmp(name, "max-repo-count"))
114 ctx.cfg.max_repo_count = atoi(value); 114 ctx.cfg.max_repo_count = atoi(value);
115 else if (!strcmp(name, "max-commit-count")) 115 else if (!strcmp(name, "max-commit-count"))
116 ctx.cfg.max_commit_count = atoi(value); 116 ctx.cfg.max_commit_count = atoi(value);
117 else if (!strcmp(name, "source-filter")) 117 else if (!strcmp(name, "source-filter"))
118 ctx.cfg.source_filter = new_filter(value, 1); 118 ctx.cfg.source_filter = new_filter(value, 1);
119 else if (!strcmp(name, "summary-log")) 119 else if (!strcmp(name, "summary-log"))
120 ctx.cfg.summary_log = atoi(value); 120 ctx.cfg.summary_log = atoi(value);
121 else if (!strcmp(name, "summary-branches")) 121 else if (!strcmp(name, "summary-branches"))
122 ctx.cfg.summary_branches = atoi(value); 122 ctx.cfg.summary_branches = atoi(value);
123 else if (!strcmp(name, "summary-tags")) 123 else if (!strcmp(name, "summary-tags"))
124 ctx.cfg.summary_tags = atoi(value); 124 ctx.cfg.summary_tags = atoi(value);
125 else if (!strcmp(name, "agefile")) 125 else if (!strcmp(name, "agefile"))
126 ctx.cfg.agefile = xstrdup(value); 126 ctx.cfg.agefile = xstrdup(value);
127 else if (!strcmp(name, "renamelimit")) 127 else if (!strcmp(name, "renamelimit"))
128 ctx.cfg.renamelimit = atoi(value); 128 ctx.cfg.renamelimit = atoi(value);
129 else if (!strcmp(name, "robots")) 129 else if (!strcmp(name, "robots"))
130 ctx.cfg.robots = xstrdup(value); 130 ctx.cfg.robots = xstrdup(value);
131 else if (!strcmp(name, "clone-prefix")) 131 else if (!strcmp(name, "clone-prefix"))
132 ctx.cfg.clone_prefix = xstrdup(value); 132 ctx.cfg.clone_prefix = xstrdup(value);
133 else if (!strcmp(name, "local-time")) 133 else if (!strcmp(name, "local-time"))
134 ctx.cfg.local_time = atoi(value); 134 ctx.cfg.local_time = atoi(value);
135 else if (!prefixcmp(name, "mimetype.")) 135 else if (!prefixcmp(name, "mimetype."))
136 add_mimetype(name + 9, value); 136 add_mimetype(name + 9, value);
137 else if (!strcmp(name, "repo.group")) 137 else if (!strcmp(name, "repo.group"))
138 ctx.cfg.repo_group = xstrdup(value); 138 ctx.cfg.repo_group = xstrdup(value);
139 else if (!strcmp(name, "repo.url")) 139 else if (!strcmp(name, "repo.url"))
140 ctx.repo = cgit_add_repo(value); 140 ctx.repo = cgit_add_repo(value);
141 else if (!strcmp(name, "repo.name")) 141 else if (!strcmp(name, "repo.name"))
142 ctx.repo->name = xstrdup(value); 142 ctx.repo->name = xstrdup(value);
143 else if (ctx.repo && !strcmp(name, "repo.path")) 143 else if (ctx.repo && !strcmp(name, "repo.path"))
144 ctx.repo->path = trim_end(value, '/'); 144 ctx.repo->path = trim_end(value, '/');
145 else if (ctx.repo && !strcmp(name, "repo.clone-url")) 145 else if (ctx.repo && !strcmp(name, "repo.clone-url"))
146 ctx.repo->clone_url = xstrdup(value); 146 ctx.repo->clone_url = xstrdup(value);
147 else if (ctx.repo && !strcmp(name, "repo.desc")) 147 else if (ctx.repo && !strcmp(name, "repo.desc"))
148 ctx.repo->desc = xstrdup(value); 148 ctx.repo->desc = xstrdup(value);
149 else if (ctx.repo && !strcmp(name, "repo.owner")) 149 else if (ctx.repo && !strcmp(name, "repo.owner"))
150 ctx.repo->owner = xstrdup(value); 150 ctx.repo->owner = xstrdup(value);
151 else if (ctx.repo && !strcmp(name, "repo.defbranch")) 151 else if (ctx.repo && !strcmp(name, "repo.defbranch"))
152 ctx.repo->defbranch = xstrdup(value); 152 ctx.repo->defbranch = xstrdup(value);
153 else if (ctx.repo && !strcmp(name, "repo.snapshots")) 153 else if (ctx.repo && !strcmp(name, "repo.snapshots"))
154 ctx.repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */ 154 ctx.repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */
155 else if (ctx.repo && !strcmp(name, "repo.enable-log-filecount")) 155 else if (ctx.repo && !strcmp(name, "repo.enable-log-filecount"))
156 ctx.repo->enable_log_filecount = ctx.cfg.enable_log_filecount * atoi(value); 156 ctx.repo->enable_log_filecount = ctx.cfg.enable_log_filecount * atoi(value);
157 else if (ctx.repo && !strcmp(name, "repo.enable-log-linecount")) 157 else if (ctx.repo && !strcmp(name, "repo.enable-log-linecount"))
158 ctx.repo->enable_log_linecount = ctx.cfg.enable_log_linecount * atoi(value); 158 ctx.repo->enable_log_linecount = ctx.cfg.enable_log_linecount * atoi(value);
159 else if (ctx.repo && !strcmp(name, "repo.max-stats")) 159 else if (ctx.repo && !strcmp(name, "repo.max-stats"))
160 ctx.repo->max_stats = cgit_find_stats_period(value, NULL); 160 ctx.repo->max_stats = cgit_find_stats_period(value, NULL);
161 else if (ctx.repo && !strcmp(name, "repo.module-link")) 161 else if (ctx.repo && !strcmp(name, "repo.module-link"))
162 ctx.repo->module_link= xstrdup(value); 162 ctx.repo->module_link= xstrdup(value);
163 else if (ctx.repo && !strcmp(name, "repo.about-filter")) 163 else if (ctx.repo && !strcmp(name, "repo.about-filter"))
164 ctx.repo->about_filter = new_filter(value, 0); 164 ctx.repo->about_filter = new_filter(value, 0);
165 else if (ctx.repo && !strcmp(name, "repo.commit-filter")) 165 else if (ctx.repo && !strcmp(name, "repo.commit-filter"))
166 ctx.repo->commit_filter = new_filter(value, 0); 166 ctx.repo->commit_filter = new_filter(value, 0);
167 else if (ctx.repo && !strcmp(name, "repo.source-filter")) 167 else if (ctx.repo && !strcmp(name, "repo.source-filter"))
168 ctx.repo->source_filter = new_filter(value, 1); 168 ctx.repo->source_filter = new_filter(value, 1);
169 else if (ctx.repo && !strcmp(name, "repo.readme") && value != NULL) { 169 else if (ctx.repo && !strcmp(name, "repo.readme") && value != NULL) {
170 if (*value == '/') 170 if (*value == '/')
171 ctx.repo->readme = xstrdup(value); 171 ctx.repo->readme = xstrdup(value);
172 else 172 else
173 ctx.repo->readme = xstrdup(fmt("%s/%s", ctx.repo->path, value)); 173 ctx.repo->readme = xstrdup(fmt("%s/%s", ctx.repo->path, value));
174 } else if (!strcmp(name, "include")) 174 } else if (!strcmp(name, "include"))
175 parse_configfile(value, config_cb); 175 parse_configfile(value, config_cb);
176} 176}
177 177
178static void querystring_cb(const char *name, const char *value) 178static void querystring_cb(const char *name, const char *value)
179{ 179{
180 if (!strcmp(name,"r")) { 180 if (!strcmp(name,"r")) {
181 ctx.qry.repo = xstrdup(value); 181 ctx.qry.repo = xstrdup(value);
182 ctx.repo = cgit_get_repoinfo(value); 182 ctx.repo = cgit_get_repoinfo(value);
183 } else if (!strcmp(name, "p")) { 183 } else if (!strcmp(name, "p")) {
184 ctx.qry.page = xstrdup(value); 184 ctx.qry.page = xstrdup(value);
185 } else if (!strcmp(name, "url")) { 185 } else if (!strcmp(name, "url")) {
186 ctx.qry.url = xstrdup(value); 186 ctx.qry.url = xstrdup(value);
187 cgit_parse_url(value); 187 cgit_parse_url(value);
188 } else if (!strcmp(name, "qt")) { 188 } else if (!strcmp(name, "qt")) {
189 ctx.qry.grep = xstrdup(value); 189 ctx.qry.grep = xstrdup(value);
190 } else if (!strcmp(name, "q")) { 190 } else if (!strcmp(name, "q")) {
191 ctx.qry.search = xstrdup(value); 191 ctx.qry.search = xstrdup(value);
192 } else if (!strcmp(name, "h")) { 192 } else if (!strcmp(name, "h")) {
193 ctx.qry.head = xstrdup(value); 193 ctx.qry.head = xstrdup(value);
194 ctx.qry.has_symref = 1; 194 ctx.qry.has_symref = 1;
195 } else if (!strcmp(name, "id")) { 195 } else if (!strcmp(name, "id")) {
196 ctx.qry.sha1 = xstrdup(value); 196 ctx.qry.sha1 = xstrdup(value);
197 ctx.qry.has_sha1 = 1; 197 ctx.qry.has_sha1 = 1;
198 } else if (!strcmp(name, "id2")) { 198 } else if (!strcmp(name, "id2")) {
199 ctx.qry.sha2 = xstrdup(value); 199 ctx.qry.sha2 = xstrdup(value);
200 ctx.qry.has_sha1 = 1; 200 ctx.qry.has_sha1 = 1;
201 } else if (!strcmp(name, "ofs")) { 201 } else if (!strcmp(name, "ofs")) {
202 ctx.qry.ofs = atoi(value); 202 ctx.qry.ofs = atoi(value);
203 } else if (!strcmp(name, "path")) { 203 } else if (!strcmp(name, "path")) {
204 ctx.qry.path = trim_end(value, '/'); 204 ctx.qry.path = trim_end(value, '/');
205 } else if (!strcmp(name, "name")) { 205 } else if (!strcmp(name, "name")) {
206 ctx.qry.name = xstrdup(value); 206 ctx.qry.name = xstrdup(value);
207 } else if (!strcmp(name, "mimetype")) { 207 } else if (!strcmp(name, "mimetype")) {
208 ctx.qry.mimetype = xstrdup(value); 208 ctx.qry.mimetype = xstrdup(value);
209 } else if (!strcmp(name, "s")){ 209 } else if (!strcmp(name, "s")){
210 ctx.qry.sort = xstrdup(value); 210 ctx.qry.sort = xstrdup(value);
211 } else if (!strcmp(name, "showmsg")) { 211 } else if (!strcmp(name, "showmsg")) {
212 ctx.qry.showmsg = atoi(value); 212 ctx.qry.showmsg = atoi(value);
213 } else if (!strcmp(name, "period")) { 213 } else if (!strcmp(name, "period")) {
214 ctx.qry.period = xstrdup(value); 214 ctx.qry.period = xstrdup(value);
215 } 215 }
216} 216}
217 217
218static void prepare_context(struct cgit_context *ctx) 218static void prepare_context(struct cgit_context *ctx)
219{ 219{
220 memset(ctx, 0, sizeof(ctx)); 220 memset(ctx, 0, sizeof(ctx));
221 ctx->cfg.agefile = "info/web/last-modified"; 221 ctx->cfg.agefile = "info/web/last-modified";
222 ctx->cfg.nocache = 0; 222 ctx->cfg.nocache = 0;
223 ctx->cfg.cache_size = 0; 223 ctx->cfg.cache_size = 0;
224 ctx->cfg.cache_dynamic_ttl = 5; 224 ctx->cfg.cache_dynamic_ttl = 5;
225 ctx->cfg.cache_max_create_time = 5; 225 ctx->cfg.cache_max_create_time = 5;
226 ctx->cfg.cache_repo_ttl = 5; 226 ctx->cfg.cache_repo_ttl = 5;
227 ctx->cfg.cache_root = CGIT_CACHE_ROOT; 227 ctx->cfg.cache_root = CGIT_CACHE_ROOT;
228 ctx->cfg.cache_root_ttl = 5; 228 ctx->cfg.cache_root_ttl = 5;
229 ctx->cfg.cache_static_ttl = -1; 229 ctx->cfg.cache_static_ttl = -1;
230 ctx->cfg.css = "/cgit.css"; 230 ctx->cfg.css = "/cgit.css";
231 ctx->cfg.logo = "/git-logo.png"; 231 ctx->cfg.logo = "/cgit.png";
232 ctx->cfg.local_time = 0; 232 ctx->cfg.local_time = 0;
233 ctx->cfg.max_repo_count = 50; 233 ctx->cfg.max_repo_count = 50;
234 ctx->cfg.max_commit_count = 50; 234 ctx->cfg.max_commit_count = 50;
235 ctx->cfg.max_lock_attempts = 5; 235 ctx->cfg.max_lock_attempts = 5;
236 ctx->cfg.max_msg_len = 80; 236 ctx->cfg.max_msg_len = 80;
237 ctx->cfg.max_repodesc_len = 80; 237 ctx->cfg.max_repodesc_len = 80;
238 ctx->cfg.max_stats = 0; 238 ctx->cfg.max_stats = 0;
239 ctx->cfg.module_link = "./?repo=%s&page=commit&id=%s"; 239 ctx->cfg.module_link = "./?repo=%s&page=commit&id=%s";
240 ctx->cfg.renamelimit = -1; 240 ctx->cfg.renamelimit = -1;
241 ctx->cfg.robots = "index, nofollow"; 241 ctx->cfg.robots = "index, nofollow";
242 ctx->cfg.root_title = "Git repository browser"; 242 ctx->cfg.root_title = "Git repository browser";
243 ctx->cfg.root_desc = "a fast webinterface for the git dscm"; 243 ctx->cfg.root_desc = "a fast webinterface for the git dscm";
244 ctx->cfg.script_name = CGIT_SCRIPT_NAME; 244 ctx->cfg.script_name = CGIT_SCRIPT_NAME;
245 ctx->cfg.summary_branches = 10; 245 ctx->cfg.summary_branches = 10;
246 ctx->cfg.summary_log = 10; 246 ctx->cfg.summary_log = 10;
247 ctx->cfg.summary_tags = 10; 247 ctx->cfg.summary_tags = 10;
248 ctx->page.mimetype = "text/html"; 248 ctx->page.mimetype = "text/html";
249 ctx->page.charset = PAGE_ENCODING; 249 ctx->page.charset = PAGE_ENCODING;
250 ctx->page.filename = NULL; 250 ctx->page.filename = NULL;
251 ctx->page.size = 0; 251 ctx->page.size = 0;
252 ctx->page.modified = time(NULL); 252 ctx->page.modified = time(NULL);
253 ctx->page.expires = ctx->page.modified; 253 ctx->page.expires = ctx->page.modified;
254 ctx->page.etag = NULL; 254 ctx->page.etag = NULL;
255 memset(&ctx->cfg.mimetypes, 0, sizeof(struct string_list)); 255 memset(&ctx->cfg.mimetypes, 0, sizeof(struct string_list));
256} 256}
257 257
258struct refmatch { 258struct refmatch {
259 char *req_ref; 259 char *req_ref;
260 char *first_ref; 260 char *first_ref;
261 int match; 261 int match;
262}; 262};
263 263
264int find_current_ref(const char *refname, const unsigned char *sha1, 264int find_current_ref(const char *refname, const unsigned char *sha1,
265 int flags, void *cb_data) 265 int flags, void *cb_data)
266{ 266{
267 struct refmatch *info; 267 struct refmatch *info;
268 268
269 info = (struct refmatch *)cb_data; 269 info = (struct refmatch *)cb_data;
270 if (!strcmp(refname, info->req_ref)) 270 if (!strcmp(refname, info->req_ref))
271 info->match = 1; 271 info->match = 1;
272 if (!info->first_ref) 272 if (!info->first_ref)
273 info->first_ref = xstrdup(refname); 273 info->first_ref = xstrdup(refname);
274 return info->match; 274 return info->match;
275} 275}
276 276
277char *find_default_branch(struct cgit_repo *repo) 277char *find_default_branch(struct cgit_repo *repo)
278{ 278{
279 struct refmatch info; 279 struct refmatch info;
280 char *ref; 280 char *ref;
281 281
282 info.req_ref = repo->defbranch; 282 info.req_ref = repo->defbranch;
283 info.first_ref = NULL; 283 info.first_ref = NULL;
284 info.match = 0; 284 info.match = 0;
285 for_each_branch_ref(find_current_ref, &info); 285 for_each_branch_ref(find_current_ref, &info);
286 if (info.match) 286 if (info.match)
287 ref = info.req_ref; 287 ref = info.req_ref;
288 else 288 else
289 ref = info.first_ref; 289 ref = info.first_ref;
290 if (ref) 290 if (ref)
291 ref = xstrdup(ref); 291 ref = xstrdup(ref);
292 return ref; 292 return ref;
293} 293}
294 294
295static int prepare_repo_cmd(struct cgit_context *ctx) 295static int prepare_repo_cmd(struct cgit_context *ctx)
296{ 296{
297 char *tmp; 297 char *tmp;
298 unsigned char sha1[20]; 298 unsigned char sha1[20];
299 int nongit = 0; 299 int nongit = 0;
300 300
301 setenv("GIT_DIR", ctx->repo->path, 1); 301 setenv("GIT_DIR", ctx->repo->path, 1);
302 setup_git_directory_gently(&nongit); 302 setup_git_directory_gently(&nongit);
303 if (nongit) { 303 if (nongit) {
304 ctx->page.title = fmt("%s - %s", ctx->cfg.root_title, 304 ctx->page.title = fmt("%s - %s", ctx->cfg.root_title,
305 "config error"); 305 "config error");
306 tmp = fmt("Not a git repository: '%s'", ctx->repo->path); 306 tmp = fmt("Not a git repository: '%s'", ctx->repo->path);
307 ctx->repo = NULL; 307 ctx->repo = NULL;
308 cgit_print_http_headers(ctx); 308 cgit_print_http_headers(ctx);
309 cgit_print_docstart(ctx); 309 cgit_print_docstart(ctx);
310 cgit_print_pageheader(ctx); 310 cgit_print_pageheader(ctx);
311 cgit_print_error(tmp); 311 cgit_print_error(tmp);
312 cgit_print_docend(); 312 cgit_print_docend();
313 return 1; 313 return 1;
314 } 314 }
315 ctx->page.title = fmt("%s - %s", ctx->repo->name, ctx->repo->desc); 315 ctx->page.title = fmt("%s - %s", ctx->repo->name, ctx->repo->desc);
316 316
317 if (!ctx->qry.head) { 317 if (!ctx->qry.head) {
318 ctx->qry.nohead = 1; 318 ctx->qry.nohead = 1;
319 ctx->qry.head = find_default_branch(ctx->repo); 319 ctx->qry.head = find_default_branch(ctx->repo);
320 ctx->repo->defbranch = ctx->qry.head; 320 ctx->repo->defbranch = ctx->qry.head;
321 } 321 }
322 322
323 if (!ctx->qry.head) { 323 if (!ctx->qry.head) {
324 cgit_print_http_headers(ctx); 324 cgit_print_http_headers(ctx);
325 cgit_print_docstart(ctx); 325 cgit_print_docstart(ctx);
326 cgit_print_pageheader(ctx); 326 cgit_print_pageheader(ctx);
327 cgit_print_error("Repository seems to be empty"); 327 cgit_print_error("Repository seems to be empty");
328 cgit_print_docend(); 328 cgit_print_docend();
329 return 1; 329 return 1;
330 } 330 }
331 331
332 if (get_sha1(ctx->qry.head, sha1)) { 332 if (get_sha1(ctx->qry.head, sha1)) {
333 tmp = xstrdup(ctx->qry.head); 333 tmp = xstrdup(ctx->qry.head);
334 ctx->qry.head = ctx->repo->defbranch; 334 ctx->qry.head = ctx->repo->defbranch;
335 ctx->page.status = 404; 335 ctx->page.status = 404;
336 ctx->page.statusmsg = "not found"; 336 ctx->page.statusmsg = "not found";
337 cgit_print_http_headers(ctx); 337 cgit_print_http_headers(ctx);
338 cgit_print_docstart(ctx); 338 cgit_print_docstart(ctx);
339 cgit_print_pageheader(ctx); 339 cgit_print_pageheader(ctx);
340 cgit_print_error(fmt("Invalid branch: %s", tmp)); 340 cgit_print_error(fmt("Invalid branch: %s", tmp));
341 cgit_print_docend(); 341 cgit_print_docend();
342 return 1; 342 return 1;
343 } 343 }
344 return 0; 344 return 0;
345} 345}
346 346
347static void process_request(void *cbdata) 347static void process_request(void *cbdata)
348{ 348{
349 struct cgit_context *ctx = cbdata; 349 struct cgit_context *ctx = cbdata;
350 struct cgit_cmd *cmd; 350 struct cgit_cmd *cmd;
351 351
352 cmd = cgit_get_cmd(ctx); 352 cmd = cgit_get_cmd(ctx);
353 if (!cmd) { 353 if (!cmd) {
354 ctx->page.title = "cgit error"; 354 ctx->page.title = "cgit error";
355 cgit_print_http_headers(ctx); 355 cgit_print_http_headers(ctx);
356 cgit_print_docstart(ctx); 356 cgit_print_docstart(ctx);
357 cgit_print_pageheader(ctx); 357 cgit_print_pageheader(ctx);
358 cgit_print_error("Invalid request"); 358 cgit_print_error("Invalid request");
359 cgit_print_docend(); 359 cgit_print_docend();
360 return; 360 return;
361 } 361 }
362 362
363 if (cmd->want_repo && !ctx->repo) { 363 if (cmd->want_repo && !ctx->repo) {
364 cgit_print_http_headers(ctx); 364 cgit_print_http_headers(ctx);
365 cgit_print_docstart(ctx); 365 cgit_print_docstart(ctx);
366 cgit_print_pageheader(ctx); 366 cgit_print_pageheader(ctx);
367 cgit_print_error(fmt("No repository selected")); 367 cgit_print_error(fmt("No repository selected"));
368 cgit_print_docend(); 368 cgit_print_docend();
369 return; 369 return;
370 } 370 }
371 371
372 if (ctx->repo && prepare_repo_cmd(ctx)) 372 if (ctx->repo && prepare_repo_cmd(ctx))
373 return; 373 return;
374 374
375 if (cmd->want_layout) { 375 if (cmd->want_layout) {
376 cgit_print_http_headers(ctx); 376 cgit_print_http_headers(ctx);
377 cgit_print_docstart(ctx); 377 cgit_print_docstart(ctx);
378 cgit_print_pageheader(ctx); 378 cgit_print_pageheader(ctx);
379 } 379 }
380 380
381 cmd->fn(ctx); 381 cmd->fn(ctx);
382 382
383 if (cmd->want_layout) 383 if (cmd->want_layout)
384 cgit_print_docend(); 384 cgit_print_docend();
385} 385}
386 386
387int cmp_repos(const void *a, const void *b) 387int cmp_repos(const void *a, const void *b)
388{ 388{
389 const struct cgit_repo *ra = a, *rb = b; 389 const struct cgit_repo *ra = a, *rb = b;
390 return strcmp(ra->url, rb->url); 390 return strcmp(ra->url, rb->url);
391} 391}
392 392
393void print_repo(struct cgit_repo *repo) 393void print_repo(struct cgit_repo *repo)
394{ 394{
395 printf("repo.url=%s\n", repo->url); 395 printf("repo.url=%s\n", repo->url);
396 printf("repo.name=%s\n", repo->name); 396 printf("repo.name=%s\n", repo->name);
397 printf("repo.path=%s\n", repo->path); 397 printf("repo.path=%s\n", repo->path);
398 if (repo->owner) 398 if (repo->owner)
399 printf("repo.owner=%s\n", repo->owner); 399 printf("repo.owner=%s\n", repo->owner);
400 if (repo->desc) 400 if (repo->desc)
401 printf("repo.desc=%s\n", repo->desc); 401 printf("repo.desc=%s\n", repo->desc);
402 if (repo->readme) 402 if (repo->readme)
403 printf("repo.readme=%s\n", repo->readme); 403 printf("repo.readme=%s\n", repo->readme);
404 printf("\n"); 404 printf("\n");
405} 405}
406 406
407void print_repolist(struct cgit_repolist *list) 407void print_repolist(struct cgit_repolist *list)
408{ 408{
409 int i; 409 int i;
410 410
411 for(i = 0; i < list->count; i++) 411 for(i = 0; i < list->count; i++)
412 print_repo(&list->repos[i]); 412 print_repo(&list->repos[i]);
413} 413}
414 414
415 415
416static void cgit_parse_args(int argc, const char **argv) 416static void cgit_parse_args(int argc, const char **argv)
417{ 417{
418 int i; 418 int i;
419 int scan = 0; 419 int scan = 0;
420 420
421 for (i = 1; i < argc; i++) { 421 for (i = 1; i < argc; i++) {
422 if (!strncmp(argv[i], "--cache=", 8)) { 422 if (!strncmp(argv[i], "--cache=", 8)) {
423 ctx.cfg.cache_root = xstrdup(argv[i]+8); 423 ctx.cfg.cache_root = xstrdup(argv[i]+8);
424 } 424 }
425 if (!strcmp(argv[i], "--nocache")) { 425 if (!strcmp(argv[i], "--nocache")) {
426 ctx.cfg.nocache = 1; 426 ctx.cfg.nocache = 1;
427 } 427 }
428 if (!strncmp(argv[i], "--query=", 8)) { 428 if (!strncmp(argv[i], "--query=", 8)) {
429 ctx.qry.raw = xstrdup(argv[i]+8); 429 ctx.qry.raw = xstrdup(argv[i]+8);
430 } 430 }
431 if (!strncmp(argv[i], "--repo=", 7)) { 431 if (!strncmp(argv[i], "--repo=", 7)) {
432 ctx.qry.repo = xstrdup(argv[i]+7); 432 ctx.qry.repo = xstrdup(argv[i]+7);
433 } 433 }
434 if (!strncmp(argv[i], "--page=", 7)) { 434 if (!strncmp(argv[i], "--page=", 7)) {
435 ctx.qry.page = xstrdup(argv[i]+7); 435 ctx.qry.page = xstrdup(argv[i]+7);
436 } 436 }
437 if (!strncmp(argv[i], "--head=", 7)) { 437 if (!strncmp(argv[i], "--head=", 7)) {
438 ctx.qry.head = xstrdup(argv[i]+7); 438 ctx.qry.head = xstrdup(argv[i]+7);
439 ctx.qry.has_symref = 1; 439 ctx.qry.has_symref = 1;
440 } 440 }
441 if (!strncmp(argv[i], "--sha1=", 7)) { 441 if (!strncmp(argv[i], "--sha1=", 7)) {
442 ctx.qry.sha1 = xstrdup(argv[i]+7); 442 ctx.qry.sha1 = xstrdup(argv[i]+7);
443 ctx.qry.has_sha1 = 1; 443 ctx.qry.has_sha1 = 1;
444 } 444 }
445 if (!strncmp(argv[i], "--ofs=", 6)) { 445 if (!strncmp(argv[i], "--ofs=", 6)) {
446 ctx.qry.ofs = atoi(argv[i]+6); 446 ctx.qry.ofs = atoi(argv[i]+6);
447 } 447 }
448 if (!strncmp(argv[i], "--scan-tree=", 12)) { 448 if (!strncmp(argv[i], "--scan-tree=", 12)) {
449 scan++; 449 scan++;
450 scan_tree(argv[i] + 12); 450 scan_tree(argv[i] + 12);
451 } 451 }
452 } 452 }
453 if (scan) { 453 if (scan) {
454 qsort(cgit_repolist.repos, cgit_repolist.count, 454 qsort(cgit_repolist.repos, cgit_repolist.count,
455 sizeof(struct cgit_repo), cmp_repos); 455 sizeof(struct cgit_repo), cmp_repos);
456 print_repolist(&cgit_repolist); 456 print_repolist(&cgit_repolist);
457 exit(0); 457 exit(0);
458 } 458 }
459} 459}
460 460
461static int calc_ttl() 461static int calc_ttl()
462{ 462{
463 if (!ctx.repo) 463 if (!ctx.repo)
464 return ctx.cfg.cache_root_ttl; 464 return ctx.cfg.cache_root_ttl;
465 465
466 if (!ctx.qry.page) 466 if (!ctx.qry.page)
467 return ctx.cfg.cache_repo_ttl; 467 return ctx.cfg.cache_repo_ttl;
468 468
469 if (ctx.qry.has_symref) 469 if (ctx.qry.has_symref)
470 return ctx.cfg.cache_dynamic_ttl; 470 return ctx.cfg.cache_dynamic_ttl;
471 471
472 if (ctx.qry.has_sha1) 472 if (ctx.qry.has_sha1)
473 return ctx.cfg.cache_static_ttl; 473 return ctx.cfg.cache_static_ttl;
474 474
475 return ctx.cfg.cache_repo_ttl; 475 return ctx.cfg.cache_repo_ttl;
476} 476}
477 477
478int main(int argc, const char **argv) 478int main(int argc, const char **argv)
479{ 479{
480 const char *cgit_config_env = getenv("CGIT_CONFIG"); 480 const char *cgit_config_env = getenv("CGIT_CONFIG");
481 const char *method = getenv("REQUEST_METHOD"); 481 const char *method = getenv("REQUEST_METHOD");
482 const char *path; 482 const char *path;
483 char *qry; 483 char *qry;
484 int err, ttl; 484 int err, ttl;
485 485
486 prepare_context(&ctx); 486 prepare_context(&ctx);
487 cgit_repolist.length = 0; 487 cgit_repolist.length = 0;
488 cgit_repolist.count = 0; 488 cgit_repolist.count = 0;
489 cgit_repolist.repos = NULL; 489 cgit_repolist.repos = NULL;
490 490
491 if (getenv("SCRIPT_NAME")) 491 if (getenv("SCRIPT_NAME"))
492 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME")); 492 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME"));
493 if (getenv("QUERY_STRING")) 493 if (getenv("QUERY_STRING"))
494 ctx.qry.raw = xstrdup(getenv("QUERY_STRING")); 494 ctx.qry.raw = xstrdup(getenv("QUERY_STRING"));
495 cgit_parse_args(argc, argv); 495 cgit_parse_args(argc, argv);
496 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG, 496 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG,
497 config_cb); 497 config_cb);
498 ctx.repo = NULL; 498 ctx.repo = NULL;
499 http_parse_querystring(ctx.qry.raw, querystring_cb); 499 http_parse_querystring(ctx.qry.raw, querystring_cb);
500 500
501 /* If virtual-root isn't specified in cgitrc, lets pretend 501 /* If virtual-root isn't specified in cgitrc, lets pretend
502 * that virtual-root equals SCRIPT_NAME. 502 * that virtual-root equals SCRIPT_NAME.
503 */ 503 */
504 if (!ctx.cfg.virtual_root) 504 if (!ctx.cfg.virtual_root)
505 ctx.cfg.virtual_root = ctx.cfg.script_name; 505 ctx.cfg.virtual_root = ctx.cfg.script_name;
506 506
507 /* If no url parameter is specified on the querystring, lets 507 /* If no url parameter is specified on the querystring, lets
508 * use PATH_INFO as url. This allows cgit to work with virtual 508 * use PATH_INFO as url. This allows cgit to work with virtual
509 * urls without the need for rewriterules in the webserver (as 509 * urls without the need for rewriterules in the webserver (as
510 * long as PATH_INFO is included in the cache lookup key). 510 * long as PATH_INFO is included in the cache lookup key).
511 */ 511 */
512 path = getenv("PATH_INFO"); 512 path = getenv("PATH_INFO");
513 if (!ctx.qry.url && path) { 513 if (!ctx.qry.url && path) {
514 if (path[0] == '/') 514 if (path[0] == '/')
515 path++; 515 path++;
516 ctx.qry.url = xstrdup(path); 516 ctx.qry.url = xstrdup(path);
517 if (ctx.qry.raw) { 517 if (ctx.qry.raw) {
518 qry = ctx.qry.raw; 518 qry = ctx.qry.raw;
519 ctx.qry.raw = xstrdup(fmt("%s?%s", path, qry)); 519 ctx.qry.raw = xstrdup(fmt("%s?%s", path, qry));
520 free(qry); 520 free(qry);
521 } else 521 } else
522 ctx.qry.raw = ctx.qry.url; 522 ctx.qry.raw = ctx.qry.url;
523 cgit_parse_url(ctx.qry.url); 523 cgit_parse_url(ctx.qry.url);
524 } 524 }
525 525
526 ttl = calc_ttl(); 526 ttl = calc_ttl();
527 ctx.page.expires += ttl*60; 527 ctx.page.expires += ttl*60;
528 if (method && !strcmp(method, "HEAD")) 528 if (method && !strcmp(method, "HEAD"))
529 ctx.cfg.nocache = 1; 529 ctx.cfg.nocache = 1;
530 if (ctx.cfg.nocache) 530 if (ctx.cfg.nocache)
531 ctx.cfg.cache_size = 0; 531 ctx.cfg.cache_size = 0;
532 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root, 532 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root,
533 ctx.qry.raw, ttl, process_request, &ctx); 533 ctx.qry.raw, ttl, process_request, &ctx);
534 if (err) 534 if (err)
535 cgit_print_error(fmt("Error processing page: %s (%d)", 535 cgit_print_error(fmt("Error processing page: %s (%d)",
536 strerror(err), err)); 536 strerror(err), err));
537 return err; 537 return err;
538} 538}
diff --git a/cgitrc.5.txt b/cgitrc.5.txt
index 4d656fe..54490eb 100644
--- a/cgitrc.5.txt
+++ b/cgitrc.5.txt
@@ -1,454 +1,454 @@
1CGITRC(5) 1CGITRC(5)
2======== 2========
3 3
4 4
5NAME 5NAME
6---- 6----
7cgitrc - runtime configuration for cgit 7cgitrc - runtime configuration for cgit
8 8
9 9
10SYNOPSIS 10SYNOPSIS
11-------- 11--------
12Cgitrc contains all runtime settings for cgit, including the list of git 12Cgitrc contains all runtime settings for cgit, including the list of git
13repositories, formatted as a line-separated list of NAME=VALUE pairs. Blank 13repositories, formatted as a line-separated list of NAME=VALUE pairs. Blank
14lines, and lines starting with '#', are ignored. 14lines, and lines starting with '#', are ignored.
15 15
16 16
17GLOBAL SETTINGS 17GLOBAL SETTINGS
18--------------- 18---------------
19about-filter:: 19about-filter::
20 Specifies a command which will be invoked to format the content of 20 Specifies a command which will be invoked to format the content of
21 about pages (both top-level and for each repository). The command will 21 about pages (both top-level and for each repository). The command will
22 get the content of the about-file on its STDIN, and the STDOUT from the 22 get the content of the about-file on its STDIN, and the STDOUT from the
23 command will be included verbatim on the about page. Default value: 23 command will be included verbatim on the about page. Default value:
24 none. 24 none.
25 25
26agefile:: 26agefile::
27 Specifies a path, relative to each repository path, which can be used 27 Specifies a path, relative to each repository path, which can be used
28 to specify the date and time of the youngest commit in the repository. 28 to specify the date and time of the youngest commit in the repository.
29 The first line in the file is used as input to the "parse_date" 29 The first line in the file is used as input to the "parse_date"
30 function in libgit. Recommended timestamp-format is "yyyy-mm-dd 30 function in libgit. Recommended timestamp-format is "yyyy-mm-dd
31 hh:mm:ss". Default value: "info/web/last-modified". 31 hh:mm:ss". Default value: "info/web/last-modified".
32 32
33cache-root:: 33cache-root::
34 Path used to store the cgit cache entries. Default value: 34 Path used to store the cgit cache entries. Default value:
35 "/var/cache/cgit". 35 "/var/cache/cgit".
36 36
37cache-dynamic-ttl:: 37cache-dynamic-ttl::
38 Number which specifies the time-to-live, in minutes, for the cached 38 Number which specifies the time-to-live, in minutes, for the cached
39 version of repository pages accessed without a fixed SHA1. Default 39 version of repository pages accessed without a fixed SHA1. Default
40 value: "5". 40 value: "5".
41 41
42cache-repo-ttl:: 42cache-repo-ttl::
43 Number which specifies the time-to-live, in minutes, for the cached 43 Number which specifies the time-to-live, in minutes, for the cached
44 version of the repository summary page. Default value: "5". 44 version of the repository summary page. Default value: "5".
45 45
46cache-root-ttl:: 46cache-root-ttl::
47 Number which specifies the time-to-live, in minutes, for the cached 47 Number which specifies the time-to-live, in minutes, for the cached
48 version of the repository index page. Default value: "5". 48 version of the repository index page. Default value: "5".
49 49
50cache-size:: 50cache-size::
51 The maximum number of entries in the cgit cache. Default value: "0" 51 The maximum number of entries in the cgit cache. Default value: "0"
52 (i.e. caching is disabled). 52 (i.e. caching is disabled).
53 53
54cache-static-ttl:: 54cache-static-ttl::
55 Number which specifies the time-to-live, in minutes, for the cached 55 Number which specifies the time-to-live, in minutes, for the cached
56 version of repository pages accessed with a fixed SHA1. Default value: 56 version of repository pages accessed with a fixed SHA1. Default value:
57 "5". 57 "5".
58 58
59clone-prefix:: 59clone-prefix::
60 Space-separated list of common prefixes which, when combined with a 60 Space-separated list of common prefixes which, when combined with a
61 repository url, generates valid clone urls for the repository. This 61 repository url, generates valid clone urls for the repository. This
62 setting is only used if `repo.clone-url` is unspecified. Default value: 62 setting is only used if `repo.clone-url` is unspecified. Default value:
63 none. 63 none.
64 64
65commit-filter:: 65commit-filter::
66 Specifies a command which will be invoked to format commit messages. 66 Specifies a command which will be invoked to format commit messages.
67 The command will get the message on its STDIN, and the STDOUT from the 67 The command will get the message on its STDIN, and the STDOUT from the
68 command will be included verbatim as the commit message, i.e. this can 68 command will be included verbatim as the commit message, i.e. this can
69 be used to implement bugtracker integration. Default value: none. 69 be used to implement bugtracker integration. Default value: none.
70 70
71css:: 71css::
72 Url which specifies the css document to include in all cgit pages. 72 Url which specifies the css document to include in all cgit pages.
73 Default value: "/cgit.css". 73 Default value: "/cgit.css".
74 74
75embedded:: 75embedded::
76 Flag which, when set to "1", will make cgit generate a html fragment 76 Flag which, when set to "1", will make cgit generate a html fragment
77 suitable for embedding in other html pages. Default value: none. See 77 suitable for embedding in other html pages. Default value: none. See
78 also: "noheader". 78 also: "noheader".
79 79
80enable-index-links:: 80enable-index-links::
81 Flag which, when set to "1", will make cgit generate extra links for 81 Flag which, when set to "1", will make cgit generate extra links for
82 each repo in the repository index (specifically, to the "summary", 82 each repo in the repository index (specifically, to the "summary",
83 "commit" and "tree" pages). Default value: "0". 83 "commit" and "tree" pages). Default value: "0".
84 84
85enable-log-filecount:: 85enable-log-filecount::
86 Flag which, when set to "1", will make cgit print the number of 86 Flag which, when set to "1", will make cgit print the number of
87 modified files for each commit on the repository log page. Default 87 modified files for each commit on the repository log page. Default
88 value: "0". 88 value: "0".
89 89
90enable-log-linecount:: 90enable-log-linecount::
91 Flag which, when set to "1", will make cgit print the number of added 91 Flag which, when set to "1", will make cgit print the number of added
92 and removed lines for each commit on the repository log page. Default 92 and removed lines for each commit on the repository log page. Default
93 value: "0". 93 value: "0".
94 94
95favicon:: 95favicon::
96 Url used as link to a shortcut icon for cgit. If specified, it is 96 Url used as link to a shortcut icon for cgit. If specified, it is
97 suggested to use the value "/favicon.ico" since certain browsers will 97 suggested to use the value "/favicon.ico" since certain browsers will
98 ignore other values. Default value: none. 98 ignore other values. Default value: none.
99 99
100footer:: 100footer::
101 The content of the file specified with this option will be included 101 The content of the file specified with this option will be included
102 verbatim at the bottom of all pages (i.e. it replaces the standard 102 verbatim at the bottom of all pages (i.e. it replaces the standard
103 "generated by..." message. Default value: none. 103 "generated by..." message. Default value: none.
104 104
105head-include:: 105head-include::
106 The content of the file specified with this option will be included 106 The content of the file specified with this option will be included
107 verbatim in the html HEAD section on all pages. Default value: none. 107 verbatim in the html HEAD section on all pages. Default value: none.
108 108
109header:: 109header::
110 The content of the file specified with this option will be included 110 The content of the file specified with this option will be included
111 verbatim at the top of all pages. Default value: none. 111 verbatim at the top of all pages. Default value: none.
112 112
113include:: 113include::
114 Name of a configfile to include before the rest of the current config- 114 Name of a configfile to include before the rest of the current config-
115 file is parsed. Default value: none. 115 file is parsed. Default value: none.
116 116
117index-header:: 117index-header::
118 The content of the file specified with this option will be included 118 The content of the file specified with this option will be included
119 verbatim above the repository index. This setting is deprecated, and 119 verbatim above the repository index. This setting is deprecated, and
120 will not be supported by cgit-1.0 (use root-readme instead). Default 120 will not be supported by cgit-1.0 (use root-readme instead). Default
121 value: none. 121 value: none.
122 122
123index-info:: 123index-info::
124 The content of the file specified with this option will be included 124 The content of the file specified with this option will be included
125 verbatim below the heading on the repository index page. This setting 125 verbatim below the heading on the repository index page. This setting
126 is deprecated, and will not be supported by cgit-1.0 (use root-desc 126 is deprecated, and will not be supported by cgit-1.0 (use root-desc
127 instead). Default value: none. 127 instead). Default value: none.
128 128
129local-time:: 129local-time::
130 Flag which, if set to "1", makes cgit print commit and tag times in the 130 Flag which, if set to "1", makes cgit print commit and tag times in the
131 servers timezone. Default value: "0". 131 servers timezone. Default value: "0".
132 132
133logo:: 133logo::
134 Url which specifies the source of an image which will be used as a logo 134 Url which specifies the source of an image which will be used as a logo
135 on all cgit pages. 135 on all cgit pages. Default value: "/cgit.png".
136 136
137logo-link:: 137logo-link::
138 Url loaded when clicking on the cgit logo image. If unspecified the 138 Url loaded when clicking on the cgit logo image. If unspecified the
139 calculated url of the repository index page will be used. Default 139 calculated url of the repository index page will be used. Default
140 value: none. 140 value: none.
141 141
142max-commit-count:: 142max-commit-count::
143 Specifies the number of entries to list per page in "log" view. Default 143 Specifies the number of entries to list per page in "log" view. Default
144 value: "50". 144 value: "50".
145 145
146max-message-length:: 146max-message-length::
147 Specifies the maximum number of commit message characters to display in 147 Specifies the maximum number of commit message characters to display in
148 "log" view. Default value: "80". 148 "log" view. Default value: "80".
149 149
150max-repo-count:: 150max-repo-count::
151 Specifies the number of entries to list per page on therepository 151 Specifies the number of entries to list per page on therepository
152 index page. Default value: "50". 152 index page. Default value: "50".
153 153
154max-repodesc-length:: 154max-repodesc-length::
155 Specifies the maximum number of repo description characters to display 155 Specifies the maximum number of repo description characters to display
156 on the repository index page. Default value: "80". 156 on the repository index page. Default value: "80".
157 157
158max-stats:: 158max-stats::
159 Set the default maximum statistics period. Valid values are "week", 159 Set the default maximum statistics period. Valid values are "week",
160 "month", "quarter" and "year". If unspecified, statistics are 160 "month", "quarter" and "year". If unspecified, statistics are
161 disabled. Default value: none. See also: "repo.max-stats". 161 disabled. Default value: none. See also: "repo.max-stats".
162 162
163mimetype.<ext>:: 163mimetype.<ext>::
164 Set the mimetype for the specified filename extension. This is used 164 Set the mimetype for the specified filename extension. This is used
165 by the `plain` command when returning blob content. 165 by the `plain` command when returning blob content.
166 166
167module-link:: 167module-link::
168 Text which will be used as the formatstring for a hyperlink when a 168 Text which will be used as the formatstring for a hyperlink when a
169 submodule is printed in a directory listing. The arguments for the 169 submodule is printed in a directory listing. The arguments for the
170 formatstring are the path and SHA1 of the submodule commit. Default 170 formatstring are the path and SHA1 of the submodule commit. Default
171 value: "./?repo=%s&page=commit&id=%s" 171 value: "./?repo=%s&page=commit&id=%s"
172 172
173nocache:: 173nocache::
174 If set to the value "1" caching will be disabled. This settings is 174 If set to the value "1" caching will be disabled. This settings is
175 deprecated, and will not be honored starting with cgit-1.0. Default 175 deprecated, and will not be honored starting with cgit-1.0. Default
176 value: "0". 176 value: "0".
177 177
178noplainemail:: 178noplainemail::
179 If set to "1" showing full author email adresses will be disabled. 179 If set to "1" showing full author email adresses will be disabled.
180 Default value: "0". 180 Default value: "0".
181 181
182noheader:: 182noheader::
183 Flag which, when set to "1", will make cgit omit the standard header 183 Flag which, when set to "1", will make cgit omit the standard header
184 on all pages. Default value: none. See also: "embedded". 184 on all pages. Default value: none. See also: "embedded".
185 185
186renamelimit:: 186renamelimit::
187 Maximum number of files to consider when detecting renames. The value 187 Maximum number of files to consider when detecting renames. The value
188 "-1" uses the compiletime value in git (for further info, look at 188 "-1" uses the compiletime value in git (for further info, look at
189 `man git-diff`). Default value: "-1". 189 `man git-diff`). Default value: "-1".
190 190
191repo.group:: 191repo.group::
192 A value for the current repository group, which all repositories 192 A value for the current repository group, which all repositories
193 specified after this setting will inherit. Default value: none. 193 specified after this setting will inherit. Default value: none.
194 194
195robots:: 195robots::
196 Text used as content for the "robots" meta-tag. Default value: 196 Text used as content for the "robots" meta-tag. Default value:
197 "index, nofollow". 197 "index, nofollow".
198 198
199root-desc:: 199root-desc::
200 Text printed below the heading on the repository index page. Default 200 Text printed below the heading on the repository index page. Default
201 value: "a fast webinterface for the git dscm". 201 value: "a fast webinterface for the git dscm".
202 202
203root-readme:: 203root-readme::
204 The content of the file specified with this option will be included 204 The content of the file specified with this option will be included
205 verbatim below the "about" link on the repository index page. Default 205 verbatim below the "about" link on the repository index page. Default
206 value: none. 206 value: none.
207 207
208root-title:: 208root-title::
209 Text printed as heading on the repository index page. Default value: 209 Text printed as heading on the repository index page. Default value:
210 "Git Repository Browser". 210 "Git Repository Browser".
211 211
212snapshots:: 212snapshots::
213 Text which specifies the default (and allowed) set of snapshot formats 213 Text which specifies the default (and allowed) set of snapshot formats
214 supported by cgit. The value is a space-separated list of zero or more 214 supported by cgit. The value is a space-separated list of zero or more
215 of the following values: 215 of the following values:
216 "tar" uncompressed tar-file 216 "tar" uncompressed tar-file
217 "tar.gz"gzip-compressed tar-file 217 "tar.gz"gzip-compressed tar-file
218 "tar.bz2"bzip-compressed tar-file 218 "tar.bz2"bzip-compressed tar-file
219 "zip" zip-file 219 "zip" zip-file
220 Default value: none. 220 Default value: none.
221 221
222source-filter:: 222source-filter::
223 Specifies a command which will be invoked to format plaintext blobs 223 Specifies a command which will be invoked to format plaintext blobs
224 in the tree view. The command will get the blob content on its STDIN 224 in the tree view. The command will get the blob content on its STDIN
225 and the name of the blob as its only command line argument. The STDOUT 225 and the name of the blob as its only command line argument. The STDOUT
226 from the command will be included verbatim as the blob contents, i.e. 226 from the command will be included verbatim as the blob contents, i.e.
227 this can be used to implement e.g. syntax highlighting. Default value: 227 this can be used to implement e.g. syntax highlighting. Default value:
228 none. 228 none.
229 229
230summary-branches:: 230summary-branches::
231 Specifies the number of branches to display in the repository "summary" 231 Specifies the number of branches to display in the repository "summary"
232 view. Default value: "10". 232 view. Default value: "10".
233 233
234summary-log:: 234summary-log::
235 Specifies the number of log entries to display in the repository 235 Specifies the number of log entries to display in the repository
236 "summary" view. Default value: "10". 236 "summary" view. Default value: "10".
237 237
238summary-tags:: 238summary-tags::
239 Specifies the number of tags to display in the repository "summary" 239 Specifies the number of tags to display in the repository "summary"
240 view. Default value: "10". 240 view. Default value: "10".
241 241
242virtual-root:: 242virtual-root::
243 Url which, if specified, will be used as root for all cgit links. It 243 Url which, if specified, will be used as root for all cgit links. It
244 will also cause cgit to generate 'virtual urls', i.e. urls like 244 will also cause cgit to generate 'virtual urls', i.e. urls like
245 '/cgit/tree/README' as opposed to '?r=cgit&p=tree&path=README'. Default 245 '/cgit/tree/README' as opposed to '?r=cgit&p=tree&path=README'. Default
246 value: none. 246 value: none.
247 NOTE: cgit has recently learned how to use PATH_INFO to achieve the 247 NOTE: cgit has recently learned how to use PATH_INFO to achieve the
248 same kind of virtual urls, so this option will probably be deprecated. 248 same kind of virtual urls, so this option will probably be deprecated.
249 249
250REPOSITORY SETTINGS 250REPOSITORY SETTINGS
251------------------- 251-------------------
252repo.about-filter:: 252repo.about-filter::
253 Override the default about-filter. Default value: <about-filter>. 253 Override the default about-filter. Default value: <about-filter>.
254 254
255repo.clone-url:: 255repo.clone-url::
256 A list of space-separated urls which can be used to clone this repo. 256 A list of space-separated urls which can be used to clone this repo.
257 Default value: none. 257 Default value: none.
258 258
259repo.commit-filter:: 259repo.commit-filter::
260 Override the default commit-filter. Default value: <commit-filter>. 260 Override the default commit-filter. Default value: <commit-filter>.
261 261
262repo.defbranch:: 262repo.defbranch::
263 The name of the default branch for this repository. If no such branch 263 The name of the default branch for this repository. If no such branch
264 exists in the repository, the first branch name (when sorted) is used 264 exists in the repository, the first branch name (when sorted) is used
265 as default instead. Default value: "master". 265 as default instead. Default value: "master".
266 266
267repo.desc:: 267repo.desc::
268 The value to show as repository description. Default value: none. 268 The value to show as repository description. Default value: none.
269 269
270repo.enable-log-filecount:: 270repo.enable-log-filecount::
271 A flag which can be used to disable the global setting 271 A flag which can be used to disable the global setting
272 `enable-log-filecount'. Default value: none. 272 `enable-log-filecount'. Default value: none.
273 273
274repo.enable-log-linecount:: 274repo.enable-log-linecount::
275 A flag which can be used to disable the global setting 275 A flag which can be used to disable the global setting
276 `enable-log-linecount'. Default value: none. 276 `enable-log-linecount'. Default value: none.
277 277
278repo.max-stats:: 278repo.max-stats::
279 Override the default maximum statistics period. Valid values are equal 279 Override the default maximum statistics period. Valid values are equal
280 to the values specified for the global "max-stats" setting. Default 280 to the values specified for the global "max-stats" setting. Default
281 value: none. 281 value: none.
282 282
283repo.name:: 283repo.name::
284 The value to show as repository name. Default value: <repo.url>. 284 The value to show as repository name. Default value: <repo.url>.
285 285
286repo.owner:: 286repo.owner::
287 A value used to identify the owner of the repository. Default value: 287 A value used to identify the owner of the repository. Default value:
288 none. 288 none.
289 289
290repo.path:: 290repo.path::
291 An absolute path to the repository directory. For non-bare repositories 291 An absolute path to the repository directory. For non-bare repositories
292 this is the .git-directory. Default value: none. 292 this is the .git-directory. Default value: none.
293 293
294repo.readme:: 294repo.readme::
295 A path (relative to <repo.path>) which specifies a file to include 295 A path (relative to <repo.path>) which specifies a file to include
296 verbatim as the "About" page for this repo. Default value: none. 296 verbatim as the "About" page for this repo. Default value: none.
297 297
298repo.snapshots:: 298repo.snapshots::
299 A mask of allowed snapshot-formats for this repo, restricted by the 299 A mask of allowed snapshot-formats for this repo, restricted by the
300 "snapshots" global setting. Default value: <snapshots>. 300 "snapshots" global setting. Default value: <snapshots>.
301 301
302repo.source-filter:: 302repo.source-filter::
303 Override the default source-filter. Default value: <source-filter>. 303 Override the default source-filter. Default value: <source-filter>.
304 304
305repo.url:: 305repo.url::
306 The relative url used to access the repository. This must be the first 306 The relative url used to access the repository. This must be the first
307 setting specified for each repo. Default value: none. 307 setting specified for each repo. Default value: none.
308 308
309 309
310EXAMPLE CGITRC FILE 310EXAMPLE CGITRC FILE
311------------------- 311-------------------
312 312
313.... 313....
314# Enable caching of up to 1000 output entriess 314# Enable caching of up to 1000 output entriess
315cache-size=1000 315cache-size=1000
316 316
317 317
318# Specify some default clone prefixes 318# Specify some default clone prefixes
319clone-prefix=git://foobar.com ssh://foobar.com/pub/git http://foobar.com/git 319clone-prefix=git://foobar.com ssh://foobar.com/pub/git http://foobar.com/git
320 320
321# Specify the css url 321# Specify the css url
322css=/css/cgit.css 322css=/css/cgit.css
323 323
324 324
325# Show extra links for each repository on the index page 325# Show extra links for each repository on the index page
326enable-index-links=1 326enable-index-links=1
327 327
328 328
329# Show number of affected files per commit on the log pages 329# Show number of affected files per commit on the log pages
330enable-log-filecount=1 330enable-log-filecount=1
331 331
332 332
333# Show number of added/removed lines per commit on the log pages 333# Show number of added/removed lines per commit on the log pages
334enable-log-linecount=1 334enable-log-linecount=1
335 335
336 336
337# Add a cgit favicon 337# Add a cgit favicon
338favicon=/favicon.ico 338favicon=/favicon.ico
339 339
340 340
341# Use a custom logo 341# Use a custom logo
342logo=/img/mylogo.png 342logo=/img/mylogo.png
343 343
344 344
345# Enable statistics per week, month and quarter 345# Enable statistics per week, month and quarter
346max-stats=quarter 346max-stats=quarter
347 347
348 348
349# Set the title and heading of the repository index page 349# Set the title and heading of the repository index page
350root-title=foobar.com git repositories 350root-title=foobar.com git repositories
351 351
352 352
353# Set a subheading for the repository index page 353# Set a subheading for the repository index page
354root-desc=tracking the foobar development 354root-desc=tracking the foobar development
355 355
356 356
357# Include some more info about foobar.com on the index page 357# Include some more info about foobar.com on the index page
358root-readme=/var/www/htdocs/about.html 358root-readme=/var/www/htdocs/about.html
359 359
360 360
361# Allow download of tar.gz, tar.bz2 and zip-files 361# Allow download of tar.gz, tar.bz2 and zip-files
362snapshots=tar.gz tar.bz2 zip 362snapshots=tar.gz tar.bz2 zip
363 363
364 364
365## 365##
366## List of common mimetypes 366## List of common mimetypes
367## 367##
368 368
369mimetype.git=image/git 369mimetype.git=image/git
370mimetype.html=text/html 370mimetype.html=text/html
371mimetype.jpg=image/jpeg 371mimetype.jpg=image/jpeg
372mimetype.jpeg=image/jpeg 372mimetype.jpeg=image/jpeg
373mimetype.pdf=application/pdf 373mimetype.pdf=application/pdf
374mimetype.png=image/png 374mimetype.png=image/png
375mimetype.svg=image/svg+xml 375mimetype.svg=image/svg+xml
376 376
377 377
378## 378##
379## List of repositories. 379## List of repositories.
380## PS: Any repositories listed when repo.group is unset will not be 380## PS: Any repositories listed when repo.group is unset will not be
381## displayed under a group heading 381## displayed under a group heading
382## PPS: This list could be kept in a different file (e.g. '/etc/cgitrepos') 382## PPS: This list could be kept in a different file (e.g. '/etc/cgitrepos')
383## and included like this: 383## and included like this:
384## include=/etc/cgitrepos 384## include=/etc/cgitrepos
385## 385##
386 386
387 387
388repo.url=foo 388repo.url=foo
389repo.path=/pub/git/foo.git 389repo.path=/pub/git/foo.git
390repo.desc=the master foo repository 390repo.desc=the master foo repository
391repo.owner=fooman@foobar.com 391repo.owner=fooman@foobar.com
392repo.readme=info/web/about.html 392repo.readme=info/web/about.html
393 393
394 394
395repo.url=bar 395repo.url=bar
396repo.path=/pub/git/bar.git 396repo.path=/pub/git/bar.git
397repo.desc=the bars for your foo 397repo.desc=the bars for your foo
398repo.owner=barman@foobar.com 398repo.owner=barman@foobar.com
399repo.readme=info/web/about.html 399repo.readme=info/web/about.html
400 400
401 401
402# The next repositories will be displayed under the 'extras' heading 402# The next repositories will be displayed under the 'extras' heading
403repo.group=extras 403repo.group=extras
404 404
405 405
406repo.url=baz 406repo.url=baz
407repo.path=/pub/git/baz.git 407repo.path=/pub/git/baz.git
408repo.desc=a set of extensions for bar users 408repo.desc=a set of extensions for bar users
409 409
410repo.url=wiz 410repo.url=wiz
411repo.path=/pub/git/wiz.git 411repo.path=/pub/git/wiz.git
412repo.desc=the wizard of foo 412repo.desc=the wizard of foo
413 413
414 414
415# Add some mirrored repositories 415# Add some mirrored repositories
416repo.group=mirrors 416repo.group=mirrors
417 417
418 418
419repo.url=git 419repo.url=git
420repo.path=/pub/git/git.git 420repo.path=/pub/git/git.git
421repo.desc=the dscm 421repo.desc=the dscm
422 422
423 423
424repo.url=linux 424repo.url=linux
425repo.path=/pub/git/linux.git 425repo.path=/pub/git/linux.git
426repo.desc=the kernel 426repo.desc=the kernel
427 427
428# Disable adhoc downloads of this repo 428# Disable adhoc downloads of this repo
429repo.snapshots=0 429repo.snapshots=0
430 430
431# Disable line-counts for this repo 431# Disable line-counts for this repo
432repo.enable-log-linecount=0 432repo.enable-log-linecount=0
433 433
434# Restrict the max statistics period for this repo 434# Restrict the max statistics period for this repo
435repo.max-stats=month 435repo.max-stats=month
436.... 436....
437 437
438 438
439BUGS 439BUGS
440---- 440----
441Comments currently cannot appear on the same line as a setting; the comment 441Comments currently cannot appear on the same line as a setting; the comment
442will be included as part of the value. E.g. this line: 442will be included as part of the value. E.g. this line:
443 443
444 robots=index # allow indexing 444 robots=index # allow indexing
445 445
446will generate the following html element: 446will generate the following html element:
447 447
448 <meta name='robots' content='index # allow indexing'/> 448 <meta name='robots' content='index # allow indexing'/>
449 449
450 450
451 451
452AUTHOR 452AUTHOR
453------ 453------
454Lars Hjemli <hjemli@gmail.com> 454Lars Hjemli <hjemli@gmail.com>