#!/usr/local/cpanel/3rdparty/bin/perl
# cpanel - scripts/fix_pear_registry Copyright 2022 cPanel, L.L.C.
# All rights reserved.
# copyright@cpanel.net http://cpanel.net
# This code is subject to the cPanel license. Unauthorized copying is prohibited
use strict;
use Cpanel::Imports;
use Cpanel::Binaries ();
use Cpanel::Pkgr ();
use Cpanel::SafeRun::Object ();
# This script is intended to clean up after the Horde and other PEAR RPMs that had
# a faulty %postun. This has since been fixed, but the script is needed as a one-time
# operation to repair the PEAR registry, since the %postun of the old RPM is executed
# after the new RPM's %post has already run. This could be handled with a %triggerpostun,
# but that would add a new element to the long-term behavior of the RPMs, which is
# undesirable for this one-time cleanup measure.
exit run(@ARGV) unless caller;
sub run {
my %report;
my $pear_bin = Cpanel::Binaries::CPANEL_PHP_BIN() . '/pear'; # Don't use Cpanel::FindBin for this. It may grab the system pear, which we don't want.
my $php_rpm_prefix = 'cpanel-php' . Cpanel::Binaries::get_php_version( flat => 1 ) . '*';
my @installed_pear_packages = keys( %{ Cpanel::Pkgr::installed_packages($php_rpm_prefix) } );
$report{installed} = @installed_pear_packages;
for my $pkg ( sort @installed_pear_packages ) {
my $xmlfile = Cpanel::Binaries::CPANEL_PHP . '/lib/pear/' . $pkg . '.xml';
if ( -f $xmlfile ) {
my $package_is_registered = is_registered($pkg);
if ($package_is_registered) {
logger->info( locale->maketext( 'The “[_1]” package is already in the [asis,PEAR] registry.', $pkg ) );
$report{already_registered}++;
next;
}
$report{not_already_registered}++;
my @pear_args = ( qw(install --nodeps --force --register-only), $xmlfile );
my $run = Cpanel::SafeRun::Object->new(
program => $pear_bin,
args => \@pear_args,
);
if ( $run->CHILD_ERROR || !is_registered($pkg) ) {
logger->warn( locale->maketext( 'The system failed to add the [asis,PEAR] registry entry for “[_1]”: [_2]', $pkg, $run->stdout ) ); # pear errors are on stdout, and pear may exit zero without actually doing anything
logger->warn("Command: $pear_bin @pear_args");
$report{failed}++;
}
else {
print $run->stdout;
logger->info( locale->maketext( 'The system added the [asis,PEAR] registry entry for “[_1]”.', $pkg ) );
$report{succeeded}++;
}
}
else {
logger->info( locale->maketext( 'The system could not find the [asis,XML] file for “[_1]”.', $pkg ) );
$report{no_xml}++;
}
}
logger->info(
locale->maketext(
'There are [numf,_1] installed [asis,PEAR] packages. Of those packages, [numf,_2] already existed in the [asis,PEAR] registry, [numf,_3] did not, and [numf,_4] did not contain an [asis,XML] file. Of the [quant,_3,package,packages] not listed in the registry, the system successfully added [numf,_5], and failed to add [numf,_6].',
map { $_ || 0 } @report{qw(installed already_registered not_already_registered no_xml succeeded failed )}
)
);
# We can't count on every single PEAR package to always register successfully, so only exit nonzero if something
# very bad like the pear binary being missing occurred.
return 0;
}
sub is_registered {
my ($pkg) = @_;
my $pear_bin = Cpanel::Binaries::THIRD_PARTY_BIN . '/pear';
return scalar grep {
my $channel = $_;
my @pear_args = ( qw(list -c horde), "$channel/$pkg" );
my $run = Cpanel::SafeRun::Object->new(
program => $pear_bin,
args => \@pear_args,
);
!$run->CHILD_ERROR;
} 'horde', 'pear';
}