blame: allow restriction to line range
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
diff --git a/src/blame.c b/src/blame.c
index 33477d6..a4ac84f 100644
--- a/src/blame.c
+++ b/src/blame.c
@@ -179,6 +179,10 @@ static void normalize_options(
if (git_oid_iszero(&out->newest_commit)) {
git_reference_name_to_id(&out->newest_commit, repo, "HEAD");
}
+
+ /* min_line 0 really means 1 */
+ if (!out->min_line) out->min_line = 1;
+ /* max_line 0 really means N, but we don't know N yet */
}
static git_blame_hunk *split_hunk_in_vector(
@@ -265,7 +269,14 @@ static int walk_and_mark(git_blame *blame)
ent = git__calloc(1, sizeof(*ent));
ent->num_lines = prepare_lines(&sb);
+ ent->lno = blame->options.min_line - 1;
+ ent->num_lines = ent->num_lines - blame->options.min_line + 1;
+ if (blame->options.max_line > 0) {
+ ent->num_lines = blame->options.max_line - blame->options.min_line + 1;
+ }
+ ent->s_lno = ent->lno;
ent->suspect = o;
+
sb.ent = ent;
sb.path = blame->path;
sb.blame = blame;
diff --git a/tests-clar/blame/simple.c b/tests-clar/blame/simple.c
index 4e2d886..1de4a51 100644
--- a/tests-clar/blame/simple.c
+++ b/tests-clar/blame/simple.c
@@ -201,4 +201,72 @@ void test_blame_simple__trivial_libgit2(void)
git_repository_free(repo);
}
+
+/*
+ * $ git blame -n b.txt -L 8
+ * orig line no final line no
+ * commit V author timestamp V
+ * 63d671eb 8 (Ben Straub 2013-02-12 15:13:04 -0800 8
+ * 63d671eb 9 (Ben Straub 2013-02-12 15:13:04 -0800 9
+ * 63d671eb 10 (Ben Straub 2013-02-12 15:13:04 -0800 10
+ * aa06ecca 6 (Ben Straub 2013-02-12 15:14:46 -0800 11
+ * aa06ecca 7 (Ben Straub 2013-02-12 15:14:46 -0800 12
+ * aa06ecca 8 (Ben Straub 2013-02-12 15:14:46 -0800 13
+ * aa06ecca 9 (Ben Straub 2013-02-12 15:14:46 -0800 14
+ * aa06ecca 10 (Ben Straub 2013-02-12 15:14:46 -0800 15
+ *
+ * $ git blame -n b.txt -L ,6
+ * orig line no final line no
+ * commit V author timestamp V
+ * da237394 1 (Ben Straub 2013-02-12 15:11:30 -0800 1
+ * da237394 2 (Ben Straub 2013-02-12 15:11:30 -0800 2
+ * da237394 3 (Ben Straub 2013-02-12 15:11:30 -0800 3
+ * da237394 4 (Ben Straub 2013-02-12 15:11:30 -0800 4
+ * ^b99f7ac 1 (Ben Straub 2013-02-12 15:10:12 -0800 5
+ * 63d671eb 6 (Ben Straub 2013-02-12 15:13:04 -0800 6
+ *
+ * $ git blame -n b.txt -L 2,7
+ * orig line no final line no
+ * commit V author timestamp V
+ * da237394 2 (Ben Straub 2013-02-12 15:11:30 -0800 2
+ * da237394 3 (Ben Straub 2013-02-12 15:11:30 -0800 3
+ * da237394 4 (Ben Straub 2013-02-12 15:11:30 -0800 4
+ * ^b99f7ac 1 (Ben Straub 2013-02-12 15:10:12 -0800 5
+ * 63d671eb 6 (Ben Straub 2013-02-12 15:13:04 -0800 6
+ * 63d671eb 7 (Ben Straub 2013-02-12 15:13:04 -0800 7
+ */
+void test_blame_simple__can_restrict_to_lines(void)
+{
+ git_blame *blame = NULL;
+ git_repository *repo;
+ git_blame_options opts = GIT_BLAME_OPTIONS_INIT;
+
+ cl_git_pass(git_repository_open(&repo, cl_fixture("blametest.git")));
+
+ opts.min_line = 8;
+ cl_git_pass(git_blame_file(&blame, repo, "b.txt", &opts));
+ cl_assert_equal_i(2, git_blame_get_hunk_count(blame));
+ check_blame_hunk_index(repo, blame, 0, 8, 3, "63d671eb", "b.txt");
+ check_blame_hunk_index(repo, blame, 1, 11, 5, "aa06ecca", "b.txt");
+
+ opts.min_line = 0;
+ opts.max_line = 6;
+ cl_git_pass(git_blame_file(&blame, repo, "b.txt", &opts));
+ cl_assert_equal_i(3, git_blame_get_hunk_count(blame));
+ check_blame_hunk_index(repo, blame, 0, 1, 4, "da237394", "b.txt");
+ check_blame_hunk_index(repo, blame, 1, 5, 1, "b99f7ac0", "b.txt");
+ check_blame_hunk_index(repo, blame, 2, 6, 1, "63d671eb", "b.txt");
+
+ opts.min_line = 2;
+ opts.max_line = 7;
+ cl_git_pass(git_blame_file(&blame, repo, "b.txt", &opts));
+ cl_assert_equal_i(3, git_blame_get_hunk_count(blame));
+ check_blame_hunk_index(repo, blame, 0, 2, 3, "da237394", "b.txt");
+ check_blame_hunk_index(repo, blame, 1, 5, 1, "b99f7ac0", "b.txt");
+ check_blame_hunk_index(repo, blame, 2, 6, 2, "63d671eb", "b.txt");
+
+ git_blame_free(blame);
+ git_repository_free(repo);
+}
+
/* TODO: no newline at end of file? */