Youri::Media::URPM - URPM-based media implementation


Youri-Media documentation Contained in the Youri-Media distribution.

Index


Code Index:

NAME

Top

Youri::Media::URPM - URPM-based media implementation

DESCRIPTION

Top

This is an URPM-based Youri::Media implementation.

It can be created either from local or remote full (hdlist) or partial (synthesis) compressed header files, or from a package directory. File-based inputs are only usable with this latest option.

CLASS METHODS

Top

new(%args)

Creates and returns a new Youri::Media::URPM object.

Specific parameters:

synthesis $synthesis

Path, URL or list of path or URL of synthesis file used for creating this media. If a list is given, the first successfully accessed will be used, so as to allow better reliability.

hdlist $hdlist

Path, URL or list of path or URL of hdlist file used for creating this media. If a list is given, the first successfully accessed will be used, so as to allow better reliability.

path $path

Path of package directory used for creating this media.

max_age $age

Maximum age of packages for this media.

rpmlint_config $file

rpmlint configuration file for this media.

preload $level

Allows to parse headers at object creation, rather than lazily if really needed. (default: Youri::Media::URPM::None)

cache true/false

Cache headers once parsed (default: true)

In case of multiple synthesis, hdlist and path options given, they will be tried in this order, so as to minimize parsing time.

INSTANCE METHODS

Top

get_hdlist()

Returns hdlist used for creating this media, if any.

get_path()

Returns path used for creating this media, if any.

COPYRIGHT AND LICENSE

Top


Youri-Media documentation Contained in the Youri-Media distribution.
# $Id: /mirror/youri/soft/Media/trunk/lib/Youri/Media/URPM.pm 2336 2007-03-23T09:11:26.464388Z guillomovitch  $
package Youri::Media::URPM;

use strict;
use warnings;
use Carp;
use File::Find;
use File::Temp qw/tempfile/;
use LWP::Simple;
use URPM;
use Youri::Package::RPM::URPM;

use constant {
    NONE    => 0,
    PARTIAL => 1,
    FULL    => 2
};

use base 'Youri::Media';

sub _init {
    my $self   = shift;

    my %options = (
        hdlist    => '',
        synthesis => '',
        path      => '',
        preload   => NONE,
        cache     => 1,
        @_
    );

    # check options
    if ($options{path}) {
        if (! -d $options{path}) {
            carp "non-existing directory $options{path}, dropping";
        } elsif (! -r $options{path}) {
            carp "non-readable directory $options{path}, dropping";
        } else {
            $self->{_path} = $options{path};
        }
    }

    # find source
    SOURCE: {
        if ($options{synthesis}) {
            foreach my $file (
                ref $options{synthesis} eq 'ARRAY' ?
                    @{$options{synthesis}} :
                    $options{synthesis}
            ) {
                my $synthesis = $self->_get_file($file);
                if ($synthesis) {
                    $self->{_synthesis} = $synthesis;
                    last SOURCE;
                }
            }
        }

        if ($options{hdlist}) { 
            foreach my $file (
                ref $options{hdlist} eq 'ARRAY' ?
                    @{$options{hdlist}} :
                    $options{hdlist}
            ) {
                my $hdlist = $self->_get_file($file);
                if ($hdlist) {
                    $self->{_hdlist} = $hdlist;
                    last SOURCE;
                }
            }
        }

        if ($self->{_path}) {
            last SOURCE;
        }
        
        croak "no source specified";
    }

    if ($options{preload} && !$options{cache}) {
        carp "Forcing caching as preloading specified\n";
        $options{cache} = 1;
    }

    $self->_parse_headers($options{preload}) if $options{preload};
    $self->{_level} = $options{preload};
    $self->{_cache} = $options{cache};

    return $self;
}

sub _remove_all_archs {
    my ($self) = @_;
    croak "Not a class method" unless ref $self;

    $self->{_urpm}->{depslist} = [];
}

sub _remove_archs {
    my ($self, $skip_archs) = @_;
    croak "Not a class method" unless ref $self;

    my $urpm = $self->{_urpm};
    $urpm->{depslist} = [
         grep { ! $skip_archs->{$_->arch()} } @{$urpm->{depslist}}
    ];
}

