# ex:ts=8 sw=4: # $OpenBSD: PkgSpec.pm,v 1.15 2007/06/04 20:48:23 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::PkgSpec; # all the shit that does handle package specifications sub compare_pseudo_numbers { my ($n, $m) = @_; my ($n1, $m1); if ($n =~ m/^(\d+)(.*)$/o) { $n1 = $1; $n = $2; } if ($m =~ m/^(\d+)(.*)$/o) { $m1 = $1; $m = $2; } if ($n1 == $m1) { return $n cmp $m; } else { return $n1 <=> $m1; } } sub dewey_compare { my ($a, $b) = @_; my ($pa, $pb); unless ($b =~ m/p\d+$/o) { # does the Dewey hold a p ? $a =~ s/p\d+$//o; # No -> strip it from version. } return 0 if $a =~ /^$b$/; # bare equality if ($a =~ s/p(\d+)$//o) { # extract patchlevels $pa = $1; } if ($b =~ s/p(\d+)$//o) { $pb = $1; } my @a = split(/\./o, $a); push @a, $pa if defined $pa; # ... and restore them my @b = split(/\\\./o, $b); push @b, $pb if defined $pb; while (@a > 0 && @b > 0) { my $va = shift @a; my $vb = shift @b; next if $va eq $vb; return compare_pseudo_numbers($va, $vb); } if (@a > 0) { return 1; } else { return -1; } } sub check_version { my ($v, $spec) = @_; local $_; # any version spec return 1 if $spec eq '.*'; my @specs = split(/\,/o, $spec); for (grep /^\d/o, @specs) { # exact number: check match return 1 if $v =~ /^$_$/; return 1 if $v =~ /^${_}p\d+$/; # allows for recent patches } # Last chance: dewey specs ? my @deweys = grep !/^\d/o, @specs; for (@deweys) { if (m/^(\<\=|\>\=|\<|\>)(.*)$/o) { my ($op, $dewey) = ($1, $2); my $compare = dewey_compare($v, $dewey); 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; } else { return 0; # unknown spec type } } return @deweys == 0 ? 0 : 1; } sub check_1flavor { my ($f, $spec) = @_; local $_; for (split /\-/o, $spec) { # must not be here if (m/^\!(.*)$/o) { return 0 if $f->{$1}; # must be here } else { return 0 unless $f->{$_}; } } return 1; } sub check_flavor { my ($f, $spec) = @_; local $_; # no flavor constraints return 1 if $spec eq ''; $spec =~ s/^-//o; # retrieve all flavors my %f = map +($_, 1), split /\-/o, $f; # check each flavor constraint for (split /\,/o, $spec) { if (check_1flavor(\%f, $_)) { return 1; } } return 0; } sub subpattern_match { my ($p, $list) = @_; local $_; my ($stemspec, $vspec, $flavorspec); # then, guess at where the version number is if any, # this finds patterns like -<=2.3,>=3.4.p1- # the only constraint is that the actual number # - must start with a digit, # - not contain - or , if ($p =~ m/^(.*?)\-((?:\>|\>\=|\<|\<\=)?\d[^-]*)(.*)$/o) { ($stemspec, $vspec, $flavorspec) = ($1, $2, $3); # `any version' matcher } elsif ($p =~ m/^(.*?)\-\*(.*)$/o) { ($stemspec, $vspec, $flavorspec) = ($1, '*', $2); # okay, so no version marker. Assume no flavor spec. } else { ($stemspec, $vspec, $flavorspec) = ($p, '', ''); } $stemspec =~ s/\./\\\./go; $stemspec =~ s/\+/\\\+/go; $stemspec =~ s/\*/\.\*/go; $stemspec =~ s/\?/\./go; $vspec =~ s/\./\\\./go; $vspec =~ s/\+/\\\+/go; $vspec =~ s/\*/\.\*/go; $vspec =~ s/\?/\./go; $p = $stemspec; $p.="-.*" if $vspec ne ''; # First trim down the list my @l = grep {/^$p$/} @$list; my @result = (); # Now, have to extract the version number, and the flavor... for (@l) { my ($stem, $v, $flavor); if (m/^(.*?)\-(\d[^-]*)(.*)$/o) { ($stem, $v, $flavor) = ($1, $2, $3); if ($stem =~ m/^$stemspec$/ && check_version($v, $vspec) && check_flavor($flavor, $flavorspec)) { push(@result, $_); } } else { if ($vspec eq '') { push(@result, $_); } } } return @result; } 1;