home *** CD-ROM | disk | FTP | other *** search
Text File | 2000-03-22 | 84.3 KB | 2,351 lines |
- package PPM;
- require 5.004;
- require Exporter;
-
- @ISA = qw(Exporter);
- @EXPORT = qw(PPMdat PPMERR InstalledPackageProperties ListOfRepositories RemoveRepository AddRepository GetPPMOptions SetPPMOptions InstallPackage RemovePackage VerifyPackage UpgradePackage RepositoryPackages RepositoryPackageProperties QueryInstalledPackages QueryPPD RepositorySummary);
-
- use LWP::UserAgent;
- use LWP::Simple;
-
- use File::Basename;
- use File::Copy;
- use File::Path;
- use ExtUtils::Install;
- use Cwd;
- use Config;
- use PPM::RelocPerl;
- use PPM::SOAPClient;
-
- use XML::PPD;
- use XML::PPMConfig;
- use XML::Parser;
- use Archive::Tar;
-
- use strict;
-
- #set Debug to 1 to debug PPMdat file reading
- # 2 to debug parsing PPDs
- #
- # values may be or'ed together.
- #
- my $Debug = 0;
-
- my $PPMERR;
-
- my ($PPM_ver, $CPU, $OS_VALUE, $OS_VERSION, $LANGUAGE);
-
- # options from data file.
- my ($build_dir, $Ignorecase, $Clean, $Force_install, $Confirm, $Root, $More, $Trace, $TraceFile, $Verbose);
-
- my $TraceStarted = 0;
-
- my %repositories;
- my %cached_ppd_list;
-
- my ($current_root, $orig_root);
-
- # Keys for this hash are package names. It is filled in by a successful
- # call to read_config(). Each package is a hash with the following keys:
- # LOCATION, INST_DATE, INST_ROOT, INST_PACKLIST and INST_PPD.
- my %installed_packages = ();
-
- # Keys for this hash are CODEBASE, INSTALL_HREF, INSTALL_EXEC,
- # INSTALL_SCRIPT, NAME, VERSION, TITLE, ABSTRACT, LICENSE, AUTHOR,
- # UNINSTALL_HREF, UNINSTALL_EXEC, UNINSTALL_SCRIPT, PERLCORE_VER and DEPEND.
- # It is filled in after a successful call to parsePPD().
- my %current_package = ();
- my @current_package_stack;
-
- # this may get overridden by the config file.
- my @required_packages = ('PPM', 'SOAP', 'libnet', 'Archive-Tar', 'Compress-Zlib', 'libwww-perl', 'XML-Parser', 'XML-Element');
-
- # Packages that can't be upgraded on Win9x
- my @Win9x_denied = qw(xml-parser compress-zlib);
- my %Win9x_denied;
- @Win9x_denied{@Win9x_denied} = ();
-
- # ppm.xml location is in the environment variable 'PPM_DAT', else it is in
- # the same place as this script.
- my ($basename, $path) = fileparse($0);
-
- if (defined $ENV{'PPM_DAT'} && -f $ENV{'PPM_DAT'})
- {
- $PPM::PPMdat = $ENV{'PPM_DAT'};
- }
- elsif (-f "$Config{'installsitelib'}/ppm.xml")
- {
- $PPM::PPMdat = "$Config{'installsitelib'}/ppm.xml";
- }
- elsif (-f "$Config{'installprivlib'}/ppm.xml")
- {
- $PPM::PPMdat = "$Config{'installprivlib'}/ppm.xml";
- }
- elsif (-f $path . "/ppm.xml")
- {
- $PPM::PPMdat = $path . $PPM::PPMdat;
- }
- else
- {
- &Trace("Failed to load PPM_DAT file") if $Trace;
- print "Failed to load PPM_DAT file\n";
- return -1;
- }
-
- &Trace("Using config file: $PPM::PPMdat") if $Trace;
-
- my $init = 0;
- chmod(0600, $PPM::PPMdat);
-
- #
- # Exported subs
- #
-
- sub InstalledPackageProperties
- {
- my (%ret_hash, $dep);
- read_config();
- foreach $_ (keys %installed_packages) {
- parsePPD(%{ $installed_packages{$_}{'INST_PPD'} } );
- $ret_hash{$_}{'NAME'} = $_;
- $ret_hash{$_}{'DATE'} = $installed_packages{$_}{'INST_DATE'};
- $ret_hash{$_}{'TITLE'} = $current_package{'TITLE'};
- $ret_hash{$_}{'AUTHOR'} = $current_package{'AUTHOR'};
- $ret_hash{$_}{'VERSION'} = $current_package{'VERSION'};
- $ret_hash{$_}{'ABSTRACT'} = $current_package{'ABSTRACT'};
- $ret_hash{$_}{'PERLCORE_VER'} = $current_package{'PERLCORE_VER'};
- foreach $dep (keys %{$current_package{'DEPEND'}}) {
- push @{$ret_hash{$_}{'DEPEND'}}, $dep;
- }
- }
- return %ret_hash;
- }
-
- sub ListOfRepositories
- {
- my %reps;
-
- read_config();
- foreach (keys %repositories) {
- $reps{$_} = $repositories{$_}{'LOCATION'};
- }
- return %reps;
- # return %repositories;
- }
-
- sub RemoveRepository
- {
- my (%argv) = @_;
- my ($arg, $repository, $save, $loc);
- foreach $arg (keys %argv) {
- if ($arg eq 'repository') { $repository = $argv{$arg}; }
- if ($arg eq 'save') { $save = $argv{$arg}; }
- }
-
- read_config();
- foreach $_ (keys %repositories) {
- if ($_ =~ /^\Q$repository\E$/) {
- &Trace("Removed repository $repositories{$repository}") if $Trace;
- delete $repositories{$repository};
- last;
- }
- }
- if (defined $save && $save != 0) { save_options(); }
- }
-
- sub AddRepository
- {
- my (%argv) = @_;
- my ($arg, $repository, $location, $username, $password, $save);
- foreach $arg (keys %argv) {
- if ($arg eq 'repository') { $repository = $argv{$arg}; }
- if ($arg eq 'location') { $location = $argv{$arg}; }
- if ($arg eq 'username') { $username = $argv{$arg}; }
- if ($arg eq 'password') { $password = $argv{$arg}; }
- if ($arg eq 'save') { $save = $argv{$arg}; }
- }
-
- read_config();
- $repositories{$repository}{'LOCATION'} = $location;
- $repositories{$repository}{'USERNAME'} = $username if defined $username;
- $repositories{$repository}{'PASSWORD'} = $password if defined $password;
- # $repositories{$repository} = $location;
- &Trace("Added repository $location") if $Trace;
- if (defined $save && $save != 0) { save_options(); }
- }
-
- sub GetPPMOptions
- {
- my %ret_hash;
- read_config();
- $ret_hash{'IGNORECASE'} = $Ignorecase;
- $ret_hash{'CLEAN'} = $Clean;
- $ret_hash{'FORCE_INSTALL'} = $Force_install;
- $ret_hash{'CONFIRM'} = $Confirm;
- $ret_hash{'ROOT'} = $Root;
- $ret_hash{'BUILDDIR'} = $build_dir;
- $ret_hash{'MORE'} = $More;
- $ret_hash{'TRACE'} = $Trace;
- $ret_hash{'TRACEFILE'} = $TraceFile;
- $ret_hash{'VERBOSE'} = $Verbose;
- return %ret_hash;
- }
-
- sub SetPPMOptions
- {
- my (%argv) = @_;
- my ($arg, %opts, $save);
- foreach $arg (keys %argv) {
- if ($arg eq 'options') { %opts = %{$argv{$arg}}; }
- if ($arg eq 'save') { $save = $argv{$arg}; }
- }
-
- # set reasonable defaults for any options not explicitly set in %opts
- read_config();
-
- $Ignorecase = $opts{'IGNORECASE'} if defined $opts{'IGNORECASE'};
- $Clean = $opts{'CLEAN'} if defined $opts{'CLEAN'};
- $Force_install = $opts{'FORCE_INSTALL'} if defined $opts{'FORCE_INSTALL'};
- $Confirm = $opts{'CONFIRM'} if defined $opts{'CONFIRM'};
- $Root = $opts{'ROOT'} if defined $opts{'ROOT'};
- $build_dir = $opts{'BUILDDIR'} if defined $opts{'BUILDDIR'};
- $More = $opts{'MORE'} if defined $opts{'MORE'};
- $Trace = $opts{'TRACE'} if defined $opts{'TRACE'};
- $TraceFile = $opts{'TRACEFILE'} if defined $opts{'TRACEFILE'};
- $Verbose = $opts{'VERBOSE'} if defined $opts{'VERBOSE'};
-
- if (defined $save && $save != 0) { save_options(); }
- }
-
- sub UpgradePackage
- {
- my (%argv) = @_;
- my ($arg, $package, $location);
- foreach $arg (keys %argv) {
- if ($arg eq 'package') { $package = $argv{$arg}; }
- if ($arg eq 'location') { $location = $argv{$arg}; }
- }
- return VerifyPackage("package" => $package, "location" => $location, "upgrade" => 1);
- }
-
- # Returns 1 on success, 0 and sets $PPMERR on failure.
- sub InstallPackage
- {
- my (%argv) = @_;
- my ($arg, $package, $location, $root);
- foreach $arg (keys %argv) {
- if ($arg eq 'package') { $package = $argv{$arg}; }
- if ($arg eq 'location') { $location = $argv{$arg}; }
- if ($arg eq 'root') { $root = $argv{$arg}; }
- }
- my ($PPDfile,%PPD);
- $root = $current_root if (!defined($root) && defined($current_root));
-
- read_config();
-
- if (!defined($package) && -d "blib" && -f "Makefile") {
- unless (open MAKEFILE, "< Makefile") {
- $PPM::PPMERR = "Couldn't open Makefile for reading: $!";
- return 0;
- }
- while (<MAKEFILE>) {
- if (/^DISTNAME\s*=\s*(\S+)/) {
- $package = $1;
- $PPDfile = "$1.ppd";
- last;
- }
- }
- close MAKEFILE;
- unless (defined $PPDfile) {
- $PPM::PPMERR = "Couldn't determine local package name";
- return 0;
- }
- system("$Config{make} ppd");
- %PPD = readPPDfile($PPDfile, 'XML::PPD');
- if (!%PPD) { return 0; }
- parsePPD(%PPD);
- $Clean = 0;
- goto InstallBlib;
- }
-
- my $packagefile = $package;
- if (!defined($PPDfile = locatePPDfile($packagefile, $location))) {
- &Trace("Could not locate a PPD file for package $package") if $Trace;
- $PPM::PPMERR = "Could not locate a PPD file for package $package";
- return 0;
- }
- if ($Config{'osname'} eq 'MSWin32' &&
- !&Win32::IsWinNT &&
- exists $Win9x_denied{lc($package)}) {
- $PPM::PPMERR = "Package '$package' cannot be installed with PPM on Win9x--see http://www.ActiveState.com/ppm for details";
- return 0;
- }
- %PPD = readPPDfile($PPDfile, 'XML::PPD');
- if (!%PPD) { return 0; }
-
- parsePPD(%PPD);
- if (!$current_package{'CODEBASE'} && !$current_package{'INSTALL_HREF'}) {
- &Trace("Could not locate a PPM binary of '$package' for this platform") if $Trace;
- $PPM::PPMERR = "Could not locate a PPM binary of '$package' for this platform";
- return 0;
- }
-
- if (defined $current_package{'DEPEND'}) {
- my ($dep);
- push(@current_package_stack, [%current_package]);
- foreach $dep (keys %{$current_package{'DEPEND'}}) {
- # Has PPM already installed it?
- if(!defined $installed_packages{$dep}) {
- # Has *anybody* installed it, or is it part of core Perl?
- my $p = $dep;
- $p =~ s@-@/@g;
- my $found = grep -f, map "$_/$p.pm", @INC;
- if (!$found) {
- &Trace("Installing dependency '$dep'...") if $Trace;
- # print "Installing dependency '$dep'...\n";
- if(!InstallPackage("package" => $dep, "location" => $location)) {
- &Trace("Error installing dependency: $PPM::PPMERR") if $Trace;
- $PPM::PPMERR = "Error installing dependency: $PPM::PPMERR\n";
- if ($Force_install eq "No") {
- return 0;
- }
- }
- }
- }
- # make sure minimum version is installed, if necessary
- elsif (defined $current_package{'DEPEND'}{$dep}) {
- my (@comp) = split (',', $current_package{'DEPEND'}{$dep});
- # parsePPD fills in %current_package
- push(@current_package_stack, [%current_package]);
- parsePPD(%{$installed_packages{$dep}{'INST_PPD'}});
- my (@inst) = split (',', $current_package{'VERSION'});
-
- foreach(0..3) {
- if ($comp[$_] > $inst[$_]) {
- VerifyPackage("package" => $dep, "upgrade" => 1);
- last;
- }
- last if ($comp[$_] < $inst[$_]);
- }
- %current_package = @{pop @current_package_stack};
- }
- }
- %current_package = @{pop @current_package_stack};
- }
- my ($basename, $path) = fileparse($PPDfile);
- # strip the trailing path separator
- my $chr = substr($path, -1, 1);
- if ($chr eq '/' || $chr eq '\\') {
- chop $path;
- }
-
- if ($path =~ /^file:\/\/.*\|/i) {
- # $path is a local directory, let's avoid LWP by changing
- # it to a pathname.
- $path =~ s@^file://@@i;
- $path =~ s@^localhost/@@i;
- $path =~ s@\|@:@;
- }
-
-
- # get the code and put it in $build_dir
- my $install_dir = $build_dir . "/" . $current_package{'NAME'};
- File::Path::rmtree($install_dir,0,0);
- if(!-d $install_dir && !File::Path::mkpath($install_dir, 0, 0755)) {
- &Trace("Could not create $install_dir: $!") if $Trace;
- $PPM::PPMERR = "Could not create $install_dir: $!";
- return 0;
- }
- ($basename) = fileparse($current_package{'CODEBASE'});
- # CODEBASE is a URL
- if ($current_package{'CODEBASE'} =~ m@^...*://@i) {
- if (PPM_getstore("source" => "$current_package{'CODEBASE'}",
- "target" => "$install_dir/$basename") != 0) {
- return 0;
- }
- }
- # CODEBASE is a full pathname
- elsif (-f $current_package{'CODEBASE'}) {
- &Trace("Copying $current_package{'CODEBASE'} to $install_dir/$basename") if $Trace > 1;
- copy($current_package{'CODEBASE'}, "$install_dir/$basename");
- }
- # CODEBASE is relative to the directory location of the PPD
- elsif (-f "$path/$current_package{'CODEBASE'}") {
- &Trace("Copying $path/$current_package{'CODEBASE'} to $install_dir/$basename") if $Trace > 1;
- copy("$path/$current_package{'CODEBASE'}", "$install_dir/$basename");
- }
- # CODEBASE is relative to the URL location of the PPD
- else {
- if (PPM_getstore("source" => "$path/$current_package{'CODEBASE'}",
- "target" => "$install_dir/$basename") != 0) {
- return 0;
- }
- }
-
- my $cwd = getcwd();
- chdir($install_dir);
-
- my $tar;
- if ($basename =~ /\.gz$/i) {
- $tar = Archive::Tar->new($basename,1);
- }
- else {
- $tar = Archive::Tar->new($basename,0);
- }
- $tar->extract($tar->list_files);
- $basename =~ /(.*).tar/i;
- chdir($1);
- RelocPerl('.') if ($Config{'osname'} ne 'MSWin32');
-
- InstallBlib:
- my $inst_archlib = $Config{installsitearch};
- my $inst_root = $Config{prefix};
- my $packlist = MM->catfile("$Config{installsitearch}/auto", split(/-/, $current_package{'NAME'}), ".packlist");
-
- # copied from ExtUtils::Install
- my $INST_LIB = MM->catdir(MM->curdir,"blib","lib");
- my $INST_ARCHLIB = MM->catdir(MM->curdir,"blib","arch");
- my $INST_BIN = MM->catdir(MM->curdir,'blib','bin');
- my $INST_SCRIPT = MM->catdir(MM->curdir,'blib','script');
- my $INST_MAN1DIR = MM->catdir(MM->curdir,'blib','man1');
- my $INST_MAN3DIR = MM->catdir(MM->curdir,'blib','man3');
- my $INST_HTMLDIR = MM->catdir(MM->curdir,'blib','html');
- my $INST_HTMLHELPDIR = MM->catdir(MM->curdir,'blib','htmlhelp');
-
- my ($inst_lib, $inst_bin, $inst_script, $inst_man1dir, $inst_man3dir,
- $inst_htmldir, $inst_htmlhelpdir);
- $inst_script = $Config{installscript};
- $inst_man1dir = $Config{installman1dir};
- $inst_man3dir = $Config{installman3dir};
- $inst_bin = $Config{installbin};
- $inst_htmldir = $Config{installhtmldir};
- $inst_htmlhelpdir = $Config{installhtmlhelpdir};
- $inst_lib = $Config{installsitelib};
-
- if (defined $root && ($root !~ /^\Q$inst_root\E$/i)) {
- if ($packlist =~ m#[/|\\]lib[/|\\](.*)#) {
- $packlist = "$root/lib/$1";
- }
- if ($inst_lib =~ m#[/|\\]lib[/|\\](.*)#) {
- $inst_lib = "$root/lib/$1";
- }
- if ($inst_archlib =~ m#[/|\\]site[/|\\](.*)#) {
- $inst_archlib = "$root/site/$1";
- }
- $inst_bin =~ s/\Q$inst_root/$root\E/i;
- $inst_script =~ s/\Q$inst_root/$root\E/i;
- $inst_man1dir =~ s/\Q$inst_root/$root\E/i;
- $inst_man3dir =~ s/\Q$inst_root/$root\E/i;
- $inst_root = $root;
- }
-
- while (1) {
- my $cwd = getcwd();
- &Trace("Calling ExtUtils::Install::install") if $Trace > 1;
- eval {
- ExtUtils::Install::install({
- "read" => $packlist, "write" => $packlist,
- $INST_LIB => $inst_lib, $INST_ARCHLIB => $inst_archlib,
- $INST_BIN => $inst_bin, $INST_SCRIPT => $inst_script,
- $INST_MAN1DIR => $inst_man1dir, $INST_MAN3DIR => $inst_man3dir,
- $INST_HTMLDIR => $inst_htmldir, $INST_HTMLHELPDIR => $inst_htmlhelpdir},0,0,0);
- };
- # install might have croaked in another directory
- chdir($cwd);
- # Can't remove some DLLs, but we can rename them and try again.
- if ($@ && $@ =~ /Cannot forceunlink (\S+)/) {
- &Trace("$@...attempting rename") if $Trace;
- my $oldname = $1;
- $oldname =~ s/:$//;
- my $newname = $oldname . "." . time();
- if(!rename($oldname, $newname)) {
- &Trace("$!") if $Trace;
- $PPM::PPMERR = $@;
- return 0;
- }
- }
- # Some other error
- elsif($@) {
- &Trace("$@") if $Trace;
- $PPM::PPMERR = $@;
- return 0;
- }
- else { last; }
- }
-
- if (defined $current_package{'INSTALL_SCRIPT'}) {
- run_script("script" => $current_package{'INSTALL_SCRIPT'},
- "scriptHREF" => $current_package{'INSTALL_HREF'},
- "exec" => $current_package{'INSTALL_EXEC'},
- "inst_root" => $inst_root, "inst_archlib" => $inst_archlib);
- }
-
- chdir($cwd);
-
- # ask to store this location as default for this package?
- PPMdat_add_package($path, $packlist, $inst_root);
- # if 'install.ppm' exists, don't remove; system()
- # has probably not finished with it yet.
- if ($Clean eq "Yes" && !-f "$install_dir/install.ppm") {
- File::Path::rmtree($install_dir,0,0);
- }
- &Trace("Package $package successfully installed") if $Trace;
- reread_config();
-
- return 1;
- }
-
- # Returns a hash with key $location, and elements of arrays of package names.
- # Uses '%repositories' if $location is not specified.
- sub RepositoryPackages
- {
- my (%argv) = @_;
- my ($arg, $location, %ppds);
- foreach $arg (keys %argv) {
- if ($arg eq 'location') { $location = $argv{$arg}; }
- }
-
- if (defined $location) {
- @{$ppds{$location}} = list_available("location" => $location);
- }
- else {
- read_config(); # need repositories
- foreach $_ (keys %repositories) {
- $location = $repositories{$_}{'LOCATION'};
- @{$ppds{$location}} = list_available("location" => $location);
- }
- }
-
- return %ppds;
- }
-
- sub RepositoryPackageProperties
- {
- my (%argv) = @_;
- my ($arg, $package, $location, $PPDfile);
- foreach $arg (keys %argv) {
- if ($arg eq 'package') { $package = $argv{$arg}; }
- if ($arg eq 'location') { $location = $argv{$arg}; }
- }
- read_config();
- if (!defined($PPDfile = locatePPDfile($package, $location))) {
- &Trace("RepositoryPackageProperties: Could not locate a PPD file for package $package") if $Trace;
- $PPM::PPMERR = "Could not locate a PPD file for package $package";
- return;
- }
-
- my %PPD = readPPDfile($PPDfile, 'XML::PPD');
- if (!%PPD) { return; }
-
- parsePPD(%PPD);
-
- my (%ret_hash, $dep);
- $ret_hash{'NAME'} = $current_package{'NAME'};
- $ret_hash{'TITLE'} = $current_package{'TITLE'};
- $ret_hash{'AUTHOR'} = $current_package{'AUTHOR'};
- $ret_hash{'VERSION'} = $current_package{'VERSION'};
- $ret_hash{'ABSTRACT'} = $current_package{'ABSTRACT'};
- $ret_hash{'PERLCORE_VER'} = $current_package{'PERLCORE_VER'};
- foreach $dep (keys %{$current_package{'DEPEND'}}) {
- push @{$ret_hash{'DEPEND'}}, $dep;
- }
-
- return %ret_hash;
- }
-
- # Returns 1 on success, 0 and sets $PPMERR on failure.
- sub RemovePackage
- {
- my (%argv) = @_;
- my ($arg, $package, $force);
- foreach $arg (keys %argv) {
- if ($arg eq 'package') { $package = $argv{$arg}; }
- if ($arg eq 'force') { $force = $argv{$arg}; }
- }
- my %PPD;
-
- read_config();
- if (!defined $installed_packages{$package}) {
- my $pattern = $package;
- undef $package;
- # Do another lookup, ignoring case
- foreach $_ (keys %installed_packages) {
- if (/^$pattern$/i) {
- $package = $_;
- last;
- }
- }
- if (!defined $package) {
- &Trace("Package '$pattern' has not been installed by PPM") if $Trace;
- $PPM::PPMERR = "Package '$pattern' has not been installed by PPM";
- return 0;
- }
- }
-
- # Don't let them remove PPM itself, libnet, Archive-Tar, etc.
- # but we can force removal if we're upgrading
- unless (defined $force) {
- foreach (@required_packages) {
- if ($_ eq $package) {
- &Trace("Package '$package' is required by PPM and cannot be removed") if $Trace;
- $PPM::PPMERR = "Package '$package' is required by PPM and cannot be removed";
- return 0;
- }
- }
- }
-
- my $install_dir = $build_dir . "/" . $package;
-
- %PPD = %{ $installed_packages{$package}{'INST_PPD'} };
- parsePPD(%PPD);
- my $cwd = getcwd();
- if (defined $current_package{'UNINSTALL_SCRIPT'}) {
- if (!chdir($install_dir)) {
- &Trace("Could not chdir() to $install_dir: $!") if $Trace;
- $PPM::PPMERR = "Could not chdir() to $install_dir: $!";
- return 0;
- }
- run_script("script" => $current_package{'UNINSTALL_SCRIPT'},
- "scriptHREF" => $current_package{'UNINSTALL_HREF'},
- "exec" => $current_package{'UNINSTALL_EXEC'});
- chdir($cwd);
- }
- else {
- if (-f $installed_packages{$package}{'INST_PACKLIST'}) {
- &Trace("Calling ExtUtils::Install::uninstall") if $Trace > 1;
- eval {
- ExtUtils::Install::uninstall("$installed_packages{$package}{'INST_PACKLIST'}", 0, 0);
- };
- warn $@ if $@;
- }
- }
-
- File::Path::rmtree($install_dir,0,0);
- PPMdat_remove_package($package);
- &Trace("Package $package removed") if $Trace;
- reread_config();
- return 1;
- }
-
- # returns "0" if package is up-to-date; "1" if an upgrade is available;
- # undef and sets $PPMERR on error; and the new VERSION string if a package
- # was upgraded.
- sub VerifyPackage
- {
- my (%argv) = @_;
- my ($arg, $package, $location, $upgrade, $force);
- foreach $arg (keys %argv) {
- if ($arg eq 'package') { $package = $argv{$arg}; }
- if ($arg eq 'location') { $location = $argv{$arg}; }
- if ($arg eq 'upgrade') { $upgrade = $argv{$arg}; }
- if ($arg eq 'force') { $force = $argv{$arg}; }
- }
-
- my ($installedPPDfile, $comparePPDfile, %installedPPD, %comparePPD);
-
- read_config();
-
- if (!defined $installed_packages{$package}) {
- my $pattern = $package;
- undef $package;
- # Do another lookup, ignoring case
- foreach $_ (keys %installed_packages) {
- if (/^$pattern$/i) {
- $package = $_;
- last;
- }
- }
- if (!defined $package) {
- &Trace("Package '$pattern' has not been installed by PPM") if $Trace;
- $PPM::PPMERR = "Package '$pattern' has not been installed by PPM";
- return undef;
- }
- }
-
- %installedPPD = %{ $installed_packages{$package}{'INST_PPD'} };
-
- if (!defined($comparePPDfile = locatePPDfile($package, $location))) {
- &Trace("VerifyPackage: Could not locate a PPD file for $package") if $Trace;
- $PPM::PPMERR = "Could not locate a PPD file for $package";
- return undef;
- }
- %comparePPD = readPPDfile($comparePPDfile, 'XML::PPD');
- if (!%comparePPD) { return undef; }
-
- parsePPD(%installedPPD);
- my ($inst_version) = $current_package{'VERSION'};
- my ($inst_major, $inst_minor,
- $inst_patch1, $inst_patch2) = split (',', $inst_version);
- my ($inst_root) = $installed_packages{$package}{'INST_ROOT'};
- parsePPD(%comparePPD);
- my ($comp_version) = $current_package{'VERSION'};
- my ($comp_major, $comp_minor,
- $comp_patch1, $comp_patch2) = split (',', $comp_version);
-
- if ($force || $comp_major > $inst_major ||
- ($comp_major == $inst_major && $comp_minor > $inst_minor) ||
- ($comp_major == $inst_major && $comp_minor == $inst_minor &&
- $comp_patch1 > $inst_patch1) ||
- ($comp_major == $inst_major && $comp_minor == $inst_minor &&
- $comp_patch1 == $inst_patch1 && $comp_patch2 > $inst_patch2)) {
- &Trace("Upgrade to $package is available") if $Trace > 1;
- if ($upgrade) {
- if ($Config{'osname'} eq 'MSWin32' &&
- !&Win32::IsWinNT &&
- exists $Win9x_denied{lc($package)}) {
- $PPM::PPMERR = "Package '$package' cannot be upgraded with PPM on Win9x--see http://www.ActiveState.com/ppm for details";
- return undef;
- }
-
- if (!defined $location) {
- # need to remember the $location, because once we remove the
- # package, it's unavailable.
- $location = $installed_packages{$package}{'LOCATION'};
- }
- unless (defined locatePPDfile($package, $location)) {
- &Trace("VerifyPackage: Could not locate a PPD file for $package") if $Trace;
- $PPM::PPMERR = "Could not locate a PPD file for $package";
- return undef;
- }
- RemovePackage("package" => $package, "force" => 1);
- InstallPackage("package" => $package, "location" => $location,
- "root" => $inst_root) or return undef;
- return $comp_version;
- }
- return '1';
- }
- else {
- # package is up to date
- return '0';
- }
- }
-
- # Changes where the packages are installed.
- # Returns previous root on success, undef and sets $PPMERR on failure.
- sub chroot
- {
- my (%argv) = @_;
- my ($arg, $location, $previous_root);
- foreach $arg (keys %argv) {
- if ($arg eq 'location') { $location = $argv{$arg}; }
- }
-
- if (!-d $location) {
- &Trace("'$location' does not exist.") if $Trace;
- $PPM::PPMERR = "'$location' does not exist.\n";
- return undef;
- }
-
- if (!defined $orig_root) {
- $orig_root = $Config{installbin};
- $orig_root =~ s/bin.*$//i;
- chop $orig_root;
- $current_root = $orig_root;
- }
- # mjn: move this to front-end?
- $previous_root = $current_root;
- $current_root = $location;
- return $previous_root;
- }
-
- sub QueryInstalledPackages
- {
- my (%argv) = @_;
- my ($searchRE, $searchtag, $ignorecase, $package, %ret_hash);
- $PPM::PPMERR = undef;
- my ($arg);
- foreach $arg (keys %argv) {
- if ($arg eq 'searchRE' && defined $argv{$arg}) {
- $searchRE = $argv{$arg};
- eval { $searchRE =~ /$searchRE/ };
- if ($@) {
- &Trace("'$searchRE': invalid regular expression.") if $Trace;
- $PPM::PPMERR = "'$searchRE': invalid regular expression.";
- return ();
- }
- }
- if ($arg eq 'searchtag' && (defined $argv{$arg})) { $searchtag = uc($argv{$arg}); }
- if ($arg eq 'ignorecase') { $ignorecase = $argv{$arg}; }
- }
- if (!defined $ignorecase) {
- $ignorecase = $Ignorecase;
- }
-
- read_config();
- foreach $package (keys %installed_packages) {
- my $results;
- if (defined $searchtag) {
- my %Package = %{ $installed_packages{$package} };
- parsePPD( %{ $Package{'INST_PPD'} } );
- $results = $current_package{$searchtag};
- }
- else {
- $results = $package;
- }
- if (!defined $searchRE) {
- $ret_hash{$package} = $results;
- }
- elsif ($results =~ /$searchRE/) {
- $ret_hash{$package} = $results;
- }
- elsif ($ignorecase eq "Yes" && ($results =~ /$searchRE/i)) {
- $ret_hash{$package} = $results;
- }
- }
-
- return %ret_hash;
- }
-
- # Returns the matched string on success, "" on no match, and undef
- # on error.
- sub QueryPPD
- {
- my (%argv) = @_;
- my ($location, $searchRE, $searchtag, $ignorecase, $package);
- my ($arg, $PPDfile, $string);
- foreach $arg (keys %argv) {
- if ($arg eq 'location') { $location = $argv{$arg}; }
- if ($arg eq 'searchRE' && defined $argv{$arg}) {
- $searchRE = $argv{$arg};
- eval { $searchRE =~ /$searchRE/ };
- if ($@) {
- &Trace("'$searchRE': invalid regular expression") if $Trace;
- $PPM::PPMERR = "'$searchRE': invalid regular expression.";
- return undef;
- }
- }
- if ($arg eq 'searchtag') { $searchtag = $argv{$arg}; }
- if ($arg eq 'ignorecase') { $ignorecase = $argv{$arg}; }
- if ($arg eq 'package') { $package = $argv{$arg}; }
- }
- if (!defined $ignorecase) {
- $ignorecase = $Ignorecase;
- }
-
- if (!$location) {
- read_config();
- }
-
- if (!defined($PPDfile = locatePPDfile($package, $location))) {
- &Trace("QueryPPD: Could not locate a PPD file for package $package") if $Trace;
- $PPM::PPMERR = "Could not locate a PPD file for package $package";
- return undef;
- }
- my %PPD = readPPDfile($PPDfile, 'XML::PPD');
- if (!%PPD) { return undef; }
-
- parsePPD(%PPD);
-
- my $retval = "";
-
- if ($searchtag eq 'abstract') {
- $string = $current_package{'ABSTRACT'}
- }
- elsif ($searchtag eq 'author') {
- $string = $current_package{'AUTHOR'}
- }
- elsif ($searchtag eq 'title') {
- $string = $current_package{'TITLE'}
- }
-
- if (!$searchRE) {
- $retval = $string;
- }
- elsif ($ignorecase eq "Yes") {
- if ($string =~ /$searchRE/i) {
- $retval = $string;
- }
- }
- elsif ($string =~ /$searchRE/) {
- $retval = $string;
- }
-
- return $retval;
- }
-
- # Returns a summary of available packages for all repositories.
- # Returned hash has the following structure:
- #
- # $hash{repository}{package_name}{NAME}
- # $hash{repository}{package_name}{VERSION}
- # etc.
- #
- sub RepositorySummary {
- my (%argv) = @_;
- my ($arg, $location, %summary, %locations);
-
- ###########################################################################
- # See if we were given the location of a repository to get the summary for.
- ###########################################################################
- foreach $arg (keys %argv) {
- if ($arg eq 'location') { $location = $argv{$arg}; }
- }
-
- ###########################################################################
- # If we weren't given the location of a repository to query the summary
- # for, check all of the repositories that we know about.
- ###########################################################################
- if (!defined $location) {
- read_config(); # need repositories
- foreach (keys %repositories) {
- $locations{$repositories{$_}{'LOCATION'}} = $repositories{$_}{'SUMMARYFILE'};
- }
- }
- ###########################################################################
- # Otherwise, we were given a repository to query, figure out where we can
- # find the summary file for that repository.
- ###########################################################################
- else {
- foreach (keys %repositories) {
- if ($location =~ /\Q$repositories{$_}{'LOCATION'}\E/i) {
- $locations{$repositories{$_}{'LOCATION'}} = $repositories{$_}{'SUMMARYFILE'};
- last;
- }
- }
- }
-
- ###########################################################################
- # Check all of the summary file locations that we were able to find.
- ###########################################################################
- foreach $location (keys %locations) {
- my $summaryfile = $locations{$location};
- if (!$summaryfile) {
- &Trace("RepositorySummary: No summary available from $location.") if $Trace;
- $PPM::PPMERR = "No summary available from $location.\n";
- next;
- }
-
- # Todo: The following code should really be a call to
- # readPPDfile( "$location/$summaryfile", 'XML::RepositorySummary' );
- # for validation.
- if ($summaryfile ne 'fetch_summary' && !valid_URL_or_file("$location/$summaryfile")) {
- &Trace("RepositorySummary: No summary available from $location.") if $Trace;
- $PPM::PPMERR = "No summary available from $location.\n";
- }
- else {
- my ($data);
- if ($location =~ m@^...*://@i) {
- next if (!defined ($data = read_href("href" => "$location/$summaryfile", "request" => 'GET')));
- } else {
- local $/;
- next if (!open (DATAFILE, "$location/$summaryfile"));
- $data = <DATAFILE>;
- close(DATAFILE);
- }
-
- # take care of '&'
- $data =~ s/&(?!\w+;)/&/go;
-
- my $parser = new XML::Parser( Style => 'Objects', Pkg => 'XML::RepositorySummary' );
- my @parsed = @{ $parser->parse( $data ) };
-
- my $packages = ${$parsed[0]}{Kids};
-
- foreach my $package (@{$packages}) {
- my $elem_type = ref $package;
- $elem_type =~ s/.*:://;
- next if ($elem_type eq 'Characters');
-
- if ($elem_type eq 'SOFTPKG') {
- my %ret_hash;
- parsePPD(%{$package});
- $ret_hash{'NAME'} = $current_package{'NAME'};
- $ret_hash{'TITLE'} = $current_package{'TITLE'};
- $ret_hash{'AUTHOR'} = $current_package{'AUTHOR'};
- $ret_hash{'VERSION'} = $current_package{'VERSION'};
- $ret_hash{'ABSTRACT'} = $current_package{'ABSTRACT'};
- $ret_hash{'PERLCORE_VER'} = $current_package{'PERLCORE_VER'};
- foreach my $dep (keys %{$current_package{'DEPEND'}}) {
- push @{$ret_hash{'DEPEND'}}, $dep;
- }
- $summary{$location}{$current_package{'NAME'}} = \%ret_hash;
- }
- }
- }
- }
-
- return %summary;
- }
-
- #
- # Internal subs
- #
-
- sub PPM_getstore {
- my (%argv) = @_;
- my ($arg, $source, $target, $username, $password);
- foreach $arg (keys %argv) {
- if ($arg eq 'source') { $source = $argv{$arg}; }
- if ($arg eq 'target') { $target = $argv{$arg}; }
- }
- if (!defined $source || !defined $target) {
- &Trace("PPM_getstore: No source or no target") if $Trace;
- $PPM::PPMERR = "No source or no target\n";
- return 1;
- }
-
- # Do we need to do authorization?
- # This is a hack, but will have to do for now.
- foreach (keys %repositories) {
- if ($source =~ /^$repositories{$_}{'LOCATION'}/i) {
- $username = $repositories{$_}{'USERNAME'};
- $password = $repositories{$_}{'PASSWORD'};
- last;
- }
- }
-
- if (defined $ENV{HTTP_proxy} || defined $username) {
- my $ua = new LWP::UserAgent;
- $ua->agent("$0/0.1 " . $ua->agent);
- my $proxy_user = $ENV{HTTP_proxy_user};
- my $proxy_pass = $ENV{HTTP_proxy_pass};
- $ua->env_proxy;
- my $req = new HTTP::Request('GET' => $source);
- if (defined $proxy_user && defined $proxy_pass) {
- &Trace("PPM_getstore: calling proxy_authorization_basic($proxy_user, $proxy_pass)") if $Trace > 1;
-
- $req->proxy_authorization_basic("$proxy_user", "$proxy_pass");
- }
- if (defined $username && defined $password) {
- &Trace("PPM_getstore: calling proxy_authorization_basic($username, $password)") if $Trace > 1;
- $req->authorization_basic($username, $password);
- }
- my $res = $ua->request($req);
- if ($res->is_success) {
- if (!open(OUT, ">$target")) {
- &Trace("PPM_getstore: Couldn't open $target for writing") if $Trace;
- $PPM::PPMERR = "Couldn't open $target for writing\n";
- return 1;
- }
- binmode(OUT);
- print OUT $res->content;
- close(OUT);
- } else {
- &Trace("PPM_getstore: Error reading $source: " . $res->code . " " . $res->message) if $Trace;
- $PPM::PPMERR = "Error reading $source: " . $res->code . " " . $res->message . "\n";
- return 1;
- }
- }
- else {
- my $status = LWP::Simple::getstore($source, $target);
- if ($status < 200 || $status > 299) {
- &Trace("PPM_getstore: Read of $source failed") if $Trace;
- $PPM::PPMERR = "Read of $source failed";
- return 1;
- }
- }
- return 0;
- }
-
- sub save_options
- {
- read_config();
- # Read in the existing PPM configuration file
- my %PPMConfig = readPPDfile( $PPM::PPMdat, 'XML::PPMConfig' );
- if (!%PPMConfig) { return 0; }
-
- # Remove all of the declarations for REPOSITORY and PPMPRECIOUS;
- # we'll output these from the lists we've got in memory instead.
- my $idx;
- foreach $idx (0 .. @{$PPMConfig{Kids}})
- {
- my $elem = $PPMConfig{Kids}[$idx];
- my $elem_type = ref $elem;
- if ($elem_type =~ /::REPOSITORY$|::PPMPRECIOUS$/o)
- {
- splice( @{$PPMConfig{Kids}}, $idx, 1 );
- redo; # Restart again so we don't miss any
- }
- }
-
- # Traverse the info we read in and replace the values in it with the new
- # config options that we've got.
- my $elem;
- foreach $elem (@{ $PPMConfig{Kids} })
- {
- my $elem_type = ref $elem;
- $elem_type =~ s/.*:://;
- next if ($elem_type ne 'OPTIONS');
-
- $elem->{IGNORECASE} = $Ignorecase;
- $elem->{CLEAN} = $Clean;
- $elem->{CONFIRM} = $Confirm;
- $elem->{FORCEINSTALL} = $Force_install;
- $elem->{ROOT} = $Root;
- $elem->{BUILDDIR} = $build_dir;
- $elem->{MORE} = $More;
- $elem->{TRACE} = $Trace;
- $elem->{TRACEFILE} = $TraceFile;
- $elem->{VERBOSE} = $Verbose;
- }
-
- # Find out where the package listings start and insert our PPMPRECIOUS and
- # updated list of REPOSITORYs.
- foreach $idx (0 .. @{$PPMConfig{Kids}})
- {
- my $elem = $PPMConfig{Kids}[$idx];
- my $elem_type = ref $elem;
- $elem_type =~ s/.*:://;
- next unless (($elem_type eq 'PACKAGE') or
- ($idx == $#{$PPMConfig{Kids}}));
-
- # Insert our PPMPRECIOUS
- my $chardata = new XML::PPMConfig::Characters;
- $chardata->{Text} = join( ';', @required_packages );
- my $precious = new XML::PPMConfig::PPMPRECIOUS;
- push( @{$precious->{Kids}}, $chardata );
- splice( @{$PPMConfig{Kids}}, $idx, 0, $precious );
-
- # Insert the list of repositories we've got
- my $rep_name;
- foreach $rep_name (keys %repositories)
- {
- my $repository = new XML::PPMConfig::REPOSITORY;
- $repository->{NAME} = $rep_name;
- $repository->{LOCATION} = $repositories{ $rep_name }{'LOCATION'};
- $repository->{USERNAME} = $repositories{ $rep_name }{'USERNAME'} if defined $repositories{ $rep_name }{'USERNAME'};
- $repository->{PASSWORD} = $repositories{ $rep_name }{'PASSWORD'} if defined $repositories{ $rep_name }{'PASSWORD'};
- $repository->{SUMMARYFILE} = $repositories{ $rep_name }{'SUMMARYFILE'} if defined $repositories{ $rep_name }{'SUMMARYFILE'};
- splice( @{$PPMConfig{Kids}}, $idx, 0, $repository );
- }
- last;
- }
- # Take the data structure we've got and bless it into a PPMCONFIG object so
- # that we can output it.
- my $cfg = bless \%PPMConfig, 'XML::PPMConfig::PPMCONFIG';
-
- # Open the output file and output the PPM config file
- if (!open( DAT, ">$PPM::PPMdat" )) {
- &Trace("open of $PPM::PPMdat failed: $!") if $Trace;
- $PPM::PPMERR = "open of $PPM::PPMdat failed: $!\n";
- return 1;
- }
- my $oldout = select DAT;
- $cfg->output();
- select $oldout;
- close( DAT );
- &Trace("Wrote config file") if $Trace > 1;
- }
-
- ###############################################################################
- # Subroutine: list_available
- ###############################################################################
- # Gets a listing of all of the packages available in the repository. If an
- # argument of 'location' is provided in %argv, it is used as the repository to
- # query. This method returns to the caller a complete list of all of the
- # available packages at the repository in a list context, returning 'undef' if
- # any errors occurred.
- ###############################################################################
- sub list_available
- {
- my (%argv) = @_;
- my ($arg, $location, @ppds);
- foreach $arg (keys %argv) {
- if ($arg eq 'location') { $location = $argv{$arg}; }
- }
-
- if ($location =~ /^file:\/\/.*\|/i) {
- # $location is a local directory, let's avoid LWP by changing
- # it to a pathname.
- $location =~ s@^file://@@i;
- $location =~ s@^localhost/@@i;
- $location =~ s@\|@:@;
- }
-
- # URL in UNC notation
- if ($location =~ /^file:\/\/\/\//i) {
- $location =~ s@^file://@@i;
- }
-
- # directory or UNC
- if (-d $location || $location =~ /^\\\\/ || $location =~ /^\/\//) {
- opendir(PPDDIR, $location) or return undef;
- my ($file);
- @ppds = grep { /\.ppd$/i && -f "$location/$_" } readdir(PPDDIR);
- foreach $file (@ppds) {
- $file =~ s/\.ppd//i;
- }
- }
- elsif ($location =~ m@^...*://@i) {
- if ($cached_ppd_list{$location}) {
- return @{$cached_ppd_list{$location}};
- }
-
- #######################################################################
- # If we're accessing a SOAP server, do things differently than we would
- # for FTP, HTTP, etc.
- #######################################################################
- if ($location =~ m#^soap://#i)
- {
- my $client = new PPM::SOAPClient( $location );
- @ppds = $client->packages();
- }
- else
- {
- my $doc = read_href("href" => $location, "request" => 'GET');
- if (!defined $doc) {
- return undef;
- }
-
- if ($doc =~ /^<head><title>/) {
- # read an IIS format directory listing
- @ppds = grep { /\.ppd/i } split('<br>', $doc);
- my $file;
- foreach $file (@ppds) {
- $file =~ s/\.ppd<.*$//is;
- $file =~ s@.*>@@is;
- }
- }
- elsif ($doc =~ /<BODY BGCOLOR=FFFFFF>\n\n<form name=VPMform/s) {
- # read output of default.prk over an HTTP connection
- @ppds = grep { /^<!--Key:.*-->$/ } split('\n', $doc);
- my $file;
- foreach $file (@ppds) {
- if ($file =~ /^<!--Key:(.*)-->$/) {
- $file = $1;
- }
- }
- }
- else {
- # read an Apache (?) format directory listing
- @ppds = grep { /\.ppd/i } split('\n', $doc);
- my $file;
- foreach $file (@ppds) {
- $file =~ s/^.*>(.*?)\.ppd<.*$/$1/i;
- }
- }
- }
-
- #######################################################################
- # All done, take the list of PPDs that we've queried and cache it for
- # later re-use, then return it to the caller.
- #######################################################################
- @{$cached_ppd_list{$location}} = sort @ppds;
- return @{$cached_ppd_list{$location}};
- }
- return sort @ppds;
- }
-
- sub read_href
- {
- my (%argv) = @_;
- my ($arg, $href, $request, $proxy_user, $proxy_pass);
- foreach $arg (keys %argv) {
- if ($arg eq 'href') { $href = $argv{$arg}; }
- if ($arg eq 'request') { $request = $argv{$arg}; }
- }
- # If this is a SOAP URL, handle it differently than FTP/HTTP/file.
- if ($href =~ m#^soap://#io)
- {
- my ($srvr, $fcn) = ($href =~ m@(.*)/(.+?)$@o);
- my $content;
-
- my $client = new PPM::SOAPClient( $srvr );
- return $client->fetch_summary() if ($fcn eq 'fetch_summary');
- return $client->fetch_ppd( $fcn ) if (uc($request) eq 'HEAD');
- return;
- }
-
- # Otherwise its a standard URL, go ahead and request it using LWP.
- my $ua = new LWP::UserAgent;
- $ua->agent("$0/0.1 " . $ua->agent);
- if (defined $ENV{HTTP_proxy}) {
- $proxy_user = $ENV{HTTP_proxy_user};
- $proxy_pass = $ENV{HTTP_proxy_pass};
- &Trace("read_href: calling env_proxy: $ENV{'HTTP_proxy'}") if $Trace > 1;
- $ua->env_proxy;
- }
- my $req = new HTTP::Request $request => $href;
- if (defined $proxy_user && defined $proxy_pass) {
- &Trace("read_href: calling proxy_authorization_basic($proxy_user, $proxy_pass)") if $Trace > 1;
- $req->proxy_authorization_basic("$proxy_user", "$proxy_pass");
- }
-
- # Do we need to do authorization?
- # This is a hack, but will have to do for now.
- foreach (keys %repositories) {
- if ($href =~ /^$repositories{$_}{'LOCATION'}/i) {
- my $username = $repositories{$_}{'USERNAME'};
- my $password = $repositories{$_}{'PASSWORD'};
- if (defined $username && defined $password) {
- &Trace("read_href: calling proxy_authorization_basic($username, $password)") if $Trace > 1;
- $req->authorization_basic($username, $password);
- last;
- }
- }
- }
-
- # send request
- my $res = $ua->request($req);
-
- # check the outcome
- if ($res->is_success) {
- return $res->content;
- } else {
- &Trace("read_href: Error reading $href: " . $res->code . " " . $res->message) if $Trace;
- $PPM::PPMERR = "Error reading $href: " . $res->code . " " . $res->message . "\n";
- return undef;
- }
- }
-
- sub reread_config
- {
- %current_package = ();
- %installed_packages = ();
- $init = 0;
- read_config();
- }
-
- # returns 0 on success, 1 and sets $PPMERR on error.
- sub PPMdat_add_package
- {
- my ($location, $packlist, $inst_root) = @_;
- my $package = $current_package{'NAME'};
- my $time_str = localtime;
-
- # If we already have this package installed, remove it from the PPM
- # Configuration file so we can put the new one in.
- if (defined $installed_packages{$package} ) {
- # remove the existing entry for this package.
- PPMdat_remove_package($package);
- }
-
- # Build the new SOFTPKG data structure for this package we're adding.
- my $softpkg =
- new XML::PPMConfig::SOFTPKG( NAME => $package,
- VERSION => $current_package{VERSION}
- );
-
- if (defined $current_package{TITLE})
- {
- my $chardata =
- new XML::PPMConfig::Characters( Text => $current_package{TITLE} );
- my $newelem = new XML::PPMConfig::TITLE;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$softpkg->{Kids}}, $newelem );
- }
-
- if (defined $current_package{ABSTRACT})
- {
- my $chardata =
- new XML::PPMConfig::Characters( Text => $current_package{ABSTRACT});
- my $newelem = new XML::PPMConfig::ABSTRACT;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$softpkg->{Kids}}, $newelem );
- }
-
- if (defined $current_package{AUTHOR})
- {
- my $chardata =
- new XML::PPMConfig::Characters( Text => $current_package{AUTHOR} );
- my $newelem = new XML::PPMConfig::AUTHOR;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$softpkg->{Kids}}, $newelem );
- }
-
- if (defined $current_package{LICENSE})
- {
- my $chardata =
- new XML::PPMConfig::Characters( Text => $current_package{LICENSE});
- my $newelem = new XML::PPMConfig::LICENSE;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$softpkg->{Kids}}, $newelem );
- }
-
- my $impl = new XML::PPMConfig::IMPLEMENTATION;
- push( @{$softpkg->{Kids}}, $impl );
-
- if (defined $current_package{PERLCORE_VER})
- {
- my $newelem = new XML::PPMConfig::PERLCORE( VERSION => $current_package{PERLCORE_VER} );
- push( @{$impl->{Kids}}, $newelem );
- }
-
- foreach $_ (keys %{$current_package{DEPEND}})
- {
- my $newelem = new XML::PPMConfig::DEPENDENCY( NAME => $_, VERSION => $current_package{DEPEND}{$_} );
- push( @{$impl->{Kids}}, $newelem );
- }
-
- my $codebase = new XML::PPMConfig::CODEBASE( HREF => $current_package{CODEBASE} );
- push( @{$impl->{Kids}}, $codebase );
-
- my $inst = new XML::PPMConfig::INSTALL;
- push( @{$impl->{Kids}}, $inst );
- if (defined $current_package{INSTALL_EXEC})
- { $inst->{EXEC} = $current_package{INSTALL_EXEC}; }
- if (defined $current_package{INSTALL_HREF})
- { $inst->{HREF} = $current_package{INSTALL_HREF}; }
- if (defined $current_package{INSTALL_SCRIPT})
- {
- my $chardata =
- new XML::PPMConfig::Characters( Text => $current_package{INSTALL_SCRIPT} );
- push( @{$inst->{Kids}}, $chardata );
- }
-
- my $uninst = new XML::PPMConfig::UNINSTALL;
- push( @{$impl->{Kids}}, $uninst );
- if (defined $current_package{UNINSTALL_EXEC})
- { $uninst->{EXEC} = $current_package{UNINSTALL_EXEC}; }
- if (defined $current_package{UNINSTALL_HREF})
- { $uninst->{HREF} = $current_package{UNINSTALL_HREF}; }
- if (defined $current_package{UNINSTALL_SCRIPT})
- {
- my $chardata =
- new XML::PPMConfig::Characters( Text => $current_package{UNINSTALL_SCRIPT} );
- push( @{$uninst->{Kids}}, $chardata );
- }
-
- # Then, build the PACKAGE object and stick the SOFTPKG inside of it.
- my $pkg = new XML::PPMConfig::PACKAGE( NAME => $package );
-
- if (defined $location)
- {
- my $chardata = new XML::PPMConfig::Characters( Text => $location );
- my $newelem = new XML::PPMConfig::LOCATION;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$pkg->{Kids}}, $newelem );
- }
-
- if (defined $packlist)
- {
- my $chardata = new XML::PPMConfig::Characters( Text => $packlist );
- my $newelem = new XML::PPMConfig::INSTPACKLIST;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$pkg->{Kids}}, $newelem );
- }
-
- if (defined $inst_root)
- {
- my $chardata = new XML::PPMConfig::Characters( Text => $inst_root );
- my $newelem = new XML::PPMConfig::INSTROOT;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$pkg->{Kids}}, $newelem );
- }
-
- if (defined $time_str)
- {
- my $chardata = new XML::PPMConfig::Characters( Text => $time_str);
- my $newelem = new XML::PPMConfig::INSTDATE;
- push( @{$newelem->{Kids}}, $chardata );
- push( @{$pkg->{Kids}}, $newelem );
- }
-
- my $instppd = new XML::PPMConfig::INSTPPD;
- push( @{$instppd->{Kids}}, $softpkg );
- push( @{$pkg->{Kids}}, $instppd );
-
- # Now that we've got the structure built, read in the existing PPM
- # Configuration file, add this to it, and spit it back out.
- my %PPMConfig = readPPDfile( $PPM::PPMdat, 'XML::PPMConfig' );
- if (!%PPMConfig) { return 1; }
- push( @{$PPMConfig{Kids}}, $pkg );
- my $cfg = bless \%PPMConfig, 'XML::PPMConfig::PPMCONFIG';
-
- if (!open( DAT, ">$PPM::PPMdat" )) {
- &Trace("open of $PPM::PPMdat failed: $!") if $Trace;
- $PPM::PPMERR = "open of $PPM::PPMdat failed: $!\n";
- return 1;
- }
- my $oldout = select DAT;
- $cfg->output();
- select $oldout;
- close( DAT );
- &Trace("PPMdat_add_package: wrote $PPM::PPMdat") if $Trace > 1;
-
- return 0;
- }
-
- # returns 0 on success, 1 and sets $PPMERR on error.
- sub PPMdat_remove_package
- {
- my ($package) = @_;
-
- # Read in the existing PPM configuration file
- my %PPMConfig = readPPDfile( $PPM::PPMdat, 'XML::PPMConfig' );
- if (!%PPMConfig) { return 1; }
-
- # Try to find the package that we're supposed to be removing, and yank it
- # out of the list of installed packages.
- my $idx;
- foreach $idx (0 .. @{$PPMConfig{Kids}})
- {
- my $elem = $PPMConfig{Kids}[$idx];
- my $elem_type = ref $elem;
- next if ($elem_type !~ /::PACKAGE$/o);
- next if ($elem->{NAME} ne $package);
- splice( @{$PPMConfig{Kids}}, $idx, 1 );
- }
-
- # Take the data structure we've got and bless it into a PPMCONFIG object so
- # that we can output it again.
- my $cfg = bless \%PPMConfig, 'XML::PPMConfig::PPMCONFIG';
-
- # Now that we've removed the package, save the configuration file back out.
- if (!open( DAT, ">$PPM::PPMdat" )) {
- $PPM::PPMERR = "open of $PPM::PPMdat failed: $!\n";
- return 1;
- }
- my $oldout = select DAT;
- $cfg->output();
- select $oldout;
- close( DAT );
- &Trace("PPMdat_remove_package: wrote $PPM::PPMdat") if $Trace > 1;
-
- return 0;
- }
-
- # Run $script using system(). If $scriptHREF is specified, its contents are
- # used as the script. If $exec is specified, the script is saved to a
- # temporary file and executed by $exec.
- sub run_script
- {
- my (%argv) = @_;
- my ($arg, $script, $scriptHREF, $exec, $inst_root, $inst_archlib);
- foreach $arg (keys %argv) {
- if ($arg eq 'script') { $script = $argv{$arg}; }
- if ($arg eq 'scriptHREF') { $scriptHREF = $argv{$arg}; }
- if ($arg eq 'exec') { $exec = $argv{$arg}; }
- if ($arg eq 'inst_root') { $inst_root = $argv{$arg}; }
- if ($arg eq 'inst_archlib') { $inst_archlib = $argv{$arg}; }
- }
-
- my (@commands, $tmpname);
-
- if ($scriptHREF) {
- if ($exec) {
- # store in a temp file.
- $tmpname = $build_dir . "/PPM-" . time();
- LWP::Simple::getstore($scriptHREF, $tmpname);
- }
- else {
- my $doc = LWP::Simple::get $scriptHREF;
- if (!defined $doc) {
- print $PPM::PPMERR;
- }
- @commands = split("\n", $doc);
- if (!defined $commands[0]) {
- print $PPM::PPMERR;
- return 0;
- }
- }
- }
- else {
- if (-f $script) {
- $tmpname = $script;
- }
- else {
- # change any escaped chars
- $script =~ s/</</gi;
- $script =~ s/>/>/gi;
-
- @commands = split(';;', $script);
- if ($exec) {
- my $command;
-
- # store in a temp file.
- $tmpname = $build_dir . "/PPM-" . time();
- open(TMP, ">$tmpname");
- foreach $command (@commands) {
- print TMP "$command\n";
- }
- close(TMP);
- }
- }
- }
- $ENV{'PPM_INSTROOT'} = $inst_root;
- $ENV{'PPM_INSTARCHLIB'} = $inst_archlib;
- if ($exec) {
- if ($exec =~ /^PPM_PERL$/i) {
- $exec = $^X;
- }
- system("start $exec $tmpname");
- }
- else {
- my $command;
- for $command (@commands) {
- system($command);
- }
- }
- }
-
- sub parsePPD
- {
- my %PPD = @_;
- my $pkg;
-
- %current_package = ();
-
- # Get the package name and version from the attributes and stick it
- # into the 'current package' global var
- $current_package{NAME} = $PPD{NAME};
- $current_package{VERSION} = $PPD{VERSION};
-
- # Get all the information for this package and put it into the 'current
- # package' global var.
- my $got_implementation = 0;
- my $elem;
-
- foreach $elem (@{$PPD{Kids}})
- {
- my $elem_type = ref $elem;
- $elem_type =~ s/.*:://;
- next if ($elem_type eq 'Characters');
-
- if ($elem_type eq 'TITLE')
- {
- # Get the package title out of our _only_ char data child
- $current_package{TITLE} = $elem->{Kids}[0]{Text};
- }
- elsif ($elem_type eq 'LICENSE')
- {
- # Get the HREF for the license out of our attribute
- $current_package{LICENSE} = $elem->{HREF};
- }
- elsif ($elem_type eq 'ABSTRACT')
- {
- # Get the package abstract out of our _only_ char data child
- $current_package{ABSTRACT} = $elem->{Kids}[0]{Text};
- }
- elsif ($elem_type eq 'AUTHOR')
- {
- # Get the authors name out of our _only_ char data child
- $current_package{AUTHOR} = $elem->{Kids}[0]{Text};
- }
- elsif ($elem_type eq 'IMPLEMENTATION')
- {
- # If we don't have a valid implementation yet, check if this is
- # it.
- next if ($got_implementation);
- $got_implementation = implementation( @{ $elem->{Kids} } );
- }
- else
- {
- &Trace("Unknown element '$elem_type' found inside SOFTPKG") if $Trace;
- die "Unknown element '$elem_type' found inside SOFTPKG.";
- }
- } # End of "for each child element inside the PPD"
-
- if ($Trace > 3 and (%current_package) ) {
- &Trace("Read a PPD:");
- my $elem;
- foreach $elem (keys %current_package) {
- &Trace("\t$elem:\t$current_package{$elem}");
- }
- }
-
- if ( ($Debug & 2) and (%current_package) )
- {
- print "Read a PPD...\n";
- my $elem;
- foreach $elem (keys %current_package)
- { print "\t$elem:\t$current_package{$elem}\n"; }
- }
- }
-
- # Tests the passed IMPLEMENTATION for suitability on the current platform.
- # Fills in the CODEBASE, INSTALL_HREF, INSTALL_EXEC, INSTALL_SCRIPT,
- # UNINSTALL_HREF, UNINSTALL_EXEC, UNINSTALL_SCRIPT and DEPEND keys of
- # %current_package. Returns 1 on success, 0 otherwise.
- sub implementation
- {
- my @impl = @_;
-
- # Declare the tmp vars we're going to use to hold onto things.
- my ($ImplProcessor, $ImplOS, $ImplOSVersion, $ImplLanguage, $ImplCodebase);
- my ($ImplInstallHREF, $ImplInstallEXEC, $ImplInstallScript);
- my ($ImplUninstallHREF, $ImplUninstallEXEC, $ImplUninstallScript);
- my ($ImplArch, $ImplPerlCoreVer, %ImplDepend);
-
- my $elem;
- foreach $elem (@impl)
- {
- my $elem_type = ref $elem;
- $elem_type =~ s/.*:://;
- next if ($elem_type eq 'Characters');
-
- if ($elem_type eq 'CODEBASE')
- {
- # Get the reference to the codebase out of our attributes.
- $ImplCodebase = $elem->{HREF};
- }
- elsif ($elem_type eq 'DEPENDENCY')
- {
- # Get the name of any dependencies we have out of our attributes.
- # Dependencies in old PPDs might not have version info.
- $ImplDepend{$elem->{NAME}} = (defined $elem->{VERSION} && $elem->{VERSION} ne "") ? $elem->{VERSION} : "0,0,0,0";
- }
- elsif ($elem_type eq 'LANGUAGE')
- {
- # Get the language out of our attributes (if we don't already have
- # the right one).
- if ($ImplLanguage && ($ImplLanguage ne $LANGUAGE))
- { $ImplLanguage = $elem->{VALUE}; }
- }
- elsif ($elem_type eq 'ARCHITECTURE') {
- $ImplArch = $elem->{VALUE};
- }
- elsif ($elem_type eq 'OS')
- {
- # Get the OS value out of our attribute.
- $ImplOS = $elem->{VALUE};
- }
- elsif ($elem_type eq 'OSVERSION')
- {
- # Get the OS version value out of our attribute
- $ImplOSVersion = $elem->{VALUE};
- }
- elsif ($elem_type eq 'PERLCORE')
- {
- # Get the compiled Perl core value out of our attributes
- $ImplPerlCoreVer = $elem->{VERSION};
- }
- elsif ($elem_type eq 'PROCESSOR')
- {
- # Get the processor value out of our attribute
- $ImplProcessor = $elem->{VALUE};
- }
- elsif ($elem_type eq 'INSTALL')
- {
- # Get anything which might have been an attribute
- $ImplInstallHREF = $elem->{HREF};
- $ImplInstallEXEC = $elem->{EXEC};
- # Get any raw Perl script out of here (if we've got any)
- if ( (exists $elem->{Kids}) and (exists $elem->{Kids}[0]{Text}) )
- { $ImplInstallScript = $elem->{Kids}[0]{Text}; }
- }
- elsif ($elem_type eq 'UNINSTALL')
- {
- # Get anything which might have been an attribute
- $ImplUninstallHREF = $elem->{HREF};
- $ImplUninstallEXEC = $elem->{EXEC};
- # Get any raw Perl script out of here (if we've got any)
- if ( (exists $elem->{Kids}) and (exists $elem->{Kids}[0]{Text}) )
- { $ImplUninstallScript = $elem->{Kids}[0]{Text}; }
- }
- else
- {
- die "Unknown element '$elem_type' found inside of IMPLEMENTATION.";
- }
- } # end of 'for every element inside IMPLEMENTATION'
-
- # Check to see if we've found a valid IMPLEMENTATION for the target
- # machine.
- if ( (defined $ImplArch) and ($ImplArch ne $Config{'archname'}) )
- { return 0; }
- if ( (defined $ImplProcessor) and ($ImplProcessor ne $CPU) )
- { return 0; }
- if ( (defined $ImplLanguage) and ($ImplLanguage ne $LANGUAGE) )
- { return 0; }
- if ( (defined $ImplOS) and ($ImplOS ne $OS_VALUE) )
- { return 0; }
- if ( (defined $ImplOSVersion) and ($ImplOSVersion ne $OS_VERSION) )
- { return 0; }
-
- # Got a valid IMPLEMENTATION, stuff all the values we just read in into the
- # 'current package' global var.
- $current_package{PERLCORE_VER} = $ImplPerlCoreVer
- if (defined $ImplPerlCoreVer);
- $current_package{CODEBASE} = $ImplCodebase
- if (defined $ImplCodebase);
- $current_package{INSTALL_HREF} = $ImplInstallHREF
- if (defined $ImplInstallHREF);
- $current_package{INSTALL_EXEC} = $ImplInstallEXEC
- if (defined $ImplInstallEXEC);
- $current_package{INSTALL_SCRIPT} = $ImplInstallScript
- if (defined $ImplInstallScript);
- $current_package{UNINSTALL_HREF} = $ImplUninstallHREF
- if (defined $ImplUninstallHREF);
- $current_package{UNINSTALL_EXEC} = $ImplUninstallEXEC
- if (defined $ImplUninstallEXEC);
- $current_package{UNINSTALL_SCRIPT} = $ImplUninstallScript
- if (defined $ImplUninstallScript);
- %{$current_package{DEPEND}} = %ImplDepend
- if (%ImplDepend);
-
- return 1;
- }
-
- sub valid_URL_or_file
- {
- my ($File) = @_;
- if ($File =~ /^file:\/\/.*\|/i) {
- # $File is a local directory, let's avoid LWP by changing
- # it to a pathname.
- $File =~ s@^file://@@i;
- $File =~ s@^localhost/@@i;
- $File =~ s@\|@:@;
- }
- return 1 if (-f $File);
- return 1 if ($File =~ m@^...*://@i && defined read_href("href" => $File, "request" => 'HEAD'));
- &Trace("valid_URL_or_file: $File is not valid") if $Trace;
- return 0;
- }
-
- # Builds a fully qualified pathname or URL to a PPD for $Package.
- # If '$location' argument is given, that is used. Otherwise, the
- # '<LOCATION>' tag for a previously installed version is used, and
- # if that fails, the default locations are looked at.
- #
- # returns undef if it can't find a valid file or URL.
- #
- sub locatePPDfile
- {
- my ($Package, $location) = @_;
- my $PPDfile;
- if (defined($location)) {
- if ($location =~ /[^\/]$/) { $location .= "/"; }
- $PPDfile = $location . $Package . ".ppd";
- if (!valid_URL_or_file($PPDfile)) {
- # not good.
- undef $PPDfile;
- }
- }
- else {
- # Is $Package a filename or URL?
- if (valid_URL_or_file($Package)) {
- $PPDfile = $Package;
- }
- # does the package have a <LOCATION> in $PPM::PPMdat?
- elsif (defined( $installed_packages{$Package} )) {
- $location = $installed_packages{$Package}{'LOCATION'};
- if ($location =~ /[^\/]$/) { $location .= "/"; }
- $PPDfile = $location . $Package . ".ppd";
- if (!valid_URL_or_file($PPDfile)) {
- # not good.
- undef $PPDfile;
- }
- }
-
- if (!defined $PPDfile) {
- # No, try the defaults.
- my $location;
-
- foreach $_ (keys %repositories) {
- $location = $repositories{$_}{'LOCATION'};
- if ($location =~ /[^\/]$/) { $location .= "/"; }
- $PPDfile = $location . $Package . ".ppd";
-
- if (valid_URL_or_file($PPDfile)) {
- last;
- }
- undef $PPDfile;
- }
- }
- }
-
- # return the fully qualified pathname or HREF
- return $PPDfile;
- }
-
- # reads and parses $PPDfile, which can be a file or HREF
- sub readPPDfile
- {
- my ($PPDfile, $Pkg) = @_;
- my (@DATA, $CONTENTS);
- $Pkg = 'XML::PPD' unless (defined $Pkg);
-
- if ($PPDfile =~ /^file:\/\/.*\|/i) {
- # $PPDfile is a local directory, let's avoid LWP by changing
- # it to a pathname.
- $PPDfile =~ s@^file://@@i;
- $PPDfile =~ s@^localhost/@@i;
- $PPDfile =~ s@\|@:@;
- }
-
- # if $PPDfile is a SOAP request
- if ($PPDfile =~ m#^soap://#i) {
- my ($srvr, $pkg) = ($PPDfile =~ m@(.*)/(.+?)$@o);
- my $client = new PPM::SOAPClient( $srvr );
- $CONTENTS = $client->fetch_ppd( $pkg );
- }
- # if $PPDfile is an HREF
- elsif ($PPDfile =~ m@^...*://@i) {
- if (!(@DATA = read_href("href" => $PPDfile, "request" => 'GET'))) {
- &Trace("readPPDfile: read_href of $PPDfile failed") if $Trace;
- return undef;
- }
- $CONTENTS = join('', @DATA);
- } else {
- # else $PPDfile is a regular file
- if (!open (DATAFILE, $PPDfile)) {
- &Trace("readPPDfile: open of $PPDfile failed") if $Trace;
- $PPM::PPMERR = "open of $PPDfile failed: $!\n";
- return undef;
- }
- @DATA = <DATAFILE>;
- close(DATAFILE);
- $CONTENTS = join('', @DATA);
- }
-
- # take care of '&'
- $CONTENTS =~ s/&(?!\w+;)/&/go;
-
- # Got everything we need, no just parse the PPD file
- my $parser = new XML::Parser( Style => 'Objects', Pkg => $Pkg );
- my @parsed = @{ $parser->parse( $CONTENTS ) };
-
- # Validate what we've read in and output an error if something wasn't
- # parsed properly. As well, if we found any errors, return undef.
- if (!$parsed[0]->rvalidate( \&PPM::parse_err ))
- { return undef; }
-
- # Done, and things did seem to get parsed properly, return the structure.
- return %{$parsed[0]};
- }
-
- # Spits out the error from parsing, and sets our global error message
- # accordingly.
- sub parse_err
- {
- &Trace("parse_err: @_") if $Trace;
- warn @_;
- $PPM::PPMERR = 'Errors found while parsing document.';
- }
-
- # reads and parses the PPM data file $PPM::PPMdat. Stores config information in
- # $PPM_ver, $build_dir, %repositories, $CPU, $OS_VALUE, and $OS_VERSION.
- # Stores information about individual packages in the hash %installed_packages.
- sub read_config
- {
- if ($init) { return; }
- else { $init++; }
-
- my %PPMConfig = readPPDfile($PPM::PPMdat, 'XML::PPMConfig');
- if (!%PPMConfig) { return 0; }
-
- my $elem;
- foreach $elem (@{$PPMConfig{Kids}})
- {
- my $subelem = ref $elem;
- $subelem =~ s/.*:://;
- next if ($subelem eq 'Characters');
-
- if ($subelem eq 'PPMVER')
- {
- # Get the value out of our _only_ character data element.
- $PPM_ver = $elem->{Kids}[0]{Text};
- }
- elsif ($subelem eq 'PPMPRECIOUS')
- {
- # Get the value out of our _only_ character data element.
- @required_packages = split( ';', $elem->{Kids}[0]{Text} );
- }
- elsif ($subelem eq 'PLATFORM')
- {
- # Get values out of our attributes
- $CPU = $elem->{CPU};
- $OS_VALUE = $elem->{OSVALUE};
- $OS_VERSION = $elem->{OSVERSION};
- $LANGUAGE = $elem->{LANGUAGE};
- }
- elsif ($subelem eq 'REPOSITORY')
- {
- # Get a repository out of the element attributes
- my ($name);
- $name = $elem->{NAME};
- $repositories{ $name }{'LOCATION'} = $elem->{LOCATION};
- $repositories{ $name }{'USERNAME'} = $elem->{USERNAME};
- $repositories{ $name }{'PASSWORD'} = $elem->{PASSWORD};
- $repositories{ $name }{'SUMMARYFILE'} = $elem->{SUMMARYFILE};
- }
- elsif ($subelem eq 'OPTIONS')
- {
- # Get our options out of the element attributes
- $Ignorecase = $elem->{IGNORECASE};
- $Clean = $elem->{CLEAN};
- $Confirm = $elem->{CONFIRM};
- $Force_install = $elem->{FORCEINSTALL};
- $Root = $elem->{ROOT};
- $More = $elem->{MORE};
- $Trace = defined $elem->{TRACE} ? $elem->{TRACE} : "0";
- $TraceFile = defined $elem->{TRACEFILE} ? $elem->{TRACEFILE} : "PPM.LOG";
- $Verbose = defined $elem->{VERBOSE} ? $elem->{VERBOSE} : "1";
-
- #if (defined $Root)
- # { PPM::chroot( 'location' => $Root ); }
-
- $build_dir = $elem->{BUILDDIR};
- # Strip trailing separator
- my $chr = substr( $build_dir, -1, 1 );
- if ($chr eq '/' || $chr eq '\\')
- { chop $build_dir; }
-
- if ($Trace && !$TraceStarted) {
- $TraceFile = "PPM.log" if (!defined $TraceFile);
- open(PPMTRACE, ">>$TraceFile");
- my $oldfh = select(PPMTRACE);
- $| = 1;
- select($oldfh);
- &Trace("starting up...");
- $TraceStarted = 1;
- }
- }
- elsif ($subelem eq 'PACKAGE')
- {
- # Get our package name out of our attributes
- my $pkg = $elem->{NAME};
-
- # Gather the information on this package from the child elements.
- my ($loc, $instdate, $root, $packlist, $ppd);
- my $child;
- foreach $child (@{$elem->{Kids}})
- {
- my $child_type = ref $child;
- $child_type =~ s/.*:://;
- next if ($child_type eq 'Characters');
-
- if ($child_type eq 'LOCATION')
- { $loc = $child->{Kids}[0]{Text}; }
- elsif ($child_type eq 'INSTDATE')
- { $instdate = $child->{Kids}[0]{Text}; }
- elsif ($child_type eq 'INSTROOT')
- { $root = $child->{Kids}[0]{Text}; }
- elsif ($child_type eq 'INSTPACKLIST')
- { $packlist = $child->{Kids}[0]{Text}; }
- elsif ($child_type eq 'INSTPPD')
- {
- # Find the SOFTPKG inside here and hang onto it
- my $tmp;
- foreach $tmp (@{$child->{Kids}})
- {
- if ((ref $tmp) =~ /::SOFTPKG$/o)
- { $ppd = $tmp; }
- }
- }
- else
- {
- die "Unknown element inside of $pkg PACKAGE; $child";
- }
- } # end 'foreach $child...'
-
- my %package_details = (
- LOCATION => $loc,
- INST_DATE => $instdate,
- INST_ROOT => $root,
- INST_PACKLIST => $packlist,
- INST_PPD => $ppd
- );
- $installed_packages{ $pkg } = \%package_details;
- }
- else
- {
- die "Unknown element found in PPD_DAT file; $subelem";
- }
- }
- if ($Debug & 1) {
- print "This is ppm, version $PPM_ver.\nRepository locations:\n";
- foreach (keys %repositories) {
- print "\t$_: $repositories{$_}{'LOCATION'}\n"
- }
- print "Platform is $OS_VALUE version $OS_VERSION on a $CPU CPU.\n";
- print "Packages will be built in $build_dir\n";
- print "Commands will " . ($Confirm eq "Yes" ? "" : "not ") . "be confirmed.\n";
- print "Temporary files will " . ($Clean eq "Yes" ? "" : "not ") . "be deleted.\n";
- print "Installations will " . ($Force_install eq "Yes" ? "" : "not ") . "continue if a dependency cannot be installed.\n";
- print "Screens will " . ($More > 0 ? "pause after each $More lines.\n" : "not pause after the screen is full.\n");
- print "Tracing info will " . ($Trace > 0 ? "be written to $TraceFile.\n" : "not be written.\n");
- print "Case-" . ($Ignorecase eq "Yes" ? "in" : "") . "sensitive searches will be performed.\n";
-
- my $pkg;
- foreach $pkg (keys %installed_packages)
- {
- print "\nFound installed package $pkg, " .
- "installed on $installed_packages{$pkg}{INST_DATE}\n" .
- "in directory root $installed_packages{$pkg}{INST_ROOT} " .
- "from $installed_packages{$pkg}{'LOCATION'}.\n\n";
- }
- }
- }
-
- sub Trace
- {
- print PPMTRACE "$0: @_ at ", scalar localtime(), "\n";
- }
-
- 1;
-
- __END__
-
- =head1 NAME
-
- ppm - PPM (Perl Package Management)
-
- =head1 SYNOPSIS
-
- use PPM;
-
- PPM::InstallPackage("package" => $package, "location" => $location, "root" => $root);
- PPM::RemovePackage("package" => $package, "force" => $force);
- PPM::VerifyPackage("package" => $package, "location" => $location, "upgrade" => $upgrade);
- PPM::QueryInstalledPackages("searchRE" => $searchRE, "searchtag" => $searchtag, "ignorecase" => $ignorecase);
- PPM::InstalledPackageProperties();
- PPM::QueryPPD("location" => $location, "searchRE" => $searchRE, "searchtag" => $searchtag, "ignorecase" => $ignorecase, "package" => $package);
-
- PPM::ListOfRepositories();
- PPM::RemoveRepository("repository" => $repository, "save" => $save);
- PPM::AddRepository("repository" => $repository, "location" => $location, "save" => $save);
- PPM::RepositoryPackages("location" => $location);
- PPM::RepositoryPackageProperties("package" => $package, "location" => $location);
- PPM::RepositorySummary("location" => $location);
-
- PPM::GetPPMOptions();
- PPM::SetPPMOptions("options" => %options, "save" => $save);
-
- =head1 DESCRIPTION
-
- PPM is a group of functions intended to simplify the tasks of locating,
- installing, upgrading and removing software 'packages'. It can determine
- if the most recent version of a software package is installed on a system,
- and can install or upgrade that package from a local or remote host.
-
- PPM uses files containing a modified form of the Open Software Distribution
- (OSD) specification for information about software packages.
- These description files, which are written in Extensible Markup
- Language (XML) code, are referred to as 'PPD' files. Information about
- OSD can be found at the W3C web site (at the time of this writing,
- http://www.w3.org/TR/NOTE-OSD.html). The modifications to OSD used by PPM
- are documented in PPM::ppd.
-
- PPD files for packages are generated from POD files using the pod2ppd
- command.
-
- =head1 USAGE
-
- =over 4
-
- =item PPM::InstallPackage("package" => $package, "location" => $location, "root" => $root);
-
- Installs the specified package onto the local system. 'package' may
- be a simple package name ('foo'), a pathname (P:\PACKAGES\FOO.PPD) or
- a URL (HTTP://www.ActiveState.com/packages/foo.ppd). In the case of a
- simple package name, the function will look for the package's PPD file
- at 'location', if provided; otherwise, it will use information stored
- in the PPM data file (see 'Files' section below) to locate the PPD file
- for the requested package. The package's files will be installed under
- the directory specified in 'root'; if not specified the default value
- of 'root' will be used.
-
- The function uses the values stored in the PPM data file to determine the
- local operating system, operating system version and CPU type. If the PPD
- for this package contains implementations for different platforms, these
- values will be used to determine which one is installed.
-
- InstallPackage() updates the PPM data file with information about the package
- installation. It stores a copy of the PPD used for installation, as well
- as the location from which this PPD was obtained. This location will
- become the default PPD location for this package.
-
- During an installation, the following actions are performed:
-
- - the PPD file for the package is read
- - a directory for this package is created in the directory specified in
- <BUILDDIR> in the PPM data file.
- - the file specified with the <CODEBASE> tag in the PPD file is
- retrieved/copied into the directory created above.
- - the package is unarchived in the directory created for this package
- - individual files from the archive are installed in the appropriate
- directories of the local Perl installation.
- - perllocal.pod is updated with the install information.
- - if provided, the <INSTALL> script from the PPD is executed in the
- directory created above.
- - information about the installation is stored in the PPM data file.
-
- =item PPM::RemovePackage("package" => $package, "force" => $force)
-
- Removes the specified package from the system. Reads the package's PPD
- (stored during installation) for removal details. If 'force' is
- specified, even a package required by PPM will be removed (useful
- when installing an upgrade).
-
- =item PPM::VerifyPackage("package" => $package, "location" => $location, "upgrade" => $upgrade)
-
- Reads a PPD file for 'package', and compares the currently installed
- version of 'package' to the version available according to the PPD.
- The PPD file is expected to be on a local directory or remote site
- specified either in the PPM data file or in the 'location' argument.
- The 'location' argument may be a directory location or a URL.
- The 'upgrade' argument forces an upgrade if the installed package is
- not up-to-date.
-
- The PPD file for each package will initially be searched for at
- 'location', and if not found will then be searched for using the
- locations specified in the PPM data file.
-
- =item PPM::QueryInstalledPackages("searchRE" => $searchRE, "searchtag" => $searchtag, "ignorecase" => $ignorecase);
-
- Returns a hash containing information about all installed packages.
- By default, a list of all installed packages is returned. If a regular
- expression 'searchRE' is specified, only packages matching it are
- returned. If 'searchtag' is specified, the pattern match is applied
- to the appropriate tag (e.g., ABSTRACT).
-
- The data comes from the PPM data file, which contains installation
- information about each installed package.
-
- =item PPM::InstalledPackageProperties();
-
- Returns a hash with package names as keys, and package properties as
- attributes.
-
- =item PPM::QueryPPD("location" => $location, "searchRE" => $searchRE, "searchtag" => $searchtag, "ignorecase" => $ignorecase, "package" => $package);
-
- Searches for 'searchRE' (a regular expression) in the <ABSTRACT>,
- <AUTHOR> or <TITLE> tags of the PPD file for 'package' at 'location'.
- 'location' may be either a remote address or a directory path, and if
- it is not provided, the default location as specified in the PPM data
- file will be used. If the 'ignorecase' option is specified, it overrides
- the current global case-sensitivity setting. On success, the matching
- string is returned.
-
- =item PPM::RepositoryPackages("location" => $location);
-
- Returns a hash, with 'location' being the key, and arrays of all packages
- with package description (PPD) files available at 'location' as its
- elements. 'location' may be either a remote address or a directory path.
- If 'location' is not specified, the default location as specified in
- the PPM data file will be used.
-
- =item PPM::ListOfRepositories();
-
- Returns a hash containing the name of the repository and its location.
- These repositories will be searched if an explicit location is not
- provided in any function needing to locate a PPD.
-
- =item PPM::RemoveRepository("repository" => $repository, "save" => $save);
-
- Removes the repository named 'repository' from the list of available
- repositories. If 'save' is not specified, the change is for the current
- session only.
-
- =item PPM::AddRepository("repository" => $repository, "location" => $location, "save" => $save);
-
- Adds the repository named 'repository' to the list of available repositories.
- If 'save' is not specified, the change is for the current session only.
-
- =item PPM::RepositoryPackageProperties("package" => $package, "location" => $location);
-
- Reads the PPD file for 'package', from 'location' or the default repository,
- and returns a hash with keys being the various tags from the PPD (e.g.
- 'ABSTRACT', 'AUTHOR', etc.).
-
- =item PPM::RepositorySummary("location" => $location);
-
- Attempts to retrieve the summary file associated with the specified repository,
- or from all repositories if 'location' is not specified. The return value
- is a hash with the key being the repository, and the data being another
- hash of package name keys, and package detail data.
-
- =item PPM::GetPPMOptions();
-
- Returns a hash containing values for all PPM internal options ('IGNORECASE',
- 'CLEAN', 'CONFIRM', 'ROOT', 'BUILDDIR').
-
- =item PPM::SetPPMOptions("options" => %options, "save" => $save);
-
- Sets internal PPM options as specified in the 'options' hash, which is
- expected to be the hash previously returned by a call to GetPPMOptions().
-
- =back
-
- =head1 EXAMPLES
-
- =over 4
-
- =item PPM::AddRepository("repository" => 'ActiveState', "location" => "http://www.ActiveState.com/packages", "save" => 1);
-
- Adds a repository to the list of available repositories, and saves it in
- the PPM options file.
-
- =item PPM::InstallPackage("package" => 'http://www.ActiveState.com/packages/foo.ppd');
-
- Installs the software package 'foo' based on the information in the PPD
- obtained from the specified URL.
-
- =item PPM::VerifyPackage("package" => 'foo', "upgrade" => true)
-
- Compares the currently installed version of the software package 'foo' to
- the one available according to the PPD obtained from the package-specific
- location provided in the PPM data file, and upgrades to a newer
- version if available. If a location for this specific package is not
- given in PPM data file, a default location is searched.
-
- =item PPM::VerifyPackage("package" => 'foo', "location" => 'P:\PACKAGES', "upgrade" => true);
-
- Compares the currently installed version of the software package 'foo'
- to the one available according to the PPD obtained from the specified
- directory, and upgrades to a newer version if available.
-
- =item PPM::VerifyPackage("package" => 'PerlDB');
-
- Verifies that package 'PerlDB' is up to date, using package locations specified
- in the PPM data file.
-
- =item PPM::RepositoryPackages("location" => http://www.ActiveState.com/packages);
-
- Returns a hash keyed on 'location', with its elements being an array of
- packages with PPD files available at the specified location.
-
- =item PPM::QueryPPD("location" => 'P:\PACKAGES', "searchRE" => 'ActiveState', "searchtag" => 'author');
-
- Searches the specified location for any package with an <AUTHOR> tag
- containing the string 'ActiveState'. On a successful search, the matching
- string is returned.
-
- =item %opts = PPM::GetPPMOptions();
-
- =item $opts{'CONFIRM'} = 'No';
-
- =item PPM::SetPPMOptions("options" => \%opts, "save" => 1);
-
- Sets and saves the value of the option 'CONFIRM' to 'No'.
-
- =back
-
- =head1 ENVIRONMENT VARIABLES
-
- =over 4
-
- =item HTTP_proxy
-
- If the environment variable 'HTTP_proxy' is set, then it will
- be used as the address of a proxy for accessing the Internet.
- If the environment variables 'HTTP_proxy_user' and 'HTTP_proxy_pass'
- are set, they will be used as the login and password for the
- proxy server.
-
- =back
-
- =head1 FILES
-
- =over 4
-
- =item package.ppd
-
- A description of a software package, in Perl Package Distribution (PPD)
- format. More information on this file format can be found in L<XML::PPD>.
- PPM stores a copy of the PPD it uses to install or upgrade any software
- package.
-
- =item ppm.xml - PPM data file.
-
- The XML format file in which PPM stores configuration and package
- installation information. This file is created when PPM is installed,
- and under normal circumstances should never require modification other
- than by PPM itself. For more information on this file, refer to
- L<XML::PPMConfig>.
-
- =back
-
- =head1 AUTHOR
-
- Murray Nesbitt, E<lt>F<murray@activestate.com>E<gt>
-
- =head1 SEE ALSO
-
- L<XML::PPMConfig>
- .
-
- =cut
-
-