summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c4
-rw-r--r--cgit.h1
-rw-r--r--ui-shared.c2
3 files changed, 6 insertions, 1 deletions
diff --git a/cgit.c b/cgit.c
index a92383d..36777b5 100644
--- a/cgit.c
+++ b/cgit.c
@@ -192,128 +192,129 @@ static void querystring_cb(const char *name, const char *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
218char *xstrdupn(const char *str) 218char *xstrdupn(const char *str)
219{ 219{
220 return (str ? xstrdup(str) : NULL); 220 return (str ? xstrdup(str) : NULL);
221} 221}
222 222
223static void prepare_context(struct cgit_context *ctx) 223static void prepare_context(struct cgit_context *ctx)
224{ 224{
225 memset(ctx, 0, sizeof(ctx)); 225 memset(ctx, 0, sizeof(ctx));
226 ctx->cfg.agefile = "info/web/last-modified"; 226 ctx->cfg.agefile = "info/web/last-modified";
227 ctx->cfg.nocache = 0; 227 ctx->cfg.nocache = 0;
228 ctx->cfg.cache_size = 0; 228 ctx->cfg.cache_size = 0;
229 ctx->cfg.cache_dynamic_ttl = 5; 229 ctx->cfg.cache_dynamic_ttl = 5;
230 ctx->cfg.cache_max_create_time = 5; 230 ctx->cfg.cache_max_create_time = 5;
231 ctx->cfg.cache_repo_ttl = 5; 231 ctx->cfg.cache_repo_ttl = 5;
232 ctx->cfg.cache_root = CGIT_CACHE_ROOT; 232 ctx->cfg.cache_root = CGIT_CACHE_ROOT;
233 ctx->cfg.cache_root_ttl = 5; 233 ctx->cfg.cache_root_ttl = 5;
234 ctx->cfg.cache_static_ttl = -1; 234 ctx->cfg.cache_static_ttl = -1;
235 ctx->cfg.css = "/cgit.css"; 235 ctx->cfg.css = "/cgit.css";
236 ctx->cfg.logo = "/git-logo.png"; 236 ctx->cfg.logo = "/git-logo.png";
237 ctx->cfg.local_time = 0; 237 ctx->cfg.local_time = 0;
238 ctx->cfg.max_repo_count = 50; 238 ctx->cfg.max_repo_count = 50;
239 ctx->cfg.max_commit_count = 50; 239 ctx->cfg.max_commit_count = 50;
240 ctx->cfg.max_lock_attempts = 5; 240 ctx->cfg.max_lock_attempts = 5;
241 ctx->cfg.max_msg_len = 80; 241 ctx->cfg.max_msg_len = 80;
242 ctx->cfg.max_repodesc_len = 80; 242 ctx->cfg.max_repodesc_len = 80;
243 ctx->cfg.max_stats = 0; 243 ctx->cfg.max_stats = 0;
244 ctx->cfg.module_link = "./?repo=%s&page=commit&id=%s"; 244 ctx->cfg.module_link = "./?repo=%s&page=commit&id=%s";
245 ctx->cfg.renamelimit = -1; 245 ctx->cfg.renamelimit = -1;
246 ctx->cfg.robots = "index, nofollow"; 246 ctx->cfg.robots = "index, nofollow";
247 ctx->cfg.root_title = "Git repository browser"; 247 ctx->cfg.root_title = "Git repository browser";
248 ctx->cfg.root_desc = "a fast webinterface for the git dscm"; 248 ctx->cfg.root_desc = "a fast webinterface for the git dscm";
249 ctx->cfg.script_name = CGIT_SCRIPT_NAME; 249 ctx->cfg.script_name = CGIT_SCRIPT_NAME;
250 ctx->cfg.summary_branches = 10; 250 ctx->cfg.summary_branches = 10;
251 ctx->cfg.summary_log = 10; 251 ctx->cfg.summary_log = 10;
252 ctx->cfg.summary_tags = 10; 252 ctx->cfg.summary_tags = 10;
253 ctx->env.cgit_config = xstrdupn(getenv("CGIT_CONFIG")); 253 ctx->env.cgit_config = xstrdupn(getenv("CGIT_CONFIG"));
254 ctx->env.http_host = xstrdupn(getenv("HTTP_HOST")); 254 ctx->env.http_host = xstrdupn(getenv("HTTP_HOST"));
255 ctx->env.https = xstrdupn(getenv("HTTPS")); 255 ctx->env.https = xstrdupn(getenv("HTTPS"));
256 ctx->env.no_http = xstrdupn(getenv("NO_HTTP"));
256 ctx->env.path_info = xstrdupn(getenv("PATH_INFO")); 257 ctx->env.path_info = xstrdupn(getenv("PATH_INFO"));
257 ctx->env.query_string = xstrdupn(getenv("QUERY_STRING")); 258 ctx->env.query_string = xstrdupn(getenv("QUERY_STRING"));
258 ctx->env.request_method = xstrdupn(getenv("REQUEST_METHOD")); 259 ctx->env.request_method = xstrdupn(getenv("REQUEST_METHOD"));
259 ctx->env.script_name = xstrdupn(getenv("SCRIPT_NAME")); 260 ctx->env.script_name = xstrdupn(getenv("SCRIPT_NAME"));
260 ctx->env.server_name = xstrdupn(getenv("SERVER_NAME")); 261 ctx->env.server_name = xstrdupn(getenv("SERVER_NAME"));
261 ctx->env.server_port = xstrdupn(getenv("SERVER_PORT")); 262 ctx->env.server_port = xstrdupn(getenv("SERVER_PORT"));
262 ctx->page.mimetype = "text/html"; 263 ctx->page.mimetype = "text/html";
263 ctx->page.charset = PAGE_ENCODING; 264 ctx->page.charset = PAGE_ENCODING;
264 ctx->page.filename = NULL; 265 ctx->page.filename = NULL;
265 ctx->page.size = 0; 266 ctx->page.size = 0;
266 ctx->page.modified = time(NULL); 267 ctx->page.modified = time(NULL);
267 ctx->page.expires = ctx->page.modified; 268 ctx->page.expires = ctx->page.modified;
268 ctx->page.etag = NULL; 269 ctx->page.etag = NULL;
269 memset(&ctx->cfg.mimetypes, 0, sizeof(struct string_list)); 270 memset(&ctx->cfg.mimetypes, 0, sizeof(struct string_list));
270 if (ctx->env.script_name) 271 if (ctx->env.script_name)
271 ctx->cfg.script_name = ctx->env.script_name; 272 ctx->cfg.script_name = ctx->env.script_name;
272 if (ctx->env.query_string) 273 if (ctx->env.query_string)
273 ctx->qry.raw = ctx->env.query_string; 274 ctx->qry.raw = ctx->env.query_string;
274 if (!ctx->env.cgit_config) 275 if (!ctx->env.cgit_config)
275 ctx->env.cgit_config = CGIT_CONFIG; 276 ctx->env.cgit_config = CGIT_CONFIG;
276} 277}
277 278
278struct refmatch { 279struct refmatch {
279 char *req_ref; 280 char *req_ref;
280 char *first_ref; 281 char *first_ref;
281 int match; 282 int match;
282}; 283};
283 284
284int find_current_ref(const char *refname, const unsigned char *sha1, 285int find_current_ref(const char *refname, const unsigned char *sha1,
285 int flags, void *cb_data) 286 int flags, void *cb_data)
286{ 287{
287 struct refmatch *info; 288 struct refmatch *info;
288 289
289 info = (struct refmatch *)cb_data; 290 info = (struct refmatch *)cb_data;
290 if (!strcmp(refname, info->req_ref)) 291 if (!strcmp(refname, info->req_ref))
291 info->match = 1; 292 info->match = 1;
292 if (!info->first_ref) 293 if (!info->first_ref)
293 info->first_ref = xstrdup(refname); 294 info->first_ref = xstrdup(refname);
294 return info->match; 295 return info->match;
295} 296}
296 297
297char *find_default_branch(struct cgit_repo *repo) 298char *find_default_branch(struct cgit_repo *repo)
298{ 299{
299 struct refmatch info; 300 struct refmatch info;
300 char *ref; 301 char *ref;
301 302
302 info.req_ref = repo->defbranch; 303 info.req_ref = repo->defbranch;
303 info.first_ref = NULL; 304 info.first_ref = NULL;
304 info.match = 0; 305 info.match = 0;
305 for_each_branch_ref(find_current_ref, &info); 306 for_each_branch_ref(find_current_ref, &info);
306 if (info.match) 307 if (info.match)
307 ref = info.req_ref; 308 ref = info.req_ref;
308 else 309 else
309 ref = info.first_ref; 310 ref = info.first_ref;
310 if (ref) 311 if (ref)
311 ref = xstrdup(ref); 312 ref = xstrdup(ref);
312 return ref; 313 return ref;
313} 314}
314 315
315static int prepare_repo_cmd(struct cgit_context *ctx) 316static int prepare_repo_cmd(struct cgit_context *ctx)
316{ 317{
317 char *tmp; 318 char *tmp;
318 unsigned char sha1[20]; 319 unsigned char sha1[20];
319 int nongit = 0; 320 int nongit = 0;
@@ -384,128 +385,131 @@ static void process_request(void *cbdata)
384 cgit_print_http_headers(ctx); 385 cgit_print_http_headers(ctx);
385 cgit_print_docstart(ctx); 386 cgit_print_docstart(ctx);
386 cgit_print_pageheader(ctx); 387 cgit_print_pageheader(ctx);
387 cgit_print_error(fmt("No repository selected")); 388 cgit_print_error(fmt("No repository selected"));
388 cgit_print_docend(); 389 cgit_print_docend();
389 return; 390 return;
390 } 391 }
391 392
392 if (ctx->repo && prepare_repo_cmd(ctx)) 393 if (ctx->repo && prepare_repo_cmd(ctx))
393 return; 394 return;
394 395
395 if (cmd->want_layout) { 396 if (cmd->want_layout) {
396 cgit_print_http_headers(ctx); 397 cgit_print_http_headers(ctx);
397 cgit_print_docstart(ctx); 398 cgit_print_docstart(ctx);
398 cgit_print_pageheader(ctx); 399 cgit_print_pageheader(ctx);
399 } 400 }
400 401
401 cmd->fn(ctx); 402 cmd->fn(ctx);
402 403
403 if (cmd->want_layout) 404 if (cmd->want_layout)
404 cgit_print_docend(); 405 cgit_print_docend();
405} 406}
406 407
407int cmp_repos(const void *a, const void *b) 408int cmp_repos(const void *a, const void *b)
408{ 409{
409 const struct cgit_repo *ra = a, *rb = b; 410 const struct cgit_repo *ra = a, *rb = b;
410 return strcmp(ra->url, rb->url); 411 return strcmp(ra->url, rb->url);
411} 412}
412 413
413void print_repo(struct cgit_repo *repo) 414void print_repo(struct cgit_repo *repo)
414{ 415{
415 printf("repo.url=%s\n", repo->url); 416 printf("repo.url=%s\n", repo->url);
416 printf("repo.name=%s\n", repo->name); 417 printf("repo.name=%s\n", repo->name);
417 printf("repo.path=%s\n", repo->path); 418 printf("repo.path=%s\n", repo->path);
418 if (repo->owner) 419 if (repo->owner)
419 printf("repo.owner=%s\n", repo->owner); 420 printf("repo.owner=%s\n", repo->owner);
420 if (repo->desc) 421 if (repo->desc)
421 printf("repo.desc=%s\n", repo->desc); 422 printf("repo.desc=%s\n", repo->desc);
422 if (repo->readme) 423 if (repo->readme)
423 printf("repo.readme=%s\n", repo->readme); 424 printf("repo.readme=%s\n", repo->readme);
424 printf("\n"); 425 printf("\n");
425} 426}
426 427
427void print_repolist(struct cgit_repolist *list) 428void print_repolist(struct cgit_repolist *list)
428{ 429{
429 int i; 430 int i;
430 431
431 for(i = 0; i < list->count; i++) 432 for(i = 0; i < list->count; i++)
432 print_repo(&list->repos[i]); 433 print_repo(&list->repos[i]);
433} 434}
434 435
435 436
436static void cgit_parse_args(int argc, const char **argv) 437static void cgit_parse_args(int argc, const char **argv)
437{ 438{
438 int i; 439 int i;
439 int scan = 0; 440 int scan = 0;
440 441
441 for (i = 1; i < argc; i++) { 442 for (i = 1; i < argc; i++) {
442 if (!strncmp(argv[i], "--cache=", 8)) { 443 if (!strncmp(argv[i], "--cache=", 8)) {
443 ctx.cfg.cache_root = xstrdup(argv[i]+8); 444 ctx.cfg.cache_root = xstrdup(argv[i]+8);
444 } 445 }
445 if (!strcmp(argv[i], "--nocache")) { 446 if (!strcmp(argv[i], "--nocache")) {
446 ctx.cfg.nocache = 1; 447 ctx.cfg.nocache = 1;
447 } 448 }
449 if (!strcmp(argv[i], "--nohttp")) {
450 ctx.env.no_http = "1";
451 }
448 if (!strncmp(argv[i], "--query=", 8)) { 452 if (!strncmp(argv[i], "--query=", 8)) {
449 ctx.qry.raw = xstrdup(argv[i]+8); 453 ctx.qry.raw = xstrdup(argv[i]+8);
450 } 454 }
451 if (!strncmp(argv[i], "--repo=", 7)) { 455 if (!strncmp(argv[i], "--repo=", 7)) {
452 ctx.qry.repo = xstrdup(argv[i]+7); 456 ctx.qry.repo = xstrdup(argv[i]+7);
453 } 457 }
454 if (!strncmp(argv[i], "--page=", 7)) { 458 if (!strncmp(argv[i], "--page=", 7)) {
455 ctx.qry.page = xstrdup(argv[i]+7); 459 ctx.qry.page = xstrdup(argv[i]+7);
456 } 460 }
457 if (!strncmp(argv[i], "--head=", 7)) { 461 if (!strncmp(argv[i], "--head=", 7)) {
458 ctx.qry.head = xstrdup(argv[i]+7); 462 ctx.qry.head = xstrdup(argv[i]+7);
459 ctx.qry.has_symref = 1; 463 ctx.qry.has_symref = 1;
460 } 464 }
461 if (!strncmp(argv[i], "--sha1=", 7)) { 465 if (!strncmp(argv[i], "--sha1=", 7)) {
462 ctx.qry.sha1 = xstrdup(argv[i]+7); 466 ctx.qry.sha1 = xstrdup(argv[i]+7);
463 ctx.qry.has_sha1 = 1; 467 ctx.qry.has_sha1 = 1;
464 } 468 }
465 if (!strncmp(argv[i], "--ofs=", 6)) { 469 if (!strncmp(argv[i], "--ofs=", 6)) {
466 ctx.qry.ofs = atoi(argv[i]+6); 470 ctx.qry.ofs = atoi(argv[i]+6);
467 } 471 }
468 if (!strncmp(argv[i], "--scan-tree=", 12)) { 472 if (!strncmp(argv[i], "--scan-tree=", 12)) {
469 scan++; 473 scan++;
470 scan_tree(argv[i] + 12); 474 scan_tree(argv[i] + 12);
471 } 475 }
472 } 476 }
473 if (scan) { 477 if (scan) {
474 qsort(cgit_repolist.repos, cgit_repolist.count, 478 qsort(cgit_repolist.repos, cgit_repolist.count,
475 sizeof(struct cgit_repo), cmp_repos); 479 sizeof(struct cgit_repo), cmp_repos);
476 print_repolist(&cgit_repolist); 480 print_repolist(&cgit_repolist);
477 exit(0); 481 exit(0);
478 } 482 }
479} 483}
480 484
481static int calc_ttl() 485static int calc_ttl()
482{ 486{
483 if (!ctx.repo) 487 if (!ctx.repo)
484 return ctx.cfg.cache_root_ttl; 488 return ctx.cfg.cache_root_ttl;
485 489
486 if (!ctx.qry.page) 490 if (!ctx.qry.page)
487 return ctx.cfg.cache_repo_ttl; 491 return ctx.cfg.cache_repo_ttl;
488 492
489 if (ctx.qry.has_symref) 493 if (ctx.qry.has_symref)
490 return ctx.cfg.cache_dynamic_ttl; 494 return ctx.cfg.cache_dynamic_ttl;
491 495
492 if (ctx.qry.has_sha1) 496 if (ctx.qry.has_sha1)
493 return ctx.cfg.cache_static_ttl; 497 return ctx.cfg.cache_static_ttl;
494 498
495 return ctx.cfg.cache_repo_ttl; 499 return ctx.cfg.cache_repo_ttl;
496} 500}
497 501
498int main(int argc, const char **argv) 502int main(int argc, const char **argv)
499{ 503{
500 const char *path; 504 const char *path;
501 char *qry; 505 char *qry;
502 int err, ttl; 506 int err, ttl;
503 507
504 prepare_context(&ctx); 508 prepare_context(&ctx);
505 cgit_repolist.length = 0; 509 cgit_repolist.length = 0;
506 cgit_repolist.count = 0; 510 cgit_repolist.count = 0;
507 cgit_repolist.repos = NULL; 511 cgit_repolist.repos = NULL;
508 512
509 cgit_parse_args(argc, argv); 513 cgit_parse_args(argc, argv);
510 parse_configfile(ctx.env.cgit_config, config_cb); 514 parse_configfile(ctx.env.cgit_config, config_cb);
511 ctx.repo = NULL; 515 ctx.repo = NULL;
diff --git a/cgit.h b/cgit.h
index 893231d..d90ccdc 100644
--- a/cgit.h
+++ b/cgit.h
@@ -150,128 +150,129 @@ struct cgit_config {
150 char *favicon; 150 char *favicon;
151 char *footer; 151 char *footer;
152 char *head_include; 152 char *head_include;
153 char *header; 153 char *header;
154 char *index_header; 154 char *index_header;
155 char *index_info; 155 char *index_info;
156 char *logo; 156 char *logo;
157 char *logo_link; 157 char *logo_link;
158 char *module_link; 158 char *module_link;
159 char *repo_group; 159 char *repo_group;
160 char *robots; 160 char *robots;
161 char *root_title; 161 char *root_title;
162 char *root_desc; 162 char *root_desc;
163 char *root_readme; 163 char *root_readme;
164 char *script_name; 164 char *script_name;
165 char *virtual_root; 165 char *virtual_root;
166 int cache_size; 166 int cache_size;
167 int cache_dynamic_ttl; 167 int cache_dynamic_ttl;
168 int cache_max_create_time; 168 int cache_max_create_time;
169 int cache_repo_ttl; 169 int cache_repo_ttl;
170 int cache_root_ttl; 170 int cache_root_ttl;
171 int cache_static_ttl; 171 int cache_static_ttl;
172 int embedded; 172 int embedded;
173 int enable_index_links; 173 int enable_index_links;
174 int enable_log_filecount; 174 int enable_log_filecount;
175 int enable_log_linecount; 175 int enable_log_linecount;
176 int local_time; 176 int local_time;
177 int max_repo_count; 177 int max_repo_count;
178 int max_commit_count; 178 int max_commit_count;
179 int max_lock_attempts; 179 int max_lock_attempts;
180 int max_msg_len; 180 int max_msg_len;
181 int max_repodesc_len; 181 int max_repodesc_len;
182 int max_stats; 182 int max_stats;
183 int nocache; 183 int nocache;
184 int noplainemail; 184 int noplainemail;
185 int noheader; 185 int noheader;
186 int renamelimit; 186 int renamelimit;
187 int snapshots; 187 int snapshots;
188 int summary_branches; 188 int summary_branches;
189 int summary_log; 189 int summary_log;
190 int summary_tags; 190 int summary_tags;
191 struct string_list mimetypes; 191 struct string_list mimetypes;
192 struct cgit_filter *about_filter; 192 struct cgit_filter *about_filter;
193 struct cgit_filter *commit_filter; 193 struct cgit_filter *commit_filter;
194 struct cgit_filter *source_filter; 194 struct cgit_filter *source_filter;
195}; 195};
196 196
197struct cgit_page { 197struct cgit_page {
198 time_t modified; 198 time_t modified;
199 time_t expires; 199 time_t expires;
200 size_t size; 200 size_t size;
201 char *mimetype; 201 char *mimetype;
202 char *charset; 202 char *charset;
203 char *filename; 203 char *filename;
204 char *etag; 204 char *etag;
205 char *title; 205 char *title;
206 int status; 206 int status;
207 char *statusmsg; 207 char *statusmsg;
208}; 208};
209 209
210struct cgit_environment { 210struct cgit_environment {
211 char *cgit_config; 211 char *cgit_config;
212 char *http_host; 212 char *http_host;
213 char *https; 213 char *https;
214 char *no_http;
214 char *path_info; 215 char *path_info;
215 char *query_string; 216 char *query_string;
216 char *request_method; 217 char *request_method;
217 char *script_name; 218 char *script_name;
218 char *server_name; 219 char *server_name;
219 char *server_port; 220 char *server_port;
220}; 221};
221 222
222struct cgit_context { 223struct cgit_context {
223 struct cgit_environment env; 224 struct cgit_environment env;
224 struct cgit_query qry; 225 struct cgit_query qry;
225 struct cgit_config cfg; 226 struct cgit_config cfg;
226 struct cgit_repo *repo; 227 struct cgit_repo *repo;
227 struct cgit_page page; 228 struct cgit_page page;
228}; 229};
229 230
230struct cgit_snapshot_format { 231struct cgit_snapshot_format {
231 const char *suffix; 232 const char *suffix;
232 const char *mimetype; 233 const char *mimetype;
233 write_archive_fn_t write_func; 234 write_archive_fn_t write_func;
234 int bit; 235 int bit;
235}; 236};
236 237
237extern const char *cgit_version; 238extern const char *cgit_version;
238 239
239extern struct cgit_repolist cgit_repolist; 240extern struct cgit_repolist cgit_repolist;
240extern struct cgit_context ctx; 241extern struct cgit_context ctx;
241extern const struct cgit_snapshot_format cgit_snapshot_formats[]; 242extern const struct cgit_snapshot_format cgit_snapshot_formats[];
242 243
243extern struct cgit_repo *cgit_add_repo(const char *url); 244extern struct cgit_repo *cgit_add_repo(const char *url);
244extern struct cgit_repo *cgit_get_repoinfo(const char *url); 245extern struct cgit_repo *cgit_get_repoinfo(const char *url);
245extern void cgit_repo_config_cb(const char *name, const char *value); 246extern void cgit_repo_config_cb(const char *name, const char *value);
246 247
247extern int chk_zero(int result, char *msg); 248extern int chk_zero(int result, char *msg);
248extern int chk_positive(int result, char *msg); 249extern int chk_positive(int result, char *msg);
249extern int chk_non_negative(int result, char *msg); 250extern int chk_non_negative(int result, char *msg);
250 251
251extern char *trim_end(const char *str, char c); 252extern char *trim_end(const char *str, char c);
252extern char *strlpart(char *txt, int maxlen); 253extern char *strlpart(char *txt, int maxlen);
253extern char *strrpart(char *txt, int maxlen); 254extern char *strrpart(char *txt, int maxlen);
254 255
255extern void cgit_add_ref(struct reflist *list, struct refinfo *ref); 256extern void cgit_add_ref(struct reflist *list, struct refinfo *ref);
256extern int cgit_refs_cb(const char *refname, const unsigned char *sha1, 257extern int cgit_refs_cb(const char *refname, const unsigned char *sha1,
257 int flags, void *cb_data); 258 int flags, void *cb_data);
258 259
259extern void *cgit_free_commitinfo(struct commitinfo *info); 260extern void *cgit_free_commitinfo(struct commitinfo *info);
260 261
261extern int cgit_diff_files(const unsigned char *old_sha1, 262extern int cgit_diff_files(const unsigned char *old_sha1,
262 const unsigned char *new_sha1, 263 const unsigned char *new_sha1,
263 unsigned long *old_size, unsigned long *new_size, 264 unsigned long *old_size, unsigned long *new_size,
264 int *binary, linediff_fn fn); 265 int *binary, linediff_fn fn);
265 266
266extern void cgit_diff_tree(const unsigned char *old_sha1, 267extern void cgit_diff_tree(const unsigned char *old_sha1,
267 const unsigned char *new_sha1, 268 const unsigned char *new_sha1,
268 filepair_fn fn, const char *prefix); 269 filepair_fn fn, const char *prefix);
269 270
270extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); 271extern void cgit_diff_commit(struct commit *commit, filepair_fn fn);
271 272
272extern char *fmt(const char *format,...); 273extern char *fmt(const char *format,...);
273 274
274extern struct commitinfo *cgit_parse_commit(struct commit *commit); 275extern struct commitinfo *cgit_parse_commit(struct commit *commit);
275extern struct taginfo *cgit_parse_tag(struct tag *tag); 276extern struct taginfo *cgit_parse_tag(struct tag *tag);
276extern void cgit_parse_url(const char *url); 277extern void cgit_parse_url(const char *url);
277 278
diff --git a/ui-shared.c b/ui-shared.c
index 538ddda..4175bd8 100644
--- a/ui-shared.c
+++ b/ui-shared.c
@@ -394,129 +394,129 @@ void cgit_object_link(struct object *obj)
394 name = fmt("%s %s...", typename(obj->type), shortrev); 394 name = fmt("%s %s...", typename(obj->type), shortrev);
395 reporevlink(page, name, NULL, NULL, ctx.qry.head, fullrev, NULL); 395 reporevlink(page, name, NULL, NULL, ctx.qry.head, fullrev, NULL);
396} 396}
397 397
398void cgit_print_date(time_t secs, char *format, int local_time) 398void cgit_print_date(time_t secs, char *format, int local_time)
399{ 399{
400 char buf[64]; 400 char buf[64];
401 struct tm *time; 401 struct tm *time;
402 402
403 if (!secs) 403 if (!secs)
404 return; 404 return;
405 if(local_time) 405 if(local_time)
406 time = localtime(&secs); 406 time = localtime(&secs);
407 else 407 else
408 time = gmtime(&secs); 408 time = gmtime(&secs);
409 strftime(buf, sizeof(buf)-1, format, time); 409 strftime(buf, sizeof(buf)-1, format, time);
410 html_txt(buf); 410 html_txt(buf);
411} 411}
412 412
413void cgit_print_age(time_t t, time_t max_relative, char *format) 413void cgit_print_age(time_t t, time_t max_relative, char *format)
414{ 414{
415 time_t now, secs; 415 time_t now, secs;
416 416
417 if (!t) 417 if (!t)
418 return; 418 return;
419 time(&now); 419 time(&now);
420 secs = now - t; 420 secs = now - t;
421 421
422 if (secs > max_relative && max_relative >= 0) { 422 if (secs > max_relative && max_relative >= 0) {
423 cgit_print_date(t, format, ctx.cfg.local_time); 423 cgit_print_date(t, format, ctx.cfg.local_time);
424 return; 424 return;
425 } 425 }
426 426
427 if (secs < TM_HOUR * 2) { 427 if (secs < TM_HOUR * 2) {
428 htmlf("<span class='age-mins'>%.0f min.</span>", 428 htmlf("<span class='age-mins'>%.0f min.</span>",
429 secs * 1.0 / TM_MIN); 429 secs * 1.0 / TM_MIN);
430 return; 430 return;
431 } 431 }
432 if (secs < TM_DAY * 2) { 432 if (secs < TM_DAY * 2) {
433 htmlf("<span class='age-hours'>%.0f hours</span>", 433 htmlf("<span class='age-hours'>%.0f hours</span>",
434 secs * 1.0 / TM_HOUR); 434 secs * 1.0 / TM_HOUR);
435 return; 435 return;
436 } 436 }
437 if (secs < TM_WEEK * 2) { 437 if (secs < TM_WEEK * 2) {
438 htmlf("<span class='age-days'>%.0f days</span>", 438 htmlf("<span class='age-days'>%.0f days</span>",
439 secs * 1.0 / TM_DAY); 439 secs * 1.0 / TM_DAY);
440 return; 440 return;
441 } 441 }
442 if (secs < TM_MONTH * 2) { 442 if (secs < TM_MONTH * 2) {
443 htmlf("<span class='age-weeks'>%.0f weeks</span>", 443 htmlf("<span class='age-weeks'>%.0f weeks</span>",
444 secs * 1.0 / TM_WEEK); 444 secs * 1.0 / TM_WEEK);
445 return; 445 return;
446 } 446 }
447 if (secs < TM_YEAR * 2) { 447 if (secs < TM_YEAR * 2) {
448 htmlf("<span class='age-months'>%.0f months</span>", 448 htmlf("<span class='age-months'>%.0f months</span>",
449 secs * 1.0 / TM_MONTH); 449 secs * 1.0 / TM_MONTH);
450 return; 450 return;
451 } 451 }
452 htmlf("<span class='age-years'>%.0f years</span>", 452 htmlf("<span class='age-years'>%.0f years</span>",
453 secs * 1.0 / TM_YEAR); 453 secs * 1.0 / TM_YEAR);
454} 454}
455 455
456void cgit_print_http_headers(struct cgit_context *ctx) 456void cgit_print_http_headers(struct cgit_context *ctx)
457{ 457{
458 if (ctx->cfg.embedded) 458 if (ctx->env.no_http && !strcmp(ctx->env.no_http, "1"))
459 return; 459 return;
460 460
461 if (ctx->page.status) 461 if (ctx->page.status)
462 htmlf("Status: %d %s\n", ctx->page.status, ctx->page.statusmsg); 462 htmlf("Status: %d %s\n", ctx->page.status, ctx->page.statusmsg);
463 if (ctx->page.mimetype && ctx->page.charset) 463 if (ctx->page.mimetype && ctx->page.charset)
464 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype, 464 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype,
465 ctx->page.charset); 465 ctx->page.charset);
466 else if (ctx->page.mimetype) 466 else if (ctx->page.mimetype)
467 htmlf("Content-Type: %s\n", ctx->page.mimetype); 467 htmlf("Content-Type: %s\n", ctx->page.mimetype);
468 if (ctx->page.size) 468 if (ctx->page.size)
469 htmlf("Content-Length: %ld\n", ctx->page.size); 469 htmlf("Content-Length: %ld\n", ctx->page.size);
470 if (ctx->page.filename) 470 if (ctx->page.filename)
471 htmlf("Content-Disposition: inline; filename=\"%s\"\n", 471 htmlf("Content-Disposition: inline; filename=\"%s\"\n",
472 ctx->page.filename); 472 ctx->page.filename);
473 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified)); 473 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified));
474 htmlf("Expires: %s\n", http_date(ctx->page.expires)); 474 htmlf("Expires: %s\n", http_date(ctx->page.expires));
475 if (ctx->page.etag) 475 if (ctx->page.etag)
476 htmlf("ETag: \"%s\"\n", ctx->page.etag); 476 htmlf("ETag: \"%s\"\n", ctx->page.etag);
477 html("\n"); 477 html("\n");
478 if (ctx->env.request_method && !strcmp(ctx->env.request_method, "HEAD")) 478 if (ctx->env.request_method && !strcmp(ctx->env.request_method, "HEAD"))
479 exit(0); 479 exit(0);
480} 480}
481 481
482void cgit_print_docstart(struct cgit_context *ctx) 482void cgit_print_docstart(struct cgit_context *ctx)
483{ 483{
484 if (ctx->cfg.embedded) 484 if (ctx->cfg.embedded)
485 return; 485 return;
486 486
487 char *host = cgit_hosturl(); 487 char *host = cgit_hosturl();
488 html(cgit_doctype); 488 html(cgit_doctype);
489 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n"); 489 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n");
490 html("<head>\n"); 490 html("<head>\n");
491 html("<title>"); 491 html("<title>");
492 html_txt(ctx->page.title); 492 html_txt(ctx->page.title);
493 html("</title>\n"); 493 html("</title>\n");
494 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version); 494 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version);
495 if (ctx->cfg.robots && *ctx->cfg.robots) 495 if (ctx->cfg.robots && *ctx->cfg.robots)
496 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots); 496 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots);
497 html("<link rel='stylesheet' type='text/css' href='"); 497 html("<link rel='stylesheet' type='text/css' href='");
498 html_attr(ctx->cfg.css); 498 html_attr(ctx->cfg.css);
499 html("'/>\n"); 499 html("'/>\n");
500 if (ctx->cfg.favicon) { 500 if (ctx->cfg.favicon) {
501 html("<link rel='shortcut icon' href='"); 501 html("<link rel='shortcut icon' href='");
502 html_attr(ctx->cfg.favicon); 502 html_attr(ctx->cfg.favicon);
503 html("'/>\n"); 503 html("'/>\n");
504 } 504 }
505 if (host && ctx->repo) { 505 if (host && ctx->repo) {
506 html("<link rel='alternate' title='Atom feed' href='"); 506 html("<link rel='alternate' title='Atom feed' href='");
507 html(cgit_httpscheme()); 507 html(cgit_httpscheme());
508 html_attr(cgit_hosturl()); 508 html_attr(cgit_hosturl());
509 html_attr(cgit_fileurl(ctx->repo->url, "atom", ctx->qry.path, 509 html_attr(cgit_fileurl(ctx->repo->url, "atom", ctx->qry.path,
510 fmt("h=%s", ctx->qry.head))); 510 fmt("h=%s", ctx->qry.head)));
511 html("' type='application/atom+xml'/>\n"); 511 html("' type='application/atom+xml'/>\n");
512 } 512 }
513 if (ctx->cfg.head_include) 513 if (ctx->cfg.head_include)
514 html_include(ctx->cfg.head_include); 514 html_include(ctx->cfg.head_include);
515 html("</head>\n"); 515 html("</head>\n");
516 html("<body>\n"); 516 html("<body>\n");
517 if (ctx->cfg.header) 517 if (ctx->cfg.header)
518 html_include(ctx->cfg.header); 518 html_include(ctx->cfg.header);
519} 519}
520 520
521void cgit_print_docend() 521void cgit_print_docend()
522{ 522{