aboutsummaryrefslogtreecommitdiff
path: root/contrib/blameview/blameview.perl
blob: 5e9a67c1235cd04cb51f2166b1dd2cf9fdada8eb (plain)
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
108
109
110
111
112
113
114
115
116
117
118
119
#!/usr/bin/perl

use Gtk2 -init;
use Gtk2::SimpleList;

my $fn = shift or die "require filename to blame";

Gtk2::Rc->parse_string(<<'EOS');
style "treeview_style"
{
  GtkTreeView::vertical-separator = 0
}
class "GtkTreeView" style "treeview_style"
EOS

my $window = Gtk2::Window->new('toplevel');
$window->signal_connect(destroy => sub { Gtk2->main_quit });
my $scrolled_window = Gtk2::ScrolledWindow->new;
$window->add($scrolled_window);
my $fileview = Gtk2::SimpleList->new(
    'Commit' => 'text',
    'CommitInfo' => 'text',
    'FileLine' => 'text',
    'Data' => 'text'
);
$scrolled_window->add($fileview);
$fileview->get_column(0)->set_spacing(0);
$fileview->set_size_request(1024, 768);
$fileview->set_rules_hint(1);

my $fh;
open($fh, '-|', "git cat-file blob HEAD:$fn")
  or die "unable to open $fn: $!";
while(<$fh>) {
  chomp;
  $fileview->{data}->[$.] = ['HEAD', '?', "$fn:$.", $_];
}

my $blame;
open($blame, '-|', qw(git blame --incremental --), $fn)
    or die "cannot start git-blame $fn";

Glib::IO->add_watch(fileno($blame), 'in', \&read_blame_line);

$window->show_all;
Gtk2->main;
exit 0;

my %commitinfo = ();

sub flush_blame_line {
	my ($attr) = @_;

	return unless defined $attr;

	my ($commit, $s_lno, $lno, $cnt) =
	    @{$attr}{qw(COMMIT S_LNO LNO CNT)};

	my ($filename, $author, $author_time, $author_tz) =
	    @{$commitinfo{$commit}}{qw(FILENAME AUTHOR AUTHOR-TIME AUTHOR-TZ)};
	my $info = $author . ' ' . format_time($author_time, $author_tz);

	for(my $i = 0; $i < $cnt; $i++) {
		@{$fileview->{data}->[$lno+$i-1]}[0,1,2] =
		    (substr($commit, 0, 8), $info,
		     $filename . ':' . ($s_lno+$i));
	}
}

my $buf;
my $current;
sub read_blame_line {

	my $r = sysread($blame, $buf, 1024, length($buf));
	die "I/O error" unless defined $r;

	if ($r == 0) {
		flush_blame_line($current);
		$current = undef;
		return 0;
	}

	while ($buf =~ s/([^\n]*)\n//) {
		my $line = $1;

		if (($commit, $s_lno, $lno, $cnt) =
		    ($line =~ /^([0-9a-f]{40}) (\d+) (\d+) (\d+)$/)) {
			flush_blame_line($current);
			$current = +{
				COMMIT => $1,
				S_LNO => $2,
				LNO => $3,
				CNT => $4,
			};
			next;
		}

		# extended attribute values
		if ($line =~ /^(author|author-mail|author-time|author-tz|committer|committer-mail|committer-time|committer-tz|summary|filename) (.*)$/) {
			my $commit = $current->{COMMIT};
			$commitinfo{$commit}{uc($1)} = $2;
			next;
		}
	}
	return 1;
}

sub format_time {
  my $time = shift;
  my $tz = shift;

  my $minutes = $tz < 0 ? 0-$tz : $tz;
  $minutes = ($minutes / 100)*60 + ($minutes % 100);
  $minutes = $tz < 0 ? 0-$minutes : $minutes;
  $time += $minutes * 60;
  my @t = gmtime($time);
  return sprintf('%04d-%02d-%02d %02d:%02d:%02d %s',
		 $t[5] + 1900, @t[4,3,2,1,0], $tz);
}