X-Git-Url: http://git.vpit.fr/?a=blobdiff_plain;f=lib%2FCPANPLUS%2FDist%2FGentoo%2FAtom.pm;h=543ece4f61f0dd3aa35aa4afb5ccd62c2d29678c;hb=d77142c8216a27e216565391d32d2cf3f321f63c;hp=ebe6a4e64d8e393c58c5af2ae3c7d6608becf8d0;hpb=9c0162fc729f9b0cc7bff16ef1c4aa07cf9c2ab9;p=perl%2Fmodules%2FCPANPLUS-Dist-Gentoo.git diff --git a/lib/CPANPLUS/Dist/Gentoo/Atom.pm b/lib/CPANPLUS/Dist/Gentoo/Atom.pm index ebe6a4e..543ece4 100644 --- a/lib/CPANPLUS/Dist/Gentoo/Atom.pm +++ b/lib/CPANPLUS/Dist/Gentoo/Atom.pm @@ -3,19 +3,47 @@ package CPANPLUS::Dist::Gentoo::Atom; use strict; use warnings; -our $VERSION = '0.08'; +=head1 NAME + +CPANPLUS::Dist::Gentoo::Atom - Gentoo atom object. + +=head1 VERSION + +Version 0.10 + +=cut + +our $VERSION = '0.10'; + +=head1 DESCRIPTION + +This class models Gentoo atoms. + +=cut use Carp (); use Scalar::Util (); use overload ( - '<=>' => \&cmp, - '""' => \&as_string, + '<=>' => \&_spaceship, + 'cmp' => \&_cmp, + '""' => \&_stringify, ); use CPANPLUS::Dist::Gentoo::Version; -my $version_rx = $CPANPLUS::Dist::Gentoo::Version::version_rx; +my $range_rx = qr/(?:<|<=|=|>=|>)/; +my $name_rx = qr/[a-zA-Z0-9_+-]+/; +my $category_rx = $name_rx; +my $version_rx = $CPANPLUS::Dist::Gentoo::Version::version_rx; + +=head1 METHODS + +=head2 C<< new category => $category, name => $name [, version => $version, range => $range, ebuild => $ebuild ] >> + +Creates a new L object from the supplied C<$category>, C<$name>, C<$version>, C<$range> and C<$ebuild>. + +=cut sub new { my $class = shift; @@ -23,114 +51,284 @@ sub new { my %args = @_; - my ($category, $name, $version); + my ($range, $category, $name, $version); if (defined $args{name}) { - ($category, $name, $version) = @args{qw/category name version/}; + ($range, $category, $name, $version) = @args{qw}; Carp::confess('Category unspecified') unless defined $category; - /[^\w-]/ and Carp::confess('Invalid argument') for $name, $category; + Carp::confess('Invalid category') unless $category =~ /^$category_rx$/o; + Carp::confess('Invalid name') unless $name =~ /^$name_rx$/o; } elsif (defined $args{atom}) { my $atom = $args{atom}; - $atom =~ m{^([\w-]+)/([\w-]+)-v?($version_rx)$} + $atom =~ m{^($range_rx)?($category_rx)/($name_rx)(?:-($version_rx))?$}o or Carp::confess('Invalid atom'); - ($category, $name, $version) = ($1, $2, $3); - } elsif (defined $args{ebuild}) { - my $ebuild = $args{ebuild}; - $ebuild =~ m{/([\w-]+)/([\w-]+)/\2-v?($version_rx)\.ebuild$} - or Carp::confess('Invalid ebuild'); - ($category, $name, $version) = ($1, $2, $3); + ($range, $category, $name, $version) = ($1, $2, $3, $4); } else { Carp::confess('Not enough information for building an atom object'); } if (defined $version) { unless (Scalar::Util::blessed($version) - and $_->isa('CPANPLUS::Dist::Gentoo::Version')) { + and $version->isa('CPANPLUS::Dist::Gentoo::Version')) { $version = CPANPLUS::Dist::Gentoo::Version->new($version); } } - Carp::confess('Minimum atoms require a valid version') if not defined $version - and $args{minimum}; + if (defined $version) { + if (defined $range) { + Carp::confess("Invalid range $range") unless $range =~ /^$range_rx$/o; + } else { + $range = '>='; + } + } else { + Carp::confess('Range atoms require a valid version') + if defined $range and length $range; + } bless { category => $category, name => $name, version => $version, - minimum => $args{minimum}, + range => $range, ebuild => $args{ebuild}, }, $class; } +=head2 C + +Creates a new L object by inferring the category, name and version from the given C<$ebuild> + +=cut + +sub new_from_ebuild { + my $class = shift; + $class = ref($class) || $class; + + my $ebuild = shift; + $ebuild = '' unless defined $ebuild; + + $ebuild =~ m{/($category_rx)/($name_rx)/\2-($version_rx)\.ebuild$}o + or Carp::confess('Invalid ebuild'); + my ($category, $name, $version) = ($1, $2, $3); + + return $class->new( + category => $category, + name => $name, + version => $version, + ebuild => $ebuild, + ); +} + BEGIN { - eval "sub $_ { \$_[0]->{$_} }" for qw/category name version minimum ebuild/; + eval "sub $_ { \$_[0]->{$_} }" for qw; } -sub cmp { +=head2 C + +Read-only accessor to the atom category. + +=head2 C + +Read-only accessor to the atom name. + +=head2 C + +Read-only accessor to the L object associated with the atom. + +=head2 C + +Read-only accessor to the atom range. + +=head2 C + +Read-only accessor to the path of an optional ebuild associated with the atom. + +=head2 C + +Returns the qualified name for the atom, i.e. C<$category/$name>. + +=cut + +sub qualified_name { join '/', $_[0]->category, $_[0]->name } + +sub _spaceship { my ($a1, $a2, $r) = @_; my $v1 = $a1->version; my $v2; - if (Scalar::Util::blessed($a2) and $a2->isa(__PACKAGE__)) { - Carp::confess('Can\'t compare atoms of different packages') - if $a1->category ne $a2->category or $a1->name ne $a2->name; + my $blessed = Scalar::Util::blessed($a2); + unless ($blessed and $a2->isa(__PACKAGE__)) { + if ($blessed and $a2->isa('CPANPLUS::Dist::Gentoo::Version')) { + $v2 = $a2; + $a2 = undef; + } else { + my $maybe_atom = eval { __PACKAGE__->new(atom => $a2) }; + if (my $err = $@) { + $v2 = eval { CPANPLUS::Dist::Gentoo::Version->new($a2) }; + Carp::confess("Can't compare an atom against something that's not an atom, an atom string ($err), a version or a version string ($@)") if $@; + $a2 = undef; + } else { + $a2 = $maybe_atom; + } + } + } + + if (defined $a2) { $v2 = $a2->version; - } else { - $v2 = $a2; + + my $p1 = $a1->qualified_name; + my $p2 = $a2->qualified_name; + Carp::confess("Atoms for different packages $p1 and $p2") unless $p1 eq $p2; } ($v1, $v2) = ($v2, $v1) if $r; + return (defined $v1 or 0) <=> (defined $v2 or 0) unless defined $v1 + and defined $v2; + return $v1 <=> $v2; } -sub as_string { +sub _cmp { + my ($a1, $a2, $r) = @_; + + if (defined $a2) { + my $p1 = $a1->qualified_name; + + unless (Scalar::Util::blessed($a2) && $a2->isa(__PACKAGE__)) { + $a2 = eval { __PACKAGE__->new(atom => $a2) }; + Carp::confess("Can't compare an atom against something that's not an atom or an atom string ($@)") if $@; + } + my $p2 = $a2->qualified_name; + + if (my $c = $p1 cmp $p2) { + return $r ? -$c : $c; + } + } + + $a1 <=> $a2; +} + +sub _stringify { my ($a) = @_; - my $atom = $a->category . '/' . $a->name; + my $atom = $a->qualified_name; my $version = $a->version; - if (defined $version) { - $atom = "=$atom-$version"; - $atom = ">$atom" if $a->minimum; - } + $atom = $a->range . $atom . '-' . $version if defined $version; return $atom; } +my %order = ( + '<' => -2, + '<=' => -1, + '=' => 0, + '>=' => 1, + '>' => 2, +); + +=head2 C + +Compute the ranged atom representing the logical AND between C<@atoms> with the same category and name. + +=cut + +sub and { + shift unless length ref $_[0]; + + my $a1 = shift; + return $a1 unless @_; + + my $a2 = shift; + $a2 = $a2->and(@_) if @_; + + my $p1 = $a1->qualified_name; + my $p2 = $a2->qualified_name; + Carp::confess("Atoms for different packages $p1 and $p2") unless $p1 eq $p2; + + my $v1 = $a1->version; + return $a2 unless defined $v1; + my $r1 = $a1->range; # Defined if $v1 is defined + + my $v2 = $a2->version; + return $a1 unless defined $v2; + my $r2 = $a2->range; # defined if $v2 is defined + + my $o1 = $order{$r1}; + my $o2 = $order{$r2}; + + Carp::confess("Incompatible ranges $r1$p1 and $r2$p2") if $o1 * $o2 < 0; + + if ($r2 eq '=') { + ($a1, $a2) = ($a2, $a1); + ($v1, $v2) = ($v2, $v1); + ($r1, $r2) = ($r2, $r1); + ($o1, $o2) = ($o2, $o1); + } + + if ($r1 eq '=') { + my $r = $r2 eq '=' ? '==' : $r2; + Carp::confess("Version mismatch $v1 $r $v2") unless eval "\$a1 $r \$a2"; + return $a1; + } elsif ($o1 > 0) { + return $a1 < $a2 ? $a2 : $a1; + } else { + return $a1 < $a2 ? $a1 : $a2; + } +} + +=head2 C + +Returns a list built from C<@atoms> but where there's only one atom for a given category and name. + +=cut + sub fold { shift unless length ref $_[0]; my %seen; for my $atom (@_) { - my ($category, $name, $version) = map $atom->$_, qw/category name version/; - my $key = join '/', $category, $name; + my $key = $atom->qualified_name; + my $cur = $seen{$key}; + $seen{$key} = defined $cur ? $cur->and($atom) : $atom; + } - unless (defined $cur) { - $seen{$key} = $atom; - next; - } + return map $seen{$_}, sort keys %seen; +} - next unless defined $version; +=pod - if (not defined $cur->version) { - $seen{$key} = $atom; - next; - } +This class provides overloaded methods for numerical comparison, string comparison and strigification. - if ($atom->minimum) { - if ($cur->minimum) { - $seen{$key} = $atom < $cur ? $cur : $atom; - } else { - Carp::confess('Version mismatch') if $atom > $cur; - } - } elsif ($cur->minimum) { - Carp::confess('Version mismatch') if $cur > $atom; - } - } +=head1 SEE ALSO - return values %seen; -} +L, L. + +=head1 AUTHOR + +Vincent Pit, C<< >>, L. + +You can contact me by mail or on C (vincent). + +=head1 BUGS + +Please report any bugs or feature requests to C, or through the web interface at L. +I will be notified, and then you'll automatically be notified of progress on your bug as I make changes. + +=head1 SUPPORT + +You can find documentation for this module with the perldoc command. + + perldoc CPANPLUS::Dist::Gentoo + +=head1 COPYRIGHT & LICENSE + +Copyright 2009,2010 Vincent Pit, all rights reserved. + +This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. + +=cut -1; +1; # End of CPANPLUS::Dist::Gentoo::Atom