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
@@ -50,16 +50,17 @@ EXTLIBS = 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
diff --git a/cgit.h b/cgit.h
index b01fa31..a1fa841 100644
--- a/cgit.h
+++ b/cgit.h
@@ -19,16 +19,17 @@
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)
diff --git a/cmd.c b/cmd.c
index fe0ea8f..c0e4db3 100644
--- a/cmd.c
+++ b/cmd.c
@@ -5,28 +5,34 @@
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
@@ -97,16 +103,17 @@ static void tree_fn(struct cgit_context *ctx)
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),
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
@@ -29,16 +29,31 @@ static char *http_date(time_t t)
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
@@ -423,16 +438,17 @@ void cgit_print_http_headers(struct cgit_context *ctx)
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)
@@ -440,16 +456,23 @@ void cgit_print_docstart(struct cgit_context *ctx)
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)
diff --git a/ui-shared.h b/ui-shared.h
index 07da4b4..f4123d3 100644
--- a/ui-shared.h
+++ b/ui-shared.h
@@ -1,11 +1,12 @@
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);