From: Thomas Rast <trast@student.ethz.ch>
To: Jeff King <peff@peff.net>
Cc: git@vger.kernel.org
Subject: [RFC PATCH] git-add--interactive: manual hunk editing mode v2
Date: Sun, 1 Jun 2008 02:41:48 +0200 [thread overview]
Message-ID: <200806010241.51464.trast@student.ethz.ch> (raw)
In-Reply-To: <20080529185808.GA2140@sigill.intra.peff.net>
Adds a new option 'e' to the 'add -p' command loop that lets you
edit the current hunk in your favourite editor.
---
This is a draft of Jeff King's idea
Jeff King wrote:
> But I find the interface a bit clunky. I would much rather get dumped in
> my favorite editor, which happens to be quite fast at removing a subset
> of lines. After editing, any lines remaining would be staged.
>
> We would have to figure out what happens if lines are added or edited,
> of course. It may be right to signal an error, or maybe there is some
> other useful functionality that can come of that.
in the hope that I might get some more ideas and pointers on the details.
The current implementation rejects edits that break the (whole) patch
(thanks Junio for pointing out --check...), but it does help the user
in one aspect: @ lines are edited to reflect their hunk contents, even
inferring the starting line numbers from the last hunk if they are
missing. This means you can insert a line consisting just of an @,
and it will silently be fixed to start a new hunk at that point.
Some things that might need improvement/fixing:
- Does anyone need a way to force a broken diff past the git-apply
check? I don't know what use such a diff might have, but who knows,
perhaps it applies cleanly once you exclude a hunk or two...
- Perhaps the instructions should go to the bottom, but then the odds
are they would not be visible in many cases.
- Should I try to come up with a unique filename to support concurrent
edits? (git-commit doesn't...)
- Perl's autovivification is out to get me. I've fixed a few, but
there are probably still bugs.
Thomas
---
git-add--interactive.perl | 171 +++++++++++++++++++++++++++++++++++++++++++++
1 files changed, 171 insertions(+), 0 deletions(-)
diff --git a/git-add--interactive.perl b/git-add--interactive.perl
index 903953e..c752e20 100755
--- a/git-add--interactive.perl
+++ b/git-add--interactive.perl
@@ -18,6 +18,18 @@ my ($fraginfo_color) =
$diff_use_color ? (
$repo->get_color('color.diff.frag', 'cyan'),
) : ();
+my ($diff_plain_color) =
+ $diff_use_color ? (
+ $repo->get_color('color.diff.plain', ''),
+ ) : ();
+my ($diff_old_color) =
+ $diff_use_color ? (
+ $repo->get_color('color.diff.old', 'red'),
+ ) : ();
+my ($diff_new_color) =
+ $diff_use_color ? (
+ $repo->get_color('color.diff.new', 'green'),
+ ) : ();
my $normal_color = $repo->get_color("", "reset");
@@ -770,6 +782,158 @@ sub coalesce_overlapping_hunks {
return @out;
}
+sub edit_hunk_manually {
+ my @oldtext;
+ for (@_) {
+ push @oldtext, @{$_->{TEXT}};
+ }
+
+ # use a .diff file to help editors with highlighting
+ my $editpath = $repo->repo_path() . "/ADDP_HUNK_EDIT.diff";
+ my $fh;
+ open $fh, '>', $editpath
+ or die "failed to open hunk edit file for writing: " . $!;
+ print $fh <<EOF;
+# MANUAL HUNK EDIT MODE
+#
+# You can change the hunk to your heart's content, but it will be
+# refused if the end result (the entire patch including your edited
+# hunk) does not apply cleanly.
+#
+# To remove '-' lines, make them ' ' lines (context).
+# To remove '+' lines, delete them.
+# Empty lines and lines starting with # will be removed.
+#
+# Lines starting with @ start a new hunk. Line counts will be adjusted
+# according to contents. If the line numbers are missing altogether,
+# they will be inferred from the previous hunk.
+EOF
+ print $fh @oldtext;
+ close $fh;
+
+ my $editor = $ENV{GIT_EDITOR} || $repo->config("core.editor")
+ || $ENV{VISUAL} || $ENV{EDITOR} || "vi";
+ system('sh', '-c', $editor.' "$@"', $editor, $editpath);
+
+ open $fh, '<', $editpath
+ or die "failed to open hunk edit file for reading: " . $!;
+ my @newtext;
+ while (<$fh>) {
+ push (@newtext, $_) unless /^#/ || /^$/;
+ }
+ close $fh;
+ my @heads = ();
+ my ($o_ofs, $n_ofs);
+ my $o_cnt = 0;
+ my $n_cnt = 0;
+ my ($guess_o_ofs, undef, $guess_n_ofs, undef) = parse_hunk_header($oldtext[0]);
+ for (my $i = 0; $i < @newtext; $i++) {
+ if ((scalar @heads) == 0 && $newtext[$i] =~ /^[ +-]/) {
+ splice @newtext, $i, 0, $oldtext[0];
+ push @heads, $i;
+ }
+ elsif ($newtext[$i] =~ /^ /) {
+ $o_cnt++;
+ $n_cnt++;
+ }
+ elsif ($newtext[$i] =~ /^-/) {
+ $o_cnt++;
+ }
+ elsif ($newtext[$i] =~ /^\+/) {
+ $n_cnt++;
+ }
+ elsif ($newtext[$i] =~ /^@/) {
+ if (@heads > 0) {
+ # fix up the previous header first
+ ($o_ofs, undef, $n_ofs, undef)
+ = parse_hunk_header($newtext[$heads[-1]]);
+ $o_ofs = $guess_o_ofs unless defined $o_ofs;
+ $n_ofs = $guess_n_ofs unless defined $n_ofs;
+ $newtext[$heads[-1]] = (
+ "@@ -$o_ofs" . (($o_cnt != 1) ? ",$o_cnt" : '')
+ . " +$n_ofs" . (($n_cnt != 1) ? ",$n_cnt" : '')
+ . " @@\n");
+ $guess_o_ofs = $o_ofs + $o_cnt;
+ $guess_n_ofs = $n_ofs + $n_cnt;
+ }
+ $o_cnt = 0;
+ $n_cnt = 0;
+ push @heads, $i;
+ }
+ }
+ ($o_ofs, undef, $n_ofs, undef)
+ = parse_hunk_header($newtext[$heads[-1]]);
+ $o_ofs = $guess_o_ofs unless defined $o_ofs;
+ $n_ofs = $guess_n_ofs unless defined $n_ofs;
+ $newtext[$heads[-1]] = (
+ "@@ -$o_ofs" . (($o_cnt != 1) ? ",$o_cnt" : '')
+ . " +$n_ofs" . (($n_cnt != 1) ? ",$n_cnt" : '')
+ . " @@\n");
+
+ push @heads, (scalar @newtext);
+ my (@hunks) = ();
+ for (my $i = 0; $i < @heads-1; $i++) {
+ my @hunktext = @newtext[$heads[$i]..$heads[$i+1]-1];
+ my @hunkdisplay = ();
+ for (@hunktext) {
+ if (/^@/) {
+ push @hunkdisplay, (colored $fraginfo_color, $_);
+ }
+ elsif (/^\+/) {
+ push @hunkdisplay, (colored $diff_new_color, $_);
+ }
+ elsif (/^-/) {
+ push @hunkdisplay, (colored $diff_old_color, $_);
+ }
+ else {
+ push @hunkdisplay, (colored $diff_plain_color, $_);
+ }
+ }
+ push @hunks, {TEXT => \@hunktext, DISPLAY => \@hunkdisplay};
+ }
+
+ return @hunks;
+}
+
+sub edit_hunk_loop {
+ my ($head, $hunks, $ix) = @_;
+
+ my @newhunks = ($hunks->[$ix]);
+
+ EDIT:
+ while (1) {
+ @newhunks = edit_hunk_manually(@newhunks);
+ my $fh;
+ open $fh, '| git apply --cached --check';
+ for my $h ($head,
+ $ix > 0 ? @$hunks[0..$ix-1] : (),
+ @newhunks,
+ $ix < (scalar @$hunks)-2 ? @$hunks[$ix+1..@$hunks] : ()) {
+ for (@{$h->{TEXT}}) {
+ print $fh $_;
+ }
+ }
+ if (!close $fh) {
+ # didn't apply cleanly
+ while (1) {
+ print colored $prompt_color, "Your edited hunk does not apply. Edit again (saying \"no\" discards!) [y/n]? ";
+ my $line = <STDIN>;
+ if ($line =~ /^y/) {
+ redo EDIT;
+ }
+ elsif ($line =~ /^n/) {
+ return $hunks->[$ix];
+ }
+ }
+ }
+ if (1 < @newhunks) {
+ print colored $header_color, "Manually edited into ",
+ scalar(@newhunks), " hunks.\n";
+ }
+ return @newhunks;
+ }
+}
+
sub help_patch_cmd {
print colored $help_color, <<\EOF ;
y - stage this hunk
@@ -781,6 +945,7 @@ J - leave this hunk undecided, see next hunk
k - leave this hunk undecided, see previous undecided hunk
K - leave this hunk undecided, see previous hunk
s - split the current hunk into smaller hunks
+e - manually edit the current hunk
? - print help
EOF
}
@@ -885,6 +1050,7 @@ sub patch_update_file {
if (hunk_splittable($hunk[$ix]{TEXT})) {
$other .= '/s';
}
+ $other .= '/e';
for (@{$hunk[$ix]{DISPLAY}}) {
print;
}
@@ -949,6 +1115,11 @@ sub patch_update_file {
$num = scalar @hunk;
next;
}
+ elsif ($line =~ /^e/) {
+ splice @hunk, $ix, 1, edit_hunk_loop($head, \@hunk, $ix);
+ $num = scalar @hunk;
+ next;
+ }
else {
help_patch_cmd($other);
next;
--
1.5.6.rc0.159.g710c6
next prev parent reply other threads:[~2008-06-01 0:42 UTC|newest]
Thread overview: 62+ messages / expand[flat|nested] mbox.gz Atom feed top
2008-05-23 20:21 [PATCH] git-add--interactive: manual hunk editing mode Thomas Rast
2008-05-24 1:24 ` Ping Yin
2008-05-29 15:37 ` Thomas Rast
2008-05-29 16:12 ` Johannes Schindelin
2008-05-29 19:10 ` Thomas Rast
2008-05-29 19:16 ` Thomas Rast
2008-05-29 18:58 ` Jeff King
2008-05-30 9:49 ` Johannes Schindelin
2008-05-30 10:46 ` Jakub Narebski
2008-05-30 12:21 ` Thomas Rast
2008-05-30 21:35 ` Junio C Hamano
2008-06-01 0:41 ` Thomas Rast [this message]
2008-06-01 14:50 ` [RFC PATCH] git-add--interactive: manual hunk editing mode v2.1 Thomas Rast
2008-06-01 15:14 ` Jeff King
2008-06-05 1:46 ` [RFC PATCH] git-add--interactive: manual hunk editing mode v2 Jeff King
2008-06-05 7:53 ` Thomas Rast
2008-06-05 8:11 ` Jeff King
2008-06-05 9:04 ` Thomas Rast
2008-06-05 9:20 ` Jeff King
2008-06-05 9:38 ` Thomas Rast
2008-06-05 9:46 ` Jeff King
2008-06-05 8:16 ` Junio C Hamano
2008-06-05 8:56 ` Jeff King
2008-06-05 10:28 ` Johannes Schindelin
2008-06-06 5:10 ` Jeff King
2008-06-06 6:03 ` Jeff King
2008-06-08 22:33 ` Thomas Rast
2008-06-08 23:06 ` Johannes Schindelin
2008-06-06 14:31 ` Johannes Schindelin
2008-06-08 22:18 ` Thomas Rast
2008-06-08 23:02 ` Johannes Schindelin
2008-06-05 12:38 ` [WIP PATCH v2] git-add--interactive: manual hunk editing mode Thomas Rast
2008-06-08 22:32 ` [PATCH v3] " Thomas Rast
2008-06-08 23:19 ` Johannes Schindelin
2008-06-09 5:46 ` Johan Herland
2008-06-09 12:29 ` Jeff King
2008-06-09 16:13 ` Johannes Schindelin
2008-06-09 19:59 ` Junio C Hamano
2008-06-09 17:31 ` Johan Herland
2008-06-09 20:17 ` Jeff King
2008-06-09 21:19 ` Johan Herland
2008-06-10 11:05 ` Jeff King
2008-06-11 9:02 ` Thomas Rast
2008-06-12 4:49 ` Jeff King
2008-06-12 6:55 ` Thomas Rast
2008-06-12 7:13 ` Jeff King
2008-06-13 15:48 ` [PATCH v4] " Thomas Rast
2008-06-23 18:38 ` Jeff King
2008-06-23 18:54 ` Johannes Schindelin
2008-06-23 19:57 ` Jeff King
2008-06-23 21:16 ` apply --recount, was " Johannes Schindelin
2008-06-24 5:09 ` Jeff King
2008-06-24 19:07 ` [PATCH 0/3] Manual editing for 'add' and 'add -p' Thomas Rast
2008-06-24 19:53 ` Miklos Vajna
2008-06-24 19:08 ` [PATCH 1/3] Allow git-apply to ignore the hunk headers (AKA recountdiff) Thomas Rast
2008-06-24 23:35 ` Junio C Hamano
2008-06-25 5:45 ` Jeff King
2008-06-27 17:43 ` Johannes Schindelin
2008-06-24 19:08 ` [PATCH 2/3] git-add: introduce --edit (to edit the diff vs. the index) Thomas Rast
2008-06-24 19:08 ` [PATCH 3/3] git-add--interactive: manual hunk editing mode Thomas Rast
2008-06-10 11:19 ` [PATCH v3] " Andreas Ericsson
2008-06-05 9:02 ` [PATCH] " Thomas Rast
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=200806010241.51464.trast@student.ethz.ch \
--to=trast@student.ethz.ch \
--cc=git@vger.kernel.org \
--cc=peff@peff.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).