# ex:ts=8 sw=4: # $OpenBSD: PackageName.pm,v 1.33 2009/03/07 12:04:13 espie Exp $ # # Copyright (c) 2003-2007 Marc Espie # # Permission to use, copy, modify, and distribute this software for any # purpose with or without fee is hereby granted, provided that the above # copyright notice and this permission notice appear in all copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. use strict; use warnings; package OpenBSD::PackageName; sub url2pkgname($) { my $name = $_[0]; $name =~ s|.*/||; $name =~ s|\.tgz$||; return $name; } # see packages-specs(7) sub splitname { my $_ = shift; if (/^(.*?)\-(\d.*)$/o) { my $stem = $1; my $rest = $2; my @all = split /\-/o, $rest; return ($stem, @all); } else { return ($_); } } sub from_string { my ($class, $_) = @_; if (/^(.*?)\-(\d.*)$/o) { my $stem = $1; my $rest = $2; my @all = split /\-/o, $rest; my $version = OpenBSD::PackageName::version->from_string(shift @all); my %flavors = map {($_,1)} @all; return bless { stem => $stem, version => $version, flavors => \%flavors, }, "OpenBSD::PackageName::Name"; } else { return bless { stem => $_, }, "OpenBSD::PackageName::Stem"; } } sub splitstem { my $_ = shift; if (/^(.*?)\-\d/o) { return $1; } else { return $_; } } sub is_stem { my $_ = shift; if (m/\-\d/o || $_ eq '-') { return 0; } else { return 1; } } sub splitp { my $_ = shift; if (/^(.*\-\d[^-]*)p(\d+)(.*)$/o) { return ($1.$3, $2); } else { return ($_,-1); } } sub rebuildp { my ($pkg, $p) = @_; if ($p == -1) { return $pkg; } if ($pkg =~ m/^(.*?)(\-\d[^-v]*)(.*)$/o) { return "$1$2p$p$3"; } else { return $pkg."p".$p; } } sub keep_most_recent { my $h = {}; for my $pkgname (@_) { my ($p, $v) = splitp($pkgname); if (!defined $h->{$p} || $h->{$p} < $v) { $h->{$p} = $v; } } my @list = (); while (my ($p, $v) = each %$h) { push(@list, rebuildp($p, $v)); } return @list; } sub compile_stemlist { my $hash = {}; for my $n (@_) { my $stem = splitstem($n); $hash->{$stem} = {} unless defined $hash->{$stem}; $hash->{$stem}->{$n} = 1; } bless $hash, "OpenBSD::PackageLocator::_compiled_stemlist"; } sub avail2stems { my @avail = @_; if (@avail == 0) { require OpenBSD::Error; OpenBSD::Error::Warn("No packages available in the PKG_PATH\n"); } return OpenBSD::PackageName::compile_stemlist(@avail); } package OpenBSD::PackageLocator::_compiled_stemlist; sub find { my ($self, $stem) = @_; return keys %{$self->{$stem}}; } sub add { my ($self, $pkgname) = @_; my $stem = OpenBSD::PackageName::splitstem($pkgname); $self->{$stem}->{$pkgname} = 1; } sub delete { my ($self, $pkgname) = @_; my $stem = OpenBSD::PackageName::splitstem($pkgname); delete $self->{$stem}->{$pkgname}; if(keys %{$self->{$stem}} == 0) { delete $self->{$stem}; } } sub find_partial { my ($self, $partial) = @_; my @result = (); while (my ($stem, $pkgs) = each %$self) { next unless $stem =~ /\Q$partial\E/i; push(@result, keys %$pkgs); } return @result; } package OpenBSD::PackageName::version; sub make_dewey { my $o = shift; $o->{deweys} = [ split(/\./o, $o->{string}) ]; for my $suffix (qw(rc beta pre pl)) { if ($o->{deweys}->[-1] =~ m/^(\d+)$suffix(\d*)$/) { $o->{deweys}->[-1] = $1; $o->{$suffix} = $2; } } } sub from_string { my ($class, $string) = @_; my $vnum = -1; my $pnum = -1; if ($string =~ m/^(.*)v(\d+)$/o) { $vnum = $2; $string = $1; } if ($string =~ m/^(.*)p(\d+)$/o) { $pnum = $2; $string = $1; } my $o = bless { pnum => $pnum, vnum => $vnum, string => $string, }, $class; $o->make_dewey; return $o; } sub to_string { my $o = shift; my $string = $o->{string}; if ($o->{pnum} > -1) { $string .= 'p'.$o->{pnum}; } if ($o->{vnum} > -1) { $string .= 'v'.$o->{vnum}; } return $string; } sub pnum_compare { my ($a, $b) = @_; return $a->{pnum} <=> $b->{pnum} } sub compare { my ($a, $b) = @_; # Simple case: epoch number if ($a->{vnum} != $b->{vnum}) { return $a->{vnum} <=> $b->{vnum}; } # Simple case: only p number differs if ($a->{string} eq $b->{string}) { return $a->pnum_compare($b); } # Try a diff in dewey numbers first for (my $i = 0; ; $i++) { if (!defined $a->{deweys}->[$i]) { if (!defined $b->{deweys}->[$i]) { return 0; } else { return -1; } } if (!defined $b->{deweys}->[$i]) { return 1; } my $r = dewey_compare($a->{deweys}->[$i], $b->{deweys}->[$i]); return $r if $r != 0; } # finally try all the usual suspects # release candidates and beta and pre releases. for my $suffix (qw(rc beta pre pl)) { my $result = $suffix eq 'pl' ? 1 : -1; if (defined $a->{$suffix} && defined $b->{$suffix}) { return $a->{$suffix} <=> $b->{$suffix}; } if (defined $a->{$suffix} && !defined $b->{$suffix}) { return $result; } if (!defined $a->{$suffix} && defined $b->{$suffix}) { return -$result; } } # give up: we don't know how to make a difference return 0; } sub dewey_compare { my ($a, $b) = @_; # numerical comparison if ($a =~ m/^\d+$/o and $b =~ m/^\d+$/o) { return $a <=> $b; } # added lowercase letter if ("$a.$b" =~ m/^(\d+)([a-z]?)\.(\d+)([a-z]?)$/o) { my ($an, $al, $bn, $bl) = ($1, $2, $3, $4); if ($an != $bn) { return $an <=> $bn; } else { return $al cmp $bl; } } return $a cmp $b; } package OpenBSD::PackageName::versionspec; our @ISA = qw(OpenBSD::PackageName::version); sub from_string { my ($class, $s) = @_; my ($op, $version) = ('=', $s); if ($s =~ m/^(\>\=|\>|\<\=|\<|\=)(.*)$/) { ($op, $version) = ($1, $2); } my $self = $class->SUPER::from_string($version); $self->{op} = $op; return $self; } sub pnum_compare { my ($spec, $b) = @_; if ($spec->{pnum} == -1) { return 0; } else { return $spec->SUPER::pnum_compare($b); } } sub match { my ($self, $b) = @_; my $op = $self->{op}; my $compare = - $self->compare($b); return 0 if $op eq '<' && $compare >= 0; return 0 if $op eq '<=' && $compare > 0; return 0 if $op eq '>' && $compare <= 0; return 0 if $op eq '>=' && $compare < 0; return 0 if $op eq '=' && $compare != 0; return 1; } package OpenBSD::PackageName::Stem; sub to_string { my $o = shift; return $o->{stem}; } sub to_pattern { my $o = shift; return $o->{stem}.'-*'; } package OpenBSD::PackageName::Name; sub to_string { my $o = shift; return join('-', $o->{stem}, $o->{version}->to_string, sort keys %{$o->{flavors}}); } sub to_pattern { my $o = shift; return join('-', $o->{stem}, '*', sort keys %{$o->{flavors}}); } 1;