sub get_package_class {
    my ($self) = @_;
    croak "Not a class method" unless ref $self;

    return "Youri::Package::RPM::URPM";
}

sub traverse_files {
    my ($self, $function) = @_;
    croak "Not a class method" unless ref $self;
    croak "No files for this media" unless $self->{_path};

    my $callback = sub {
        return unless -f $File::Find::name;
        return unless -r $File::Find::name;
        return unless $_ =~ /\.rpm$/;

        my $package = Youri::Package::RPM::URPM->new(file => $File::Find::name);
        return if $self->{_skip_archs}->{$package->get_arch()};

        $function->($File::Find::name, $package);
    };

    find($callback, $self->{_path});
}

sub traverse_headers {
    my ($self, $function) = @_;
    croak "Not a class method" unless ref $self;

    # lazy initialisation
    if ($self->{_level} < PARTIAL) {
        $self->_parse_headers(PARTIAL);
        $self->{_level} = PARTIAL;
    }

    $self->{_urpm}->traverse(sub {
        $function->(Youri::Package::RPM::URPM->new(header => $_[0]));
    });

    # remove headers if needed
    if (!$self->{_cache}) {
        $self->{_urpm} = undef;
        $self->{_level} = NONE;
    }
}

sub traverse_full_headers {
    my ($self, $function) = @_;
    croak "Not a class method" unless ref $self;

    # lazy initialisation
    if ($self->{_level} < FULL) {
        $self->_parse_headers(FULL);
        $self->{_level} = FULL;
    }

    $self->{_urpm}->traverse(sub {
        $function->(Youri::Package::RPM::URPM->new(header => $_[0]));
    });

    # remove headers if needed
    if (!$self->{_cache}) {
        $self->{_urpm} = undef;
        $self->{_level} = NONE;
    }
}

sub _parse_headers {
    my ($self, $level) = @_;

    # return if level is NONE
    return unless $level;

    $self->{_urpm} = URPM->new();
    CASE: {
        if ($self->{_synthesis}) {
            print "Parsing synthesis $self->{_synthesis}\n"
                if $self->{_verbose};
            $self->{_urpm}->parse_synthesis(
                $self->{_synthesis}, keep_all_tags => ($level == FULL)
            );
            last CASE;
        }

        if ($self->{_hdlist}) {
            print "Parsing hdlist $self->{_hdlist}\n"
                if $self->{_verbose};
            $self->{_urpm}->parse_hdlist(
                $self->{_hdlist}, keep_all_tags => ($level == FULL)
            );
            last CASE;
        }

        if ($self->{_path}) {
            print "Parsing directory $self->{_path} content\n"
                if $self->{_verbose};

            my $pattern = qr/\.rpm$/;

            my $parse = sub {
                return unless -f $File::Find::name;
                return unless -r $File::Find::name;
                return unless $_ =~ $pattern;

                $self->{_urpm}->parse_rpm(
                    $File::Find::name, keep_all_tags => ($level == FULL)
                );
            };

            find($parse, $self->{_path});
            last CASE;
        }
    }
}


sub get_hdlist {
    my ($self) = @_;
    croak "Not a class method" unless ref $self;

    return $self->{_hdlist};
}

sub get_path {
    my ($self) = @_;
    croak "Not a class method" unless ref $self;

    return $self->{_path};
}

sub _get_file {
    my ($self, $file) = @_;

    if ($file =~ /^(?:http|ftp):\/\/.*$/) {
        my ($handle, $name) = tempfile(UNLINK =>1 );
        print "Attempting to download $file as $name\n" if $self->{_verbose};
        my $status = getstore($file, $name);
        unless (is_success($status)) {
            carp "invalid URL $file: $status";
            return;
        }
        return $name;
    } else {
        unless (-f $file) {
            carp "non-existing file $file";
            return;
        }
        unless (-r $file) {
            carp "non-readable file $file";
            return;
        }
        return $file;
    }
}

1;