The Perl and Raku Conference 2025: Greenville, South Carolina - June 27-29 Learn more

$Git::Version::Compare::VERSION = '1.005';
use strict;
use Carp ();
my @ops = qw( lt gt le ge eq ne );
our @ISA = qw(Exporter);
our @EXPORT_OK = ( looks_like_git => map "${_}_git", cmp => @ops );
our %EXPORT_TAGS = ( ops => [ map "${_}_git", @ops ], all => \@EXPORT_OK );
# A few versions have two tags, or non-standard numbering:
# - the left-hand side is what `git --version` reports
# - the right-hand side is an internal canonical name
#
# We turn versions into strings, so we can use the fast `eq` and `gt`.
# The 6 elements are integers padded with 0:
# - the 4 parts of the dotted version (padded with as many .0 as needed)
# - '.000' if not an RC, or '-xxx' if an RC (- sorts before . in ascii)
# - the number of commits since the previous tag (for dev versions)
#
# The special cases are pre-computed below, the rest is computed as needed.
my %version_alias = (
'0.99.7a' => '00.99.07.01.00.0000',
'0.99.7b' => '00.99.07.02.00.0000',
'0.99.7c' => '00.99.07.03.00.0000',
'0.99.7d' => '00.99.07.04.00.0000',
'0.99.8a' => '00.99.08.01.00.0000',
'0.99.8b' => '00.99.08.02.00.0000',
'0.99.8c' => '00.99.08.03.00.0000',
'0.99.8d' => '00.99.08.04.00.0000',
'0.99.8e' => '00.99.08.05.00.0000',
'0.99.8f' => '00.99.08.06.00.0000',
'0.99.8g' => '00.99.08.07.00.0000',
'0.99.9a' => '00.99.09.01.00.0000',
'0.99.9b' => '00.99.09.02.00.0000',
'0.99.9c' => '00.99.09.03.00.0000',
'0.99.9d' => '00.99.09.04.00.0000',
'0.99.9e' => '00.99.09.05.00.0000',
'0.99.9f' => '00.99.09.06.00.0000',
'0.99.9g' => '00.99.09.07.00.0000',
'0.99.9h' => '00.99.09.08.00.0000', # 1.0.rc1
'1.0.rc1' => '00.99.09.08.00.0000',
'0.99.9i' => '00.99.09.09.00.0000', # 1.0.rc2
'1.0.rc2' => '00.99.09.09.00.0000',
'0.99.9j' => '00.99.09.10.00.0000', # 1.0.rc3
'1.0.rc3' => '00.99.09.10.00.0000',
'0.99.9k' => '00.99.09.11.00.0000',
'0.99.9l' => '00.99.09.12.00.0000', # 1.0.rc4
'1.0.rc4' => '00.99.09.12.00.0000',
'0.99.9m' => '00.99.09.13.00.0000', # 1.0.rc5
'1.0.rc5' => '00.99.09.13.00.0000',
'0.99.9n' => '00.99.09.14.00.0000', # 1.0.rc6
'1.0.rc6' => '00.99.09.14.00.0000',
'1.0.0a' => '01.00.01.00.00.0000',
'1.0.0b' => '01.00.02.00.00.0000',
);
sub looks_like_git {
return scalar $_[0] =~
/^(?:v|git\ version\ )? # prefix
[0-9]+(?:[.-](?:0[ab]?|[1-9][0-9a-z]*|[a-zA-Z]+))* # x.y.z.*
(?:[.-]?[a-z]+[0-9]+)? # rc or vendor specific suffixes
(?:[.-](GIT|[1-9][0-9]*[.-]g[A-Fa-f0-9]+))? # devel
(?:\ .*)? # comment
$/x;
}
sub _normalize {
my ($v) = @_;
return undef if !defined $v;
# minimal consistency check
Carp::croak "$v does not look like a Git version" if !looks_like_git($v);
# reformat git.git tag names, output of `git --version`
$v =~ s/^v|^git version |\.[a-zA-Z]+\..*|[\012\015]+\z//g;
$v =~ y/-/./;
$v =~ s/0rc/0.rc/;
($v) = split / /, $v; # drop anything after the version
# can't use exists() because the assignment in the @ops created the slot
return $version_alias{$v} if defined $version_alias{$v};
# split the dotted version string
my @v = split /\./, $v;
my ( $r, $c ) = ( 0, 0 );
# commit count since the previous tag
($c) = ( 1, splice @v, -1 ) if $v[-1] eq 'GIT'; # before 1.4
($c) = splice @v, -2 if substr( $v[-1], 0, 1 ) eq 'g'; # after 1.4
# release candidate number
($r) = splice @v, -1 if substr( $v[-1], 0, 2 ) eq 'rc';
$r &&= do { $r =~ s/rc//; sprintf '-%02d', $r };
# compute and cache normalized string
return $version_alias{$v} =
join( '.', map sprintf( '%02d', $_ ), ( @v, 0, 0, 0 )[ 0 .. 3 ] )
. ( $r || '.00' )
. sprintf( '.%04d', $c );
}
for my $op (@ops) {
no strict 'refs';
*{"${op}_git"} = eval << "OP";
sub {
my ( \$v1, \$v2 ) = \@_;
\$_ = \$version_alias{\$_} ||= _normalize( \$_ ) for \$v1, \$v2;
return \$v1 $op \$v2;
}
OP
}
sub cmp_git ($$) {
my ( $v1, $v2 ) = @_;
$_ = $version_alias{$_} ||= _normalize( $_ ) for $v1, $v2;
return $v1 cmp $v2;
}
1;
__END__
=head1 NAME
Git::Version::Compare - Functions to compare Git versions
=head1 SYNOPSIS
use Git::Version::Compare qw( cmp_git );
# result: 1.2.3 1.7.0.rc0 1.7.4.rc1 1.8.3.4 1.9.3 2.0.0.rc2 2.0.3 2.3.0.rc1
my @versions = sort cmp_git qw(
1.7.4.rc1 1.9.3 1.7.0.rc0 2.0.0.rc2 1.2.3 1.8.3.4 2.3.0.rc1 2.0.3
);
=head1 DESCRIPTION
L<Git::Version::Compare> contains a selection of subroutines that make
dealing with Git-related things (like versions) a little bit easier.
The strings to compare can be version numbers, tags from C<git.git>
or the output of C<git version> or C<git describe>.
These routines collect the knowledge about Git versions that
was accumulated while developing L<Git::Repository>.
=head1 AVAILABLE FUNCTIONS
By default L<Git::Version::Compare> does not export any subroutines.
All the comparison version functions die when given strings that do not
look like Git version numbers (the check is done with L</looks_like_git>).
=head2 lt_git
if ( lt_git( $v1, $v2 ) ) { ... }
A Git-aware version of the C<lt> operator.
=head2 gt_git
if ( gt_git( $v1, $v2 ) ) { ... }
A Git-aware version of the C<gt> operator.
=head2 le_git
if ( le_git( $v1, $v2 ) ) { ... }
A Git-aware version of the C<le> operator.
=head2 ge_git
if ( ge_git( $v1, $v2 ) ) { ... }
A Git-aware version of the C<ge> operator.
=head2 eq_git
if ( eq_git( $v1, $v2 ) ) { ... }
A Git-aware version of the C<eq> operator.
=head2 ne_git
if ( ne_git( $v1, $v2 ) ) { ... }
A Git-aware version of the C<ne> operator.
=head2 cmp_git
@versions = sort cmp_git @versions;
A Git-aware version of the C<cmp> operator.
=head2 looks_like_git
# true
looks_like_git(`git version`); # duh
# false
looks_like_git('v1.7.3_02'); # no _ in git versions
Given a string, returns true if it looks like a Git version number
(and can therefore be parsed by C<Git::Version::Number>) and false
otherwise.
It accepts the version strings from all standard Git versions and from some
non-standard Gits as well, such as GitLab's embedded Git which uses a special
suffix like C<.gl1>.
=head1 EXPORT TAGS
=head2 :ops
Exports C<lt_git>, C<gt_git>, C<le_git>, C<ge_git>, C<eq_git>, and C<ne_git>.
=head2 :all
Exports C<lt_git>, C<gt_git>, C<le_git>, C<ge_git>, C<eq_git>, C<ne_git>,
C<cmp_git>, and C<looks_like_git>.
=head1 EVERYTHING YOU EVER WANTED TO KNOW ABOUT GIT VERSION NUMBERS
=head1 Version numbers
Version numbers as returned by C<git version> are in the following
formats (since the C<1.4> series, in 2006):
# stable version
1.6.0
2.7.1
# maintenance release
1.8.5.6
# release candidate
1.6.0.rc2
# development version
# (the last two elements come from `git describe`)
1.7.1.209.gd60ad
1.8.5.1.21.gb2a0afd
2.3.0.rc0.36.g63a0e83
In the C<git.git> repository, several commits have multiple tags
(e.g. C<v1.0.1> and C<v1.0.2> point respectively to C<v1.0.0a>
and C<v1.0.0b>). Pre-1.0.0 versions also have non-standard formats
like C<0.99.9j> or C<1.0rc2>.
This explains why:
# this is true
eq_git( '0.99.9l', '1.0rc4' );
eq_git( '1.0.0a', '1.0.1' );
# this is false
ge_git( '1.0rc3', '0.99.9m' );
C<git version> appeared in version C<1.3.0>.
C<git --version> appeared in version C<0.99.7>. Before that, there is no
way to know which version of Git one is dealing with.
C<Git::Version::Compare> converts all version numbers to an internal
format before performing a simple string comparison.
=head2 Development versions
Prior to C<1.4.0-rc1> (June 2006), compiling a development version of Git
would lead C<git --version> to output C<1.x-GIT> (with C<x> in C<0 .. 3>),
which would make comparing versions that are very close a futile exercise.
Other issues exist when comparing development version numbers with one
another. For example, C<1.7.1.1> is greater than both C<1.7.1.1.gc8c07>
and C<1.7.1.1.g5f35a>, and C<1.7.1> is less than both. Obviously,
C<1.7.1.1.gc8c07> will compare as greater than C<1.7.1.1.g5f35a>
(asciibetically), but in fact these two version numbers cannot be
compared, as they are two siblings children of the commit tagged
C<v1.7.1>). For practical purposes, the version-comparison methods
declares them equal.
Therefore:
# this is true
lt_git( '1.8.5.4.8.g7c9b668', '1.8.5.4.19.g5032098' );
gt_git( '1.3.GIT', '1.3.0' );
# this is false
ne_git( '1.7.1.1.gc8c07', '1.7.1.1.g5f35a' );
gt_git( '1.3.GIT', '1.3.1' );
If one were to compute the set of all possible version numbers (as returned
by C<git --version>) for all git versions that can be compiled from each
commit in the F<git.git> repository, the result would not be a totally ordered
set. Big deal.
Also, don't be too precise when requiring the minimum version of Git that
supported a given feature. The precise commit in git.git at which a given
feature was added doesn't mean as much as the release branch in which that
commit was merged.
=head1 SEE ALSO
L<Test::Requires::Git>, for defining Git version requirements in test
scripts that need B<git>.
=head1 COPYRIGHT
Copyright 2016-2023 Philippe Bruhat (BooK), all rights reserved.
=head1 LICENSE
This program is free software; you can redistribute it and/or modify it
under the same terms as Perl itself.
=cut