Re: [PATCH v3] diff.c: keep arrow(=) on show_stats()'s shortened filename part to make rename visible.

2013-10-15 Thread Yoshioka Tsuneo
Hello Duy

Thank you very much your suggestion.
As you suggested, I tried to reuse intermediate result of pprint_rename(), 
instead of
parsing the output again.
I just posted the new patch as PATCH v4

Thanks !

---
Tsuneo Yoshioka (吉岡 恒夫)
yoshiokatsu...@gmail.com




On Oct 14, 2013, at 10:04 PM, Duy Nguyen pclo...@gmail.com wrote:

 On Sun, Oct 13, 2013 at 3:48 AM, Yoshioka Tsuneo
 yoshiokatsu...@gmail.com wrote:
 git diff -M --stat can detect rename and show renamed file name like
 foofoofoo = barbarbar, but if destination filename is long the line
 is shortened like ...barbarbar so there is no way to know whether the
 file is renamed or existed in the source commit.
 This commit makes it visible like ...foo = ...bar.
 
 Signed-off-by: Tsuneo Yoshioka yoshiokatsu...@gmail.com
 ---
 diff.c | 58 +++---
 1 file changed, 51 insertions(+), 7 deletions(-)
 
 diff --git a/diff.c b/diff.c
 index a04a34d..3aeaf3e 100644
 --- a/diff.c
 +++ b/diff.c
 @@ -1643,13 +1643,57 @@ static void show_stats(struct diffstat_t *data, 
 struct diff_options *options)
