forked from acme/git-pureperl
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCommit.pm
192 lines (154 loc) · 5.48 KB
/
Commit.pm
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
package Git::PurePerl::Object::Commit;
use Moose;
use MooseX::StrictConstructor;
use Moose::Util::TypeConstraints;
use Encode qw/decode/;
use namespace::autoclean;
extends 'Git::PurePerl::Object';
has 'kind' =>
( is => 'ro', isa => 'ObjectKind', required => 1, default => 'commit' );
has 'tree_sha1' => ( is => 'rw', isa => 'Str', required => 0 );
has 'parent_sha1s' => ( is => 'rw', isa => 'ArrayRef[Str]', required => 0, default => sub { [] });
has 'author' => ( is => 'rw', isa => 'Git::PurePerl::Actor', required => 0 );
has 'authored_time' => ( is => 'rw', isa => 'DateTime', required => 0 );
has 'committer' =>
( is => 'rw', isa => 'Git::PurePerl::Actor', required => 0 );
has 'committed_time' => ( is => 'rw', isa => 'DateTime', required => 0 );
has 'comment' => ( is => 'rw', isa => 'Str', required => 0 );
has 'encoding' => ( is => 'rw', isa => 'Str', required => 0 );
has 'gpg_signature' => ( is => 'rw', isa => 'Str', required => 0 );
my %method_map = (
'tree' => 'tree_sha1',
'parent' => '_push_parent_sha1',
'author' => 'authored_time',
'committer' => 'committed_time',
'gpgsig' => 'gpg_signature',
);
sub BUILD {
my $self = shift;
return unless $self->content;
my @lines = split "\n", $self->content;
my %header;
while ( my $line = shift @lines ) {
# Apparent format is roughly:
#
# <token><space><DATA>
# <space><DATA> # repeated
#
# And a line not leading with <space> ends the token.
#
# Though, at present, git itself has this special-cased for GPG Signatures.
#
# Its probably extendable to support any value of <token> though.
if ( $line =~ /^gpgsig (.*$)/ ) {
my $sig = "$1";
while ( $line = $lines[0] ) {
last unless $line =~ /^ (.*$)/;
$sig .= "$1\n";
shift @lines;
}
push @{ $header{gpgsig} }, $sig;
}
last unless $line;
my ( $key, $value ) = split ' ', $line, 2;
push @{$header{$key}}, $value;
}
$header{encoding}
||= [ $self->git->config->get(key => "i18n.commitEncoding") || "utf-8" ];
my $encoding = $header{encoding}->[-1];
for my $key (keys %header) {
for my $value (@{$header{$key}}) {
$value = decode($encoding, $value);
if ( $key eq 'committer' or $key eq 'author' ) {
my @data = split ' ', $value;
my ( $email, $epoch, $tz ) = splice( @data, -3 );
$email = substr( $email, 1, -1 );
my $name = join ' ', @data;
my $actor
= Git::PurePerl::Actor->new( name => $name, email => $email );
$self->$key($actor);
$key = $method_map{$key};
my $dt
= DateTime->from_epoch( epoch => $epoch, time_zone => $tz );
$self->$key($dt);
} else {
$key = $method_map{$key} || $key;
$self->$key($value);
}
}
}
$self->comment( decode($encoding, join "\n", @lines) );
}
=head1 METHODS
=head2 tree
Returns the L<< C<::Tree>|Git::PurePerl::Object::Tree >> associated with this commit.
=cut
sub tree {
my $self = shift;
return $self->git->get_object( $self->tree_sha1 );
}
sub _push_parent_sha1 {
my ($self, $sha1) = @_;
push(@{$self->parent_sha1s}, $sha1);
}
=head2 parent_sha1
Returns the C<sha1> for the first parent of this this commit.
=cut
sub parent_sha1 {
return shift->parent_sha1s->[0];
}
=head2 parent
Returns the L<< C<::Commit>|Git::PurePerl::Object::Commit >> for this commits first parent.
=cut
sub parent {
my $self = shift;
return $self->git->get_object( $self->parent_sha1 );
}
=head2 parents
Returns L<< C<::Commit>s|Git::PurePerl::Object::Commit >> for all this commits parents.
=cut
sub parents {
my $self = shift;
return map { $self->git->get_object( $_ ) } @{$self->parent_sha1s};
}
=head2 has_ancestor_sha1
Traverses up the parentage of the object graph to find out if the given C<sha1> appears as an ancestor.
if ( $commit_object->has_ancestor_sha1( 'deadbeef' x 5 ) ) {
...
}
=cut
sub has_ancestor_sha1 {
my ( $self, $sha1 ) = @_;
# This may seem redundant, but its not entirely.
# However, its a penalty paid for the branch shortening optimization.
#
# x^, y^ , z^ , y[ y^ , y... ] , z[ z^ , z... ]
#
# Will still be faster than
#
# x^, y[ y^ , y... ] , z[ z^ , z... ]
#
# In the event y is very long.
return 1 if $self->sha1 eq $sha1;
# This is a slight optimization of sorts,
# as it means
# x->{ y->{ y' } , z->{ z' } }
# has a check order of:
# x^, y^ , z^ , y[ y^ , ... ], z[ z^, ... ]
# instead of
# x^, y[ y^, y... ], z[ z^, z... ]
# Which will probably make things a bit faster if y is incredibly large
# and you just want to check if a given commit x has a direct ancestor i.
for my $parent ( @{ $self->parent_sha1s } ) {
return 1 if $parent eq $sha1;
}
# Depth First.
# TODO perhaps make it breadth first? could be very useful on very long repos
# where the given ancestor might not be in the "first-parent" ancestry line.
# But if somebody wants this feature, they'll have to provide the benchmarks, the code, or both.
for my $parent ( $self->parents ) {
return 1 if $parent->has_ancestor_sha1( $sha1, );
}
return;
}
__PACKAGE__->meta->make_immutable;