author | Lars Hjemli <hjemli@gmail.com> | 2007-06-17 16:12:03 (UTC) |
---|---|---|
committer | Lars Hjemli <hjemli@gmail.com> | 2007-06-17 16:12:03 (UTC) |
commit | 4a0be586662843382ecfa53af34a13b291312bc0 (patch) (unidiff) | |
tree | 01e0cd725fe249df3449bb089aad9f8d58081f89 | |
parent | faaca447b071592c9a1e1f14b4d0d2a39b4c795a (diff) | |
download | cgit-4a0be586662843382ecfa53af34a13b291312bc0.zip cgit-4a0be586662843382ecfa53af34a13b291312bc0.tar.gz cgit-4a0be586662843382ecfa53af34a13b291312bc0.tar.bz2 |
Add cgit_diff_link()
This adds a new function used to generate links to the diff page and uses
it everywhere such links appear (expect for single files in the diffstat
displayed on the commit page: this is now a link to the tree page).
The updated diff-page now expects zero, one or two revision specifiers, in
parameters head, id and id2. Id defaults to head unless otherwise specified,
while head (as usual) defaults to repo.defbranch. If id2 isn't specified, it
defaults to the first parent of id1.
The most important change is of course that now all repo pages (summary, log,
tree, commit and diff) has support for passing on the current branch and
revision, i.e. the road is now open for a 'static' menu with links to all
of these pages.
Signed-off-by: Lars Hjemli <hjemli@gmail.com>
-rw-r--r-- | cgit.c | 3 | ||||
-rw-r--r-- | cgit.h | 7 | ||||
-rw-r--r-- | shared.c | 2 | ||||
-rw-r--r-- | ui-commit.c | 46 | ||||
-rw-r--r-- | ui-diff.c | 64 | ||||
-rw-r--r-- | ui-shared.c | 22 |
6 files changed, 75 insertions, 69 deletions
@@ -42,130 +42,129 @@ static int cgit_prepare_cache(struct cacheitem *item) | |||
42 | item->ttl = cgit_cache_static_ttl; | 42 | item->ttl = cgit_cache_static_ttl; |
43 | else | 43 | else |
44 | item->ttl = cgit_cache_repo_ttl; | 44 | item->ttl = cgit_cache_repo_ttl; |
45 | } | 45 | } |
46 | return 1; | 46 | return 1; |
47 | } | 47 | } |
48 | 48 | ||
49 | static void cgit_print_repo_page(struct cacheitem *item) | 49 | static void cgit_print_repo_page(struct cacheitem *item) |
50 | { | 50 | { |
51 | char *title; | 51 | char *title; |
52 | int show_search; | 52 | int show_search; |
53 | 53 | ||
54 | if (!cgit_query_head) | 54 | if (!cgit_query_head) |
55 | cgit_query_head = cgit_repo->defbranch; | 55 | cgit_query_head = cgit_repo->defbranch; |
56 | 56 | ||
57 | if (chdir(cgit_repo->path)) { | 57 | if (chdir(cgit_repo->path)) { |
58 | title = fmt("%s - %s", cgit_root_title, "Bad request"); | 58 | title = fmt("%s - %s", cgit_root_title, "Bad request"); |
59 | cgit_print_docstart(title, item); | 59 | cgit_print_docstart(title, item); |
60 | cgit_print_pageheader(title, 0); | 60 | cgit_print_pageheader(title, 0); |
61 | cgit_print_error(fmt("Unable to scan repository: %s", | 61 | cgit_print_error(fmt("Unable to scan repository: %s", |
62 | strerror(errno))); | 62 | strerror(errno))); |
63 | cgit_print_docend(); | 63 | cgit_print_docend(); |
64 | return; | 64 | return; |
65 | } | 65 | } |
66 | 66 | ||
67 | title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc); | 67 | title = fmt("%s - %s", cgit_repo->name, cgit_repo->desc); |
68 | show_search = 0; | 68 | show_search = 0; |
69 | setenv("GIT_DIR", cgit_repo->path, 1); | 69 | setenv("GIT_DIR", cgit_repo->path, 1); |
70 | 70 | ||
71 | if ((cgit_cmd == CMD_SNAPSHOT) && cgit_repo->snapshots) { | 71 | if ((cgit_cmd == CMD_SNAPSHOT) && cgit_repo->snapshots) { |
72 | cgit_print_snapshot(item, cgit_query_sha1, "zip", | 72 | cgit_print_snapshot(item, cgit_query_sha1, "zip", |
73 | cgit_repo->url, cgit_query_name); | 73 | cgit_repo->url, cgit_query_name); |
74 | return; | 74 | return; |
75 | } | 75 | } |
76 | 76 | ||
77 | if (cgit_cmd == CMD_BLOB) { | 77 | if (cgit_cmd == CMD_BLOB) { |
78 | cgit_print_blob(item, cgit_query_sha1, cgit_query_path); | 78 | cgit_print_blob(item, cgit_query_sha1, cgit_query_path); |
79 | return; | 79 | return; |
80 | } | 80 | } |
81 | 81 | ||
82 | show_search = (cgit_cmd == CMD_LOG); | 82 | show_search = (cgit_cmd == CMD_LOG); |
83 | cgit_print_docstart(title, item); | 83 | cgit_print_docstart(title, item); |
84 | if (!cgit_cmd) { | 84 | if (!cgit_cmd) { |
85 | cgit_print_pageheader("summary", show_search); | 85 | cgit_print_pageheader("summary", show_search); |
86 | cgit_print_summary(); | 86 | cgit_print_summary(); |
87 | cgit_print_docend(); | 87 | cgit_print_docend(); |
88 | return; | 88 | return; |
89 | } | 89 | } |
90 | 90 | ||
91 | cgit_print_pageheader(cgit_query_page, show_search); | 91 | cgit_print_pageheader(cgit_query_page, show_search); |
92 | 92 | ||
93 | switch(cgit_cmd) { | 93 | switch(cgit_cmd) { |
94 | case CMD_LOG: | 94 | case CMD_LOG: |
95 | cgit_print_log(cgit_query_sha1, cgit_query_ofs, | 95 | cgit_print_log(cgit_query_sha1, cgit_query_ofs, |
96 | cgit_max_commit_count, cgit_query_search, | 96 | cgit_max_commit_count, cgit_query_search, |
97 | cgit_query_path, 1); | 97 | cgit_query_path, 1); |
98 | break; | 98 | break; |
99 | case CMD_TREE: | 99 | case CMD_TREE: |
100 | cgit_print_tree(cgit_query_sha1, cgit_query_path); | 100 | cgit_print_tree(cgit_query_sha1, cgit_query_path); |
101 | break; | 101 | break; |
102 | case CMD_COMMIT: | 102 | case CMD_COMMIT: |
103 | cgit_print_commit(cgit_query_sha1); | 103 | cgit_print_commit(cgit_query_sha1); |
104 | break; | 104 | break; |
105 | case CMD_DIFF: | 105 | case CMD_DIFF: |
106 | cgit_print_diff(cgit_query_head, cgit_query_sha1, cgit_query_sha2, | 106 | cgit_print_diff(cgit_query_sha1, cgit_query_sha2); |
107 | cgit_query_path); | ||
108 | break; | 107 | break; |
109 | default: | 108 | default: |
110 | cgit_print_error("Invalid request"); | 109 | cgit_print_error("Invalid request"); |
111 | } | 110 | } |
112 | cgit_print_docend(); | 111 | cgit_print_docend(); |
113 | } | 112 | } |
114 | 113 | ||
115 | static void cgit_fill_cache(struct cacheitem *item, int use_cache) | 114 | static void cgit_fill_cache(struct cacheitem *item, int use_cache) |
116 | { | 115 | { |
117 | static char buf[PATH_MAX]; | 116 | static char buf[PATH_MAX]; |
118 | int stdout2; | 117 | int stdout2; |
119 | 118 | ||
120 | getcwd(buf, sizeof(buf)); | 119 | getcwd(buf, sizeof(buf)); |
121 | item->st.st_mtime = time(NULL); | 120 | item->st.st_mtime = time(NULL); |
122 | 121 | ||
123 | if (use_cache) { | 122 | if (use_cache) { |
124 | stdout2 = chk_positive(dup(STDOUT_FILENO), | 123 | stdout2 = chk_positive(dup(STDOUT_FILENO), |
125 | "Preserving STDOUT"); | 124 | "Preserving STDOUT"); |
126 | chk_zero(close(STDOUT_FILENO), "Closing STDOUT"); | 125 | chk_zero(close(STDOUT_FILENO), "Closing STDOUT"); |
127 | chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)"); | 126 | chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)"); |
128 | } | 127 | } |
129 | 128 | ||
130 | if (cgit_repo) | 129 | if (cgit_repo) |
131 | cgit_print_repo_page(item); | 130 | cgit_print_repo_page(item); |
132 | else | 131 | else |
133 | cgit_print_repolist(item); | 132 | cgit_print_repolist(item); |
134 | 133 | ||
135 | if (use_cache) { | 134 | if (use_cache) { |
136 | chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT"); | 135 | chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT"); |
137 | chk_positive(dup2(stdout2, STDOUT_FILENO), | 136 | chk_positive(dup2(stdout2, STDOUT_FILENO), |
138 | "Restoring original STDOUT"); | 137 | "Restoring original STDOUT"); |
139 | chk_zero(close(stdout2), "Closing temporary STDOUT"); | 138 | chk_zero(close(stdout2), "Closing temporary STDOUT"); |
140 | } | 139 | } |
141 | 140 | ||
142 | chdir(buf); | 141 | chdir(buf); |
143 | } | 142 | } |
144 | 143 | ||
145 | static void cgit_check_cache(struct cacheitem *item) | 144 | static void cgit_check_cache(struct cacheitem *item) |
146 | { | 145 | { |
147 | int i = 0; | 146 | int i = 0; |
148 | 147 | ||
149 | top: | 148 | top: |
150 | if (++i > cgit_max_lock_attempts) { | 149 | if (++i > cgit_max_lock_attempts) { |
151 | die("cgit_refresh_cache: unable to lock %s: %s", | 150 | die("cgit_refresh_cache: unable to lock %s: %s", |
152 | item->name, strerror(errno)); | 151 | item->name, strerror(errno)); |
153 | } | 152 | } |
154 | if (!cache_exist(item)) { | 153 | if (!cache_exist(item)) { |
155 | if (!cache_lock(item)) { | 154 | if (!cache_lock(item)) { |
156 | sleep(1); | 155 | sleep(1); |
157 | goto top; | 156 | goto top; |
158 | } | 157 | } |
159 | if (!cache_exist(item)) { | 158 | if (!cache_exist(item)) { |
160 | cgit_fill_cache(item, 1); | 159 | cgit_fill_cache(item, 1); |
161 | cache_unlock(item); | 160 | cache_unlock(item); |
162 | } else { | 161 | } else { |
163 | cache_cancel_lock(item); | 162 | cache_cancel_lock(item); |
164 | } | 163 | } |
165 | } else if (cache_expired(item) && cache_lock(item)) { | 164 | } else if (cache_expired(item) && cache_lock(item)) { |
166 | if (cache_expired(item)) { | 165 | if (cache_expired(item)) { |
167 | cgit_fill_cache(item, 1); | 166 | cgit_fill_cache(item, 1); |
168 | cache_unlock(item); | 167 | cache_unlock(item); |
169 | } else { | 168 | } else { |
170 | cache_cancel_lock(item); | 169 | cache_cancel_lock(item); |
171 | } | 170 | } |
@@ -146,88 +146,89 @@ extern char *cgit_query_path; | |||
146 | extern char *cgit_query_name; | 146 | extern char *cgit_query_name; |
147 | extern int cgit_query_ofs; | 147 | extern int cgit_query_ofs; |
148 | 148 | ||
149 | extern int htmlfd; | 149 | extern int htmlfd; |
150 | 150 | ||
151 | extern int cgit_get_cmd_index(const char *cmd); | 151 | extern int cgit_get_cmd_index(const char *cmd); |
152 | extern struct repoinfo *cgit_get_repoinfo(const char *url); | 152 | extern struct repoinfo *cgit_get_repoinfo(const char *url); |
153 | extern void cgit_global_config_cb(const char *name, const char *value); | 153 | extern void cgit_global_config_cb(const char *name, const char *value); |
154 | extern void cgit_repo_config_cb(const char *name, const char *value); | 154 | extern void cgit_repo_config_cb(const char *name, const char *value); |
155 | extern void cgit_querystring_cb(const char *name, const char *value); | 155 | extern void cgit_querystring_cb(const char *name, const char *value); |
156 | 156 | ||
157 | extern int chk_zero(int result, char *msg); | 157 | extern int chk_zero(int result, char *msg); |
158 | extern int chk_positive(int result, char *msg); | 158 | extern int chk_positive(int result, char *msg); |
159 | 159 | ||
160 | extern int hextoint(char c); | 160 | extern int hextoint(char c); |
161 | 161 | ||
162 | extern void *cgit_free_commitinfo(struct commitinfo *info); | 162 | extern void *cgit_free_commitinfo(struct commitinfo *info); |
163 | 163 | ||
164 | extern int cgit_diff_files(const unsigned char *old_sha1, | 164 | extern int cgit_diff_files(const unsigned char *old_sha1, |
165 | const unsigned char *new_sha1, | 165 | const unsigned char *new_sha1, |
166 | linediff_fn fn); | 166 | linediff_fn fn); |
167 | 167 | ||
168 | extern void cgit_diff_tree(const unsigned char *old_sha1, | 168 | extern void cgit_diff_tree(const unsigned char *old_sha1, |
169 | const unsigned char *new_sha1, | 169 | const unsigned char *new_sha1, |
170 | filepair_fn fn); | 170 | filepair_fn fn); |
171 | 171 | ||
172 | extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); | 172 | extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); |
173 | 173 | ||
174 | extern char *fmt(const char *format,...); | 174 | extern char *fmt(const char *format,...); |
175 | 175 | ||
176 | extern void html(const char *txt); | 176 | extern void html(const char *txt); |
177 | extern void htmlf(const char *format,...); | 177 | extern void htmlf(const char *format,...); |
178 | extern void html_txt(char *txt); | 178 | extern void html_txt(char *txt); |
179 | extern void html_ntxt(int len, char *txt); | 179 | extern void html_ntxt(int len, char *txt); |
180 | extern void html_attr(char *txt); | 180 | extern void html_attr(char *txt); |
181 | extern void html_hidden(char *name, char *value); | 181 | extern void html_hidden(char *name, char *value); |
182 | extern void html_link_open(char *url, char *title, char *class); | 182 | extern void html_link_open(char *url, char *title, char *class); |
183 | extern void html_link_close(void); | 183 | extern void html_link_close(void); |
184 | extern void html_filemode(unsigned short mode); | 184 | extern void html_filemode(unsigned short mode); |
185 | extern int html_include(const char *filename); | 185 | extern int html_include(const char *filename); |
186 | 186 | ||
187 | extern int cgit_read_config(const char *filename, configfn fn); | 187 | extern int cgit_read_config(const char *filename, configfn fn); |
188 | extern int cgit_parse_query(char *txt, configfn fn); | 188 | extern int cgit_parse_query(char *txt, configfn fn); |
189 | extern struct commitinfo *cgit_parse_commit(struct commit *commit); | 189 | extern struct commitinfo *cgit_parse_commit(struct commit *commit); |
190 | extern struct taginfo *cgit_parse_tag(struct tag *tag); | 190 | extern struct taginfo *cgit_parse_tag(struct tag *tag); |
191 | extern void cgit_parse_url(const char *url); | 191 | extern void cgit_parse_url(const char *url); |
192 | 192 | ||
193 | extern char *cache_safe_filename(const char *unsafe); | 193 | extern char *cache_safe_filename(const char *unsafe); |
194 | extern int cache_lock(struct cacheitem *item); | 194 | extern int cache_lock(struct cacheitem *item); |
195 | extern int cache_unlock(struct cacheitem *item); | 195 | extern int cache_unlock(struct cacheitem *item); |
196 | extern int cache_cancel_lock(struct cacheitem *item); | 196 | extern int cache_cancel_lock(struct cacheitem *item); |
197 | extern int cache_exist(struct cacheitem *item); | 197 | extern int cache_exist(struct cacheitem *item); |
198 | extern int cache_expired(struct cacheitem *item); | 198 | extern int cache_expired(struct cacheitem *item); |
199 | 199 | ||
200 | extern char *cgit_repourl(const char *reponame); | 200 | extern char *cgit_repourl(const char *reponame); |
201 | extern char *cgit_pageurl(const char *reponame, const char *pagename, | 201 | extern char *cgit_pageurl(const char *reponame, const char *pagename, |
202 | const char *query); | 202 | const char *query); |
203 | 203 | ||
204 | extern void cgit_tree_link(char *name, char *title, char *class, char *head, | 204 | extern void cgit_tree_link(char *name, char *title, char *class, char *head, |
205 | char *rev, char *path); | 205 | char *rev, char *path); |
206 | extern void cgit_log_link(char *name, char *title, char *class, char *head, | 206 | extern void cgit_log_link(char *name, char *title, char *class, char *head, |
207 | char *rev, char *path); | 207 | char *rev, char *path); |
208 | extern void cgit_commit_link(char *name, char *title, char *class, char *head, | 208 | extern void cgit_commit_link(char *name, char *title, char *class, char *head, |
209 | char *rev); | 209 | char *rev); |
210 | extern void cgit_diff_link(char *name, char *title, char *class, char *head, | ||
211 | char *new_rev, char *old_rev, char *path); | ||
210 | 212 | ||
211 | extern void cgit_print_error(char *msg); | 213 | extern void cgit_print_error(char *msg); |
212 | extern void cgit_print_date(time_t secs, char *format); | 214 | extern void cgit_print_date(time_t secs, char *format); |
213 | extern void cgit_print_age(time_t t, time_t max_relative, char *format); | 215 | extern void cgit_print_age(time_t t, time_t max_relative, char *format); |
214 | extern void cgit_print_docstart(char *title, struct cacheitem *item); | 216 | extern void cgit_print_docstart(char *title, struct cacheitem *item); |
215 | extern void cgit_print_docend(); | 217 | extern void cgit_print_docend(); |
216 | extern void cgit_print_pageheader(char *title, int show_search); | 218 | extern void cgit_print_pageheader(char *title, int show_search); |
217 | extern void cgit_print_snapshot_start(const char *mimetype, | 219 | extern void cgit_print_snapshot_start(const char *mimetype, |
218 | const char *filename, | 220 | const char *filename, |
219 | struct cacheitem *item); | 221 | struct cacheitem *item); |
220 | 222 | ||
221 | extern void cgit_print_repolist(struct cacheitem *item); | 223 | extern void cgit_print_repolist(struct cacheitem *item); |
222 | extern void cgit_print_summary(); | 224 | extern void cgit_print_summary(); |
223 | extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path, int pager); | 225 | extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *path, int pager); |
224 | extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path); | 226 | extern void cgit_print_blob(struct cacheitem *item, const char *hex, char *path); |
225 | extern void cgit_print_tree(const char *rev, char *path); | 227 | extern void cgit_print_tree(const char *rev, char *path); |
226 | extern void cgit_print_commit(const char *hex); | 228 | extern void cgit_print_commit(char *hex); |
227 | extern void cgit_print_diff(const char *head, const char *old_hex, const char *new_hex, | 229 | extern void cgit_print_diff(const char *new_hex, const char *old_hex); |
228 | char *path); | ||
229 | extern void cgit_print_snapshot(struct cacheitem *item, const char *hex, | 230 | extern void cgit_print_snapshot(struct cacheitem *item, const char *hex, |
230 | const char *format, const char *prefix, | 231 | const char *format, const char *prefix, |
231 | const char *filename); | 232 | const char *filename); |
232 | 233 | ||
233 | #endif /* CGIT_H */ | 234 | #endif /* CGIT_H */ |
@@ -299,80 +299,80 @@ int filediff_cb(void *priv, mmbuffer_t *mb, int nbuf) | |||
299 | /* Incomplete line */ | 299 | /* Incomplete line */ |
300 | diffbuf = xrealloc(diffbuf, buflen + mb[i].size); | 300 | diffbuf = xrealloc(diffbuf, buflen + mb[i].size); |
301 | memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size); | 301 | memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size); |
302 | buflen += mb[i].size; | 302 | buflen += mb[i].size; |
303 | continue; | 303 | continue; |
304 | } | 304 | } |
305 | 305 | ||
306 | /* we have a complete line */ | 306 | /* we have a complete line */ |
307 | if (!diffbuf) { | 307 | if (!diffbuf) { |
308 | ((linediff_fn)priv)(mb[i].ptr, mb[i].size); | 308 | ((linediff_fn)priv)(mb[i].ptr, mb[i].size); |
309 | continue; | 309 | continue; |
310 | } | 310 | } |
311 | diffbuf = xrealloc(diffbuf, buflen + mb[i].size); | 311 | diffbuf = xrealloc(diffbuf, buflen + mb[i].size); |
312 | memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size); | 312 | memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size); |
313 | ((linediff_fn)priv)(diffbuf, buflen + mb[i].size); | 313 | ((linediff_fn)priv)(diffbuf, buflen + mb[i].size); |
314 | free(diffbuf); | 314 | free(diffbuf); |
315 | diffbuf = NULL; | 315 | diffbuf = NULL; |
316 | buflen = 0; | 316 | buflen = 0; |
317 | } | 317 | } |
318 | if (diffbuf) { | 318 | if (diffbuf) { |
319 | ((linediff_fn)priv)(diffbuf, buflen); | 319 | ((linediff_fn)priv)(diffbuf, buflen); |
320 | free(diffbuf); | 320 | free(diffbuf); |
321 | diffbuf = NULL; | 321 | diffbuf = NULL; |
322 | buflen = 0; | 322 | buflen = 0; |
323 | } | 323 | } |
324 | return 0; | 324 | return 0; |
325 | } | 325 | } |
326 | 326 | ||
327 | int cgit_diff_files(const unsigned char *old_sha1, | 327 | int cgit_diff_files(const unsigned char *old_sha1, |
328 | const unsigned char *new_sha1, | 328 | const unsigned char *new_sha1, |
329 | linediff_fn fn) | 329 | linediff_fn fn) |
330 | { | 330 | { |
331 | mmfile_t file1, file2; | 331 | mmfile_t file1, file2; |
332 | xpparam_t diff_params; | 332 | xpparam_t diff_params; |
333 | xdemitconf_t emit_params; | 333 | xdemitconf_t emit_params; |
334 | xdemitcb_t emit_cb; | 334 | xdemitcb_t emit_cb; |
335 | 335 | ||
336 | if (!load_mmfile(&file1, old_sha1) || !load_mmfile(&file2, new_sha1)) | 336 | if (!load_mmfile(&file1, old_sha1) || !load_mmfile(&file2, new_sha1)) |
337 | return 1; | 337 | return 1; |
338 | 338 | ||
339 | diff_params.flags = XDF_NEED_MINIMAL; | 339 | diff_params.flags = XDF_NEED_MINIMAL; |
340 | emit_params.ctxlen = 3; | 340 | emit_params.ctxlen = 3; |
341 | emit_params.flags = XDL_EMIT_FUNCNAMES; | 341 | emit_params.flags = XDL_EMIT_FUNCNAMES; |
342 | emit_cb.outf = filediff_cb; | 342 | emit_cb.outf = filediff_cb; |
343 | emit_cb.priv = fn; | 343 | emit_cb.priv = fn; |
344 | xdl_diff(&file1, &file2, &diff_params, &emit_params, &emit_cb); | 344 | xdl_diff(&file1, &file2, &diff_params, &emit_params, &emit_cb); |
345 | return 0; | 345 | return 0; |
346 | } | 346 | } |
347 | 347 | ||
348 | void cgit_diff_tree(const unsigned char *old_sha1, | 348 | void cgit_diff_tree(const unsigned char *old_sha1, |
349 | const unsigned char *new_sha1, | 349 | const unsigned char *new_sha1, |
350 | filepair_fn fn) | 350 | filepair_fn fn) |
351 | { | 351 | { |
352 | struct diff_options opt; | 352 | struct diff_options opt; |
353 | int ret; | 353 | int ret; |
354 | 354 | ||
355 | diff_setup(&opt); | 355 | diff_setup(&opt); |
356 | opt.output_format = DIFF_FORMAT_CALLBACK; | 356 | opt.output_format = DIFF_FORMAT_CALLBACK; |
357 | opt.detect_rename = 1; | 357 | opt.detect_rename = 1; |
358 | opt.recursive = 1; | 358 | opt.recursive = 1; |
359 | opt.format_callback = cgit_diff_tree_cb; | 359 | opt.format_callback = cgit_diff_tree_cb; |
360 | opt.format_callback_data = fn; | 360 | opt.format_callback_data = fn; |
361 | diff_setup_done(&opt); | 361 | diff_setup_done(&opt); |
362 | 362 | ||
363 | if (old_sha1) | 363 | if (old_sha1 && !is_null_sha1(old_sha1)) |
364 | ret = diff_tree_sha1(old_sha1, new_sha1, "", &opt); | 364 | ret = diff_tree_sha1(old_sha1, new_sha1, "", &opt); |
365 | else | 365 | else |
366 | ret = diff_root_tree_sha1(new_sha1, "", &opt); | 366 | ret = diff_root_tree_sha1(new_sha1, "", &opt); |
367 | diffcore_std(&opt); | 367 | diffcore_std(&opt); |
368 | diff_flush(&opt); | 368 | diff_flush(&opt); |
369 | } | 369 | } |
370 | 370 | ||
371 | void cgit_diff_commit(struct commit *commit, filepair_fn fn) | 371 | void cgit_diff_commit(struct commit *commit, filepair_fn fn) |
372 | { | 372 | { |
373 | unsigned char *old_sha1 = NULL; | 373 | unsigned char *old_sha1 = NULL; |
374 | 374 | ||
375 | if (commit->parents) | 375 | if (commit->parents) |
376 | old_sha1 = commit->parents->item->object.sha1; | 376 | old_sha1 = commit->parents->item->object.sha1; |
377 | cgit_diff_tree(old_sha1, commit->object.sha1, fn); | 377 | cgit_diff_tree(old_sha1, commit->object.sha1, fn); |
378 | } | 378 | } |
diff --git a/ui-commit.c b/ui-commit.c index d489d7c..2679b59 100644 --- a/ui-commit.c +++ b/ui-commit.c | |||
@@ -1,242 +1,228 @@ | |||
1 | /* ui-commit.c: generate commit view | 1 | /* ui-commit.c: generate commit view |
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 | 10 | ||
11 | static int files, slots; | 11 | static int files, slots; |
12 | static int total_adds, total_rems, max_changes; | 12 | static int total_adds, total_rems, max_changes; |
13 | static int lines_added, lines_removed; | 13 | static int lines_added, lines_removed; |
14 | static char *curr_rev; | ||
14 | 15 | ||
15 | static struct fileinfo { | 16 | static struct fileinfo { |
16 | char status; | 17 | char status; |
17 | unsigned char old_sha1[20]; | 18 | unsigned char old_sha1[20]; |
18 | unsigned char new_sha1[20]; | 19 | unsigned char new_sha1[20]; |
19 | unsigned short old_mode; | 20 | unsigned short old_mode; |
20 | unsigned short new_mode; | 21 | unsigned short new_mode; |
21 | char *old_path; | 22 | char *old_path; |
22 | char *new_path; | 23 | char *new_path; |
23 | unsigned int added; | 24 | unsigned int added; |
24 | unsigned int removed; | 25 | unsigned int removed; |
25 | } *items; | 26 | } *items; |
26 | 27 | ||
27 | 28 | ||
28 | void print_fileinfo(struct fileinfo *info) | 29 | void print_fileinfo(struct fileinfo *info) |
29 | { | 30 | { |
30 | char *query, *query2; | ||
31 | char *class; | 31 | char *class; |
32 | 32 | ||
33 | switch (info->status) { | 33 | switch (info->status) { |
34 | case DIFF_STATUS_ADDED: | 34 | case DIFF_STATUS_ADDED: |
35 | class = "add"; | 35 | class = "add"; |
36 | break; | 36 | break; |
37 | case DIFF_STATUS_COPIED: | 37 | case DIFF_STATUS_COPIED: |
38 | class = "cpy"; | 38 | class = "cpy"; |
39 | break; | 39 | break; |
40 | case DIFF_STATUS_DELETED: | 40 | case DIFF_STATUS_DELETED: |
41 | class = "del"; | 41 | class = "del"; |
42 | break; | 42 | break; |
43 | case DIFF_STATUS_MODIFIED: | 43 | case DIFF_STATUS_MODIFIED: |
44 | class = "upd"; | 44 | class = "upd"; |
45 | break; | 45 | break; |
46 | case DIFF_STATUS_RENAMED: | 46 | case DIFF_STATUS_RENAMED: |
47 | class = "mov"; | 47 | class = "mov"; |
48 | break; | 48 | break; |
49 | case DIFF_STATUS_TYPE_CHANGED: | 49 | case DIFF_STATUS_TYPE_CHANGED: |
50 | class = "typ"; | 50 | class = "typ"; |
51 | break; | 51 | break; |
52 | case DIFF_STATUS_UNKNOWN: | 52 | case DIFF_STATUS_UNKNOWN: |
53 | class = "unk"; | 53 | class = "unk"; |
54 | break; | 54 | break; |
55 | case DIFF_STATUS_UNMERGED: | 55 | case DIFF_STATUS_UNMERGED: |
56 | class = "stg"; | 56 | class = "stg"; |
57 | break; | 57 | break; |
58 | default: | 58 | default: |
59 | die("bug: unhandled diff status %c", info->status); | 59 | die("bug: unhandled diff status %c", info->status); |
60 | } | 60 | } |
61 | 61 | ||
62 | html("<tr>"); | 62 | html("<tr>"); |
63 | htmlf("<td class='mode'>"); | 63 | htmlf("<td class='mode'>"); |
64 | if (is_null_sha1(info->new_sha1)) { | 64 | if (is_null_sha1(info->new_sha1)) { |
65 | html_filemode(info->old_mode); | 65 | html_filemode(info->old_mode); |
66 | } else { | 66 | } else { |
67 | html_filemode(info->new_mode); | 67 | html_filemode(info->new_mode); |
68 | } | 68 | } |
69 | 69 | ||
70 | if (info->old_mode != info->new_mode && | 70 | if (info->old_mode != info->new_mode && |
71 | !is_null_sha1(info->old_sha1) && | 71 | !is_null_sha1(info->old_sha1) && |
72 | !is_null_sha1(info->new_sha1)) { | 72 | !is_null_sha1(info->new_sha1)) { |
73 | html("<span class='modechange'>["); | 73 | html("<span class='modechange'>["); |
74 | html_filemode(info->old_mode); | 74 | html_filemode(info->old_mode); |
75 | html("]</span>"); | 75 | html("]</span>"); |
76 | } | 76 | } |
77 | htmlf("</td><td class='%s'>", class); | 77 | htmlf("</td><td class='%s'>", class); |
78 | query = fmt("id=%s&id2=%s&path=%s", sha1_to_hex(info->old_sha1), | 78 | cgit_tree_link(info->new_path, NULL, NULL, cgit_query_head, curr_rev, |
79 | sha1_to_hex(info->new_sha1), info->new_path); | 79 | info->new_path); |
80 | html_link_open(cgit_pageurl(cgit_query_repo, "diff", query), | 80 | if (info->status == DIFF_STATUS_COPIED || info->status == DIFF_STATUS_RENAMED) |
81 | NULL, NULL); | 81 | htmlf(" (%s from %s)", |
82 | if (info->status == DIFF_STATUS_COPIED || | 82 | info->status == DIFF_STATUS_COPIED ? "copied" : "renamed", |
83 | info->status == DIFF_STATUS_RENAMED) { | 83 | info->old_path); |
84 | html_txt(info->new_path); | ||
85 | htmlf("</a> (%s from ", info->status == DIFF_STATUS_COPIED ? | ||
86 | "copied" : "renamed"); | ||
87 | query2 = fmt("id=%s", sha1_to_hex(info->old_sha1)); | ||
88 | html_link_open(cgit_pageurl(cgit_query_repo, "view", query2), | ||
89 | NULL, NULL); | ||
90 | html_txt(info->old_path); | ||
91 | html("</a>)"); | ||
92 | } else { | ||
93 | html_txt(info->new_path); | ||
94 | html("</a>"); | ||
95 | } | ||
96 | html("</td><td class='right'>"); | 84 | html("</td><td class='right'>"); |
97 | htmlf("%d", info->added + info->removed); | 85 | htmlf("%d", info->added + info->removed); |
98 | html("</td><td class='graph'>"); | 86 | html("</td><td class='graph'>"); |
99 | htmlf("<table width='%d%%'><tr>", (max_changes > 100 ? 100 : max_changes)); | 87 | htmlf("<table width='%d%%'><tr>", (max_changes > 100 ? 100 : max_changes)); |
100 | htmlf("<td class='add' style='width: %.1f%%;'/>", | 88 | htmlf("<td class='add' style='width: %.1f%%;'/>", |
101 | info->added * 100.0 / max_changes); | 89 | info->added * 100.0 / max_changes); |
102 | htmlf("<td class='rem' style='width: %.1f%%;'/>", | 90 | htmlf("<td class='rem' style='width: %.1f%%;'/>", |
103 | info->removed * 100.0 / max_changes); | 91 | info->removed * 100.0 / max_changes); |
104 | htmlf("<td class='none' style='width: %.1f%%;'/>", | 92 | htmlf("<td class='none' style='width: %.1f%%;'/>", |
105 | (max_changes - info->removed - info->added) * 100.0 / max_changes); | 93 | (max_changes - info->removed - info->added) * 100.0 / max_changes); |
106 | html("</tr></table></td></tr>\n"); | 94 | html("</tr></table></td></tr>\n"); |
107 | } | 95 | } |
108 | 96 | ||
109 | void cgit_count_diff_lines(char *line, int len) | 97 | void cgit_count_diff_lines(char *line, int len) |
110 | { | 98 | { |
111 | if (line && (len > 0)) { | 99 | if (line && (len > 0)) { |
112 | if (line[0] == '+') | 100 | if (line[0] == '+') |
113 | lines_added++; | 101 | lines_added++; |
114 | else if (line[0] == '-') | 102 | else if (line[0] == '-') |
115 | lines_removed++; | 103 | lines_removed++; |
116 | } | 104 | } |
117 | } | 105 | } |
118 | 106 | ||
119 | void inspect_filepair(struct diff_filepair *pair) | 107 | void inspect_filepair(struct diff_filepair *pair) |
120 | { | 108 | { |
121 | files++; | 109 | files++; |
122 | lines_added = 0; | 110 | lines_added = 0; |
123 | lines_removed = 0; | 111 | lines_removed = 0; |
124 | cgit_diff_files(pair->one->sha1, pair->two->sha1, cgit_count_diff_lines); | 112 | cgit_diff_files(pair->one->sha1, pair->two->sha1, cgit_count_diff_lines); |
125 | if (files >= slots) { | 113 | if (files >= slots) { |
126 | if (slots == 0) | 114 | if (slots == 0) |
127 | slots = 4; | 115 | slots = 4; |
128 | else | 116 | else |
129 | slots = slots * 2; | 117 | slots = slots * 2; |
130 | items = xrealloc(items, slots * sizeof(struct fileinfo)); | 118 | items = xrealloc(items, slots * sizeof(struct fileinfo)); |
131 | } | 119 | } |
132 | items[files-1].status = pair->status; | 120 | items[files-1].status = pair->status; |
133 | hashcpy(items[files-1].old_sha1, pair->one->sha1); | 121 | hashcpy(items[files-1].old_sha1, pair->one->sha1); |
134 | hashcpy(items[files-1].new_sha1, pair->two->sha1); | 122 | hashcpy(items[files-1].new_sha1, pair->two->sha1); |
135 | items[files-1].old_mode = pair->one->mode; | 123 | items[files-1].old_mode = pair->one->mode; |
136 | items[files-1].new_mode = pair->two->mode; | 124 | items[files-1].new_mode = pair->two->mode; |
137 | items[files-1].old_path = xstrdup(pair->one->path); | 125 | items[files-1].old_path = xstrdup(pair->one->path); |
138 | items[files-1].new_path = xstrdup(pair->two->path); | 126 | items[files-1].new_path = xstrdup(pair->two->path); |
139 | items[files-1].added = lines_added; | 127 | items[files-1].added = lines_added; |
140 | items[files-1].removed = lines_removed; | 128 | items[files-1].removed = lines_removed; |
141 | if (lines_added + lines_removed > max_changes) | 129 | if (lines_added + lines_removed > max_changes) |
142 | max_changes = lines_added + lines_removed; | 130 | max_changes = lines_added + lines_removed; |
143 | total_adds += lines_added; | 131 | total_adds += lines_added; |
144 | total_rems += lines_removed; | 132 | total_rems += lines_removed; |
145 | } | 133 | } |
146 | 134 | ||
147 | 135 | ||
148 | void cgit_print_commit(const char *hex) | 136 | void cgit_print_commit(char *hex) |
149 | { | 137 | { |
150 | struct commit *commit, *parent; | 138 | struct commit *commit, *parent; |
151 | struct commitinfo *info; | 139 | struct commitinfo *info; |
152 | struct commit_list *p; | 140 | struct commit_list *p; |
153 | unsigned char sha1[20]; | 141 | unsigned char sha1[20]; |
154 | char *query; | ||
155 | char *filename; | 142 | char *filename; |
156 | char *tmp; | 143 | char *tmp; |
157 | int i; | 144 | int i; |
158 | 145 | ||
159 | if (!hex) | 146 | if (!hex) |
160 | hex = cgit_query_head; | 147 | hex = cgit_query_head; |
148 | curr_rev = hex; | ||
161 | 149 | ||
162 | if (get_sha1(hex, sha1)) { | 150 | if (get_sha1(hex, sha1)) { |
163 | cgit_print_error(fmt("Bad object id: %s", hex)); | 151 | cgit_print_error(fmt("Bad object id: %s", hex)); |
164 | return; | 152 | return; |
165 | } | 153 | } |
166 | commit = lookup_commit_reference(sha1); | 154 | commit = lookup_commit_reference(sha1); |
167 | if (!commit) { | 155 | if (!commit) { |
168 | cgit_print_error(fmt("Bad commit reference: %s", hex)); | 156 | cgit_print_error(fmt("Bad commit reference: %s", hex)); |
169 | return; | 157 | return; |
170 | } | 158 | } |
171 | info = cgit_parse_commit(commit); | 159 | info = cgit_parse_commit(commit); |
172 | 160 | ||
173 | html("<table class='commit-info'>\n"); | 161 | html("<table class='commit-info'>\n"); |
174 | html("<tr><th>author</th><td>"); | 162 | html("<tr><th>author</th><td>"); |
175 | html_txt(info->author); | 163 | html_txt(info->author); |
176 | html(" "); | 164 | html(" "); |
177 | html_txt(info->author_email); | 165 | html_txt(info->author_email); |
178 | html("</td><td class='right'>"); | 166 | html("</td><td class='right'>"); |
179 | cgit_print_date(info->author_date, FMT_LONGDATE); | 167 | cgit_print_date(info->author_date, FMT_LONGDATE); |
180 | html("</td></tr>\n"); | 168 | html("</td></tr>\n"); |
181 | html("<tr><th>committer</th><td>"); | 169 | html("<tr><th>committer</th><td>"); |
182 | html_txt(info->committer); | 170 | html_txt(info->committer); |
183 | html(" "); | 171 | html(" "); |
184 | html_txt(info->committer_email); | 172 | html_txt(info->committer_email); |
185 | html("</td><td class='right'>"); | 173 | html("</td><td class='right'>"); |
186 | cgit_print_date(info->committer_date, FMT_LONGDATE); | 174 | cgit_print_date(info->committer_date, FMT_LONGDATE); |
187 | html("</td></tr>\n"); | 175 | html("</td></tr>\n"); |
188 | html("<tr><th>tree</th><td colspan='2' class='sha1'>"); | 176 | html("<tr><th>tree</th><td colspan='2' class='sha1'>"); |
189 | tmp = xstrdup(hex); | 177 | tmp = xstrdup(hex); |
190 | cgit_tree_link(sha1_to_hex(commit->tree->object.sha1), NULL, NULL, | 178 | cgit_tree_link(sha1_to_hex(commit->tree->object.sha1), NULL, NULL, |
191 | cgit_query_head, tmp, NULL); | 179 | cgit_query_head, tmp, NULL); |
192 | html("</td></tr>\n"); | 180 | html("</td></tr>\n"); |
193 | for (p = commit->parents; p ; p = p->next) { | 181 | for (p = commit->parents; p ; p = p->next) { |
194 | parent = lookup_commit_reference(p->item->object.sha1); | 182 | parent = lookup_commit_reference(p->item->object.sha1); |
195 | if (!parent) { | 183 | if (!parent) { |
196 | html("<tr><td colspan='3'>"); | 184 | html("<tr><td colspan='3'>"); |
197 | cgit_print_error("Error reading parent commit"); | 185 | cgit_print_error("Error reading parent commit"); |
198 | html("</td></tr>"); | 186 | html("</td></tr>"); |
199 | continue; | 187 | continue; |
200 | } | 188 | } |
201 | html("<tr><th>parent</th>" | 189 | html("<tr><th>parent</th>" |
202 | "<td colspan='2' class='sha1'>"); | 190 | "<td colspan='2' class='sha1'>"); |
203 | cgit_commit_link(sha1_to_hex(p->item->object.sha1), NULL, NULL, | 191 | cgit_commit_link(sha1_to_hex(p->item->object.sha1), NULL, NULL, |
204 | cgit_query_head, sha1_to_hex(p->item->object.sha1)); | 192 | cgit_query_head, sha1_to_hex(p->item->object.sha1)); |
205 | html(" (<a href='"); | 193 | html(" ("); |
206 | query = fmt("id=%s&id2=%s", sha1_to_hex(parent->tree->object.sha1), | 194 | cgit_diff_link("diff", NULL, NULL, cgit_query_head, hex, |
207 | sha1_to_hex(commit->tree->object.sha1)); | 195 | sha1_to_hex(p->item->object.sha1), NULL); |
208 | html_attr(cgit_pageurl(cgit_query_repo, "diff", query)); | 196 | html(")</td></tr>"); |
209 | html("'>diff</a>)</td></tr>"); | ||
210 | } | 197 | } |
211 | if (cgit_repo->snapshots) { | 198 | if (cgit_repo->snapshots) { |
212 | htmlf("<tr><th>download</th><td colspan='2' class='sha1'><a href='"); | 199 | htmlf("<tr><th>download</th><td colspan='2' class='sha1'><a href='"); |
213 | filename = fmt("%s-%s.zip", cgit_query_repo, hex); | 200 | filename = fmt("%s-%s.zip", cgit_query_repo, hex); |
214 | html_attr(cgit_pageurl(cgit_query_repo, "snapshot", | 201 | html_attr(cgit_pageurl(cgit_query_repo, "snapshot", |
215 | fmt("id=%s&name=%s", hex, filename))); | 202 | fmt("id=%s&name=%s", hex, filename))); |
216 | htmlf("'>%s</a></td></tr>", filename); | 203 | htmlf("'>%s</a></td></tr>", filename); |
217 | } | 204 | } |
218 | html("</table>\n"); | 205 | html("</table>\n"); |
219 | html("<div class='commit-subject'>"); | 206 | html("<div class='commit-subject'>"); |
220 | html_txt(info->subject); | 207 | html_txt(info->subject); |
221 | html("</div>"); | 208 | html("</div>"); |
222 | html("<div class='commit-msg'>"); | 209 | html("<div class='commit-msg'>"); |
223 | html_txt(info->msg); | 210 | html_txt(info->msg); |
224 | html("</div>"); | 211 | html("</div>"); |
225 | if (!(commit->parents && commit->parents->next && commit->parents->next->next)) { | 212 | if (!(commit->parents && commit->parents->next && commit->parents->next->next)) { |
226 | html("<div class='diffstat-header'>Diffstat</div>"); | 213 | html("<div class='diffstat-header'>Diffstat</div>"); |
227 | html("<table class='diffstat'>"); | 214 | html("<table class='diffstat'>"); |
228 | max_changes = 0; | 215 | max_changes = 0; |
229 | cgit_diff_commit(commit, inspect_filepair); | 216 | cgit_diff_commit(commit, inspect_filepair); |
230 | for(i = 0; i<files; i++) | 217 | for(i = 0; i<files; i++) |
231 | print_fileinfo(&items[i]); | 218 | print_fileinfo(&items[i]); |
232 | html("</table>"); | 219 | html("</table>"); |
233 | html("<div class='diffstat-summary'>"); | 220 | html("<div class='diffstat-summary'>"); |
234 | htmlf("%d files changed, %d insertions, %d deletions (", | 221 | htmlf("%d files changed, %d insertions, %d deletions (", |
235 | files, total_adds, total_rems); | 222 | files, total_adds, total_rems); |
236 | query = fmt("h=%s", hex); | 223 | cgit_diff_link("show diff", NULL, NULL, cgit_query_head, hex, |
237 | html_link_open(cgit_pageurl(cgit_query_repo, "diff", query), NULL, NULL); | 224 | NULL, NULL); |
238 | html("show diff</a>)"); | 225 | html(")</div>"); |
239 | html("</div>"); | ||
240 | } | 226 | } |
241 | cgit_free_commitinfo(info); | 227 | cgit_free_commitinfo(info); |
242 | } | 228 | } |
@@ -28,115 +28,113 @@ static void print_line(char *line, int len) | |||
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 | ||
34 | static void header(unsigned char *sha1, char *path1, int mode1, | 34 | static void header(unsigned char *sha1, char *path1, int mode1, |
35 | unsigned char *sha2, char *path2, int mode2) | 35 | unsigned char *sha2, char *path2, int mode2) |
36 | { | 36 | { |
37 | char *abbrev1, *abbrev2; | 37 | char *abbrev1, *abbrev2; |
38 | int subproject; | 38 | int subproject; |
39 | 39 | ||
40 | subproject = (S_ISDIRLNK(mode1) || S_ISDIRLNK(mode2)); | 40 | subproject = (S_ISDIRLNK(mode1) || S_ISDIRLNK(mode2)); |
41 | html("<div class='head'>"); | 41 | html("<div class='head'>"); |
42 | html("diff --git a/"); | 42 | html("diff --git a/"); |
43 | html_txt(path1); | 43 | html_txt(path1); |
44 | html(" b/"); | 44 | html(" b/"); |
45 | html_txt(path2); | 45 | html_txt(path2); |
46 | 46 | ||
47 | if (is_null_sha1(sha1)) | 47 | if (is_null_sha1(sha1)) |
48 | path1 = "dev/null"; | 48 | path1 = "dev/null"; |
49 | if (is_null_sha1(sha2)) | 49 | if (is_null_sha1(sha2)) |
50 | path2 = "dev/null"; | 50 | path2 = "dev/null"; |
51 | 51 | ||
52 | if (mode1 == 0) | 52 | if (mode1 == 0) |
53 | htmlf("<br/>new file mode %.6o", mode2); | 53 | htmlf("<br/>new file mode %.6o", mode2); |
54 | 54 | ||
55 | if (mode2 == 0) | 55 | if (mode2 == 0) |
56 | htmlf("<br/>deleted file mode %.6o", mode1); | 56 | htmlf("<br/>deleted file mode %.6o", mode1); |
57 | 57 | ||
58 | if (!subproject) { | 58 | if (!subproject) { |
59 | abbrev1 = xstrdup(find_unique_abbrev(sha1, DEFAULT_ABBREV)); | 59 | abbrev1 = xstrdup(find_unique_abbrev(sha1, DEFAULT_ABBREV)); |
60 | abbrev2 = xstrdup(find_unique_abbrev(sha2, DEFAULT_ABBREV)); | 60 | abbrev2 = xstrdup(find_unique_abbrev(sha2, DEFAULT_ABBREV)); |
61 | htmlf("<br/>index %s..%s", abbrev1, abbrev2); | 61 | htmlf("<br/>index %s..%s", abbrev1, abbrev2); |
62 | free(abbrev1); | 62 | free(abbrev1); |
63 | free(abbrev2); | 63 | free(abbrev2); |
64 | if (mode1 != 0 && mode2 != 0) { | 64 | if (mode1 != 0 && mode2 != 0) { |
65 | htmlf(" %.6o", mode1); | 65 | htmlf(" %.6o", mode1); |
66 | if (mode2 != mode1) | 66 | if (mode2 != mode1) |
67 | htmlf("..%.6o", mode2); | 67 | htmlf("..%.6o", mode2); |
68 | } | 68 | } |
69 | html("<br/>--- a/"); | 69 | html("<br/>--- a/"); |
70 | html_txt(path1); | 70 | html_txt(path1); |
71 | html("<br/>+++ b/"); | 71 | html("<br/>+++ b/"); |
72 | html_txt(path2); | 72 | html_txt(path2); |
73 | } | 73 | } |
74 | html("</div>"); | 74 | html("</div>"); |
75 | } | 75 | } |
76 | 76 | ||
77 | static void filepair_cb(struct diff_filepair *pair) | 77 | static void filepair_cb(struct diff_filepair *pair) |
78 | { | 78 | { |
79 | header(pair->one->sha1, pair->one->path, pair->one->mode, | 79 | header(pair->one->sha1, pair->one->path, pair->one->mode, |
80 | pair->two->sha1, pair->two->path, pair->two->mode); | 80 | pair->two->sha1, pair->two->path, pair->two->mode); |
81 | if (S_ISDIRLNK(pair->one->mode) || S_ISDIRLNK(pair->two->mode)) { | 81 | if (S_ISDIRLNK(pair->one->mode) || S_ISDIRLNK(pair->two->mode)) { |
82 | if (S_ISDIRLNK(pair->one->mode)) | 82 | if (S_ISDIRLNK(pair->one->mode)) |
83 | print_line(fmt("-Subproject %s", sha1_to_hex(pair->one->sha1)), 52); | 83 | print_line(fmt("-Subproject %s", sha1_to_hex(pair->one->sha1)), 52); |
84 | if (S_ISDIRLNK(pair->two->mode)) | 84 | if (S_ISDIRLNK(pair->two->mode)) |
85 | print_line(fmt("+Subproject %s", sha1_to_hex(pair->two->sha1)), 52); | 85 | print_line(fmt("+Subproject %s", sha1_to_hex(pair->two->sha1)), 52); |
86 | return; | 86 | return; |
87 | } | 87 | } |
88 | if (cgit_diff_files(pair->one->sha1, pair->two->sha1, print_line)) | 88 | if (cgit_diff_files(pair->one->sha1, pair->two->sha1, print_line)) |
89 | cgit_print_error("Error running diff"); | 89 | cgit_print_error("Error running diff"); |
90 | } | 90 | } |
91 | 91 | ||
92 | void cgit_print_diff(const char *head, const char *old_hex, const char *new_hex, char *path) | 92 | void cgit_print_diff(const char *new_rev, const char *old_rev) |
93 | { | 93 | { |
94 | unsigned char sha1[20], sha2[20]; | 94 | unsigned char sha1[20], sha2[20]; |
95 | enum object_type type; | 95 | enum object_type type; |
96 | unsigned long size; | 96 | unsigned long size; |
97 | struct commit *commit; | 97 | struct commit *commit, *commit2; |
98 | 98 | ||
99 | html("<table class='diff'>"); | 99 | if (!new_rev) |
100 | html("<tr><td>"); | 100 | new_rev = cgit_query_head; |
101 | 101 | get_sha1(new_rev, sha1); | |
102 | if (head && !old_hex && !new_hex) { | 102 | type = sha1_object_info(sha1, &size); |
103 | get_sha1(head, sha1); | 103 | if (type == OBJ_BAD) { |
104 | commit = lookup_commit_reference(sha1); | 104 | cgit_print_error(fmt("Bad object name: %s", new_rev)); |
105 | if (commit && !parse_commit(commit)) | 105 | return; |
106 | cgit_diff_commit(commit, filepair_cb); | 106 | } |
107 | else | 107 | if (type != OBJ_COMMIT) { |
108 | cgit_print_error(fmt("Bad commit: %s", head)); | 108 | cgit_print_error(fmt("Unhandled object type: %s", |
109 | html("</td></tr>"); | 109 | typename(type))); |
110 | html("</table>"); | ||
111 | return; | 110 | return; |
112 | } | 111 | } |
113 | 112 | ||
114 | get_sha1(old_hex, sha1); | 113 | commit = lookup_commit_reference(sha1); |
115 | get_sha1(new_hex, sha2); | 114 | if (!commit || parse_commit(commit)) |
115 | cgit_print_error(fmt("Bad commit: %s", sha1_to_hex(sha1))); | ||
116 | 116 | ||
117 | type = sha1_object_info(sha1, &size); | 117 | if (old_rev) |
118 | if (type == OBJ_BAD) { | 118 | get_sha1(old_rev, sha2); |
119 | else if (commit->parents && commit->parents->item) | ||
120 | hashcpy(sha2, commit->parents->item->object.sha1); | ||
121 | else | ||
122 | hashclr(sha2); | ||
123 | |||
124 | if (!is_null_sha1(sha2)) { | ||
119 | type = sha1_object_info(sha2, &size); | 125 | type = sha1_object_info(sha2, &size); |
120 | if (type == OBJ_BAD) { | 126 | if (type == OBJ_BAD) { |
121 | cgit_print_error(fmt("Bad object names: %s, %s", old_hex, new_hex)); | 127 | cgit_print_error(fmt("Bad object name: %s", sha1_to_hex(sha2))); |
122 | return; | 128 | return; |
123 | } | 129 | } |
130 | commit2 = lookup_commit_reference(sha2); | ||
131 | if (!commit2 || parse_commit(commit2)) | ||
132 | cgit_print_error(fmt("Bad commit: %s", sha1_to_hex(sha2))); | ||
124 | } | 133 | } |
125 | 134 | ||
126 | switch(type) { | 135 | html("<table class='diff'>"); |
127 | case OBJ_BLOB: | 136 | html("<tr><td>"); |
128 | header(sha1, path, 0644, sha2, path, 0644); | 137 | cgit_diff_tree(sha2, sha1, filepair_cb); |
129 | if (cgit_diff_files(sha1, sha2, print_line)) | ||
130 | cgit_print_error("Error running diff"); | ||
131 | break; | ||
132 | case OBJ_TREE: | ||
133 | cgit_diff_tree(sha1, sha2, filepair_cb); | ||
134 | break; | ||
135 | default: | ||
136 | cgit_print_error(fmt("Unhandled object type: %s", | ||
137 | typename(type))); | ||
138 | break; | ||
139 | } | ||
140 | html("</td></tr>"); | 138 | html("</td></tr>"); |
141 | html("</table>"); | 139 | html("</table>"); |
142 | } | 140 | } |
diff --git a/ui-shared.c b/ui-shared.c index 71c899a..15d8254 100644 --- a/ui-shared.c +++ b/ui-shared.c | |||
@@ -115,128 +115,150 @@ static char *repolink(char *title, char *class, char *page, char *head, | |||
115 | html("/"); | 115 | html("/"); |
116 | if (path) | 116 | if (path) |
117 | html_attr(path); | 117 | html_attr(path); |
118 | } else { | 118 | } else { |
119 | html(cgit_script_name); | 119 | html(cgit_script_name); |
120 | html("?url="); | 120 | html("?url="); |
121 | html_attr(cgit_repo->url); | 121 | html_attr(cgit_repo->url); |
122 | if (cgit_repo->url[strlen(cgit_repo->url) - 1] != '/') | 122 | if (cgit_repo->url[strlen(cgit_repo->url) - 1] != '/') |
123 | html("/"); | 123 | html("/"); |
124 | html(page); | 124 | html(page); |
125 | html("/"); | 125 | html("/"); |
126 | if (path) | 126 | if (path) |
127 | html_attr(path); | 127 | html_attr(path); |
128 | delim = "&"; | 128 | delim = "&"; |
129 | } | 129 | } |
130 | if (head && strcmp(head, cgit_repo->defbranch)) { | 130 | if (head && strcmp(head, cgit_repo->defbranch)) { |
131 | html(delim); | 131 | html(delim); |
132 | html("h="); | 132 | html("h="); |
133 | html_attr(head); | 133 | html_attr(head); |
134 | delim = "&"; | 134 | delim = "&"; |
135 | } | 135 | } |
136 | return fmt("%s", delim); | 136 | return fmt("%s", delim); |
137 | } | 137 | } |
138 | 138 | ||
139 | static void reporevlink(char *page, char *name, char *title, char *class, | 139 | static void reporevlink(char *page, char *name, char *title, char *class, |
140 | char *head, char *rev, char *path) | 140 | char *head, char *rev, char *path) |
141 | { | 141 | { |
142 | char *delim; | 142 | char *delim; |
143 | 143 | ||
144 | delim = repolink(title, class, page, head, path); | 144 | delim = repolink(title, class, page, head, path); |
145 | if (rev && strcmp(rev, cgit_query_head)) { | 145 | if (rev && strcmp(rev, cgit_query_head)) { |
146 | html(delim); | 146 | html(delim); |
147 | html("id="); | 147 | html("id="); |
148 | html_attr(rev); | 148 | html_attr(rev); |
149 | } | 149 | } |
150 | html("'>"); | 150 | html("'>"); |
151 | html_txt(name); | 151 | html_txt(name); |
152 | html("</a>"); | 152 | html("</a>"); |
153 | } | 153 | } |
154 | 154 | ||
155 | void cgit_tree_link(char *name, char *title, char *class, char *head, | 155 | void cgit_tree_link(char *name, char *title, char *class, char *head, |
156 | char *rev, char *path) | 156 | char *rev, char *path) |
157 | { | 157 | { |
158 | reporevlink("tree", name, title, class, head, rev, path); | 158 | reporevlink("tree", name, title, class, head, rev, path); |
159 | } | 159 | } |
160 | 160 | ||
161 | void cgit_log_link(char *name, char *title, char *class, char *head, | 161 | void cgit_log_link(char *name, char *title, char *class, char *head, |
162 | char *rev, char *path) | 162 | char *rev, char *path) |
163 | { | 163 | { |
164 | reporevlink("log", name, title, class, head, rev, path); | 164 | reporevlink("log", name, title, class, head, rev, path); |
165 | } | 165 | } |
166 | 166 | ||
167 | void cgit_commit_link(char *name, char *title, char *class, char *head, | 167 | void cgit_commit_link(char *name, char *title, char *class, char *head, |
168 | char *rev) | 168 | char *rev) |
169 | { | 169 | { |
170 | if (strlen(name) > cgit_max_msg_len && cgit_max_msg_len >= 15) { | 170 | if (strlen(name) > cgit_max_msg_len && cgit_max_msg_len >= 15) { |
171 | name[cgit_max_msg_len] = '\0'; | 171 | name[cgit_max_msg_len] = '\0'; |
172 | name[cgit_max_msg_len - 1] = '.'; | 172 | name[cgit_max_msg_len - 1] = '.'; |
173 | name[cgit_max_msg_len - 2] = '.'; | 173 | name[cgit_max_msg_len - 2] = '.'; |
174 | name[cgit_max_msg_len - 3] = '.'; | 174 | name[cgit_max_msg_len - 3] = '.'; |
175 | } | 175 | } |
176 | reporevlink("commit", name, title, class, head, rev, NULL); | 176 | reporevlink("commit", name, title, class, head, rev, NULL); |
177 | } | 177 | } |
178 | 178 | ||
179 | void cgit_diff_link(char *name, char *title, char *class, char *head, | ||
180 | char *new_rev, char *old_rev, char *path) | ||
181 | { | ||
182 | char *delim; | ||
183 | |||
184 | delim = repolink(title, class, "diff", head, path); | ||
185 | if (new_rev && strcmp(new_rev, cgit_query_head)) { | ||
186 | html(delim); | ||
187 | html("id="); | ||
188 | html_attr(new_rev); | ||
189 | delim = "&"; | ||
190 | } | ||
191 | if (old_rev) { | ||
192 | html(delim); | ||
193 | html("id2="); | ||
194 | html_attr(old_rev); | ||
195 | } | ||
196 | html("'>"); | ||
197 | html_txt(name); | ||
198 | html("</a>"); | ||
199 | } | ||
200 | |||
179 | void cgit_print_date(time_t secs, char *format) | 201 | void cgit_print_date(time_t secs, char *format) |
180 | { | 202 | { |
181 | char buf[64]; | 203 | char buf[64]; |
182 | struct tm *time; | 204 | struct tm *time; |
183 | 205 | ||
184 | time = gmtime(&secs); | 206 | time = gmtime(&secs); |
185 | strftime(buf, sizeof(buf)-1, format, time); | 207 | strftime(buf, sizeof(buf)-1, format, time); |
186 | html_txt(buf); | 208 | html_txt(buf); |
187 | } | 209 | } |
188 | 210 | ||
189 | void cgit_print_age(time_t t, time_t max_relative, char *format) | 211 | void cgit_print_age(time_t t, time_t max_relative, char *format) |
190 | { | 212 | { |
191 | time_t now, secs; | 213 | time_t now, secs; |
192 | 214 | ||
193 | time(&now); | 215 | time(&now); |
194 | secs = now - t; | 216 | secs = now - t; |
195 | 217 | ||
196 | if (secs > max_relative && max_relative >= 0) { | 218 | if (secs > max_relative && max_relative >= 0) { |
197 | cgit_print_date(t, format); | 219 | cgit_print_date(t, format); |
198 | return; | 220 | return; |
199 | } | 221 | } |
200 | 222 | ||
201 | if (secs < TM_HOUR * 2) { | 223 | if (secs < TM_HOUR * 2) { |
202 | htmlf("<span class='age-mins'>%.0f min.</span>", | 224 | htmlf("<span class='age-mins'>%.0f min.</span>", |
203 | secs * 1.0 / TM_MIN); | 225 | secs * 1.0 / TM_MIN); |
204 | return; | 226 | return; |
205 | } | 227 | } |
206 | if (secs < TM_DAY * 2) { | 228 | if (secs < TM_DAY * 2) { |
207 | htmlf("<span class='age-hours'>%.0f hours</span>", | 229 | htmlf("<span class='age-hours'>%.0f hours</span>", |
208 | secs * 1.0 / TM_HOUR); | 230 | secs * 1.0 / TM_HOUR); |
209 | return; | 231 | return; |
210 | } | 232 | } |
211 | if (secs < TM_WEEK * 2) { | 233 | if (secs < TM_WEEK * 2) { |
212 | htmlf("<span class='age-days'>%.0f days</span>", | 234 | htmlf("<span class='age-days'>%.0f days</span>", |
213 | secs * 1.0 / TM_DAY); | 235 | secs * 1.0 / TM_DAY); |
214 | return; | 236 | return; |
215 | } | 237 | } |
216 | if (secs < TM_MONTH * 2) { | 238 | if (secs < TM_MONTH * 2) { |
217 | htmlf("<span class='age-weeks'>%.0f weeks</span>", | 239 | htmlf("<span class='age-weeks'>%.0f weeks</span>", |
218 | secs * 1.0 / TM_WEEK); | 240 | secs * 1.0 / TM_WEEK); |
219 | return; | 241 | return; |
220 | } | 242 | } |
221 | if (secs < TM_YEAR * 2) { | 243 | if (secs < TM_YEAR * 2) { |
222 | htmlf("<span class='age-months'>%.0f months</span>", | 244 | htmlf("<span class='age-months'>%.0f months</span>", |
223 | secs * 1.0 / TM_MONTH); | 245 | secs * 1.0 / TM_MONTH); |
224 | return; | 246 | return; |
225 | } | 247 | } |
226 | htmlf("<span class='age-years'>%.0f years</span>", | 248 | htmlf("<span class='age-years'>%.0f years</span>", |
227 | secs * 1.0 / TM_YEAR); | 249 | secs * 1.0 / TM_YEAR); |
228 | } | 250 | } |
229 | 251 | ||
230 | void cgit_print_docstart(char *title, struct cacheitem *item) | 252 | void cgit_print_docstart(char *title, struct cacheitem *item) |
231 | { | 253 | { |
232 | html("Content-Type: text/html; charset=utf-8\n"); | 254 | html("Content-Type: text/html; charset=utf-8\n"); |
233 | htmlf("Last-Modified: %s\n", http_date(item->st.st_mtime)); | 255 | htmlf("Last-Modified: %s\n", http_date(item->st.st_mtime)); |
234 | htmlf("Expires: %s\n", http_date(item->st.st_mtime + | 256 | htmlf("Expires: %s\n", http_date(item->st.st_mtime + |
235 | ttl_seconds(item->ttl))); | 257 | ttl_seconds(item->ttl))); |
236 | html("\n"); | 258 | html("\n"); |
237 | html(cgit_doctype); | 259 | html(cgit_doctype); |
238 | html("<html>\n"); | 260 | html("<html>\n"); |
239 | html("<head>\n"); | 261 | html("<head>\n"); |
240 | html("<title>"); | 262 | html("<title>"); |
241 | html_txt(title); | 263 | html_txt(title); |
242 | html("</title>\n"); | 264 | html("</title>\n"); |