len = name_width;
name_len = strlen(name);
if (name_width  name_len) {
 -   char *slash;
 -   prefix = ...;
 -   len -= 3;
 -   name += name_len - len;
 -   slash = strchr(name, '/');
 -   if (slash)
 -   name = slash;
 +   char *arrow = strstr(name,  = );
 +   if (arrow) {
 
 This looks iffy. What if  =  is part of the path name?
 file-is_renamed would be a more reliable sign. In that case I think
 you just need an ellipsis version of pprint_rename() (i.e. drop the
 result of previous pprint_rename() on the floor and create a new
 string with ... and  =  in your pprint_ellipsis_rename or
 something)
 
 +   int prefix_len = (name_width - 4) / 2;
 +   int f_omit;
 +   int f_brace = 0;
 +   char *pre_arrow = alloca(name_width + 10);
 +   char *post_arrow = arrow + 4;
 +   char *prefix_buf = alloca(name_width + 10);
 +   char *pre_arrow_slash = NULL;
 +
 +   if (arrow - name  prefix_len) {
 +   prefix_len = (int)(arrow - name);
 +   f_omit = 0;
 +   } else {
 +   prefix_len -= 3;
 +   f_omit = 1;
 +   if (name[0] == '{') {
 +   prefix_len -= 1;
 +   f_brace = 1;
 +   }
 +   }
 +   prefix_len = ((prefix_len = 0) ? prefix_len 
 : 0);
 +   strncpy(pre_arrow, arrow - prefix_len, 
 prefix_len);
 +   pre_arrow[prefix_len] = '\0';
 +   pre_arrow_slash = strchr(pre_arrow, '/');
 +   if (f_omit  pre_arrow_slash)
 +   pre_arrow = pre_arrow_slash;
 +   sprintf(prefix_buf, %s%s%s = , (f_brace ? 
 { : ), (f_omit ? ... : ), pre_arrow);
 +   prefix = prefix_buf;
 +
 +   if (strlen(post_arrow)  name_width - 
 strlen(prefix)) {
 +   char *post_arrow_slash = NULL;
 +
 +   post_arrow += strlen(post_arrow) - 
 (name_width - strlen(prefix) - 3);
 +   strcat(prefix_buf, ...);
 +   post_arrow_slash = 
 strchr(post_arrow, '/');
 +   if (post_arrow_slash)
 +   post_arrow = 
 post_arrow_slash;
 +   name = post_arrow;
 +   name_len = (int) (name_width - 
 strlen(prefix));
 +   }
 +   len -= strlen(prefix);
 +   } else {
 +   char *slash = NULL;
 +   prefix = ...;
 +   len -= 3;
 +   name += name_len - len;
 +   slash = strchr(name, '/');
 +   if (slash)
 +   name = slash;
 +   }
}
 
if (file-is_binary) {
 --
 1.8.4.475.g867697c
 
 
 --
 To unsubscribe from this list: 

Re: [PATCH v3] diff.c: keep arrow(=) on show_stats()'s shortened filename part to make rename visible.

2013-10-14 Thread Duy Nguyen
On Sun, Oct 13, 2013 at 3:48 AM, Yoshioka Tsuneo
yoshiokatsu...@gmail.com wrote:
 git diff -M --stat can detect rename and show renamed file name like
 foofoofoo = barbarbar, but if destination filename is long the line
 is shortened like ...barbarbar so there is no way to know whether the
 file is renamed or existed in the source commit.
 This commit makes it visible like ...foo = ...bar.

 Signed-off-by: Tsuneo Yoshioka yoshiokatsu...@gmail.com
 ---
  diff.c | 58 +++---
  1 file changed, 51 insertions(+), 7 deletions(-)

 diff --git a/diff.c b/diff.c
 index a04a34d..3aeaf3e 100644
 --- a/diff.c
 +++ b/diff.c
 @@ -1643,13 +1643,57 @@ static void show_stats(struct diffstat_t *data, 
 struct diff_options *options)
 len = name_width;
 name_len = strlen(name);
 if (name_width  name_len) {
 -   char *slash;
 -   prefix = ...;
 -   len -= 3;
 -   name += name_len - len;
 -   slash = strchr(name, '/');
 -   if (slash)
 -   name = slash;
 +   char *arrow = strstr(name,  = );
 +   if (arrow) {

This looks iffy. What if  =  is part of the path name?
file-is_renamed would be a more reliable sign. In that case I think
you just need an ellipsis version of pprint_rename() (i.e. drop the
result of previous pprint_rename() on the floor and create a new
string with ... and  =  in your pprint_ellipsis_rename or
something)

 +   int prefix_len = (name_width - 4) / 2;
 +   int f_omit;
 +   int f_brace = 0;
 +   char *pre_arrow = alloca(name_width + 10);
 +   char *post_arrow = arrow + 4;
 +   char *prefix_buf = alloca(name_width + 10);
 +   char *pre_arrow_slash = NULL;
 +
 +   if (arrow - name  prefix_len) {
 +   prefix_len = (int)(arrow - name);
 +   f_omit = 0;
 +   } else {
 +   prefix_len -= 3;
 +   f_omit = 1;
 +   if (name[0] == '{') {
 +   prefix_len -= 1;
 +   f_brace = 1;
 +   }
 +   }
 +   prefix_len = ((prefix_len = 0) ? prefix_len 
 : 0);
 +   strncpy(pre_arrow, arrow - prefix_len, 
 prefix_len);
 +   pre_arrow[prefix_len] = '\0';
 +   pre_arrow_slash = strchr(pre_arrow, '/');
 +   if (f_omit  pre_arrow_slash)
 +   pre_arrow = pre_arrow_slash;
 +   sprintf(prefix_buf, %s%s%s = , (f_brace ? 
 { : ), (f_omit ? ... : ), pre_arrow);
 +   prefix = prefix_buf;
 +
 +   if (strlen(post_arrow)  name_width - 
 strlen(prefix)) {
 +   char *post_arrow_slash = NULL;
 +
 +   post_arrow += strlen(post_arrow) - 
 (name_width - strlen(prefix) - 3);
 +   strcat(prefix_buf, ...);
 +   post_arrow_slash = strchr(post_arrow, 
 '/');
 +   if (post_arrow_slash)
 +   post_arrow = post_arrow_slash;
 +   name = post_arrow;
 +   name_len = (int) (name_width - 
 strlen(prefix));
 +   }
 +   len -= strlen(prefix);
 +   } else {
 +   char *slash = NULL;
 +   prefix = ...;
 +   len -= 3;
 +   name += name_len - len;
 +   slash = strchr(name, '/');
 +   if (slash)
 +   name = slash;
 +   }
 }

 if (file-is_binary) {
 --
 1.8.4.475.g867697c


 --
 To unsubscribe from this list: send the line unsubscribe git in
 the body of a message to majord...@vger.kernel.org
 More majordomo info at  http://vger.kernel.org/majordomo-info.html



-- 
Duy
--
To unsubscribe from this list: send the line unsubscribe git in
the body of a message to majord...@vger.kernel.org
More majordomo info at  

Re: [PATCH v3] diff.c: keep arrow(=) on show_stats()'s shortened filename part to make rename visible.

2013-10-13 Thread Thomas Rast
Hi,

Yoshioka Tsuneo yoshiokatsu...@gmail.com writes:

 git diff -M --stat can detect rename and show renamed file name like
 foofoofoo = barbarbar, but if destination filename is long the line
 is shortened like ...barbarbar so there is no way to know whether the
 file is renamed or existed in the source commit.

Thanks for your patch!  I think this is indeed something that should be
fixed.

Can you explain the algorithm chosen in the commit message or a block
comment in the code?  I find it much easier to follow large code blocks
(like the one you added) with a prior notion of what it tries to do.

  [As an aside, Documentation/SubmittingPatches says

The body should provide a meaningful commit message, which:

  . explains the problem the change tries to solve, iow, what is wrong
with the current code without the change.

  . justifies the way the change solves the problem, iow, why the
result with the change is better.

  . alternate solutions considered but discarded, if any.

  Observe that you explained the first item very well, but not the
  others.]

 This commit makes it visible like ...foo = ...bar.

Also, you should rewrite this to be in the imperative mood:

  Make sure there is always an arrow, e.g., ...foo = ...bar.

or some such.

  [Again from SubmittingPatches:

Describe your changes in imperative mood, e.g. make xyzzy do frotz
instead of [This patch] makes xyzzy do frotz or [I] changed xyzzy
to do frotz, as if you are giving orders to the codebase to change
its behaviour.]

 Signed-off-by: Tsuneo Yoshioka yoshiokatsu...@gmail.com
 ---
  diff.c | 58 +++---
  1 file changed, 51 insertions(+), 7 deletions(-)

Can you add a test?  Perhaps like the one below.  (You can squash it
into your commit if you like it.)

Note that in the test, the generated line looks like this:

 {..._does_not_fit_in_a_single_line = .../path1  | 0

I don't want to go all bikesheddey, but I think it's somewhat
unfortunate that the elided parts do not correspond to each other.  In
particular, I think the closing brace should not be omitted.  Perhaps
something like this would be ideal (making it up on the spot, don't
count characters):

 {...a_single_line = ..._as_the_first}/path1  | 0

diff --git a/t/t4001-diff-rename.sh b/t/t4001-diff-rename.sh
index 2f327b7..03d6371 100755
--- a/t/t4001-diff-rename.sh
+++ b/t/t4001-diff-rename.sh
@@ -156,4 +156,16 @@ test_expect_success 'rename pretty print common prefix and 
suffix overlap' '
test_i18ngrep  d/f/{ = f}/e  output
 '
 
+test_expect_success 'rename of very long path shows =' '
+   mkdir long_dirname_that_does_not_fit_in_a_single_line 
+   mkdir another_extremely_long_path_but_not_the_same_as_the_first 
+   cp path1 long_dirname*/ 
+   git add long_dirname*/path1 
+   test_commit add_long_pathname 
+   git mv long_dirname*/path1 another_extremely_*/ 
+   test_commit move_long_pathname 
+   git diff -M --stat HEAD^ HEAD output 
+   test_i18ngrep =.*path1 output
+'
+
 test_done

-- 
Thomas Rast
t...@thomasrast.ch
--
To unsubscribe from this list: send the line unsubscribe git in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH v3] diff.c: keep arrow(=) on show_stats()'s shortened filename part to make rename visible.

2013-10-12 Thread Yoshioka Tsuneo
git diff -M --stat can detect rename and show renamed file name like
foofoofoo = barbarbar, but if destination filename is long the line
is shortened like ...barbarbar so there is no way to know whether the
file is renamed or existed in the source commit.
This commit makes it visible like ...foo = ...bar.

Signed-off-by: Tsuneo Yoshioka yoshiokatsu...@gmail.com
---
 diff.c | 58 +++---
 1 file changed, 51 insertions(+), 7 deletions(-)

diff --git a/diff.c b/diff.c
index a04a34d..3aeaf3e 100644
--- a/diff.c
+++ b/diff.c
@@ -1643,13 +1643,57 @@ static void show_stats(struct diffstat_t *data, struct 
diff_options *options)
len = name_width;
name_len = strlen(name);
if (name_width  name_len) {
-   char *slash;
-   prefix = ...;
-   len -= 3;
-   name += name_len - len;
-   slash = strchr(name, '/');
-   if (slash)
-   name = slash;
+   char *arrow = strstr(name,  = );
+   if (arrow) {
+   int prefix_len = (name_width - 4) / 2;
+   int f_omit;
+   int f_brace = 0;
+   char *pre_arrow = alloca(name_width + 10);
+   char *post_arrow = arrow + 4;
+   char *prefix_buf = alloca(name_width + 10);
+   char *pre_arrow_slash = NULL;
+
+   if (arrow - name  prefix_len) {
+   prefix_len = (int)(arrow - name);
+   f_omit = 0;
+   } else {
+   prefix_len -= 3;
+   f_omit = 1;
+   if (name[0] == '{') {
+   prefix_len -= 1;
+   f_brace = 1;
+   }
+   }
+   prefix_len = ((prefix_len = 0) ? prefix_len : 
0);
+   strncpy(pre_arrow, arrow - prefix_len, 
prefix_len);
+   pre_arrow[prefix_len] = '\0';
+   pre_arrow_slash = strchr(pre_arrow, '/');
+   if (f_omit  pre_arrow_slash)
+   pre_arrow = pre_arrow_slash;
+   sprintf(prefix_buf, %s%s%s = , (f_brace ? 
{ : ), (f_omit ? ... : ), pre_arrow);
+   prefix = prefix_buf;
+
+   if (strlen(post_arrow)  name_width - 
strlen(prefix)) {
+   char *post_arrow_slash = NULL;
+
+   post_arrow += strlen(post_arrow) - 
(name_width - strlen(prefix) - 3);
+   strcat(prefix_buf, ...);
+   post_arrow_slash = strchr(post_arrow, 
'/');
+   if (post_arrow_slash)
+   post_arrow = post_arrow_slash;
+   name = post_arrow;
+   name_len = (int) (name_width - 
strlen(prefix));
+   }
+   len -= strlen(prefix);
+   } else {
+   char *slash = NULL;
+   prefix = ...;
+   len -= 3;
+   name += name_len - len;
+   slash = strchr(name, '/');
+   if (slash)
+   name = slash;
+   }
}
 
if (file-is_binary) {
-- 
1.8.4.475.g867697c


--
To unsubscribe from this list: send the line unsubscribe git in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH v3] diff.c: keep arrow(=) on show_stats()'s shortened filename part to make rename visible.

2013-10-12 Thread Yoshioka Tsuneo
git diff -M --stat can detect rename and show renamed file name like
foofoofoo = barbarbar, but if destination filename is long the line
is shortened like ...barbarbar so there is no way to know whether the
file is renamed or existed in the source commit.
This commit makes it visible like ...foo = ...bar.

Signed-off-by: Tsuneo Yoshioka yoshiokatsu...@gmail.com
---
 diff.c | 58 +++---
 1 file changed, 51 insertions(+), 7 deletions(-)

diff --git a/diff.c b/diff.c
index a04a34d..3aeaf3e 100644
--- a/diff.c
+++ b/diff.c
@@ -1643,13 +1643,57 @@ static void show_stats(struct diffstat_t *data, struct 
diff_options *options)
len = name_width;
name_len = strlen(name);
if (name_width  name_len) {
-   char *slash;
-   prefix = ...;
-   len -= 3;
-   name += name_len - len;
-   slash = strchr(name, '/');
-   if (slash)
-   name = slash;
+   char *arrow = strstr(name,  = );
+   if (arrow) {
+   int prefix_len = (name_width - 4) / 2;
+   int f_omit;
+   int f_brace = 0;
+   char *pre_arrow = alloca(name_width + 10);
+   char *post_arrow = arrow + 4;
+   char *prefix_buf = alloca(name_width + 10);
+   char *pre_arrow_slash = NULL;
+
+   if (arrow - name  prefix_len) {
+   prefix_len = (int)(arrow - name);
+   f_omit = 0;
+   } else {
+   prefix_len -= 3;
+   f_omit = 1;
+   if (name[0] == '{') {
+   prefix_len -= 1;
+   f_brace = 1;
+   }
+   }
+   prefix_len = ((prefix_len = 0) ? prefix_len : 
0);
+   strncpy(pre_arrow, arrow - prefix_len, 
prefix_len);
+   pre_arrow[prefix_len] = '\0';
+   pre_arrow_slash = strchr(pre_arrow, '/');
+   if (f_omit  pre_arrow_slash)
+   pre_arrow = pre_arrow_slash;
+   sprintf(prefix_buf, %s%s%s = , (f_brace ? 
{ : ), (f_omit ? ... : ), pre_arrow);
+   prefix = prefix_buf;
+
+   if (strlen(post_arrow)  name_width - 
strlen(prefix)) {
+   char *post_arrow_slash = NULL;
+
+   post_arrow += strlen(post_arrow) - 
(name_width - strlen(prefix) - 3);
+   strcat(prefix_buf, ...);
+   post_arrow_slash = strchr(post_arrow, 
'/');
+   if (post_arrow_slash)
+   post_arrow = post_arrow_slash;
+   name = post_arrow;
+   name_len = (int) (name_width - 
strlen(prefix));
+   }
+   len -= strlen(prefix);
+   } else {
+   char *slash = NULL;
+   prefix = ...;
+   len -= 3;
+   name += name_len - len;
+   slash = strchr(name, '/');
+   if (slash)
+   name = slash;
+   }
}
 
if (file-is_binary) {
-- 
1.8.4.475.g867697c


--
To unsubscribe from this list: send the line unsubscribe git in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html