summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2008-08-06 07:50:10 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2008-08-06 07:50:10 (UTC)
commite352a013aed6e925a10a92916500c7deccf1410a (patch) (unidiff)
tree7c49bf453bee4f624025d62bae92b4926bf83bfe
parent3c71f597cc932992d5c44196e90f4675a1d54e77 (diff)
parentb2a3d31e8839b53a623b4c99124c2c637d0e3cbb (diff)
downloadcgit-e352a013aed6e925a10a92916500c7deccf1410a.zip
cgit-e352a013aed6e925a10a92916500c7deccf1410a.tar.gz
cgit-e352a013aed6e925a10a92916500c7deccf1410a.tar.bz2
Merge branch 'lh/atom'
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--Makefile1
-rw-r--r--cgit.h1
-rw-r--r--cmd.c7
-rw-r--r--ui-atom.c129
-rw-r--r--ui-atom.h6
-rw-r--r--ui-shared.c23
-rw-r--r--ui-shared.h1
7 files changed, 168 insertions, 0 deletions
diff --git a/Makefile b/Makefile
index e1436a3..6458431 100644
--- a/Makefile
+++ b/Makefile
@@ -42,32 +42,33 @@ endif
42# 42#
43# Define a pattern rule for silent object building 43# Define a pattern rule for silent object building
44# 44#
45%.o: %.c 45%.o: %.c
46 $(QUIET_CC)$(CC) -o $*.o -c $(CFLAGS) $< 46 $(QUIET_CC)$(CC) -o $*.o -c $(CFLAGS) $<
47 47
48 48
49EXTLIBS = git/libgit.a git/xdiff/lib.a -lz -lcrypto 49EXTLIBS = git/libgit.a git/xdiff/lib.a -lz -lcrypto
50OBJECTS = 50OBJECTS =
51OBJECTS += cache.o 51OBJECTS += cache.o
52OBJECTS += cgit.o 52OBJECTS += cgit.o
53OBJECTS += cmd.o 53OBJECTS += cmd.o
54OBJECTS += configfile.o 54OBJECTS += configfile.o
55OBJECTS += html.o 55OBJECTS += html.o
56OBJECTS += parsing.o 56OBJECTS += parsing.o
57OBJECTS += shared.o 57OBJECTS += shared.o
58OBJECTS += ui-atom.o
58OBJECTS += ui-blob.o 59OBJECTS += ui-blob.o
59OBJECTS += ui-commit.o 60OBJECTS += ui-commit.o
60OBJECTS += ui-diff.o 61OBJECTS += ui-diff.o
61OBJECTS += ui-log.o 62OBJECTS += ui-log.o
62OBJECTS += ui-patch.o 63OBJECTS += ui-patch.o
63OBJECTS += ui-refs.o 64OBJECTS += ui-refs.o
64OBJECTS += ui-repolist.o 65OBJECTS += ui-repolist.o
65OBJECTS += ui-shared.o 66OBJECTS += ui-shared.o
66OBJECTS += ui-snapshot.o 67OBJECTS += ui-snapshot.o
67OBJECTS += ui-summary.o 68OBJECTS += ui-summary.o
68OBJECTS += ui-tag.o 69OBJECTS += ui-tag.o
69OBJECTS += ui-tree.o 70OBJECTS += ui-tree.o
70 71
71ifdef NEEDS_LIBICONV 72ifdef NEEDS_LIBICONV
72 EXTLIBS += -liconv 73 EXTLIBS += -liconv
73endif 74endif
diff --git a/cgit.h b/cgit.h
index b01fa31..a1fa841 100644
--- a/cgit.h
+++ b/cgit.h
@@ -11,32 +11,33 @@
11#include <tag.h> 11#include <tag.h>
12#include <diff.h> 12#include <diff.h>
13#include <diffcore.h> 13#include <diffcore.h>
14#include <refs.h> 14#include <refs.h>
15#include <revision.h> 15#include <revision.h>
16#include <log-tree.h> 16#include <log-tree.h>
17#include <archive.h> 17#include <archive.h>
18#include <xdiff/xdiff.h> 18#include <xdiff/xdiff.h>
19#include <utf8.h> 19#include <utf8.h>
20 20
21 21
22/* 22/*
23 * Dateformats used on misc. pages 23 * Dateformats used on misc. pages
24 */ 24 */
25#define FMT_LONGDATE "%Y-%m-%d %H:%M:%S (%Z)" 25#define FMT_LONGDATE "%Y-%m-%d %H:%M:%S (%Z)"
26#define FMT_SHORTDATE "%Y-%m-%d" 26#define FMT_SHORTDATE "%Y-%m-%d"
27#define FMT_ATOMDATE "%Y-%m-%dT%H:%M:%SZ"
27 28
28 29
29/* 30/*
30 * Limits used for relative dates 31 * Limits used for relative dates
31 */ 32 */
32#define TM_MIN 60 33#define TM_MIN 60
33#define TM_HOUR (TM_MIN * 60) 34#define TM_HOUR (TM_MIN * 60)
34#define TM_DAY (TM_HOUR * 24) 35#define TM_DAY (TM_HOUR * 24)
35#define TM_WEEK (TM_DAY * 7) 36#define TM_WEEK (TM_DAY * 7)
36#define TM_YEAR (TM_DAY * 365) 37#define TM_YEAR (TM_DAY * 365)
37#define TM_MONTH (TM_YEAR / 12.0) 38#define TM_MONTH (TM_YEAR / 12.0)
38 39
39 40
40/* 41/*
41 * Default encoding 42 * Default encoding
42 */ 43 */
diff --git a/cmd.c b/cmd.c
index fe0ea8f..c0e4db3 100644
--- a/cmd.c
+++ b/cmd.c
@@ -1,40 +1,46 @@
1/* cmd.c: the cgit command dispatcher 1/* cmd.c: the cgit command dispatcher
2 * 2 *
3 * Copyright (C) 2008 Lars Hjemli 3 * Copyright (C) 2008 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 "cmd.h" 10#include "cmd.h"
11#include "cache.h" 11#include "cache.h"
12#include "ui-shared.h" 12#include "ui-shared.h"
13#include "ui-atom.h"
13#include "ui-blob.h" 14#include "ui-blob.h"
14#include "ui-commit.h" 15#include "ui-commit.h"
15#include "ui-diff.h" 16#include "ui-diff.h"
16#include "ui-log.h" 17#include "ui-log.h"
17#include "ui-patch.h" 18#include "ui-patch.h"
18#include "ui-refs.h" 19#include "ui-refs.h"
19#include "ui-repolist.h" 20#include "ui-repolist.h"
20#include "ui-snapshot.h" 21#include "ui-snapshot.h"
21#include "ui-summary.h" 22#include "ui-summary.h"
22#include "ui-tag.h" 23#include "ui-tag.h"
23#include "ui-tree.h" 24#include "ui-tree.h"
24 25
26static void atom_fn(struct cgit_context *ctx)
27{
28 cgit_print_atom(ctx->qry.head, ctx->qry.path, 10);
29}
30
25static void about_fn(struct cgit_context *ctx) 31static void about_fn(struct cgit_context *ctx)
26{ 32{
27 if (ctx->repo) 33 if (ctx->repo)
28 cgit_print_repo_readme(); 34 cgit_print_repo_readme();
29 else 35 else
30 cgit_print_site_readme(); 36 cgit_print_site_readme();
31} 37}
32 38
33static void blob_fn(struct cgit_context *ctx) 39static void blob_fn(struct cgit_context *ctx)
34{ 40{
35 cgit_print_blob(ctx->qry.sha1, ctx->qry.path, ctx->qry.head); 41 cgit_print_blob(ctx->qry.sha1, ctx->qry.path, ctx->qry.head);
36} 42}
37 43
38static void commit_fn(struct cgit_context *ctx) 44static void commit_fn(struct cgit_context *ctx)
39{ 45{
40 cgit_print_commit(ctx->qry.sha1); 46 cgit_print_commit(ctx->qry.sha1);
@@ -89,32 +95,33 @@ static void summary_fn(struct cgit_context *ctx)
89static void tag_fn(struct cgit_context *ctx) 95static void tag_fn(struct cgit_context *ctx)
90{ 96{
91 cgit_print_tag(ctx->qry.sha1); 97 cgit_print_tag(ctx->qry.sha1);
92} 98}
93 99
94static void tree_fn(struct cgit_context *ctx) 100static void tree_fn(struct cgit_context *ctx)
95{ 101{
96 cgit_print_tree(ctx->qry.sha1, ctx->qry.path); 102 cgit_print_tree(ctx->qry.sha1, ctx->qry.path);
97} 103}
98 104
99#define def_cmd(name, want_repo, want_layout) \ 105#define def_cmd(name, want_repo, want_layout) \
100 {#name, name##_fn, want_repo, want_layout} 106 {#name, name##_fn, want_repo, want_layout}
101 107
102struct cgit_cmd *cgit_get_cmd(struct cgit_context *ctx) 108struct cgit_cmd *cgit_get_cmd(struct cgit_context *ctx)
103{ 109{
104 static struct cgit_cmd cmds[] = { 110 static struct cgit_cmd cmds[] = {
111 def_cmd(atom, 1, 0),
105 def_cmd(about, 0, 1), 112 def_cmd(about, 0, 1),
106 def_cmd(blob, 1, 0), 113 def_cmd(blob, 1, 0),
107 def_cmd(commit, 1, 1), 114 def_cmd(commit, 1, 1),
108 def_cmd(diff, 1, 1), 115 def_cmd(diff, 1, 1),
109 def_cmd(log, 1, 1), 116 def_cmd(log, 1, 1),
110 def_cmd(ls_cache, 0, 0), 117 def_cmd(ls_cache, 0, 0),
111 def_cmd(patch, 1, 0), 118 def_cmd(patch, 1, 0),
112 def_cmd(refs, 1, 1), 119 def_cmd(refs, 1, 1),
113 def_cmd(repolist, 0, 0), 120 def_cmd(repolist, 0, 0),
114 def_cmd(snapshot, 1, 0), 121 def_cmd(snapshot, 1, 0),
115 def_cmd(summary, 1, 1), 122 def_cmd(summary, 1, 1),
116 def_cmd(tag, 1, 1), 123 def_cmd(tag, 1, 1),
117 def_cmd(tree, 1, 1), 124 def_cmd(tree, 1, 1),
118 }; 125 };
119 int i; 126 int i;
120 127
diff --git a/ui-atom.c b/ui-atom.c
new file mode 100644
index 0000000..a6ea3ee
--- a/dev/null
+++ b/ui-atom.c
@@ -0,0 +1,129 @@
1/* ui-atom.c: functions for atom feeds
2 *
3 * Copyright (C) 2008 Lars Hjemli
4 *
5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text)
7 */
8
9#include "cgit.h"
10#include "html.h"
11#include "ui-shared.h"
12
13void add_entry(struct commit *commit, char *host)
14{
15 char delim = '&';
16 char *hex;
17 char *mail, *t, *t2;
18 struct commitinfo *info;
19
20 info = cgit_parse_commit(commit);
21 hex = sha1_to_hex(commit->object.sha1);
22 html("<entry>\n");
23 html("<title>");
24 html_txt(info->subject);
25 html("</title>\n");
26 html("<updated>");
27 cgit_print_date(info->author_date, FMT_ATOMDATE, ctx.cfg.local_time);
28 html("</updated>\n");
29 html("<author>\n");
30 if (info->author) {
31 html("<name>");
32 html_txt(info->author);
33 html("</name>\n");
34 }
35 if (info->author_email) {
36 mail = xstrdup(info->author_email);
37 t = strchr(mail, '<');
38 if (t)
39 t++;
40 else
41 t = mail;
42 t2 = strchr(t, '>');
43 if (t2)
44 *t2 = '\0';
45 html("<email>");
46 html_txt(t);
47 html("</email>\n");
48 free(mail);
49 }
50 html("</author>\n");
51 html("<published>");
52 cgit_print_date(info->author_date, FMT_ATOMDATE, ctx.cfg.local_time);
53 html("</published>\n");
54 if (host) {
55 html("<link rel='alternate' type='text/html' href='http://");
56 html_attr(host);
57 html_attr(cgit_pageurl(ctx.repo->url, "commit", NULL));
58 if (ctx.cfg.virtual_root)
59 delim = '?';
60 htmlf("%cid=%s", delim, hex);
61 html("'/>\n");
62 }
63 htmlf("<id>%s</id>\n", hex);
64 html("<content type='text'>\n");
65 html_txt(info->msg);
66 html("</content>\n");
67 html("<content type='xhtml'>\n");
68 html("<div xmlns='http://www.w3.org/1999/xhtml'>\n");
69 html("<pre>\n");
70 html_txt(info->msg);
71 html("</pre>\n");
72 html("</div>\n");
73 html("</content>\n");
74 html("</entry>\n");
75 cgit_free_commitinfo(info);
76}
77
78
79void cgit_print_atom(char *tip, char *path, int max_count)
80{
81 char *host;
82 const char *argv[] = {NULL, tip, NULL, NULL, NULL};
83 struct commit *commit;
84 struct rev_info rev;
85 int argc = 2;
86
87 if (!tip)
88 argv[1] = ctx.qry.head;
89
90 if (path) {
91 argv[argc++] = "--";
92 argv[argc++] = path;
93 }
94
95 init_revisions(&rev, NULL);
96 rev.abbrev = DEFAULT_ABBREV;
97 rev.commit_format = CMIT_FMT_DEFAULT;
98 rev.verbose_header = 1;
99 rev.show_root_diff = 0;
100 rev.max_count = max_count;
101 setup_revisions(argc, argv, &rev, NULL);
102 prepare_revision_walk(&rev);
103
104 host = cgit_hosturl();
105 ctx.page.mimetype = "text/xml";
106 ctx.page.charset = "utf-8";
107 cgit_print_http_headers(&ctx);
108 html("<feed xmlns='http://www.w3.org/2005/Atom'>\n");
109 html("<title>");
110 html_txt(ctx.repo->name);
111 html("</title>\n");
112 html("<subtitle>");
113 html_txt(ctx.repo->desc);
114 html("</subtitle>\n");
115 if (host) {
116 html("<link rel='alternate' type='text/html' href='http://");
117 html_attr(host);
118 html_attr(cgit_repourl(ctx.repo->url));
119 html("'/>\n");
120 }
121 while ((commit = get_revision(&rev)) != NULL) {
122 add_entry(commit, host);
123 free(commit->buffer);
124 commit->buffer = NULL;
125 free_commit_list(commit->parents);
126 commit->parents = NULL;
127 }
128 html("</feed>\n");
129}
diff --git a/ui-atom.h b/ui-atom.h
new file mode 100644
index 0000000..749ffd3
--- a/dev/null
+++ b/ui-atom.h
@@ -0,0 +1,6 @@
1#ifndef UI_ATOM_H
2#define UI_ATOM_H
3
4extern void cgit_print_atom(char *tip, char *path, int max_count);
5
6#endif
diff --git a/ui-shared.c b/ui-shared.c
index 197ee37..37c60b2 100644
--- a/ui-shared.c
+++ b/ui-shared.c
@@ -21,32 +21,47 @@ static char *http_date(time_t t)
21 static char month[][4] = 21 static char month[][4] =
22 {"Jan", "Feb", "Mar", "Apr", "May", "Jun", 22 {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
23 "Jul", "Aug", "Sep", "Oct", "Now", "Dec"}; 23 "Jul", "Aug", "Sep", "Oct", "Now", "Dec"};
24 struct tm *tm = gmtime(&t); 24 struct tm *tm = gmtime(&t);
25 return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday], 25 return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday],
26 tm->tm_mday, month[tm->tm_mon], 1900+tm->tm_year, 26 tm->tm_mday, month[tm->tm_mon], 1900+tm->tm_year,
27 tm->tm_hour, tm->tm_min, tm->tm_sec); 27 tm->tm_hour, tm->tm_min, tm->tm_sec);
28} 28}
29 29
30void cgit_print_error(char *msg) 30void cgit_print_error(char *msg)
31{ 31{
32 html("<div class='error'>"); 32 html("<div class='error'>");
33 html_txt(msg); 33 html_txt(msg);
34 html("</div>\n"); 34 html("</div>\n");
35} 35}
36 36
37char *cgit_hosturl()
38{
39 char *host, *port;
40
41 host = getenv("SERVER_NAME");
42 if (!host)
43 return NULL;
44 port = getenv("SERVER_PORT");
45 if (port && atoi(port) != 80)
46 host = xstrdup(fmt("%s:%d", host, atoi(port)));
47 else
48 host = xstrdup(host);
49 return host;
50}
51
37char *cgit_rooturl() 52char *cgit_rooturl()
38{ 53{
39 if (ctx.cfg.virtual_root) 54 if (ctx.cfg.virtual_root)
40 return fmt("%s/", ctx.cfg.virtual_root); 55 return fmt("%s/", ctx.cfg.virtual_root);
41 else 56 else
42 return ctx.cfg.script_name; 57 return ctx.cfg.script_name;
43} 58}
44 59
45char *cgit_repourl(const char *reponame) 60char *cgit_repourl(const char *reponame)
46{ 61{
47 if (ctx.cfg.virtual_root) { 62 if (ctx.cfg.virtual_root) {
48 return fmt("%s/%s/", ctx.cfg.virtual_root, reponame); 63 return fmt("%s/%s/", ctx.cfg.virtual_root, reponame);
49 } else { 64 } else {
50 return fmt("?r=%s", reponame); 65 return fmt("?r=%s", reponame);
51 } 66 }
52} 67}
@@ -415,49 +430,57 @@ void cgit_print_http_headers(struct cgit_context *ctx)
415{ 430{
416 if (ctx->page.mimetype && ctx->page.charset) 431 if (ctx->page.mimetype && ctx->page.charset)
417 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype, 432 htmlf("Content-Type: %s; charset=%s\n", ctx->page.mimetype,
418 ctx->page.charset); 433 ctx->page.charset);
419 else if (ctx->page.mimetype) 434 else if (ctx->page.mimetype)
420 htmlf("Content-Type: %s\n", ctx->page.mimetype); 435 htmlf("Content-Type: %s\n", ctx->page.mimetype);
421 if (ctx->page.filename) 436 if (ctx->page.filename)
422 htmlf("Content-Disposition: inline; filename=\"%s\"\n", 437 htmlf("Content-Disposition: inline; filename=\"%s\"\n",
423 ctx->page.filename); 438 ctx->page.filename);
424 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified)); 439 htmlf("Last-Modified: %s\n", http_date(ctx->page.modified));
425 htmlf("Expires: %s\n", http_date(ctx->page.expires)); 440 htmlf("Expires: %s\n", http_date(ctx->page.expires));
426 html("\n"); 441 html("\n");
427} 442}
428 443
429void cgit_print_docstart(struct cgit_context *ctx) 444void cgit_print_docstart(struct cgit_context *ctx)
430{ 445{
446 char *host = cgit_hosturl();
431 html(cgit_doctype); 447 html(cgit_doctype);
432 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n"); 448 html("<html xmlns='http://www.w3.org/1999/xhtml' xml:lang='en' lang='en'>\n");
433 html("<head>\n"); 449 html("<head>\n");
434 html("<title>"); 450 html("<title>");
435 html_txt(ctx->page.title); 451 html_txt(ctx->page.title);
436 html("</title>\n"); 452 html("</title>\n");
437 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version); 453 htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version);
438 if (ctx->cfg.robots && *ctx->cfg.robots) 454 if (ctx->cfg.robots && *ctx->cfg.robots)
439 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots); 455 htmlf("<meta name='robots' content='%s'/>\n", ctx->cfg.robots);
440 html("<link rel='stylesheet' type='text/css' href='"); 456 html("<link rel='stylesheet' type='text/css' href='");
441 html_attr(ctx->cfg.css); 457 html_attr(ctx->cfg.css);
442 html("'/>\n"); 458 html("'/>\n");
443 if (ctx->cfg.favicon) { 459 if (ctx->cfg.favicon) {
444 html("<link rel='shortcut icon' href='"); 460 html("<link rel='shortcut icon' href='");
445 html_attr(ctx->cfg.favicon); 461 html_attr(ctx->cfg.favicon);
446 html("'/>\n"); 462 html("'/>\n");
447 } 463 }
464 if (host && ctx->repo) {
465 html("<link rel='alternate' title='Atom feed' href='http://");
466 html_attr(cgit_hosturl());
467 html_attr(cgit_fileurl(ctx->repo->url, "atom", ctx->qry.path,
468 fmt("h=%s", ctx->qry.head)));
469 html("' type='application/atom+xml'/>");
470 }
448 html("</head>\n"); 471 html("</head>\n");
449 html("<body>\n"); 472 html("<body>\n");
450} 473}
451 474
452void cgit_print_docend() 475void cgit_print_docend()
453{ 476{
454 html("</div>"); 477 html("</div>");
455 if (ctx.cfg.footer) 478 if (ctx.cfg.footer)
456 html_include(ctx.cfg.footer); 479 html_include(ctx.cfg.footer);
457 else { 480 else {
458 html("<div class='footer'>generated "); 481 html("<div class='footer'>generated ");
459 cgit_print_date(time(NULL), FMT_LONGDATE, ctx.cfg.local_time); 482 cgit_print_date(time(NULL), FMT_LONGDATE, ctx.cfg.local_time);
460 htmlf(" by cgit %s", cgit_version); 483 htmlf(" by cgit %s", cgit_version);
461 html("</div>\n"); 484 html("</div>\n");
462 } 485 }
463 html("</body>\n</html>\n"); 486 html("</body>\n</html>\n");
diff --git a/ui-shared.h b/ui-shared.h
index 07da4b4..f4123d3 100644
--- a/ui-shared.h
+++ b/ui-shared.h
@@ -1,19 +1,20 @@
1#ifndef UI_SHARED_H 1#ifndef UI_SHARED_H
2#define UI_SHARED_H 2#define UI_SHARED_H
3 3
4extern char *cgit_hosturl();
4extern char *cgit_repourl(const char *reponame); 5extern char *cgit_repourl(const char *reponame);
5extern char *cgit_fileurl(const char *reponame, const char *pagename, 6extern char *cgit_fileurl(const char *reponame, const char *pagename,
6 const char *filename, const char *query); 7 const char *filename, const char *query);
7extern char *cgit_pageurl(const char *reponame, const char *pagename, 8extern char *cgit_pageurl(const char *reponame, const char *pagename,
8 const char *query); 9 const char *query);
9 10
10extern void cgit_index_link(char *name, char *title, char *class, 11extern void cgit_index_link(char *name, char *title, char *class,
11 char *pattern, int ofs); 12 char *pattern, int ofs);
12extern void cgit_tree_link(char *name, char *title, char *class, char *head, 13extern void cgit_tree_link(char *name, char *title, char *class, char *head,
13 char *rev, char *path); 14 char *rev, char *path);
14extern void cgit_log_link(char *name, char *title, char *class, char *head, 15extern void cgit_log_link(char *name, char *title, char *class, char *head,
15 char *rev, char *path, int ofs, char *grep, 16 char *rev, char *path, int ofs, char *grep,
16 char *pattern); 17 char *pattern);
17extern void cgit_commit_link(char *name, char *title, char *class, char *head, 18extern void cgit_commit_link(char *name, char *title, char *class, char *head,
18 char *rev); 19 char *rev);
19extern void cgit_patch_link(char *name, char *title, char *class, char *head, 20extern void cgit_patch_link(char *name, char *title, char *class, char *head,