blob: d696e20706770d120892d0073ce05fe7532b762e [file] [log] [blame]
Lars Hjemlid14c5f62006-12-11 17:04:19 +01001/* ui-log.c: functions for log output
2 *
Lukas Fleischerf7f26f82014-01-08 15:10:49 +01003 * Copyright (C) 2006-2014 cgit Development Team <cgit@lists.zx2c4.com>
Lars Hjemlid14c5f62006-12-11 17:04:19 +01004 *
5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text)
7 */
8
9#include "cgit.h"
John Keeping8f208792013-04-06 11:37:59 +010010#include "ui-log.h"
Lars Hjemlib1f9b9c2008-02-23 22:45:33 +010011#include "html.h"
Lars Hjemlia4d1ca12008-03-24 16:50:57 +010012#include "ui-shared.h"
Lukas Fleischer9973ef02013-11-22 13:24:52 +010013#include "argv-array.h"
Lars Hjemlid14c5f62006-12-11 17:04:19 +010014
John Keeping30304d82015-08-12 15:55:28 +010015static int files, add_lines, rem_lines, lines_counted;
Lars Hjemli48dc0032007-05-13 11:27:46 +020016
Johan Herland268b34a2010-11-15 18:39:51 +010017/*
18 * The list of available column colors in the commit graph.
19 */
20static const char *column_colors_html[] = {
21 "<span class='column1'>",
22 "<span class='column2'>",
23 "<span class='column3'>",
24 "<span class='column4'>",
25 "<span class='column5'>",
26 "<span class='column6'>",
27 "</span>",
28};
29
30#define COLUMN_COLORS_HTML_MAX (ARRAY_SIZE(column_colors_html) - 1)
31
Lukas Fleischerbafab422013-03-04 08:52:33 +010032static void count_lines(char *line, int size)
Lars Hjemli80e577c2007-05-13 17:03:27 +020033{
Lars Hjemlid04c4732007-11-06 00:35:12 +010034 if (size <= 0)
35 return;
36
37 if (line[0] == '+')
38 add_lines++;
39
40 else if (line[0] == '-')
41 rem_lines++;
Lars Hjemli80e577c2007-05-13 17:03:27 +020042}
43
Lukas Fleischerbafab422013-03-04 08:52:33 +010044static void inspect_files(struct diff_filepair *pair)
Lars Hjemli48dc0032007-05-13 11:27:46 +020045{
Lars Hjemlic495cf02009-01-31 10:40:40 +010046 unsigned long old_size = 0;
47 unsigned long new_size = 0;
48 int binary = 0;
49
Lars Hjemli48dc0032007-05-13 11:27:46 +020050 files++;
Lars Hjemlid1f3bbe2008-02-16 13:56:09 +010051 if (ctx.repo->enable_log_linecount)
Christian Hesse11695a52016-09-04 12:38:18 +020052 cgit_diff_files(&pair->one->oid, &pair->two->oid, &old_size,
Johan Herland2cc8b992010-06-24 17:52:57 +020053 &new_size, &binary, 0, ctx.qry.ignorews,
54 count_lines);
Lars Hjemli48dc0032007-05-13 11:27:46 +020055}
56
Lars Hjemlief2dc552009-01-11 12:16:18 +010057void show_commit_decorations(struct commit *commit)
58{
Christian Hesse17838ec2014-12-19 00:28:34 -070059 const struct name_decoration *deco;
Lars Hjemlief2dc552009-01-11 12:16:18 +010060 static char buf[1024];
61
62 buf[sizeof(buf) - 1] = 0;
Christian Hesse17838ec2014-12-19 00:28:34 -070063 deco = get_name_decoration(&commit->object);
Tim Nordell499b2392016-02-26 14:57:30 -060064 if (!deco)
65 return;
John Keeping927060c2013-05-18 15:54:49 +010066 html("<span class='decoration'>");
Lars Hjemlief2dc552009-01-11 12:16:18 +010067 while (deco) {
Christian Hesse0bb34ef2018-06-05 12:46:13 +020068 struct object_id peeled;
69 int is_annotated = 0;
Tim Nordell59d8fa12016-02-26 14:58:41 -060070 strncpy(buf, prettify_refname(deco->name), sizeof(buf) - 1);
71 switch(deco->type) {
72 case DECORATION_NONE:
73 /* If the git-core doesn't recognize it,
74 * don't display anything. */
75 break;
76 case DECORATION_REF_LOCAL:
Johan Herlandafc40722010-06-11 14:50:47 +020077 cgit_log_link(buf, NULL, "branch-deco", buf, NULL,
Tim Nordell59d8fa12016-02-26 14:58:41 -060078 ctx.qry.vpath, 0, NULL, NULL,
79 ctx.qry.showmsg, 0);
80 break;
81 case DECORATION_REF_TAG:
Christian Hesse0bb34ef2018-06-05 12:46:13 +020082 if (!peel_ref(deco->name, &peeled))
83 is_annotated = !oidcmp(&commit->object.oid, &peeled);
84 cgit_tag_link(buf, NULL, is_annotated ? "tag-annotated-deco" : "tag-deco", buf);
Tim Nordell59d8fa12016-02-26 14:58:41 -060085 break;
86 case DECORATION_REF_REMOTE:
Georg Müller0b6a7162012-01-03 15:30:50 +000087 if (!ctx.repo->enable_remote_branches)
Tim Nordell59d8fa12016-02-26 14:58:41 -060088 break;
Lars Hjemlief2dc552009-01-11 12:16:18 +010089 cgit_log_link(buf, NULL, "remote-deco", NULL,
Tim Nordell59d8fa12016-02-26 14:58:41 -060090 oid_to_hex(&commit->object.oid),
91 ctx.qry.vpath, 0, NULL, NULL,
92 ctx.qry.showmsg, 0);
93 break;
94 default:
Lars Hjemlief2dc552009-01-11 12:16:18 +010095 cgit_commit_link(buf, NULL, "deco", ctx.qry.head,
Tim Nordell59d8fa12016-02-26 14:58:41 -060096 oid_to_hex(&commit->object.oid),
97 ctx.qry.vpath);
98 break;
Lars Hjemlief2dc552009-01-11 12:16:18 +010099 }
100 deco = deco->next;
101 }
John Keeping927060c2013-05-18 15:54:49 +0100102 html("</span>");
Lars Hjemlief2dc552009-01-11 12:16:18 +0100103}
104
John Keeping30304d82015-08-12 15:55:28 +0100105static void handle_rename(struct diff_filepair *pair)
106{
107 /*
108 * After we have seen a rename, we generate links to the previous
109 * name of the file so that commit & diff views get fed the path
110 * that is correct for the commit they are showing, avoiding the
111 * need to walk the entire history leading back to every commit we
112 * show in order detect renames.
113 */
114 if (0 != strcmp(ctx.qry.vpath, pair->two->path)) {
115 free(ctx.qry.vpath);
116 ctx.qry.vpath = xstrdup(pair->two->path);
117 }
118 inspect_files(pair);
119}
120
121static int show_commit(struct commit *commit, struct rev_info *revs)
122{
123 struct commit_list *parents = commit->parents;
124 struct commit *parent;
125 int found = 0, saved_fmt;
Christian Hesse1dd53e32018-01-18 09:19:31 +0100126 struct diff_flags saved_flags = revs->diffopt.flags;
John Keeping30304d82015-08-12 15:55:28 +0100127
128 /* Always show if we're not in "follow" mode with a single file. */
129 if (!ctx.qry.follow)
130 return 1;
131
132 /*
133 * In "follow" mode, we don't show merges. This is consistent with
134 * "git log --follow -- <file>".
135 */
136 if (parents && parents->next)
137 return 0;
138
139 /*
140 * If this is the root commit, do what rev_info tells us.
141 */
142 if (!parents)
143 return revs->show_root_diff;
144
145 /* When we get here we have precisely one parent. */
146 parent = parents->item;
John Keepingbaa5ad12016-01-16 11:03:05 +0000147 /* If we can't parse the commit, let print_commit() report an error. */
148 if (parse_commit(parent))
149 return 1;
John Keeping30304d82015-08-12 15:55:28 +0100150
151 files = 0;
152 add_lines = 0;
153 rem_lines = 0;
154
Christian Hesse1dd53e32018-01-18 09:19:31 +0100155 revs->diffopt.flags.recursive = 1;
Christian Hesse255b78f2018-06-04 18:49:28 +0200156 diff_tree_oid(&parent->maybe_tree->object.oid,
157 &commit->maybe_tree->object.oid,
Jeff Smith86a6d352017-08-09 19:02:56 -0500158 "", &revs->diffopt);
John Keeping30304d82015-08-12 15:55:28 +0100159 diffcore_std(&revs->diffopt);
160
161 found = !diff_queue_is_empty();
162 saved_fmt = revs->diffopt.output_format;
163 revs->diffopt.output_format = DIFF_FORMAT_CALLBACK;
164 revs->diffopt.format_callback = cgit_diff_tree_cb;
165 revs->diffopt.format_callback_data = handle_rename;
166 diff_flush(&revs->diffopt);
167 revs->diffopt.output_format = saved_fmt;
168 revs->diffopt.flags = saved_flags;
169
170 lines_counted = 1;
171 return found;
172}
173
Lukas Fleischerbafab422013-03-04 08:52:33 +0100174static void print_commit(struct commit *commit, struct rev_info *revs)
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100175{
Lars Hjemli2101e262006-12-15 18:17:36 +0100176 struct commitinfo *info;
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100177 int columns = revs->graph ? 4 : 3;
Johan Herland9a8d39c2010-11-15 18:39:50 +0100178 struct strbuf graphbuf = STRBUF_INIT;
Johan Herlandad230262010-11-15 18:39:52 +0100179 struct strbuf msgbuf = STRBUF_INIT;
Johan Herland9a8d39c2010-11-15 18:39:50 +0100180
Julius Plenz2e6721e2011-03-10 17:03:22 +0100181 if (ctx.repo->enable_log_filecount)
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100182 columns++;
Julius Plenz2e6721e2011-03-10 17:03:22 +0100183 if (ctx.repo->enable_log_linecount)
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100184 columns++;
Johan Herland9a8d39c2010-11-15 18:39:50 +0100185
186 if (revs->graph) {
187 /* Advance graph until current commit */
188 while (!graph_next_line(revs->graph, &graphbuf)) {
189 /* Print graph segment in otherwise empty table row */
Johan Herlandc2bfd402010-11-15 18:39:53 +0100190 html("<tr class='nohover'><td class='commitgraph'>");
Johan Herland9a8d39c2010-11-15 18:39:50 +0100191 html(graphbuf.buf);
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100192 htmlf("</td><td colspan='%d' /></tr>\n", columns);
Johan Herland9a8d39c2010-11-15 18:39:50 +0100193 strbuf_setlen(&graphbuf, 0);
194 }
195 /* Current commit's graph segment is now ready in graphbuf */
196 }
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100197
Lars Hjemli2101e262006-12-15 18:17:36 +0100198 info = cgit_parse_commit(commit);
Johan Herlandc2bfd402010-11-15 18:39:53 +0100199 htmlf("<tr%s>", ctx.qry.showmsg ? " class='logheader'" : "");
Johan Herland9a8d39c2010-11-15 18:39:50 +0100200
201 if (revs->graph) {
202 /* Print graph segment for current commit */
203 html("<td class='commitgraph'>");
204 html(graphbuf.buf);
205 html("</td>");
206 strbuf_setlen(&graphbuf, 0);
207 }
Johan Herlandc2bfd402010-11-15 18:39:53 +0100208 else {
209 html("<td>");
John Keepingf2a901d2016-01-19 19:33:05 +0000210 cgit_print_age(info->committer_date, info->committer_tz, TM_WEEK * 2);
Johan Herlandc2bfd402010-11-15 18:39:53 +0100211 html("</td>");
212 }
Johan Herland9a8d39c2010-11-15 18:39:50 +0100213
214 htmlf("<td%s>", ctx.qry.showmsg ? " class='logsubject'" : "");
Johan Herlandad230262010-11-15 18:39:52 +0100215 if (ctx.qry.showmsg) {
216 /* line-wrap long commit subjects instead of truncating them */
217 size_t subject_len = strlen(info->subject);
218
219 if (subject_len > ctx.cfg.max_msg_len &&
220 ctx.cfg.max_msg_len >= 15) {
221 /* symbol for signaling line-wrap (in PAGE_ENCODING) */
222 const char wrap_symbol[] = { ' ', 0xE2, 0x86, 0xB5, 0 };
223 int i = ctx.cfg.max_msg_len - strlen(wrap_symbol);
224
225 /* Rewind i to preceding space character */
226 while (i > 0 && !isspace(info->subject[i]))
227 --i;
228 if (!i) /* Oops, zero spaces. Reset i */
229 i = ctx.cfg.max_msg_len - strlen(wrap_symbol);
230
231 /* add remainder starting at i to msgbuf */
232 strbuf_add(&msgbuf, info->subject + i, subject_len - i);
233 strbuf_trim(&msgbuf);
234 strbuf_add(&msgbuf, "\n\n", 2);
235
236 /* Place wrap_symbol at position i in info->subject */
237 strcpy(info->subject + i, wrap_symbol);
238 }
239 }
Lars Hjemlid14d77f2008-02-16 11:53:40 +0100240 cgit_commit_link(info->subject, NULL, NULL, ctx.qry.head,
Christian Hesse559ab5e2016-01-05 07:38:53 +0100241 oid_to_hex(&commit->object.oid), ctx.qry.vpath);
Lars Hjemlief2dc552009-01-11 12:16:18 +0100242 show_commit_decorations(commit);
Lars Hjemli5764fe92008-04-14 22:13:38 +0200243 html("</td><td>");
Jason A. Donenfeld786609b2014-01-13 16:24:40 +0100244 cgit_open_filter(ctx.repo->email_filter, info->author_email, "log");
Lars Hjemli5764fe92008-04-14 22:13:38 +0200245 html_txt(info->author);
Jason A. Donenfelda5e15532014-01-13 04:04:52 +0100246 cgit_close_filter(ctx.repo->email_filter);
Johan Herlandc2bfd402010-11-15 18:39:53 +0100247
248 if (revs->graph) {
249 html("</td><td>");
John Keepingf2a901d2016-01-19 19:33:05 +0000250 cgit_print_age(info->committer_date, info->committer_tz, TM_WEEK * 2);
Johan Herlandc2bfd402010-11-15 18:39:53 +0100251 }
252
John Keeping30304d82015-08-12 15:55:28 +0100253 if (!lines_counted && (ctx.repo->enable_log_filecount ||
254 ctx.repo->enable_log_linecount)) {
Lars Hjemlie1893442007-05-18 13:55:52 +0200255 files = 0;
Lars Hjemlid04c4732007-11-06 00:35:12 +0100256 add_lines = 0;
257 rem_lines = 0;
Johan Herland1415f3f2010-09-30 20:15:14 +0200258 cgit_diff_commit(commit, inspect_files, ctx.qry.vpath);
Lars Hjemlie1893442007-05-18 13:55:52 +0200259 }
Julius Plenz2e6721e2011-03-10 17:03:22 +0100260
261 if (ctx.repo->enable_log_filecount)
262 htmlf("</td><td>%d", files);
263 if (ctx.repo->enable_log_linecount)
Christian Hesse4fb49862016-06-29 09:37:57 +0200264 htmlf("</td><td><span class='deletions'>-%d</span>/"
265 "<span class='insertions'>+%d</span>", rem_lines, add_lines);
Julius Plenz2e6721e2011-03-10 17:03:22 +0100266
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100267 html("</td></tr>\n");
Johan Herland5a36c2a2010-11-15 18:39:49 +0100268
Christian Hesse81509a22016-07-06 22:42:36 +0200269 if ((revs->graph && !graph_is_commit_finished(revs->graph))
270 || ctx.qry.showmsg) { /* Print a second table row */
271 html("<tr class='nohover-highlight'>");
Johan Herland9a8d39c2010-11-15 18:39:50 +0100272
273 if (ctx.qry.showmsg) {
274 /* Concatenate commit message + notes in msgbuf */
275 if (info->msg && *(info->msg)) {
276 strbuf_addstr(&msgbuf, info->msg);
277 strbuf_addch(&msgbuf, '\n');
278 }
Jeff Smith86a6d352017-08-09 19:02:56 -0500279 format_display_notes(&commit->object.oid,
John Keeping849ecd92013-04-08 09:00:22 +0100280 &msgbuf, PAGE_ENCODING, 0);
Johan Herland5a36c2a2010-11-15 18:39:49 +0100281 strbuf_addch(&msgbuf, '\n');
Johan Herland9a8d39c2010-11-15 18:39:50 +0100282 strbuf_ltrim(&msgbuf);
Lars Hjemli0274b572008-11-29 18:39:41 +0100283 }
Johan Herland5a36c2a2010-11-15 18:39:49 +0100284
Johan Herland9a8d39c2010-11-15 18:39:50 +0100285 if (revs->graph) {
286 int lines = 0;
287
288 /* Calculate graph padding */
289 if (ctx.qry.showmsg) {
290 /* Count #lines in commit message + notes */
291 const char *p = msgbuf.buf;
292 lines = 1;
293 while ((p = strchr(p, '\n'))) {
294 p++;
295 lines++;
296 }
297 }
298
299 /* Print graph padding */
300 html("<td class='commitgraph'>");
301 while (lines > 0 || !graph_is_commit_finished(revs->graph)) {
302 if (graphbuf.len)
303 html("\n");
304 strbuf_setlen(&graphbuf, 0);
305 graph_next_line(revs->graph, &graphbuf);
306 html(graphbuf.buf);
307 lines--;
308 }
309 html("</td>\n");
310 }
Johan Herlandc2bfd402010-11-15 18:39:53 +0100311 else
312 html("<td/>"); /* Empty 'Age' column */
Johan Herland9a8d39c2010-11-15 18:39:50 +0100313
314 /* Print msgbuf into remainder of table row */
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100315 htmlf("<td colspan='%d'%s>\n", columns - (revs->graph ? 1 : 0),
Johan Herland9a8d39c2010-11-15 18:39:50 +0100316 ctx.qry.showmsg ? " class='logmsg'" : "");
Johan Herland5a36c2a2010-11-15 18:39:49 +0100317 html_txt(msgbuf.buf);
Lars Hjemliab671642008-11-29 19:11:26 +0100318 html("</td></tr>\n");
Lars Hjemli0274b572008-11-29 18:39:41 +0100319 }
Johan Herland5a36c2a2010-11-15 18:39:49 +0100320
Johan Herlandad230262010-11-15 18:39:52 +0100321 strbuf_release(&msgbuf);
Johan Herland9a8d39c2010-11-15 18:39:50 +0100322 strbuf_release(&graphbuf);
Lars Hjemliaaa24bd2006-12-16 14:58:20 +0100323 cgit_free_commitinfo(info);
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100324}
325
John Keepingfb3655d2013-04-06 10:28:57 +0100326static const char *disambiguate_ref(const char *ref, int *must_free_result)
Lars Hjemlia1b01b22008-12-03 17:34:23 +0100327{
Christian Hessede83de22015-07-28 10:42:01 +0200328 struct object_id oid;
John Keepingfb3655d2013-04-06 10:28:57 +0100329 struct strbuf longref = STRBUF_INIT;
Lars Hjemlia1b01b22008-12-03 17:34:23 +0100330
John Keepingfb3655d2013-04-06 10:28:57 +0100331 strbuf_addf(&longref, "refs/heads/%s", ref);
Christian Hesse1a9a75d2016-09-29 21:44:41 +0200332 if (get_oid(longref.buf, &oid) == 0) {
John Keepingfb3655d2013-04-06 10:28:57 +0100333 *must_free_result = 1;
334 return strbuf_detach(&longref, NULL);
335 }
Lars Hjemlia1b01b22008-12-03 17:34:23 +0100336
John Keepingfb3655d2013-04-06 10:28:57 +0100337 *must_free_result = 0;
338 strbuf_release(&longref);
Lars Hjemlia1b01b22008-12-03 17:34:23 +0100339 return ref;
340}
Lars Hjemli2101e262006-12-15 18:17:36 +0100341
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100342static char *next_token(char **src)
343{
344 char *result;
345
346 if (!src || !*src)
347 return NULL;
348 while (isspace(**src))
349 (*src)++;
350 if (!**src)
351 return NULL;
352 result = *src;
353 while (**src) {
354 if (isspace(**src)) {
355 **src = '\0';
356 (*src)++;
357 break;
358 }
359 (*src)++;
360 }
361 return result;
362}
363
Lars Hjemli5764fe92008-04-14 22:13:38 +0200364void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern,
Tobias Bieniek792f8132012-10-13 16:10:30 +0200365 char *path, int pager, int commit_graph, int commit_sort)
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100366{
367 struct rev_info rev;
368 struct commit *commit;
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100369 struct argv_array rev_argv = ARGV_ARRAY_INIT;
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100370 int i, columns = commit_graph ? 4 : 3;
John Keepingfb3655d2013-04-06 10:28:57 +0100371 int must_free_tip = 0;
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100372
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100373 /* rev_argv.argv[0] will be ignored by setup_revisions */
374 argv_array_push(&rev_argv, "log_rev_setup");
Lars Hjemli48dc0032007-05-13 11:27:46 +0200375
Lars Hjemlicd79c162007-06-17 14:58:45 +0200376 if (!tip)
Lars Hjemlia1b01b22008-12-03 17:34:23 +0100377 tip = ctx.qry.head;
John Keepingfb3655d2013-04-06 10:28:57 +0100378 tip = disambiguate_ref(tip, &must_free_tip);
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100379 argv_array_push(&rev_argv, tip);
Lars Hjemlicd79c162007-06-17 14:58:45 +0200380
Johan Herland52558a62010-10-28 17:05:39 +0200381 if (grep && pattern && *pattern) {
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100382 pattern = xstrdup(pattern);
Lars Hjemlia579fb02010-06-19 14:32:37 +0200383 if (!strcmp(grep, "grep") || !strcmp(grep, "author") ||
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100384 !strcmp(grep, "committer")) {
Lukas Fleischerd76c4382013-11-22 13:30:58 +0100385 argv_array_pushf(&rev_argv, "--%s=%s", grep, pattern);
386 } else if (!strcmp(grep, "range")) {
John Keepingfb3655d2013-04-06 10:28:57 +0100387 char *arg;
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100388 /* Split the pattern at whitespace and add each token
389 * as a revision expression. Do not accept other
390 * rev-list options. Also, replace the previously
391 * pushed tip (it's no longer relevant).
392 */
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100393 argv_array_pop(&rev_argv);
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100394 while ((arg = next_token(&pattern))) {
395 if (*arg == '-') {
396 fprintf(stderr, "Bad range expr: %s\n",
397 arg);
398 break;
399 }
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100400 argv_array_push(&rev_argv, arg);
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100401 }
402 }
Lars Hjemlia579fb02010-06-19 14:32:37 +0200403 }
John Keeping30304d82015-08-12 15:55:28 +0100404
405 if (!path || !ctx.cfg.enable_follow_links) {
406 /*
407 * If we don't have a path, "follow" is a no-op so make sure
408 * the variable is set to false to avoid needing to check
409 * both this and whether we have a path everywhere.
410 */
411 ctx.qry.follow = 0;
412 }
413
414 if (commit_graph && !ctx.qry.follow) {
Lukas Fleischerd76c4382013-11-22 13:30:58 +0100415 argv_array_push(&rev_argv, "--graph");
416 argv_array_push(&rev_argv, "--color");
Johan Herland268b34a2010-11-15 18:39:51 +0100417 graph_set_column_colors(column_colors_html,
418 COLUMN_COLORS_HTML_MAX);
Johan Herland9a8d39c2010-11-15 18:39:50 +0100419 }
Lars Hjemli68ca0322007-10-28 15:23:00 +0100420
Lukas Fleischerd76c4382013-11-22 13:30:58 +0100421 if (commit_sort == 1)
422 argv_array_push(&rev_argv, "--date-order");
423 else if (commit_sort == 2)
424 argv_array_push(&rev_argv, "--topo-order");
Tobias Bieniek792f8132012-10-13 16:10:30 +0200425
John Keeping30304d82015-08-12 15:55:28 +0100426 if (path && ctx.qry.follow)
427 argv_array_push(&rev_argv, "--follow");
John Keeping4046e8e2014-06-28 15:55:06 +0200428 argv_array_push(&rev_argv, "--");
429 if (path)
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100430 argv_array_push(&rev_argv, path);
Lars Hjemlia3c3c042010-11-09 20:53:36 +0100431
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100432 init_revisions(&rev, NULL);
433 rev.abbrev = DEFAULT_ABBREV;
434 rev.commit_format = CMIT_FMT_DEFAULT;
435 rev.verbose_header = 1;
436 rev.show_root_diff = 0;
John Keeping4046e8e2014-06-28 15:55:06 +0200437 rev.ignore_missing = 1;
John Keeping30304d82015-08-12 15:55:28 +0100438 rev.simplify_history = 1;
Lukas Fleischer9973ef02013-11-22 13:24:52 +0100439 setup_revisions(rev_argv.argc, rev_argv.argv, &rev, NULL);
Christian Hesse1dd53e32018-01-18 09:19:31 +0100440 load_ref_decorations(NULL, DECORATE_FULL_REFS);
Lars Hjemlief2dc552009-01-11 12:16:18 +0100441 rev.show_decorations = 1;
Jeff Smith86a6d352017-08-09 19:02:56 -0500442 rev.grep_filter.ignore_case = 1;
John Keeping30304d82015-08-12 15:55:28 +0100443
444 rev.diffopt.detect_rename = 1;
445 rev.diffopt.rename_limit = ctx.cfg.renamelimit;
446 if (ctx.qry.ignorews)
447 DIFF_XDL_SET(&rev.diffopt, IGNORE_WHITESPACE);
448
Lars Hjemlib7f33782008-10-05 19:19:59 +0200449 compile_grep_patterns(&rev.grep_filter);
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100450 prepare_revision_walk(&rev);
451
John Keeping23c17d82015-08-14 12:47:15 +0100452 if (pager) {
453 cgit_print_layout_start();
Lars Hjemli5764fe92008-04-14 22:13:38 +0200454 html("<table class='list nowrap'>");
John Keeping23c17d82015-08-14 12:47:15 +0100455 }
Lars Hjemlie1893442007-05-18 13:55:52 +0200456
Johan Herlandc2bfd402010-11-15 18:39:53 +0100457 html("<tr class='nohover'>");
Johan Herlandad230262010-11-15 18:39:52 +0100458 if (commit_graph)
Johan Herland9a8d39c2010-11-15 18:39:50 +0100459 html("<th></th>");
Johan Herlandc2bfd402010-11-15 18:39:53 +0100460 else
461 html("<th class='left'>Age</th>");
Johan Herland9a8d39c2010-11-15 18:39:50 +0100462 html("<th class='left'>Commit message");
Lars Hjemli0274b572008-11-29 18:39:41 +0100463 if (pager) {
464 html(" (");
Justin Waters1383fe32009-01-09 17:35:10 -0500465 cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL,
466 NULL, ctx.qry.head, ctx.qry.sha1,
Johan Herlandafc40722010-06-11 14:50:47 +0200467 ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep,
John Keeping30304d82015-08-12 15:55:28 +0100468 ctx.qry.search, ctx.qry.showmsg ? 0 : 1,
469 ctx.qry.follow);
Lars Hjemli0274b572008-11-29 18:39:41 +0100470 html(")");
471 }
472 html("</th><th class='left'>Author</th>");
John Keeping30304d82015-08-12 15:55:28 +0100473 if (rev.graph)
Johan Herlandc2bfd402010-11-15 18:39:53 +0100474 html("<th class='left'>Age</th>");
Lars Hjemlid1f3bbe2008-02-16 13:56:09 +0100475 if (ctx.repo->enable_log_filecount) {
Lars Hjemli5764fe92008-04-14 22:13:38 +0200476 html("<th class='left'>Files</th>");
477 columns++;
Julius Plenz2e6721e2011-03-10 17:03:22 +0100478 }
479 if (ctx.repo->enable_log_linecount) {
480 html("<th class='left'>Lines</th>");
481 columns++;
Lars Hjemlie1893442007-05-18 13:55:52 +0200482 }
Lars Hjemli5764fe92008-04-14 22:13:38 +0200483 html("</tr>\n");
Lars Hjemli420712a2006-12-14 00:40:34 +0100484
485 if (ofs<0)
486 ofs = 0;
487
John Keepingda1d4c72015-08-12 16:41:34 +0100488 for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; /* nop */) {
John Keeping30304d82015-08-12 15:55:28 +0100489 if (show_commit(commit, &rev))
490 i++;
John Keeping865afe02014-07-27 11:56:19 +0100491 free_commit_buffer(commit);
Lars Hjemli420712a2006-12-14 00:40:34 +0100492 free_commit_list(commit->parents);
493 commit->parents = NULL;
494 }
495
John Keepingda1d4c72015-08-12 16:41:34 +0100496 for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; /* nop */) {
John Keeping30304d82015-08-12 15:55:28 +0100497 /*
498 * In "follow" mode, we must count the files and lines the
499 * first time we invoke diff on a given commit, and we need
500 * to do that to see if the commit touches the path we care
501 * about, so we do it in show_commit. Hence we must clear
502 * lines_counted here.
503 *
504 * This has the side effect of avoiding running diff twice
505 * when we are both following renames and showing file
506 * and/or line counts.
507 */
508 lines_counted = 0;
509 if (show_commit(commit, &rev)) {
510 i++;
511 print_commit(commit, &rev);
512 }
John Keeping865afe02014-07-27 11:56:19 +0100513 free_commit_buffer(commit);
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100514 free_commit_list(commit->parents);
515 commit->parents = NULL;
516 }
Ondrej Jirmana9226152007-05-26 03:26:14 +0200517 if (pager) {
Lukas Fleischerb60e6bf2013-03-07 08:56:22 +0100518 html("</table><ul class='pager'>");
Ondrej Jirmana9226152007-05-26 03:26:14 +0200519 if (ofs > 0) {
Lukas Fleischerb60e6bf2013-03-07 08:56:22 +0100520 html("<li>");
Lars Hjemlid14d77f2008-02-16 11:53:40 +0100521 cgit_log_link("[prev]", NULL, NULL, ctx.qry.head,
Johan Herlandafc40722010-06-11 14:50:47 +0200522 ctx.qry.sha1, ctx.qry.vpath,
Lars Hjemlid14d77f2008-02-16 11:53:40 +0100523 ofs - cnt, ctx.qry.grep,
John Keeping30304d82015-08-12 15:55:28 +0100524 ctx.qry.search, ctx.qry.showmsg,
525 ctx.qry.follow);
Lukas Fleischerb60e6bf2013-03-07 08:56:22 +0100526 html("</li>");
Ondrej Jirmana9226152007-05-26 03:26:14 +0200527 }
Ondrej Jirmana9226152007-05-26 03:26:14 +0200528 if ((commit = get_revision(&rev)) != NULL) {
Lukas Fleischerb60e6bf2013-03-07 08:56:22 +0100529 html("<li>");
Lars Hjemlid14d77f2008-02-16 11:53:40 +0100530 cgit_log_link("[next]", NULL, NULL, ctx.qry.head,
Johan Herlandafc40722010-06-11 14:50:47 +0200531 ctx.qry.sha1, ctx.qry.vpath,
Lars Hjemlid14d77f2008-02-16 11:53:40 +0100532 ofs + cnt, ctx.qry.grep,
John Keeping30304d82015-08-12 15:55:28 +0100533 ctx.qry.search, ctx.qry.showmsg,
534 ctx.qry.follow);
Lukas Fleischerb60e6bf2013-03-07 08:56:22 +0100535 html("</li>");
Ondrej Jirmana9226152007-05-26 03:26:14 +0200536 }
Lukas Fleischerb60e6bf2013-03-07 08:56:22 +0100537 html("</ul>");
John Keeping23c17d82015-08-14 12:47:15 +0100538 cgit_print_layout_end();
Lars Hjemli5764fe92008-04-14 22:13:38 +0200539 } else if ((commit = get_revision(&rev)) != NULL) {
Lukas Fleischeref8a97d2013-03-05 15:42:14 +0100540 htmlf("<tr class='nohover'><td colspan='%d'>", columns);
Johan Herlandafc40722010-06-11 14:50:47 +0200541 cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL,
John Keeping30304d82015-08-12 15:55:28 +0100542 ctx.qry.vpath, 0, NULL, NULL, ctx.qry.showmsg,
543 ctx.qry.follow);
Lars Hjemli5764fe92008-04-14 22:13:38 +0200544 html("</td></tr>\n");
Lars Hjemli420712a2006-12-14 00:40:34 +0100545 }
John Keepingfb3655d2013-04-06 10:28:57 +0100546
547 /* If we allocated tip then it is safe to cast away const. */
548 if (must_free_tip)
549 free((char*) tip);
Lars Hjemlid14c5f62006-12-11 17:04:19 +0100550}