Blame rpmsort

Bernhard M. Wiedemann 1250f9
#! /usr/bin/perl -w
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
# This program is free software; you can redistribute it and/or
Bernhard M. Wiedemann 1250f9
# modify it under the terms of the GNU General Public License
Bernhard M. Wiedemann 1250f9
# as published by the Free Software Foundation; either version 2
Bernhard M. Wiedemann 1250f9
# of the License, or (at your option) any later version.
Bernhard M. Wiedemann 1250f9
#
Bernhard M. Wiedemann 1250f9
# This program is distributed in the hope that it will be useful,
Bernhard M. Wiedemann 1250f9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
Bernhard M. Wiedemann 1250f9
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Bernhard M. Wiedemann 1250f9
# GNU General Public License for more details.
Bernhard M. Wiedemann 1250f9
#
Bernhard M. Wiedemann 1250f9
# You should have received a copy of the GNU General Public License
Bernhard M. Wiedemann 1250f9
# along with this program; if not, write to the Free Software
Bernhard M. Wiedemann 1250f9
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307,
Bernhard M. Wiedemann 1250f9
# USA.
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
use Getopt::Long qw(:config gnu_getopt);
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
sub do_rpm_cmp_versions {
Bernhard M. Wiedemann 1250f9
    my ($evr1, $evr2) = @_;
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
    sub _rpm_cmp {
Bernhard M. Wiedemann 1250f9
	my ($s1, $s2) = @_;
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
	return defined $s1 <=> defined $s2
Bernhard M. Wiedemann 1250f9
	    unless defined $s1 && defined $s2;
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
	my ($r, $x1, $x2);
Bernhard M. Wiedemann 1250f9
	do {
Bernhard M. Wiedemann 1250f9
	    $s1 =~ s/^[^a-zA-Z0-9]+//;
Bernhard M. Wiedemann 1250f9
	    $s2 =~ s/^[^a-zA-Z0-9]+//;
Bernhard M. Wiedemann 1250f9
	    if ($s1 =~ /^\d/ || $s2 =~ /^\d/) {
Bernhard M. Wiedemann 1250f9
		$s1 =~ s/^(0*(\d*))//;  $x1 = $2;
Bernhard M. Wiedemann 1250f9
		return -1 if $1 eq '';
Bernhard M. Wiedemann 1250f9
		$s2 =~ s/^(0*(\d*))//;  $x2 = $2;
Bernhard M. Wiedemann 1250f9
		return 1 if $1 eq '';
Bernhard M. Wiedemann 1250f9
		$r = length $x1 <=> length $x2 || $x1 cmp $x2;
Bernhard M. Wiedemann 1250f9
	    } else {
Bernhard M. Wiedemann 1250f9
		$s1 =~ s/^([a-zA-Z]*)//;  $x1 = $1;
Bernhard M. Wiedemann 1250f9
		$s2 =~ s/^([a-zA-Z]*)//;  $x2 = $1;
Bernhard M. Wiedemann 1250f9
		return 0
Bernhard M. Wiedemann 1250f9
		    if $x1 eq '' && $x2 eq '';
Bernhard M. Wiedemann 1250f9
		$r = $x1 cmp $x2;
Bernhard M. Wiedemann 1250f9
	    }
Bernhard M. Wiedemann 1250f9
	} until $r;
Bernhard M. Wiedemann 1250f9
	return $r;
Bernhard M. Wiedemann 1250f9
    }
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
    my ($e1, $v1, $r1) = $evr1 =~ /^(?:(\d*):)?(.*?)(?:-([^-]*))?$/;
Bernhard M. Wiedemann 1250f9
    my ($e2, $v2, $r2) = $evr2 =~ /^(?:(\d*):)?(.*?)(?:-([^-]*))?$/;
Bernhard M. Wiedemann 1250f9
    my $r = _rpm_cmp($e1 || 0, $e2 || 0);
Bernhard M. Wiedemann 1250f9
    $r = _rpm_cmp($v1, $v2)
Bernhard M. Wiedemann 1250f9
	unless $r;
Bernhard M. Wiedemann 1250f9
    $r = _rpm_cmp($r1, $r2)
Bernhard M. Wiedemann 1250f9
	unless $r;
Bernhard M. Wiedemann 1250f9
    return $r;
Bernhard M. Wiedemann 1250f9
}
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
my $reorder = sub { return @_ };
Bernhard M. Wiedemann 1250f9
my $key = 0;
Bernhard M. Wiedemann 1250f9
my $test = 0;
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
sub rpm_cmp_versions {
Bernhard M. Wiedemann 1250f9
	my ($evr1, $evr2) = @_;
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
	chomp($evr1, $evr2);
Bernhard M. Wiedemann 1250f9
	my $res1 = do_rpm_cmp_versions($evr1, $evr2);
Bernhard M. Wiedemann 1250f9
	if ($test) {
Bernhard M. Wiedemann 1250f9
		open(my $fd, '-|', 'zypper', '--terse', 'versioncmp',
Bernhard M. Wiedemann 1250f9
			$evr1, $evr2) or die "zypper: $!\n";
Bernhard M. Wiedemann 1250f9
		my $res2 = <$fd>;
Bernhard M. Wiedemann 1250f9
		close($fd) or die "zypper: $!\n";
Bernhard M. Wiedemann 1250f9
		chomp $res2;
Bernhard M. Wiedemann 1250f9
		if ($res1 != $res2) {
Bernhard M. Wiedemann 1250f9
			my @operators = qw(< == >);
Bernhard M. Wiedemann 1250f9
			my $op1 = $operators[$res1 + 1];
Bernhard M. Wiedemann 1250f9
			my $op2 = $operators[$res2 + 1];
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
			print STDERR "BUG: $evr1 $op1 $evr2 vs. zypper: $evr1 $op2 $evr2\n";
Bernhard M. Wiedemann 1250f9
		}
Bernhard M. Wiedemann 1250f9
	}
Bernhard M. Wiedemann 1250f9
	return $res1;
Bernhard M. Wiedemann 1250f9
}
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
GetOptions ("r|reverse"	    => sub { $reorder = sub { return reverse @_ } },
Bernhard M. Wiedemann 1250f9
	    "k|key=i"	    => \$key,
Bernhard M. Wiedemann 1250f9
	    "test"        => \$test)
Bernhard M. Wiedemann 1250f9
or do {
Bernhard M. Wiedemann 1250f9
    print STDERR "Usage $0 [-r, --reverse] [-k N, --key=N] [--test]\n";
Bernhard M. Wiedemann 1250f9
    exit 1;
Bernhard M. Wiedemann 1250f9
};
Bernhard M. Wiedemann 1250f9
Bernhard M. Wiedemann 1250f9
if ($key == 0) {
Bernhard M. Wiedemann 1250f9
    # Sort by entire lines
Bernhard M. Wiedemann 1250f9
    map { print } &$reorder(sort { rpm_cmp_versions($a, $b) } <>);
Bernhard M. Wiedemann 1250f9
} else {
Bernhard M. Wiedemann 1250f9
    # Sort by field $key
Bernhard M. Wiedemann 1250f9
    my @data = map { [(split)[$key-1], $_] } <>;
Bernhard M. Wiedemann 1250f9
    map { print } &$reorder(map { $_->[1] }
Bernhard M. Wiedemann 1250f9
        sort { rpm_cmp_versions($a->[0], $b->[0]) } @data);
Bernhard M. Wiedemann 1250f9
}