From mboxrd@z Thu Jan 1 00:00:00 1970 From: lists at necoro.eu (=?UTF-8?B?UmVuw6kgTmV1bWFubg==?=) Date: Thu, 18 Apr 2013 19:31:55 +0200 Subject: [PATCH] log: allow users to follow a file In-Reply-To: <7acf47c185acc0aa326be65d69320bc566e18cc2.1366227978.git.john@keeping.me.uk> References: <516BE9CF.3060809@necoro.eu> <20130415123845.GF2278@serenity.lan> <516C3923.3060700@necoro.eu> <20130416084834.GL2278@serenity.lan> <7acf47c185acc0aa326be65d69320bc566e18cc2.1366227978.git.john@keeping.me.uk> Message-ID: <51702E0B.7030107@necoro.eu> Thanks a lot for your patch, it is a huge step in the right direction (it is way more involved than I could imagine). One issue remains: It still displays empty diffstats (example: see [1]) for the commits _before_ the rename, as they are limited to the _new_ name. This might even lead to wrong commits if you have a commit "A -> B, C -> A" and are looking at the history of the new file A. > Suggested-by: Ren? Neumann Could you change the mail address to 'necoro at necoro.eu'? - Ren? [1] https://git.necoro.eu/web/kosten.git/log/templates/layout.jinja?follow=1 > Signed-off-by: John Keeping > --- > cgit.c | 4 +++ > cgit.h | 2 ++ > cgitrc.5.txt | 4 +++ > ui-log.c | 87 ++++++++++++++++++++++++++++++++++++++++++++++++++++------- > ui-refs.c | 2 +- > ui-repolist.c | 2 +- > ui-shared.c | 20 +++++++++++--- > ui-shared.h | 2 +- > ui-tree.c | 2 +- > 9 files changed, 108 insertions(+), 17 deletions(-) > > diff --git a/cgit.c b/cgit.c > index 6f44ef2..81312bb 100644 > --- a/cgit.c > +++ b/cgit.c > @@ -171,6 +171,8 @@ static void config_cb(const char *name, const char *value) > ctx.cfg.snapshots = cgit_parse_snapshots_mask(value); > else if (!strcmp(name, "enable-filter-overrides")) > ctx.cfg.enable_filter_overrides = atoi(value); > + else if (!strcmp(name, "enable-follow-links")) > + ctx.cfg.enable_follow_links = atoi(value); > else if (!strcmp(name, "enable-http-clone")) > ctx.cfg.enable_http_clone = atoi(value); > else if (!strcmp(name, "enable-index-links")) > @@ -338,6 +340,8 @@ static void querystring_cb(const char *name, const char *value) > ctx.qry.context = atoi(value); > } else if (!strcmp(name, "ignorews")) { > ctx.qry.ignorews = atoi(value); > + } else if (!strcmp(name, "follow")) { > + ctx.qry.follow = atoi(value); > } > } > > diff --git a/cgit.h b/cgit.h > index 850b792..6c0c429 100644 > --- a/cgit.h > +++ b/cgit.h > @@ -163,6 +163,7 @@ struct cgit_query { > int show_all; > int context; > int ignorews; > + int follow; > char *vpath; > }; > > @@ -203,6 +204,7 @@ struct cgit_config { > int case_sensitive_sort; > int embedded; > int enable_filter_overrides; > + int enable_follow_links; > int enable_http_clone; > int enable_index_links; > int enable_index_owner; > diff --git a/cgitrc.5.txt b/cgitrc.5.txt > index 39b031e..48ef249 100644 > --- a/cgitrc.5.txt > +++ b/cgitrc.5.txt > @@ -121,6 +121,10 @@ enable-filter-overrides:: > Flag which, when set to "1", allows all filter settings to be > overridden in repository-specific cgitrc files. Default value: none. > > +enable-follow-links:: > + Flag which, when set to "1", allows users to follow a file in the log > + view. Default value: "0". > + > enable-http-clone:: > If set to "1", cgit will act as an dumb HTTP endpoint for git clones. > If you use an alternate way of serving git repositories, you may wish > diff --git a/ui-log.c b/ui-log.c > index 2aa12c3..d008387 100644 > --- a/ui-log.c > +++ b/ui-log.c > @@ -66,7 +66,7 @@ void show_commit_decorations(struct commit *commit) > strncpy(buf, deco->name + 11, sizeof(buf) - 1); > cgit_log_link(buf, NULL, "branch-deco", buf, NULL, > ctx.qry.vpath, 0, NULL, NULL, > - ctx.qry.showmsg); > + ctx.qry.showmsg, 0); > } > else if (!prefixcmp(deco->name, "tag: refs/tags/")) { > strncpy(buf, deco->name + 15, sizeof(buf) - 1); > @@ -83,7 +83,7 @@ void show_commit_decorations(struct commit *commit) > cgit_log_link(buf, NULL, "remote-deco", NULL, > sha1_to_hex(commit->object.sha1), > ctx.qry.vpath, 0, NULL, NULL, > - ctx.qry.showmsg); > + ctx.qry.showmsg, 0); > } > else { > strncpy(buf, deco->name, sizeof(buf) - 1); > @@ -96,6 +96,50 @@ next: > } > } > > +static int show_commit(struct commit *commit, struct rev_info *revs) > +{ > + struct commit_list *parents = commit->parents; > + struct commit *parent; > + int found = 0, saved_fmt; > + unsigned saved_flags = revs->diffopt.flags; > + > + > + /* Always show if we're not in "follow" mode with a single file. */ > + if (!ctx.qry.follow) > + return 1; > + > + /* > + * In "follow" mode, we don't show merges. This is consistent with > + * "git log --follow -- ". > + */ > + if (parents && parents->next) > + return 0; > + > + /* > + * If this is the root commit, do what rev_info tells us. > + */ > + if (!parents) > + return revs->show_root_diff; > + > + /* When we get here we have precisely one parent. */ > + parent = parents->item; > + parse_commit(parent); > + DIFF_OPT_SET(&revs->diffopt, RECURSIVE); > + diff_tree_sha1(parent->tree->object.sha1, > + commit->tree->object.sha1, > + "", &revs->diffopt); > + diffcore_std(&revs->diffopt); > + > + found = !diff_queue_is_empty(); > + saved_fmt = revs->diffopt.output_format; > + revs->diffopt.output_format = DIFF_FORMAT_NO_OUTPUT; > + diff_flush(&revs->diffopt); > + revs->diffopt.output_format = saved_fmt; > + revs->diffopt.flags = saved_flags; > + > + return found; > +} > + > static void print_commit(struct commit *commit, struct rev_info *revs) > { > struct commitinfo *info; > @@ -324,7 +368,17 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > } > } > } > - if (commit_graph) { > + > + if (!path || !ctx.cfg.enable_follow_links) { > + /* > + * If we don't have a path, "follow" is a no-op so make sure > + * the variable is set to false to avoid needing to check > + * both this and whether we have a path everywhere. > + */ > + ctx.qry.follow = 0; > + } > + > + if (commit_graph && !ctx.qry.follow) { > static const char *graph_arg = "--graph"; > static const char *color_arg = "--color"; > vector_push(&vec, &graph_arg, 0); > @@ -342,7 +396,10 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > } > > if (path) { > + static const char *follow_arg = "--follow"; > static const char *double_dash_arg = "--"; > + if (ctx.qry.follow) > + vector_push(&vec, &follow_arg, 0); > vector_push(&vec, &double_dash_arg, 0); > vector_push(&vec, &path, 0); > } > @@ -356,6 +413,7 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > rev.commit_format = CMIT_FMT_DEFAULT; > rev.verbose_header = 1; > rev.show_root_diff = 0; > + rev.simplify_history = 1; > setup_revisions(vec.count, vec.data, &rev, NULL); > load_ref_decorations(DECORATE_FULL_REFS); > rev.show_decorations = 1; > @@ -377,7 +435,8 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL, > NULL, ctx.qry.head, ctx.qry.sha1, > ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep, > - ctx.qry.search, ctx.qry.showmsg ? 0 : 1); > + ctx.qry.search, ctx.qry.showmsg ? 0 : 1, > + ctx.qry.follow); > html(")"); > } > html("Author"); > @@ -396,15 +455,20 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > if (ofs<0) > ofs = 0; > > - for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; i++) { > + for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL;) { > + if (show_commit(commit, &rev)) > + i++; > free(commit->buffer); > commit->buffer = NULL; > free_commit_list(commit->parents); > commit->parents = NULL; > } > > - for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; i++) { > - print_commit(commit, &rev); > + for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL;) { > + if (show_commit(commit, &rev)) { > + i++; > + print_commit(commit, &rev); > + } > free(commit->buffer); > commit->buffer = NULL; > free_commit_list(commit->parents); > @@ -417,7 +481,8 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > cgit_log_link("[prev]", NULL, NULL, ctx.qry.head, > ctx.qry.sha1, ctx.qry.vpath, > ofs - cnt, ctx.qry.grep, > - ctx.qry.search, ctx.qry.showmsg); > + ctx.qry.search, ctx.qry.showmsg, > + ctx.qry.follow); > html(""); > } > if ((commit = get_revision(&rev)) != NULL) { > @@ -425,14 +490,16 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern > cgit_log_link("[next]", NULL, NULL, ctx.qry.head, > ctx.qry.sha1, ctx.qry.vpath, > ofs + cnt, ctx.qry.grep, > - ctx.qry.search, ctx.qry.showmsg); > + ctx.qry.search, ctx.qry.showmsg, > + ctx.qry.follow); > html(""); > } > html(""); > } else if ((commit = get_revision(&rev)) != NULL) { > htmlf("", columns); > cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL, > - ctx.qry.vpath, 0, NULL, NULL, ctx.qry.showmsg); > + ctx.qry.vpath, 0, NULL, NULL, ctx.qry.showmsg, > + ctx.qry.follow); > html("\n"); > } > > diff --git a/ui-refs.c b/ui-refs.c > index 0ae0612..f4eefd1 100644 > --- a/ui-refs.c > +++ b/ui-refs.c > @@ -71,7 +71,7 @@ static int print_branch(struct refinfo *ref) > return 1; > html(""); > cgit_log_link(name, NULL, NULL, name, NULL, NULL, 0, NULL, NULL, > - ctx.qry.showmsg); > + ctx.qry.showmsg, 0); > html(""); > > if (ref->object->type == OBJ_COMMIT) { > diff --git a/ui-repolist.c b/ui-repolist.c > index 47ca997..d96f252 100644 > --- a/ui-repolist.c > +++ b/ui-repolist.c > @@ -314,7 +314,7 @@ void cgit_print_repolist() > html(""); > cgit_summary_link("summary", NULL, "button", NULL); > cgit_log_link("log", NULL, "button", NULL, NULL, NULL, > - 0, NULL, NULL, ctx.qry.showmsg); > + 0, NULL, NULL, ctx.qry.showmsg, 0); > cgit_tree_link("tree", NULL, "button", NULL, NULL, NULL); > html(""); > } > diff --git a/ui-shared.c b/ui-shared.c > index 519eef7..24560ba 100644 > --- a/ui-shared.c > +++ b/ui-shared.c > @@ -284,7 +284,8 @@ void cgit_plain_link(const char *name, const char *title, const char *class, > > void cgit_log_link(const char *name, const char *title, const char *class, > const char *head, const char *rev, const char *path, > - int ofs, const char *grep, const char *pattern, int showmsg) > + int ofs, const char *grep, const char *pattern, int showmsg, > + int follow) > { > char *delim; > > @@ -313,6 +314,11 @@ void cgit_log_link(const char *name, const char *title, const char *class, > if (showmsg) { > html(delim); > html("showmsg=1"); > + delim = "&"; > + } > + if (follow) { > + html(delim); > + html("follow=1"); > } > html("'>"); > html_txt(name); > @@ -452,7 +458,7 @@ static void cgit_self_link(char *name, const char *title, const char *class, > ctx->qry.has_sha1 ? ctx->qry.sha1 : NULL, > ctx->qry.path, ctx->qry.ofs, > ctx->qry.grep, ctx->qry.search, > - ctx->qry.showmsg); > + ctx->qry.showmsg, ctx->qry.follow); > else if (!strcmp(ctx->qry.page, "commit")) > cgit_commit_link(name, title, class, ctx->qry.head, > ctx->qry.has_sha1 ? ctx->qry.sha1 : NULL, > @@ -854,7 +860,7 @@ void cgit_print_pageheader(struct cgit_context *ctx) > ctx->qry.sha1, NULL); > cgit_log_link("log", NULL, hc(ctx, "log"), ctx->qry.head, > NULL, ctx->qry.vpath, 0, NULL, NULL, > - ctx->qry.showmsg); > + ctx->qry.showmsg, ctx->qry.follow); > cgit_tree_link("tree", NULL, hc(ctx, "tree"), ctx->qry.head, > ctx->qry.sha1, ctx->qry.vpath); > cgit_commit_link("commit", NULL, hc(ctx, "commit"), > @@ -906,6 +912,14 @@ void cgit_print_pageheader(struct cgit_context *ctx) > html("
"); > html("path: "); > cgit_print_path_crumbs(ctx, ctx->qry.vpath); > + if (ctx->cfg.enable_follow_links && !strcmp(ctx->qry.page, "log")) { > + html(" ("); > + ctx->qry.follow = !ctx->qry.follow; > + cgit_self_link(ctx->qry.follow ? "follow" : "unfollow", > + NULL, NULL, ctx); > + ctx->qry.follow = !ctx->qry.follow; > + html(")"); > + } > html("
"); > } > html("
"); > diff --git a/ui-shared.h b/ui-shared.h > index 5987e77..3156846 100644 > --- a/ui-shared.h > +++ b/ui-shared.h > @@ -26,7 +26,7 @@ extern void cgit_plain_link(const char *name, const char *title, > extern void cgit_log_link(const char *name, const char *title, > const char *class, const char *head, const char *rev, > const char *path, int ofs, const char *grep, > - const char *pattern, int showmsg); > + const char *pattern, int showmsg, int follow); > extern void cgit_commit_link(char *name, const char *title, > const char *class, const char *head, > const char *rev, const char *path, > diff --git a/ui-tree.c b/ui-tree.c > index aa5dee9..ebb3e9b 100644 > --- a/ui-tree.c > +++ b/ui-tree.c > @@ -169,7 +169,7 @@ static int ls_item(const unsigned char *sha1, const char *base, int baselen, > html(""); > cgit_log_link("log", NULL, "button", ctx.qry.head, > walk_tree_ctx->curr_rev, fullpath.buf, 0, NULL, NULL, > - ctx.qry.showmsg); > + ctx.qry.showmsg, 0); > if (ctx.repo->max_stats) > cgit_stats_link("stats", NULL, "button", ctx.qry.head, > fullpath.buf); >