summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2007-05-15 22:58:35 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2007-05-15 22:58:35 (UTC)
commitf9ff7df613b4ee86fe5914c4ae3400650882c03d (patch) (unidiff)
treec26fde3a4d3485943c275232f18359bebd133f1a
parenta2ddc10479ec463708e422ca5ce7ec02c22a7d02 (diff)
downloadcgit-f9ff7df613b4ee86fe5914c4ae3400650882c03d.zip
cgit-f9ff7df613b4ee86fe5914c4ae3400650882c03d.tar.gz
cgit-f9ff7df613b4ee86fe5914c4ae3400650882c03d.tar.bz2
Add support for commitdiff via h parameter
The commitdiff will be generated against the first parent, and the diff page also gets the benefit of repo.defbranch. Cleaned up some bad whitespace in cgit.h while at it. Signed-off-by: Lars Hjemli <hjemli@gmail.com>
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c3
-rw-r--r--cgit.h11
-rw-r--r--ui-diff.c14
3 files changed, 21 insertions, 7 deletions
diff --git a/cgit.c b/cgit.c
index 9b4815d..3e7e595 100644
--- a/cgit.c
+++ b/cgit.c
@@ -75,97 +75,98 @@ static void cgit_print_repo_page(struct cacheitem *item)
75 cgit_print_error(fmt("Unable to scan repository: %s", 75 cgit_print_error(fmt("Unable to scan repository: %s",
76 strerror(errno))); 76 strerror(errno)));
77 cgit_print_docend(); 77 cgit_print_docend();
78 return; 78 return;
79 } 79 }
80 80
81 title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc); 81 title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc);
82 show_search = 0; 82 show_search = 0;
83 setenv("GIT_DIR", cgit_repo->path, 1); 83 setenv("GIT_DIR", cgit_repo->path, 1);
84 84
85 if (cgit_query_page) { 85 if (cgit_query_page) {
86 if (cgit_repo->snapshots && !strcmp(cgit_query_page, "snapshot")) { 86 if (cgit_repo->snapshots && !strcmp(cgit_query_page, "snapshot")) {
87 cgit_print_snapshot(item, cgit_query_sha1, "zip", 87 cgit_print_snapshot(item, cgit_query_sha1, "zip",
88 cgit_repo->url, cgit_query_name); 88 cgit_repo->url, cgit_query_name);
89 return; 89 return;
90 } 90 }
91 if (!strcmp(cgit_query_page, "blob")) { 91 if (!strcmp(cgit_query_page, "blob")) {
92 cgit_print_blob(item, cgit_query_sha1, cgit_query_path); 92 cgit_print_blob(item, cgit_query_sha1, cgit_query_path);
93 return; 93 return;
94 } 94 }
95 } 95 }
96 96
97 if (cgit_query_page && !strcmp(cgit_query_page, "log")) 97 if (cgit_query_page && !strcmp(cgit_query_page, "log"))
98 show_search = 1; 98 show_search = 1;
99 99
100 cgit_print_docstart(title, item); 100 cgit_print_docstart(title, item);
101 101
102 102
103 if (!cgit_query_page) { 103 if (!cgit_query_page) {
104 cgit_print_pageheader("summary", show_search); 104 cgit_print_pageheader("summary", show_search);
105 cgit_print_summary(); 105 cgit_print_summary();
106 cgit_print_docend(); 106 cgit_print_docend();
107 return; 107 return;
108 } 108 }
109 109
110 cgit_print_pageheader(cgit_query_page, show_search); 110 cgit_print_pageheader(cgit_query_page, show_search);
111 111
112 if (!strcmp(cgit_query_page, "log")) { 112 if (!strcmp(cgit_query_page, "log")) {
113 cgit_print_log(cgit_query_head, cgit_query_ofs, 113 cgit_print_log(cgit_query_head, cgit_query_ofs,
114 cgit_max_commit_count, cgit_query_search, 114 cgit_max_commit_count, cgit_query_search,
115 cgit_query_path); 115 cgit_query_path);
116 } else if (!strcmp(cgit_query_page, "tree")) { 116 } else if (!strcmp(cgit_query_page, "tree")) {
117 cgit_print_tree(cgit_query_head, cgit_query_sha1, cgit_query_path); 117 cgit_print_tree(cgit_query_head, cgit_query_sha1, cgit_query_path);
118 } else if (!strcmp(cgit_query_page, "commit")) { 118 } else if (!strcmp(cgit_query_page, "commit")) {
119 cgit_print_commit(cgit_query_head); 119 cgit_print_commit(cgit_query_head);
120 } else if (!strcmp(cgit_query_page, "view")) { 120 } else if (!strcmp(cgit_query_page, "view")) {
121 cgit_print_view(cgit_query_sha1, cgit_query_path); 121 cgit_print_view(cgit_query_sha1, cgit_query_path);
122 } else if (!strcmp(cgit_query_page, "diff")) { 122 } else if (!strcmp(cgit_query_page, "diff")) {
123 cgit_print_diff(cgit_query_sha1, cgit_query_sha2, cgit_query_path); 123 cgit_print_diff(cgit_query_head, cgit_query_sha1, cgit_query_sha2,
124 cgit_query_path);
124 } else { 125 } else {
125 cgit_print_error("Invalid request"); 126 cgit_print_error("Invalid request");
126 } 127 }
127 cgit_print_docend(); 128 cgit_print_docend();
128} 129}
129 130
130static void cgit_fill_cache(struct cacheitem *item, int use_cache) 131static void cgit_fill_cache(struct cacheitem *item, int use_cache)
131{ 132{
132 static char buf[PATH_MAX]; 133 static char buf[PATH_MAX];
133 int stdout2; 134 int stdout2;
134 135
135 getcwd(buf, sizeof(buf)); 136 getcwd(buf, sizeof(buf));
136 item->st.st_mtime = time(NULL); 137 item->st.st_mtime = time(NULL);
137 138
138 if (use_cache) { 139 if (use_cache) {
139 stdout2 = chk_positive(dup(STDOUT_FILENO), 140 stdout2 = chk_positive(dup(STDOUT_FILENO),
140 "Preserving STDOUT"); 141 "Preserving STDOUT");
141 chk_zero(close(STDOUT_FILENO), "Closing STDOUT"); 142 chk_zero(close(STDOUT_FILENO), "Closing STDOUT");
142 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)"); 143 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)");
143 } 144 }
144 145
145 if (cgit_query_repo) 146 if (cgit_query_repo)
146 cgit_print_repo_page(item); 147 cgit_print_repo_page(item);
147 else 148 else
148 cgit_print_repolist(item); 149 cgit_print_repolist(item);
149 150
150 if (use_cache) { 151 if (use_cache) {
151 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT"); 152 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT");
152 chk_positive(dup2(stdout2, STDOUT_FILENO), 153 chk_positive(dup2(stdout2, STDOUT_FILENO),
153 "Restoring original STDOUT"); 154 "Restoring original STDOUT");
154 chk_zero(close(stdout2), "Closing temporary STDOUT"); 155 chk_zero(close(stdout2), "Closing temporary STDOUT");
155 } 156 }
156 157
157 chdir(buf); 158 chdir(buf);
158} 159}
159 160
160static void cgit_check_cache(struct cacheitem *item) 161static void cgit_check_cache(struct cacheitem *item)
161{ 162{
162 int i = 0; 163 int i = 0;
163 164
164 top: 165 top:
165 if (++i > cgit_max_lock_attempts) { 166 if (++i > cgit_max_lock_attempts) {
166 die("cgit_refresh_cache: unable to lock %s: %s", 167 die("cgit_refresh_cache: unable to lock %s: %s",
167 item->name, strerror(errno)); 168 item->name, strerror(errno));
168 } 169 }
169 if (!cache_exist(item)) { 170 if (!cache_exist(item)) {
170 if (!cache_lock(item)) { 171 if (!cache_lock(item)) {
171 sleep(1); 172 sleep(1);
diff --git a/cgit.h b/cgit.h
index ac710a6..764225d 100644
--- a/cgit.h
+++ b/cgit.h
@@ -112,73 +112,74 @@ extern int cgit_query_ofs;
112extern int htmlfd; 112extern int htmlfd;
113 113
114extern void cgit_global_config_cb(const char *name, const char *value); 114extern void cgit_global_config_cb(const char *name, const char *value);
115extern void cgit_repo_config_cb(const char *name, const char *value); 115extern void cgit_repo_config_cb(const char *name, const char *value);
116extern void cgit_querystring_cb(const char *name, const char *value); 116extern void cgit_querystring_cb(const char *name, const char *value);
117 117
118extern int chk_zero(int result, char *msg); 118extern int chk_zero(int result, char *msg);
119extern int chk_positive(int result, char *msg); 119extern int chk_positive(int result, char *msg);
120 120
121extern int hextoint(char c); 121extern int hextoint(char c);
122 122
123extern void *cgit_free_commitinfo(struct commitinfo *info); 123extern void *cgit_free_commitinfo(struct commitinfo *info);
124 124
125extern int cgit_diff_files(const unsigned char *old_sha1, 125extern int cgit_diff_files(const unsigned char *old_sha1,
126 const unsigned char *new_sha1, 126 const unsigned char *new_sha1,
127 linediff_fn fn); 127 linediff_fn fn);
128 128
129extern void cgit_diff_tree(const unsigned char *old_sha1, 129extern void cgit_diff_tree(const unsigned char *old_sha1,
130 const unsigned char *new_sha1, 130 const unsigned char *new_sha1,
131 filepair_fn fn); 131 filepair_fn fn);
132 132
133extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); 133extern void cgit_diff_commit(struct commit *commit, filepair_fn fn);
134 134
135extern char *fmt(const char *format,...); 135extern char *fmt(const char *format,...);
136 136
137extern void html(const char *txt); 137extern void html(const char *txt);
138extern void htmlf(const char *format,...); 138extern void htmlf(const char *format,...);
139extern void html_txt(char *txt); 139extern void html_txt(char *txt);
140extern void html_ntxt(int len, char *txt); 140extern void html_ntxt(int len, char *txt);
141extern void html_attr(char *txt); 141extern void html_attr(char *txt);
142extern void html_hidden(char *name, char *value); 142extern void html_hidden(char *name, char *value);
143extern void html_link_open(char *url, char *title, char *class); 143extern void html_link_open(char *url, char *title, char *class);
144extern void html_link_close(void); 144extern void html_link_close(void);
145extern void html_filemode(unsigned short mode); 145extern void html_filemode(unsigned short mode);
146 146
147extern int cgit_read_config(const char *filename, configfn fn); 147extern int cgit_read_config(const char *filename, configfn fn);
148extern int cgit_parse_query(char *txt, configfn fn); 148extern int cgit_parse_query(char *txt, configfn fn);
149extern struct commitinfo *cgit_parse_commit(struct commit *commit); 149extern struct commitinfo *cgit_parse_commit(struct commit *commit);
150extern struct taginfo *cgit_parse_tag(struct tag *tag); 150extern struct taginfo *cgit_parse_tag(struct tag *tag);
151 151
152extern char *cache_safe_filename(const char *unsafe); 152extern char *cache_safe_filename(const char *unsafe);
153extern int cache_lock(struct cacheitem *item); 153extern int cache_lock(struct cacheitem *item);
154extern int cache_unlock(struct cacheitem *item); 154extern int cache_unlock(struct cacheitem *item);
155extern int cache_cancel_lock(struct cacheitem *item); 155extern int cache_cancel_lock(struct cacheitem *item);
156extern int cache_exist(struct cacheitem *item); 156extern int cache_exist(struct cacheitem *item);
157extern int cache_expired(struct cacheitem *item); 157extern int cache_expired(struct cacheitem *item);
158 158
159extern char *cgit_repourl(const char *reponame); 159extern char *cgit_repourl(const char *reponame);
160extern char *cgit_pageurl(const char *reponame, const char *pagename, 160extern char *cgit_pageurl(const char *reponame, const char *pagename,
161 const char *query); 161 const char *query);
162 162
163extern void cgit_print_error(char *msg); 163extern void cgit_print_error(char *msg);
164extern void cgit_print_date(unsigned long secs); 164extern void cgit_print_date(unsigned long secs);
165extern void cgit_print_docstart(char *title, struct cacheitem *item); 165extern void cgit_print_docstart(char *title, struct cacheitem *item);
166extern void cgit_print_docend(); 166extern void cgit_print_docend();
167extern void cgit_print_pageheader(char *title, int show_search); 167extern void cgit_print_pageheader(char *title, int show_search);
168extern void cgit_print_snapshot_start(const char *mimetype, 168extern void cgit_print_snapshot_start(const char *mimetype,
169 const char *filename, 169 const char *filename,
170 struct cacheitem *item); 170 struct cacheitem *item);
171 171
172extern void cgit_print_repolist(struct cacheitem *item); 172extern void cgit_print_repolist(struct cacheitem *item);
173extern void cgit_print_summary(); 173extern void cgit_print_summary();
174extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path); 174extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path);
175extern void cgit_print_view(const char *hex, char *path); 175extern void cgit_print_view(const char *hex, char *path);
176extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path); 176extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path);
177extern void cgit_print_tree(const char *rev, const char *hex, char *path); 177extern void cgit_print_tree(const char *rev, const char *hex, char *path);
178extern void cgit_print_commit(const char *hex); 178extern void cgit_print_commit(const char *hex);
179extern void cgit_print_diff(const char *old_hex, const char *new_hex, char *path); 179extern void cgit_print_diff(const char *head, const char *old_hex, const char *new_hex,
180extern void cgit_print_snapshot(struct cacheitem *item, const char *hex, 180 char *path);
181extern void cgit_print_snapshot(struct cacheitem *item, const char *hex,
181 const char *format, const char *prefix, 182 const char *format, const char *prefix,
182 const char *filename); 183 const char *filename);
183 184
184#endif /* CGIT_H */ 185#endif /* CGIT_H */
diff --git a/ui-diff.c b/ui-diff.c
index 999b6f3..afe1c90 100644
--- a/ui-diff.c
+++ b/ui-diff.c
@@ -21,85 +21,97 @@ static void print_line(char *line, int len)
21 class = "add"; 21 class = "add";
22 else if (line[0] == '-') 22 else if (line[0] == '-')
23 class = "del"; 23 class = "del";
24 else if (line[0] == '@') 24 else if (line[0] == '@')
25 class = "hunk"; 25 class = "hunk";
26 26
27 htmlf("<div class='%s'>", class); 27 htmlf("<div class='%s'>", class);
28 line[len-1] = '\0'; 28 line[len-1] = '\0';
29 html_txt(line); 29 html_txt(line);
30 html("</div>"); 30 html("</div>");
31 line[len-1] = c; 31 line[len-1] = c;
32} 32}
33 33
34static void header(unsigned char *sha1, char *path1, 34static void header(unsigned char *sha1, char *path1,
35 unsigned char *sha2, char *path2) 35 unsigned char *sha2, char *path2)
36{ 36{
37 char *abbrev1, *abbrev2; 37 char *abbrev1, *abbrev2;
38 if (is_null_sha1(sha1)) 38 if (is_null_sha1(sha1))
39 path1 = "dev/null"; 39 path1 = "dev/null";
40 if (is_null_sha1(sha2)) 40 if (is_null_sha1(sha2))
41 path2 = "dev/null"; 41 path2 = "dev/null";
42 html("<tr><td>"); 42 html("<tr><td>");
43 html("<div class='head'>"); 43 html("<div class='head'>");
44 html("diff --git a/"); 44 html("diff --git a/");
45 html_txt(path1); 45 html_txt(path1);
46 html(" b/"); 46 html(" b/");
47 html_txt(path2); 47 html_txt(path2);
48 abbrev1 = xstrdup(find_unique_abbrev(sha1, DEFAULT_ABBREV)); 48 abbrev1 = xstrdup(find_unique_abbrev(sha1, DEFAULT_ABBREV));
49 abbrev2 = xstrdup(find_unique_abbrev(sha2, DEFAULT_ABBREV)); 49 abbrev2 = xstrdup(find_unique_abbrev(sha2, DEFAULT_ABBREV));
50 htmlf("\nindex %s..%s", abbrev1, abbrev2); 50 htmlf("\nindex %s..%s", abbrev1, abbrev2);
51 free(abbrev1); 51 free(abbrev1);
52 free(abbrev2); 52 free(abbrev2);
53 html("\n--- a/"); 53 html("\n--- a/");
54 html_txt(path1); 54 html_txt(path1);
55 html("\n+++ b/"); 55 html("\n+++ b/");
56 html_txt(path2); 56 html_txt(path2);
57 html("</div>"); 57 html("</div>");
58} 58}
59 59
60static void filepair_cb(struct diff_filepair *pair) 60static void filepair_cb(struct diff_filepair *pair)
61{ 61{
62 header(pair->one->sha1, pair->one->path, 62 header(pair->one->sha1, pair->one->path,
63 pair->two->sha1, pair->two->path); 63 pair->two->sha1, pair->two->path);
64 if (cgit_diff_files(pair->one->sha1, pair->two->sha1, print_line)) 64 if (cgit_diff_files(pair->one->sha1, pair->two->sha1, print_line))
65 cgit_print_error("Error running diff"); 65 cgit_print_error("Error running diff");
66 html("</tr></td>"); 66 html("</tr></td>");
67} 67}
68 68
69void cgit_print_diff(const char *old_hex, const char *new_hex, char *path) 69void cgit_print_diff(const char *head, const char *old_hex, const char *new_hex, char *path)
70{ 70{
71 unsigned char sha1[20], sha2[20]; 71 unsigned char sha1[20], sha2[20];
72 enum object_type type; 72 enum object_type type;
73 unsigned long size; 73 unsigned long size;
74 struct commit *commit;
75
76 if (head && !old_hex && !new_hex) {
77 get_sha1(head, sha1);
78 commit = lookup_commit_reference(sha1);
79 if (commit && !parse_commit(commit)) {
80 html("<table class='diff'>");
81 cgit_diff_commit(commit, filepair_cb);
82 html("</td></tr></table>");
83 }
84 return;
85 }
74 86
75 get_sha1(old_hex, sha1); 87 get_sha1(old_hex, sha1);
76 get_sha1(new_hex, sha2); 88 get_sha1(new_hex, sha2);
77 89
78 type = sha1_object_info(sha1, &size); 90 type = sha1_object_info(sha1, &size);
79 if (type == OBJ_BAD) { 91 if (type == OBJ_BAD) {
80 type = sha1_object_info(sha2, &size); 92 type = sha1_object_info(sha2, &size);
81 if (type == OBJ_BAD) { 93 if (type == OBJ_BAD) {
82 cgit_print_error(fmt("Bad object names: %s, %s", old_hex, new_hex)); 94 cgit_print_error(fmt("Bad object names: %s, %s", old_hex, new_hex));
83 return; 95 return;
84 } 96 }
85 } 97 }
86 98
87 html("<table class='diff'>"); 99 html("<table class='diff'>");
88 switch(type) { 100 switch(type) {
89 case OBJ_BLOB: 101 case OBJ_BLOB:
90 html("<tr><td>"); 102 html("<tr><td>");
91 header(sha1, path, sha2, path); 103 header(sha1, path, sha2, path);
92 if (cgit_diff_files(sha1, sha2, print_line)) 104 if (cgit_diff_files(sha1, sha2, print_line))
93 cgit_print_error("Error running diff"); 105 cgit_print_error("Error running diff");
94 html("</tr></td>"); 106 html("</tr></td>");
95 break; 107 break;
96 case OBJ_TREE: 108 case OBJ_TREE:
97 cgit_diff_tree(sha1, sha2, filepair_cb); 109 cgit_diff_tree(sha1, sha2, filepair_cb);
98 break; 110 break;
99 default: 111 default:
100 cgit_print_error(fmt("Unhandled object type: %s", 112 cgit_print_error(fmt("Unhandled object type: %s",
101 typename(type))); 113 typename(type)));
102 break; 114 break;
103 } 115 }
104 html("</td></tr></table>"); 116 html("</td></tr></table>");
105} 117}